Index: sys/conf/files =================================================================== RCS file: /cvsroot/src/sys/conf/files,v retrieving revision 1.1103 diff -p -u -r1.1103 files --- sys/conf/files 8 Sep 2014 14:26:16 -0000 1.1103 +++ sys/conf/files 14 Sep 2014 19:59:04 -0000 @@ -1138,9 +1138,6 @@ define drm {} include "external/bsd/drm/conf/files.drm" include "dev/drm/files.drm" -# DRM/KMS - Newer direct rendering manager with kernel mode-switching -include "external/bsd/drm2/drm/files.drmkms" - # Definitions for wscons # device attributes: display, display with emulator, keyboard, and mouse @@ -1195,6 +1192,9 @@ include "dev/rasops/files.rasops" include "dev/wsfont/files.wsfont" include "dev/wsfb/files.wsfb" +# DRM/KMS - Newer direct rendering manager with kernel mode-switching +include "external/bsd/drm2/drm/files.drmkms" + # Attributes which machine-independent bus support can be attached to. # These should be defined here, because some of these busses can have # devices which provide these attributes, and we'd like to avoid hairy Index: sys/external/bsd/drm2/drm/drmfb.c =================================================================== RCS file: sys/external/bsd/drm2/drm/drmfb.c diff -N sys/external/bsd/drm2/drm/drmfb.c --- /dev/null 1 Jan 1970 00:00:00 -0000 +++ sys/external/bsd/drm2/drm/drmfb.c 14 Sep 2014 19:59:06 -0000 @@ -0,0 +1,290 @@ +/* $NetBSD$ */ + +/*- + * Copyright (c) 2014 The NetBSD Foundation, Inc. + * All rights reserved. + * + * This code is derived from software contributed to The NetBSD Foundation + * by Taylor R. Campbell. + * + * Redistribution and use in source and binary forms, with or without + * modification, are permitted provided that the following conditions + * are met: + * 1. Redistributions of source code must retain the above copyright + * notice, this list of conditions and the following disclaimer. + * 2. Redistributions in binary form must reproduce the above copyright + * notice, this list of conditions and the following disclaimer in the + * documentation and/or other materials provided with the distribution. + * + * THIS SOFTWARE IS PROVIDED BY THE NETBSD FOUNDATION, INC. AND CONTRIBUTORS + * ``AS IS'' AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED + * TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR + * PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE FOUNDATION OR CONTRIBUTORS + * BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR + * CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF + * SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS + * INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN + * CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) + * ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE + * POSSIBILITY OF SUCH DAMAGE. + */ + +/* + * drmfb: wsdisplay support, via genfb, for any drm device. Use this + * if you're too lazy to write a hardware-accelerated framebuffer using + * wsdisplay directly. + */ + +#include +__KERNEL_RCSID(0, "$NetBSD$"); + +#ifdef _KERNEL_OPT +#include "vga.h" +#endif + +#include +#include +#include +#include +#include + +#if NVGA > 0 +/* + * XXX All we really need is vga_is_console from vgavar.h, but the + * header files are missing their own dependencies, so we need to + * explicitly drag in the other crap. + */ +#include +#include +#include +#include +#endif + +#include + +#include +#include +#include + +static int drmfb_genfb_ioctl(void *, void *, unsigned long, void *, int, + struct lwp *); +static paddr_t drmfb_genfb_mmap(void *, void *, off_t, int); +static int drmfb_genfb_enable_polling(void *); +static int drmfb_genfb_disable_polling(void *); +static bool drmfb_genfb_setmode(struct genfb_softc *, int); + +static const struct genfb_mode_callback drmfb_genfb_mode_callback = { + .gmc_setmode = drmfb_genfb_setmode, +}; + +int +drmfb_attach(struct drmfb_softc *sc, const struct drmfb_attach_args *da) +{ + const struct drm_fb_helper_surface_size *const sizes = da->da_fb_sizes; + const prop_dictionary_t dict = device_properties(da->da_dev); +#if NVGA > 0 + struct drm_device *const dev = da->da_fb_helper->dev; +#endif + static const struct genfb_ops zero_genfb_ops; + struct genfb_ops genfb_ops = zero_genfb_ops; + enum { CONS_VGA, CONS_GENFB, CONS_NONE } what_was_cons; + int error; + + /* genfb requires this. */ + KASSERTMSG((void *)&sc->sc_genfb == device_private(da->da_dev), + "drmfb_softc must be first member of device softc"); + + sc->sc_da = *da; + + prop_dictionary_set_uint32(dict, "width", sizes->surface_width); + prop_dictionary_set_uint32(dict, "height", sizes->surface_height); + prop_dictionary_set_uint8(dict, "depth", sizes->surface_bpp); + prop_dictionary_set_uint16(dict, "linebytes", + roundup2((sizes->surface_width * howmany(sizes->surface_bpp, 8)), + 64)); + prop_dictionary_set_uint32(dict, "address", 0); /* XXX >32-bit */ + CTASSERT(sizeof(uintptr_t) <= sizeof(uint64_t)); + prop_dictionary_set_uint64(dict, "virtual_address", + (uint64_t)(uintptr_t)da->da_fb_vaddr); + + prop_dictionary_set_uint64(dict, "mode_callback", + (uint64_t)(uintptr_t)&drmfb_genfb_mode_callback); + + /* XXX Whattakludge! */ +#if NVGA > 0 + if ((da->da_params->dp_is_vga_console != NULL) && + (*da->da_params->dp_is_vga_console)(dev)) { + what_was_cons = CONS_VGA; + prop_dictionary_set_bool(dict, "is_console", true); + vga_cndetach(); + if (da->da_params->dp_disable_vga) + (*da->da_params->dp_disable_vga)(dev); + } else +#endif + if (genfb_is_console() && genfb_is_enabled()) { + what_was_cons = CONS_GENFB; + prop_dictionary_set_bool(dict, "is_console", true); + } else { + what_was_cons = CONS_NONE; + prop_dictionary_set_bool(dict, "is_console", false); + } + + sc->sc_genfb.sc_dev = sc->sc_da.da_dev; + genfb_init(&sc->sc_genfb); + genfb_ops.genfb_ioctl = drmfb_genfb_ioctl; + genfb_ops.genfb_mmap = drmfb_genfb_mmap; + genfb_ops.genfb_enable_polling = drmfb_genfb_enable_polling; + genfb_ops.genfb_disable_polling = drmfb_genfb_disable_polling; + + error = genfb_attach(&sc->sc_genfb, &genfb_ops); + if (error) { + aprint_error_dev(sc->sc_da.da_dev, + "failed to attach genfb: %d\n", error); + goto fail0; + } + + /* Success! */ + return 0; + +fail0: KASSERT(error); + /* XXX Restore console... */ + switch (what_was_cons) { + case CONS_VGA: + break; + case CONS_GENFB: + break; + case CONS_NONE: + break; + default: + break; + } + return error; +} + +int +drmfb_detach(struct drmfb_softc *sc, int flags) +{ + + /* XXX genfb detach? */ + return 0; +} + +static int +drmfb_genfb_ioctl(void *v, void *vs, unsigned long cmd, void *data, int flag, + struct lwp *l) +{ + struct genfb_softc *const genfb = v; + struct drmfb_softc *const sc = container_of(genfb, struct drmfb_softc, + sc_genfb); + int error; + + if (sc->sc_da.da_params->dp_ioctl) { + error = (*sc->sc_da.da_params->dp_ioctl)(sc, cmd, data, flag, + l); + if (error != EPASSTHROUGH) + return error; + } + + switch (cmd) { + /* + * Screen blanking ioctls. Not to be confused with backlight + * (can be disabled while stuff is still drawn on the screen), + * brightness, or contrast (which we don't support). Backlight + * and brightness are done through WSDISPLAYIO_{GET,SET}PARAM. + * This toggles between DPMS ON and DPMS OFF; backlight toggles + * between DPMS ON and DPMS SUSPEND. + */ + case WSDISPLAYIO_GVIDEO: { + int *onp = (int *)data; + + /* XXX Can't really determine a single answer here. */ + *onp = 1; + return 0; + } + case WSDISPLAYIO_SVIDEO: { + const int on = *(const int *)data; + const int dpms_mode = on? DRM_MODE_DPMS_ON : DRM_MODE_DPMS_OFF; + struct drm_fb_helper *const fb_helper = sc->sc_da.da_fb_helper; + struct drm_device *const dev = fb_helper->dev; + unsigned i; + + drm_modeset_lock_all(dev); + for (i = 0; i < fb_helper->connector_count; i++) { + struct drm_connector *const connector = + fb_helper->connector_info[i]->connector; + (*connector->funcs->dpms)(connector, dpms_mode); + drm_object_property_set_value(&connector->base, + dev->mode_config.dpms_property, dpms_mode); + } + drm_modeset_unlock_all(dev); + + return 0; + } + default: + return EPASSTHROUGH; + } +} + +static paddr_t +drmfb_genfb_mmap(void *v, void *vs, off_t offset, int prot) +{ + struct genfb_softc *const genfb = v; + struct drmfb_softc *const sc = container_of(genfb, struct drmfb_softc, + sc_genfb); + + KASSERT(0 <= offset); + + if (offset < genfb->sc_fbsize) { + if (sc->sc_da.da_params->dp_mmapfb == NULL) + return -1; + return (*sc->sc_da.da_params->dp_mmapfb)(sc, offset, prot); + } else { + if (kauth_authorize_machdep(kauth_cred_get(), + KAUTH_MACHDEP_UNMANAGEDMEM, NULL, NULL, NULL, NULL) + != 0) + return -1; + if (sc->sc_da.da_params->dp_mmap == NULL) + return -1; + return (*sc->sc_da.da_params->dp_mmap)(sc, offset, prot); + } +} + +static int +drmfb_genfb_enable_polling(void *cookie) +{ + struct genfb_softc *const genfb = cookie; + struct drmfb_softc *const sc = container_of(genfb, struct drmfb_softc, + sc_genfb); + + return drm_fb_helper_debug_enter_fb(sc->sc_da.da_fb_helper); +} + +static int +drmfb_genfb_disable_polling(void *cookie) +{ + struct genfb_softc *const genfb = cookie; + struct drmfb_softc *const sc = container_of(genfb, struct drmfb_softc, + sc_genfb); + + return drm_fb_helper_debug_leave_fb(sc->sc_da.da_fb_helper); +} + +static bool +drmfb_genfb_setmode(struct genfb_softc *genfb, int mode) +{ + struct drmfb_softc *sc = container_of(genfb, struct drmfb_softc, + sc_genfb); + + if (mode == WSDISPLAYIO_MODE_EMUL) + drm_fb_helper_set_config(sc->sc_da.da_fb_helper); + + return true; +} + +bool +drmfb_shutdown(struct drmfb_softc *sc, int flags __unused) +{ + + genfb_enable_polling(sc->sc_da.da_dev); + return true; +} Index: sys/external/bsd/drm2/drm/files.drmkms =================================================================== RCS file: /cvsroot/src/sys/external/bsd/drm2/drm/files.drmkms,v retrieving revision 1.8 diff -p -u -r1.8 files.drmkms --- sys/external/bsd/drm2/drm/files.drmkms 14 Sep 2014 19:06:00 -0000 1.8 +++ sys/external/bsd/drm2/drm/files.drmkms 14 Sep 2014 19:59:06 -0000 @@ -66,4 +66,8 @@ file external/bsd/drm2/drm/drm_vma_manag file external/bsd/drm2/drm/drm_gem_vm.c drmkms file external/bsd/drm2/drm/drm_module.c drmkms +# Generic, unaccelerated drm-based framebuffer. +define drmfb: genfb +file external/bsd/drm2/drm/drmfb.c drmfb + include "external/bsd/drm2/ttm/files.ttm" Index: sys/external/bsd/drm2/i915drm/files.i915drmkms =================================================================== RCS file: /cvsroot/src/sys/external/bsd/drm2/i915drm/files.i915drmkms,v retrieving revision 1.6 diff -p -u -r1.6 files.i915drmkms --- sys/external/bsd/drm2/i915drm/files.i915drmkms 18 Aug 2014 16:55:34 -0000 1.6 +++ sys/external/bsd/drm2/i915drm/files.i915drmkms 14 Sep 2014 19:59:06 -0000 @@ -4,7 +4,7 @@ define intelfbbus { } device i915drmkms: drmkms, drmkms_pci, intelfbbus attach i915drmkms at pci -device intelfb: intelfbbus, genfb, wsemuldisplaydev +device intelfb: intelfbbus, drmfb, drmfb_pci, wsemuldisplaydev attach intelfb at intelfbbus makeoptions i915drmkms CPPFLAGS+="-I$S/external/bsd/drm2/dist/drm/i915" Index: sys/external/bsd/drm2/i915drm/intelfb.c =================================================================== RCS file: /cvsroot/src/sys/external/bsd/drm2/i915drm/intelfb.c,v retrieving revision 1.10 diff -p -u -r1.10 intelfb.c --- sys/external/bsd/drm2/i915drm/intelfb.c 14 Sep 2014 16:02:15 -0000 1.10 +++ sys/external/bsd/drm2/i915drm/intelfb.c 14 Sep 2014 19:59:06 -0000 @@ -1,4 +1,4 @@ -/* $NetBSD: intelfb.c,v 1.10 2014/09/14 16:02:15 riastradh Exp $ */ +/* $NetBSD$ */ /*- * Copyright (c) 2014 The NetBSD Foundation, Inc. @@ -30,7 +30,7 @@ */ #include -__KERNEL_RCSID(0, "$NetBSD: intelfb.c,v 1.10 2014/09/14 16:02:15 riastradh Exp $"); +__KERNEL_RCSID(0, "$NetBSD$"); #ifdef _KERNEL_OPT #include "vga.h" @@ -40,13 +40,6 @@ __KERNEL_RCSID(0, "$NetBSD: intelfb.c,v #include #include -#include -#include -#include - -#include -#include - #if NVGA > 0 /* * XXX All we really need is vga_is_console from vgavar.h, but the @@ -60,41 +53,41 @@ __KERNEL_RCSID(0, "$NetBSD: intelfb.c,v #endif #include -#include +#include "drmfb.h" +#include "drmfb_pci.h" #include "i915_drv.h" #include "i915_pci.h" #include "intelfb.h" +static int intelfb_match(device_t, cfdata_t, void *); +static void intelfb_attach(device_t, device_t, void *); +static int intelfb_detach(device_t, int); + +static void intelfb_attach_task(struct i915drmkms_task *); + +static bool intel_is_vga_console(struct drm_device *); +static bool intelfb_shutdown(device_t, int); + +static paddr_t intelfb_drmfb_mmapfb(struct drmfb_softc *, off_t, int); + struct intelfb_softc { - /* XXX genfb requires the genfb_softc to be first. */ - struct genfb_softc sc_genfb; + struct drmfb_softc sc_drmfb; /* XXX Must be first. */ device_t sc_dev; struct intelfb_attach_args sc_ifa; bus_space_handle_t sc_fb_bsh; - struct i915drmkms_task sc_setconfig_task; + struct i915drmkms_task sc_attach_task; bool sc_mapped:1; bool sc_scheduled:1; bool sc_attached:1; }; -static int intelfb_match(device_t, cfdata_t, void *); -static void intelfb_attach(device_t, device_t, void *); -static int intelfb_detach(device_t, int); - -static void intelfb_setconfig_task(struct i915drmkms_task *); - -static int intelfb_genfb_dpms(struct drm_device *, int); -static int intelfb_genfb_ioctl(void *, void *, unsigned long, void *, - int, struct lwp *); -static paddr_t intelfb_genfb_mmap(void *, void *, off_t, int); -static int intelfb_genfb_enable_polling(void *); -static int intelfb_genfb_disable_polling(void *); -static bool intelfb_genfb_shutdown(device_t, int); -static bool intelfb_genfb_setmode(struct genfb_softc *, int); - -static const struct genfb_mode_callback intelfb_genfb_mode_callback = { - .gmc_setmode = intelfb_genfb_setmode, +static const struct drmfb_params intelfb_drmfb_params = { + .dp_mmapfb = intelfb_drmfb_mmapfb, + .dp_mmap = drmfb_pci_mmap, + .dp_ioctl = drmfb_pci_ioctl, + .dp_is_vga_console = intel_is_vga_console, + .dp_disable_vga = i915_disable_vga, }; CFATTACH_DECL_NEW(intelfb, sizeof(struct intelfb_softc), @@ -126,7 +119,7 @@ intelfb_attach(device_t parent, device_t /* XXX Defer this too? */ error = bus_space_map(ifa->ifa_fb_bst, ifa->ifa_fb_addr, ifa->ifa_fb_size, - (BUS_SPACE_MAP_LINEAR | BUS_SPACE_MAP_PREFETCHABLE), + BUS_SPACE_MAP_LINEAR|BUS_SPACE_MAP_PREFETCHABLE, &sc->sc_fb_bsh); if (error) { aprint_error_dev(self, "unable to map framebuffer: %d\n", @@ -135,8 +128,8 @@ intelfb_attach(device_t parent, device_t } sc->sc_mapped = true; - i915drmkms_task_init(&sc->sc_setconfig_task, &intelfb_setconfig_task); - error = i915drmkms_task_schedule(parent, &sc->sc_setconfig_task); + i915drmkms_task_init(&sc->sc_attach_task, &intelfb_attach_task); + error = i915drmkms_task_schedule(parent, &sc->sc_attach_task); if (error) { aprint_error_dev(self, "failed to schedule mode set: %d\n", error); @@ -156,12 +149,20 @@ static int intelfb_detach(device_t self, int flags) { struct intelfb_softc *const sc = device_private(self); + int error; if (sc->sc_scheduled) return EBUSY; if (sc->sc_attached) { - /* XXX genfb detach? Help? */ + pmf_device_deregister(self); + error = drmfb_detach(&sc->sc_drmfb, flags); + if (error) { + /* XXX Ugh. */ + (void)pmf_device_register1(self, NULL, NULL, + &intelfb_shutdown); + return error; + } sc->sc_attached = false; } @@ -175,268 +176,64 @@ intelfb_detach(device_t self, int flags) } static void -intelfb_setconfig_task(struct i915drmkms_task *task) +intelfb_attach_task(struct i915drmkms_task *task) { struct intelfb_softc *const sc = container_of(task, - struct intelfb_softc, sc_setconfig_task); - const prop_dictionary_t dict = device_properties(sc->sc_dev); + struct intelfb_softc, sc_attach_task); const struct intelfb_attach_args *const ifa = &sc->sc_ifa; - const struct drm_fb_helper_surface_size *const sizes = - &ifa->ifa_fb_sizes; -#if NVGA > 0 /* XXX no workie for modules */ - struct drm_device *const dev = sc->sc_ifa.ifa_drm_dev; -#endif - enum { CONS_VGA, CONS_GENFB, CONS_NONE } what_was_cons; - static const struct genfb_ops zero_genfb_ops; - struct genfb_ops genfb_ops = zero_genfb_ops; + const struct drmfb_attach_args da = { + .da_dev = sc->sc_dev, + .da_fb_helper = ifa->ifa_fb_helper, + .da_fb_sizes = &ifa->ifa_fb_sizes, + .da_fb_vaddr = bus_space_vaddr(ifa->ifa_fb_bst, sc->sc_fb_bsh), + .da_params = &intelfb_drmfb_params, + }; int error; - KASSERT(sc->sc_scheduled); - - if (ifa->ifa_fb_zero) - bus_space_set_region_1(sc->sc_ifa.ifa_fb_bst, sc->sc_fb_bsh, 0, - 0, sc->sc_ifa.ifa_fb_size); - - /* XXX Ugh... Pass these parameters some other way! */ - prop_dictionary_set_uint32(dict, "width", sizes->surface_width); - prop_dictionary_set_uint32(dict, "height", sizes->surface_height); - prop_dictionary_set_uint8(dict, "depth", sizes->surface_bpp); - prop_dictionary_set_uint16(dict, "linebytes", - roundup2((sizes->surface_width * howmany(sizes->surface_bpp, 8)), - 64)); - prop_dictionary_set_uint32(dict, "address", 0); /* XXX >32-bit */ - CTASSERT(sizeof(uintptr_t) <= sizeof(uint64_t)); - prop_dictionary_set_uint64(dict, "virtual_address", - (uint64_t)(uintptr_t)bus_space_vaddr(sc->sc_ifa.ifa_fb_bst, - sc->sc_fb_bsh)); - - prop_dictionary_set_uint64(dict, "mode_callback", - (uint64_t)(uintptr_t)&intelfb_genfb_mode_callback); - - /* XXX Whattakludge! */ -#if NVGA > 0 - if (vga_is_console(dev->pdev->pd_pa.pa_iot, -1)) { - what_was_cons = CONS_VGA; - prop_dictionary_set_bool(dict, "is_console", true); - vga_cndetach(); - i915_disable_vga(dev); - } else -#endif - if (genfb_is_console() && genfb_is_enabled()) { - what_was_cons = CONS_GENFB; - prop_dictionary_set_bool(dict, "is_console", true); - } else { - what_was_cons = CONS_NONE; - prop_dictionary_set_bool(dict, "is_console", false); - } - - sc->sc_genfb.sc_dev = sc->sc_dev; - genfb_init(&sc->sc_genfb); - genfb_ops.genfb_ioctl = intelfb_genfb_ioctl; - genfb_ops.genfb_mmap = intelfb_genfb_mmap; - genfb_ops.genfb_enable_polling = intelfb_genfb_enable_polling; - genfb_ops.genfb_disable_polling = intelfb_genfb_disable_polling; - - error = genfb_attach(&sc->sc_genfb, &genfb_ops); + error = drmfb_attach(&sc->sc_drmfb, &da); if (error) { - aprint_error_dev(sc->sc_dev, "failed to attach genfb: %d\n", + aprint_error_dev(sc->sc_dev, "failed to attach drmfb: %d\n", error); - goto fail0; + return; } - sc->sc_attached = true; - pmf_device_register1(sc->sc_dev, NULL, NULL, - intelfb_genfb_shutdown); + if (pmf_device_register1(sc->sc_dev, NULL, NULL, &intelfb_shutdown)) + aprint_error_dev(sc->sc_dev, + "failed to register shutdown handler\n"); - /* Success! */ - sc->sc_scheduled = false; - return; - -fail0: /* XXX Restore console... */ - switch (what_was_cons) { - case CONS_VGA: - break; - case CONS_GENFB: - break; - case CONS_NONE: - break; - default: - break; - } + sc->sc_attached = true; } -static int -intelfb_genfb_dpms(struct drm_device *dev, int dpms_mode) +static bool +intel_is_vga_console(struct drm_device *dev) { - struct drm_i915_private *const dev_priv = dev->dev_private; - /* XXX What guarantees dev_priv->fbdev stays around? */ - struct drm_fb_helper *const fb_helper = &dev_priv->fbdev->helper; - unsigned i; - - drm_modeset_lock_all(dev); - for (i = 0; i < fb_helper->connector_count; i++) { - struct drm_connector *const connector = - fb_helper->connector_info[i]->connector; - (*connector->funcs->dpms)(connector, dpms_mode); - drm_object_property_set_value(&connector->base, - dev->mode_config.dpms_property, dpms_mode); - } - drm_modeset_unlock_all(dev); - return 0; + return vga_is_console(dev->pdev->pd_pa.pa_iot, -1); } -static int -intelfb_genfb_ioctl(void *v, void *vs, unsigned long cmd, void *data, int flag, - struct lwp *l) +static bool +intelfb_shutdown(device_t self, int flags) { - struct genfb_softc *const genfb = v; - struct intelfb_softc *const sc = container_of(genfb, - struct intelfb_softc, sc_genfb); - struct drm_device *const dev = sc->sc_ifa.ifa_fb_helper->dev; - const struct pci_attach_args *const pa = &dev->pdev->pd_pa; - - switch (cmd) { - case WSDISPLAYIO_GTYPE: - *(unsigned int *)data = WSDISPLAY_TYPE_PCIVGA; - return 0; - - /* PCI config read/write passthrough. */ - case PCI_IOC_CFGREAD: - case PCI_IOC_CFGWRITE: - return pci_devioctl(pa->pa_pc, pa->pa_tag, cmd, data, flag, l); - - case WSDISPLAYIO_GET_BUSID: - return wsdisplayio_busid_pci(dev->dev, pa->pa_pc, pa->pa_tag, - data); - - /* - * Screen blanking ioctls. Not to be confused with backlight - * (can be disabled while stuff is still drawn on the screen), - * brightness, or contrast (which we don't support). Backlight - * and brightness are done through WSDISPLAYIO_{GET,SET}PARAM. - * This toggles between DPMS ON and DPMS OFF; backlight toggles - * between DPMS ON and DPMS SUSPEND. - */ - case WSDISPLAYIO_GVIDEO: { - int *onp = (int *)data; - - /* XXX Can't really determine a single answer here. */ - *onp = 1; - return 0; - } - - case WSDISPLAYIO_SVIDEO: { - const int on = *(const int *)data; - - return intelfb_genfb_dpms(dev, - on? DRM_MODE_DPMS_ON : DRM_MODE_DPMS_OFF); - } + struct intelfb_softc *const sc = device_private(self); - default: - return EPASSTHROUGH; - } + return drmfb_shutdown(&sc->sc_drmfb, flags); } static paddr_t -intelfb_genfb_mmap(void *v, void *vs, off_t offset, int prot) +intelfb_drmfb_mmapfb(struct drmfb_softc *drmfb, off_t offset, int prot) { - struct genfb_softc *const genfb = v; - struct intelfb_softc *const sc = container_of(genfb, - struct intelfb_softc, sc_genfb); + struct intelfb_softc *const sc = container_of(drmfb, + struct intelfb_softc, sc_drmfb); struct drm_fb_helper *const helper = sc->sc_ifa.ifa_fb_helper; struct intel_fbdev *const fbdev = container_of(helper, struct intel_fbdev, helper); struct drm_device *const dev = helper->dev; struct drm_i915_private *const dev_priv = dev->dev_private; - const struct pci_attach_args *const pa = &dev->pdev->pd_pa; - unsigned int i; - - if (offset < 0) - return -1; - /* Treat low memory as the framebuffer itself. */ - if (offset < genfb->sc_fbsize) - return bus_space_mmap(dev->bst, - (dev_priv->gtt.mappable_base + - i915_gem_obj_ggtt_offset(fbdev->fb->obj)), - offset, prot, - (BUS_SPACE_MAP_LINEAR | BUS_SPACE_MAP_PREFETCHABLE)); - - /* XXX Cargo-culted from genfb_pci. */ - if (kauth_authorize_machdep(kauth_cred_get(), - KAUTH_MACHDEP_UNMANAGEDMEM, NULL, NULL, NULL, NULL) != 0) { - aprint_normal_dev(dev->dev, "mmap at %"PRIxMAX" rejected\n", - (uintmax_t)offset); - return -1; - } - - for (i = 0; PCI_BAR(i) <= PCI_MAPREG_ROM; i++) { - pcireg_t type; - bus_addr_t addr; - bus_size_t size; - int flags; - - /* Interrogate the BAR. */ - if (!pci_mapreg_probe(pa->pa_pc, pa->pa_tag, PCI_BAR(i), - &type)) - continue; - if (PCI_MAPREG_TYPE(type) != PCI_MAPREG_TYPE_MEM) - continue; - if (pci_mapreg_info(pa->pa_pc, pa->pa_tag, PCI_BAR(i), type, - &addr, &size, &flags)) - continue; - - /* Try to map it if it's in range. */ - if ((addr <= offset) && (offset < (addr + size))) - return bus_space_mmap(pa->pa_memt, addr, - (offset - addr), prot, flags); - - /* Skip a slot if this was a 64-bit BAR. */ - if ((PCI_MAPREG_TYPE(type) == PCI_MAPREG_TYPE_MEM) && - (PCI_MAPREG_MEM_TYPE(type) == PCI_MAPREG_MEM_TYPE_64BIT)) - i += 1; - } - - /* Failure! */ - return -1; -} - -static int -intelfb_genfb_enable_polling(void *cookie) -{ - struct genfb_softc *const genfb = cookie; - struct intelfb_softc *const sc = container_of(genfb, - struct intelfb_softc, sc_genfb); - - return drm_fb_helper_debug_enter_fb(sc->sc_ifa.ifa_fb_helper); -} - -static int -intelfb_genfb_disable_polling(void *cookie) -{ - struct genfb_softc *const genfb = cookie; - struct intelfb_softc *const sc = container_of(genfb, - struct intelfb_softc, sc_genfb); - - return drm_fb_helper_debug_leave_fb(sc->sc_ifa.ifa_fb_helper); -} - -static bool -intelfb_genfb_shutdown(device_t self, int flags) -{ - genfb_enable_polling(self); - return true; -} - -static bool -intelfb_genfb_setmode(struct genfb_softc *genfb, int mode) -{ - struct intelfb_softc *sc = (struct intelfb_softc *)genfb; - - if (mode == WSDISPLAYIO_MODE_EMUL) { - drm_fb_helper_set_config(sc->sc_ifa.ifa_fb_helper); - } + KASSERT(0 <= offset); + KASSERT(offset < fbdev->fb->obj->base.size); - return true; + return bus_space_mmap(dev->bst, dev_priv->gtt.mappable_base, + i915_gem_obj_ggtt_offset(fbdev->fb->obj) + offset, + prot, BUS_SPACE_MAP_PREFETCHABLE); } Index: sys/external/bsd/drm2/include/drm/drmfb.h =================================================================== RCS file: sys/external/bsd/drm2/include/drm/drmfb.h diff -N sys/external/bsd/drm2/include/drm/drmfb.h --- /dev/null 1 Jan 1970 00:00:00 -0000 +++ sys/external/bsd/drm2/include/drm/drmfb.h 14 Sep 2014 19:59:06 -0000 @@ -0,0 +1,92 @@ +/* $NetBSD$ */ + +/*- + * Copyright (c) 2014 The NetBSD Foundation, Inc. + * All rights reserved. + * + * This code is derived from software contributed to The NetBSD Foundation + * by Taylor R. Campbell. + * + * Redistribution and use in source and binary forms, with or without + * modification, are permitted provided that the following conditions + * are met: + * 1. Redistributions of source code must retain the above copyright + * notice, this list of conditions and the following disclaimer. + * 2. Redistributions in binary form must reproduce the above copyright + * notice, this list of conditions and the following disclaimer in the + * documentation and/or other materials provided with the distribution. + * + * THIS SOFTWARE IS PROVIDED BY THE NETBSD FOUNDATION, INC. AND CONTRIBUTORS + * ``AS IS'' AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED + * TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR + * PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE FOUNDATION OR CONTRIBUTORS + * BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR + * CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF + * SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS + * INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN + * CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) + * ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE + * POSSIBILITY OF SUCH DAMAGE. + */ + +#ifndef _DRM_DRMFB_H_ +#define _DRM_DRMFB_H_ + +#include + +#include + +struct drm_fb_helper; +struct drm_fb_helper_surface_sizes; + +struct drmfb_attach_args; +struct drmfb_params; +struct drmfb_softc; + +struct drmfb_params { + /* + * Framebuffer mmap. + */ + paddr_t (*dp_mmapfb)(struct drmfb_softc *, off_t, int); + + /* + * Other mmap, e.g. for arbitrary PCI BARs. Available only to + * users with privileges for access to unmanaged memory. + */ + paddr_t (*dp_mmap)(struct drmfb_softc *, off_t, int); + + /* + * Ioctl handler. Must handle: + * + * - WSDISPLAYIO_GET_BUSID + * - WSDISPLAYIO_GTYPE + * + * May add or override anything else. Return EPASSTHROUGH to + * defer. + */ + int (*dp_ioctl)(struct drmfb_softc *, unsigned long, + void *, int, struct lwp *); + /* XXX Kludge! */ + bool (*dp_is_vga_console)(struct drm_device *); + void (*dp_disable_vga)(struct drm_device *); +}; + +struct drmfb_attach_args { + device_t da_dev; + struct drm_fb_helper *da_fb_helper; + const struct drm_fb_helper_surface_size *da_fb_sizes; + void *da_fb_vaddr; + const struct drmfb_params *da_params; +}; + +/* Treat as opaque. Must be first member of device softc. */ +struct drmfb_softc { + struct genfb_softc sc_genfb; /* XXX Must be first. */ + struct drmfb_attach_args sc_da; +}; + +int drmfb_attach(struct drmfb_softc *, const struct drmfb_attach_args *); +int drmfb_detach(struct drmfb_softc *, int); +bool drmfb_shutdown(struct drmfb_softc *, int); + +#endif /* _DRM_DRMFB_H_ */ Index: sys/external/bsd/drm2/include/drm/drmfb_pci.h =================================================================== RCS file: sys/external/bsd/drm2/include/drm/drmfb_pci.h diff -N sys/external/bsd/drm2/include/drm/drmfb_pci.h --- /dev/null 1 Jan 1970 00:00:00 -0000 +++ sys/external/bsd/drm2/include/drm/drmfb_pci.h 14 Sep 2014 19:59:06 -0000 @@ -0,0 +1,44 @@ +/* $NetBSD$ */ + +/*- + * Copyright (c) 2014 The NetBSD Foundation, Inc. + * All rights reserved. + * + * This code is derived from software contributed to The NetBSD Foundation + * by Taylor R. Campbell. + * + * Redistribution and use in source and binary forms, with or without + * modification, are permitted provided that the following conditions + * are met: + * 1. Redistributions of source code must retain the above copyright + * notice, this list of conditions and the following disclaimer. + * 2. Redistributions in binary form must reproduce the above copyright + * notice, this list of conditions and the following disclaimer in the + * documentation and/or other materials provided with the distribution. + * + * THIS SOFTWARE IS PROVIDED BY THE NETBSD FOUNDATION, INC. AND CONTRIBUTORS + * ``AS IS'' AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED + * TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR + * PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE FOUNDATION OR CONTRIBUTORS + * BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR + * CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF + * SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS + * INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN + * CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) + * ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE + * POSSIBILITY OF SUCH DAMAGE. + */ + +#ifndef _DRM_DRMFB_PCI_H_ +#define _DRM_DRMFB_PCI_H_ + +#include + +struct drmfb_softc; +struct lwp; + +paddr_t drmfb_pci_mmap(struct drmfb_softc *, off_t, int); +int drmfb_pci_ioctl(struct drmfb_softc *, unsigned long, void *, int, + struct lwp *); + +#endif /* _DRM_DRMFB_PCI_H_ */ Index: sys/external/bsd/drm2/pci/drmfb_pci.c =================================================================== RCS file: sys/external/bsd/drm2/pci/drmfb_pci.c diff -N sys/external/bsd/drm2/pci/drmfb_pci.c --- /dev/null 1 Jan 1970 00:00:00 -0000 +++ sys/external/bsd/drm2/pci/drmfb_pci.c 14 Sep 2014 19:59:06 -0000 @@ -0,0 +1,134 @@ +/* $NetBSD$ */ + +/*- + * Copyright (c) 2014 The NetBSD Foundation, Inc. + * All rights reserved. + * + * This code is derived from software contributed to The NetBSD Foundation + * by Taylor R. Campbell. + * + * Redistribution and use in source and binary forms, with or without + * modification, are permitted provided that the following conditions + * are met: + * 1. Redistributions of source code must retain the above copyright + * notice, this list of conditions and the following disclaimer. + * 2. Redistributions in binary form must reproduce the above copyright + * notice, this list of conditions and the following disclaimer in the + * documentation and/or other materials provided with the distribution. + * + * THIS SOFTWARE IS PROVIDED BY THE NETBSD FOUNDATION, INC. AND CONTRIBUTORS + * ``AS IS'' AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED + * TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR + * PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE FOUNDATION OR CONTRIBUTORS + * BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR + * CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF + * SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS + * INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN + * CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) + * ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE + * POSSIBILITY OF SUCH DAMAGE. + */ + +/* + * drmfb_pci: drmfb hooks for PCI devices. + */ + +#include +__KERNEL_RCSID(0, "$NetBSD$"); + +#include +#include +#include +#include + +#include +#include +#include +#include + +#include +#include + +#include "drmfb.h" +#include "drmfb_pci.h" + +/* + * drmfb_pci_mmap: Implementation of drmfb_params::dp_mmap. Don't use + * this for dp_mmapfb -- how to get at the framebuffer is device- + * specific. + */ +paddr_t +drmfb_pci_mmap(struct drmfb_softc *sc, off_t offset, int prot) +{ + struct drm_device *const dev = sc->sc_da.da_fb_helper->dev; + const struct pci_attach_args *const pa = &dev->pdev->pd_pa; + unsigned i; + + for (i = 0; PCI_BAR(i) <= PCI_MAPREG_ROM; i++) { + pcireg_t type; + bus_addr_t addr; + bus_size_t size; + int flags; + + /* Interrogate the BAR. */ + if (!pci_mapreg_probe(pa->pa_pc, pa->pa_tag, PCI_BAR(i), + &type)) + continue; + if (PCI_MAPREG_TYPE(type) != PCI_MAPREG_TYPE_MEM) + continue; + if (pci_mapreg_info(pa->pa_pc, pa->pa_tag, PCI_BAR(i), type, + &addr, &size, &flags)) + continue; + + /* Try to map it if it's in range. */ + if ((addr <= offset) && (offset < (addr + size))) + return bus_space_mmap(pa->pa_memt, addr, + (offset - addr), prot, flags); + + /* Skip a slot if this was a 64-bit BAR. */ + if ((PCI_MAPREG_TYPE(type) == PCI_MAPREG_TYPE_MEM) && + (PCI_MAPREG_MEM_TYPE(type) == PCI_MAPREG_MEM_TYPE_64BIT)) + i += 1; + } + + /* Failure! */ + return -1; +} + +/* + * drmfb_pci_ioctl: Implementation of drmfb_params::dp_ioctl. Provides: + * + * - WSDISPLAY_GET_BUSID + * - WSDISPLAY_GTYPE + * + * Additionally allows access to PCI registers. + * + * XXX Do we need to provide access to PCI registers? I can't find + * anything that uses this functionality, and as is it is very + * dangerous. + */ +int +drmfb_pci_ioctl(struct drmfb_softc *sc, unsigned long cmd, void *data, + int flag, struct lwp *l) +{ + struct drm_device *const dev = sc->sc_da.da_fb_helper->dev; + struct pci_attach_args *const pa = &dev->pdev->pd_pa; + + switch (cmd) { + /* PCI config read/write passthrough. */ + case PCI_IOC_CFGREAD: + case PCI_IOC_CFGWRITE: + return pci_devioctl(pa->pa_pc, pa->pa_tag, cmd, data, flag, l); + + /* PCI-specific wsdisplay ioctls. */ + case WSDISPLAYIO_GET_BUSID: + return wsdisplayio_busid_pci(dev->dev, pa->pa_pc, pa->pa_tag, + data); + case WSDISPLAYIO_GTYPE: + *(unsigned int *)data = WSDISPLAY_TYPE_PCIVGA; + return 0; + + default: + return EPASSTHROUGH; + } +} Index: sys/external/bsd/drm2/pci/files.drmkms_pci =================================================================== RCS file: /cvsroot/src/sys/external/bsd/drm2/pci/files.drmkms_pci,v retrieving revision 1.4 diff -p -u -r1.4 files.drmkms_pci --- sys/external/bsd/drm2/pci/files.drmkms_pci 6 Aug 2014 13:54:40 -0000 1.4 +++ sys/external/bsd/drm2/pci/files.drmkms_pci 14 Sep 2014 19:59:06 -0000 @@ -10,6 +10,9 @@ file external/bsd/drm2/dist/drm/drm_agps file external/bsd/drm2/pci/drm_pci.c drmkms_pci file external/bsd/drm2/pci/drm_pci_module.c drmkms_pci +define drmfb_pci: drmfb, drmkms_pci +file external/bsd/drm2/pci/drmfb_pci.c drmfb_pci + include "external/bsd/drm2/i915drm/files.i915drmkms" include "external/bsd/drm2/radeon/files.radeon" include "external/bsd/drm2/nouveau/files.nouveau" Index: sys/external/bsd/drm2/radeon/files.radeon =================================================================== RCS file: /cvsroot/src/sys/external/bsd/drm2/radeon/files.radeon,v retrieving revision 1.5 diff -p -u -r1.5 files.radeon --- sys/external/bsd/drm2/radeon/files.radeon 28 Jul 2014 00:02:50 -0000 1.5 +++ sys/external/bsd/drm2/radeon/files.radeon 14 Sep 2014 19:59:06 -0000 @@ -5,7 +5,7 @@ device radeon: drmkms, drmkms_pci, drmkm attach radeon at pci # XXX Rename this to radeonfb when the legacy radeonfb(4) is gone. -device radeondrmkmsfb: radeonfbbus, genfb, wsemuldisplaydev +device radeondrmkmsfb: radeonfbbus, drmfb, drmfb_pci, wsemuldisplaydev attach radeondrmkmsfb at radeonfbbus makeoptions radeon CPPFLAGS+="-I$S/external/bsd/drm2/dist/drm/radeon" Index: sys/external/bsd/drm2/radeon/radeondrmkmsfb.c =================================================================== RCS file: /cvsroot/src/sys/external/bsd/drm2/radeon/radeondrmkmsfb.c,v retrieving revision 1.3 diff -p -u -r1.3 radeondrmkmsfb.c --- sys/external/bsd/drm2/radeon/radeondrmkmsfb.c 26 Jul 2014 07:02:13 -0000 1.3 +++ sys/external/bsd/drm2/radeon/radeondrmkmsfb.c 14 Sep 2014 19:59:06 -0000 @@ -1,4 +1,4 @@ -/* $NetBSD: radeondrmkmsfb.c,v 1.3 2014/07/26 07:02:13 riastradh Exp $ */ +/* $NetBSD$ */ /*- * Copyright (c) 2014 The NetBSD Foundation, Inc. @@ -29,24 +29,17 @@ * POSSIBILITY OF SUCH DAMAGE. */ - #include -__KERNEL_RCSID(0, "$NetBSD: radeondrmkmsfb.c,v 1.3 2014/07/26 07:02:13 riastradh Exp $"); +__KERNEL_RCSID(0, "$NetBSD$"); #ifdef _KERNEL_OPT #include "vga.h" #endif #include +#include #include -#include -#include -#include - -#include -#include - #if NVGA > 0 /* * XXX All we really need is vga_is_console from vgavar.h, but the @@ -60,34 +53,39 @@ __KERNEL_RCSID(0, "$NetBSD: radeondrmkms #endif #include -#include -#include -#include "radeon_drv.h" +#include "drmfb.h" +#include "drmfb_pci.h" +#include "radeon.h" #include "radeon_task.h" #include "radeondrmkmsfb.h" -struct radeonfb_softc { - /* XXX genfb requires the genfb_softc to be first. */ - struct genfb_softc sc_genfb; - device_t sc_dev; - struct radeonfb_attach_args sc_rfa; - struct radeon_task sc_setconfig_task; - bool sc_scheduled:1; - bool sc_attached:1; -}; - static int radeonfb_match(device_t, cfdata_t, void *); static void radeonfb_attach(device_t, device_t, void *); static int radeonfb_detach(device_t, int); -static void radeonfb_setconfig_task(struct radeon_task *); +static void radeonfb_attach_task(struct radeon_task *); + +static bool radeon_is_vga_console(struct drm_device *); +static bool radeonfb_shutdown(device_t, int); -static int radeonfb_genfb_ioctl(void *, void *, unsigned long, void *, - int, struct lwp *); -static paddr_t radeonfb_genfb_mmap(void *, void *, off_t, int); -static int radeonfb_genfb_enable_polling(void *); -static int radeonfb_genfb_disable_polling(void *); +static paddr_t radeonfb_drmfb_mmapfb(struct drmfb_softc *, off_t, int); + +struct radeonfb_softc { + struct drmfb_softc sc_drmfb; /* XXX Must be first. */ + struct radeonfb_attach_args sc_rfa; + device_t sc_dev; + struct radeon_task sc_attach_task; + bool sc_scheduled; + bool sc_attached; +}; + +static const struct drmfb_params radeonfb_drmfb_params = { + .dp_mmapfb = radeonfb_drmfb_mmapfb, + .dp_mmap = drmfb_pci_mmap, + .dp_ioctl = drmfb_pci_ioctl, + .dp_is_vga_console = radeon_is_vga_console, +}; CFATTACH_DECL_NEW(radeondrmkmsfb, sizeof(struct radeonfb_softc), radeonfb_match, radeonfb_attach, radeonfb_detach, NULL); @@ -114,8 +112,8 @@ radeonfb_attach(device_t parent, device_ aprint_naive("\n"); aprint_normal("\n"); - radeon_task_init(&sc->sc_setconfig_task, &radeonfb_setconfig_task); - error = radeon_task_schedule(parent, &sc->sc_setconfig_task); + radeon_task_init(&sc->sc_attach_task, &radeonfb_attach_task); + error = radeon_task_schedule(parent, &sc->sc_attach_task); if (error) { aprint_error_dev(self, "failed to schedule mode set: %d\n", error); @@ -133,12 +131,20 @@ static int radeonfb_detach(device_t self, int flags) { struct radeonfb_softc *const sc = device_private(self); + int error; if (sc->sc_scheduled) return EBUSY; if (sc->sc_attached) { - /* XXX genfb detach? Help? */ + pmf_device_deregister(self); + error = drmfb_detach(&sc->sc_drmfb, flags); + if (error) { + /* XXX Ugh. */ + (void)pmf_device_register1(self, NULL, NULL, + &radeonfb_shutdown); + return error; + } sc->sc_attached = false; } @@ -146,201 +152,67 @@ radeonfb_detach(device_t self, int flags } static void -radeonfb_setconfig_task(struct radeon_task *task) +radeonfb_attach_task(struct radeon_task *task) { struct radeonfb_softc *const sc = container_of(task, - struct radeonfb_softc, sc_setconfig_task); - const prop_dictionary_t dict = device_properties(sc->sc_dev); + struct radeonfb_softc, sc_attach_task); const struct radeonfb_attach_args *const rfa = &sc->sc_rfa; - const struct drm_fb_helper_surface_size *const sizes = - &rfa->rfa_fb_sizes; - enum { CONS_VGA, CONS_GENFB, CONS_NONE } what_was_cons; - static const struct genfb_ops zero_genfb_ops; - struct genfb_ops genfb_ops = zero_genfb_ops; + const struct drmfb_attach_args da = { + .da_dev = sc->sc_dev, + .da_fb_helper = rfa->rfa_fb_helper, + .da_fb_sizes = &rfa->rfa_fb_sizes, + .da_fb_vaddr = rfa->rfa_fb_ptr, + .da_params = &radeonfb_drmfb_params, + }; int error; - KASSERT(sc->sc_scheduled); - - /* XXX Ugh... Pass these parameters some other way! */ - prop_dictionary_set_uint32(dict, "width", sizes->fb_width); - prop_dictionary_set_uint32(dict, "height", sizes->fb_height); - prop_dictionary_set_uint8(dict, "depth", sizes->surface_bpp); - prop_dictionary_set_uint16(dict, "linebytes", - roundup2((sizes->fb_width * howmany(sizes->surface_bpp, 8)), 64)); - prop_dictionary_set_uint32(dict, "address", 0); /* XXX >32-bit */ - CTASSERT(sizeof(uintptr_t) <= sizeof(uint64_t)); - prop_dictionary_set_uint64(dict, "virtual_address", - (uint64_t)(uintptr_t)rfa->rfa_fb_ptr); - - /* XXX Whattakludge! */ -#if NVGA > 0 - if (vga_is_console(rfa->rfa_fb_helper->dev->pdev->pd_pa.pa_iot, -1)) { - what_was_cons = CONS_VGA; - prop_dictionary_set_bool(dict, "is_console", true); - vga_cndetach(); - } else -#endif - if (genfb_is_console() && genfb_is_enabled()) { - what_was_cons = CONS_GENFB; - prop_dictionary_set_bool(dict, "is_console", true); - } else { - what_was_cons = CONS_NONE; - prop_dictionary_set_bool(dict, "is_console", false); - } - - sc->sc_genfb.sc_dev = sc->sc_dev; - genfb_init(&sc->sc_genfb); - genfb_ops.genfb_ioctl = radeonfb_genfb_ioctl; - genfb_ops.genfb_mmap = radeonfb_genfb_mmap; - genfb_ops.genfb_enable_polling = radeonfb_genfb_enable_polling; - genfb_ops.genfb_disable_polling = radeonfb_genfb_disable_polling; - - error = genfb_attach(&sc->sc_genfb, &genfb_ops); + error = drmfb_attach(&sc->sc_drmfb, &da); if (error) { - aprint_error_dev(sc->sc_dev, "failed to attach genfb: %d\n", + aprint_error_dev(sc->sc_dev, "failed to attach drmfb: %d\n", error); - goto fail0; + return; } - sc->sc_attached = true; - drm_fb_helper_set_config(sc->sc_rfa.rfa_fb_helper); + if (pmf_device_register1(sc->sc_dev, NULL, NULL, &radeonfb_shutdown)) + aprint_error_dev(sc->sc_dev, + "failed to register shutdown handler\n"); - /* Success! */ - sc->sc_scheduled = false; - return; - -fail0: /* XXX Restore console... */ - switch (what_was_cons) { - case CONS_VGA: - break; - case CONS_GENFB: - break; - case CONS_NONE: - break; - default: - break; - } + sc->sc_attached = true; } -static int -radeonfb_genfb_ioctl(void *v, void *vs, unsigned long cmd, void *data, int flag, - struct lwp *l) +static bool +radeon_is_vga_console(struct drm_device *dev) { - struct genfb_softc *const genfb = v; - struct radeonfb_softc *const sc = container_of(genfb, - struct radeonfb_softc, sc_genfb); - struct drm_device *const dev = sc->sc_rfa.rfa_fb_helper->dev; - const struct pci_attach_args *const pa = &dev->pdev->pd_pa; - - switch (cmd) { - case WSDISPLAYIO_GTYPE: - *(unsigned int *)data = WSDISPLAY_TYPE_PCIVGA; - return 0; - - /* PCI config read/write passthrough. */ - case PCI_IOC_CFGREAD: - case PCI_IOC_CFGWRITE: - return pci_devioctl(pa->pa_pc, pa->pa_tag, cmd, data, flag, l); - - case WSDISPLAYIO_GET_BUSID: - return wsdisplayio_busid_pci(dev->dev, pa->pa_pc, pa->pa_tag, - data); - default: - return EPASSTHROUGH; - } + return vga_is_console(dev->pdev->pd_pa.pa_iot, -1); } -static paddr_t -radeonfb_genfb_mmap(void *v, void *vs, off_t offset, int prot) +static bool +radeonfb_shutdown(device_t self, int flags) { - struct genfb_softc *const genfb = v; - struct radeonfb_softc *const sc = container_of(genfb, - struct radeonfb_softc, sc_genfb); - struct drm_fb_helper *const helper = sc->sc_rfa.rfa_fb_helper; - struct drm_framebuffer *const fb = helper->fb; - struct radeon_framebuffer *const rfb = container_of(fb, - struct radeon_framebuffer, base); - struct drm_gem_object *const gobj = rfb->obj; - struct radeon_bo *const rbo = gem_to_radeon_bo(gobj); - struct drm_device *const dev = helper->dev; - const struct pci_attach_args *const pa = &dev->pdev->pd_pa; - unsigned int i; - - if (offset < 0) - return -1; - - /* Treat low memory as the framebuffer itself. */ - if (offset < genfb->sc_fbsize) { - const unsigned num_pages __diagused = rbo->tbo.num_pages; - int flags = 0; - - KASSERT(genfb->sc_fbsize == (num_pages << PAGE_SHIFT)); - KASSERT(rbo->tbo.mem.bus.is_iomem); - - if (ISSET(rbo->tbo.mem.placement, TTM_PL_FLAG_WC)) - flags |= BUS_SPACE_MAP_PREFETCHABLE; - - return bus_space_mmap(rbo->tbo.bdev->memt, - rbo->tbo.mem.bus.base, rbo->tbo.mem.bus.offset + offset, - prot, flags); - } - - /* XXX Cargo-culted from genfb_pci. */ - if (kauth_authorize_machdep(kauth_cred_get(), - KAUTH_MACHDEP_UNMANAGEDMEM, NULL, NULL, NULL, NULL) != 0) { - aprint_normal_dev(dev->dev, "mmap at %"PRIxMAX" rejected\n", - (uintmax_t)offset); - return -1; - } - - for (i = 0; PCI_BAR(i) <= PCI_MAPREG_ROM; i++) { - pcireg_t type; - bus_addr_t addr; - bus_size_t size; - int flags; - - /* Interrogate the BAR. */ - if (!pci_mapreg_probe(pa->pa_pc, pa->pa_tag, PCI_BAR(i), - &type)) - continue; - if (PCI_MAPREG_TYPE(type) != PCI_MAPREG_TYPE_MEM) - continue; - if (pci_mapreg_info(pa->pa_pc, pa->pa_tag, PCI_BAR(i), type, - &addr, &size, &flags)) - continue; - - /* Try to map it if it's in range. */ - if ((addr <= offset) && (offset < (addr + size))) - return bus_space_mmap(pa->pa_memt, addr, - (offset - addr), prot, flags); - - /* Skip a slot if this was a 64-bit BAR. */ - if ((PCI_MAPREG_TYPE(type) == PCI_MAPREG_TYPE_MEM) && - (PCI_MAPREG_MEM_TYPE(type) == PCI_MAPREG_MEM_TYPE_64BIT)) - i += 1; - } + struct radeonfb_softc *const sc = device_private(self); - /* Failure! */ - return -1; + return drmfb_shutdown(&sc->sc_drmfb, flags); } -static int -radeonfb_genfb_enable_polling(void *cookie) +static paddr_t +radeonfb_drmfb_mmapfb(struct drmfb_softc *drmfb, off_t offset, int prot) { - struct genfb_softc *const genfb = cookie; - struct radeonfb_softc *const sc = container_of(genfb, - struct radeonfb_softc, sc_genfb); + struct radeonfb_softc *const sc = container_of(drmfb, + struct radeonfb_softc, sc_drmfb); + struct drm_framebuffer *const fb = sc->sc_rfa.rfa_fb_helper->fb; + struct radeon_framebuffer *const rfb = container_of(fb, + struct radeon_framebuffer, base); + struct radeon_bo *const rbo = gem_to_radeon_bo(rfb->obj); + int flags = 0; - return drm_fb_helper_debug_enter_fb(sc->sc_rfa.rfa_fb_helper); -} + KASSERT(0 <= offset); + KASSERT(offset < (rbo->tbo.num_pages << PAGE_SHIFT)); + KASSERT(rbo->tbo.mem.bus.is_iomem); -static int -radeonfb_genfb_disable_polling(void *cookie) -{ - struct genfb_softc *const genfb = cookie; - struct radeonfb_softc *const sc = container_of(genfb, - struct radeonfb_softc, sc_genfb); + if (ISSET(rbo->tbo.mem.placement, TTM_PL_FLAG_WC)) + flags |= BUS_SPACE_MAP_PREFETCHABLE; - return drm_fb_helper_debug_leave_fb(sc->sc_rfa.rfa_fb_helper); + return bus_space_mmap(rbo->tbo.bdev->memt, rbo->tbo.mem.bus.base, + rbo->tbo.mem.bus.offset + offset, prot, flags); }