Path: blob/main/sys/arm/nvidia/drm2/tegra_host1x.c
39483 views
/*-1* Copyright (c) 2015 Michal Meloun2* All rights reserved.3*4* Redistribution and use in source and binary forms, with or without5* modification, are permitted provided that the following conditions6* are met:7* 1. Redistributions of source code must retain the above copyright8* notice, this list of conditions and the following disclaimer.9* 2. Redistributions in binary form must reproduce the above copyright10* notice, this list of conditions and the following disclaimer in the11* documentation and/or other materials provided with the distribution.12*13* THIS SOFTWARE IS PROVIDED BY THE AUTHOR AND CONTRIBUTORS ``AS IS'' AND14* ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE15* IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE16* ARE DISCLAIMED. IN NO EVENT SHALL THE AUTHOR OR CONTRIBUTORS BE LIABLE17* FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL18* DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS19* OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION)20* HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT21* LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY22* OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF23* SUCH DAMAGE.24*/2526#include <sys/param.h>27#include <sys/systm.h>28#include <sys/bus.h>29#include <sys/clock.h>30#include <sys/kernel.h>31#include <sys/limits.h>32#include <sys/lock.h>3334#include <sys/module.h>35#include <sys/resource.h>36#include <sys/sx.h>37#include <sys/rman.h>3839#include <machine/bus.h>40#include <machine/resource.h>4142#include <dev/clk/clk.h>43#include <dev/hwreset/hwreset.h>44#include <dev/drm2/drmP.h>45#include <dev/drm2/drm_crtc_helper.h>46#include <dev/drm2/drm_fb_helper.h>47#include <dev/fdt/simplebus.h>48#include <dev/ofw/ofw_bus.h>49#include <dev/ofw/ofw_bus_subr.h>5051#include <arm/nvidia/drm2/tegra_drm.h>5253#include "fb_if.h"54#include "tegra_drm_if.h"5556#define WR4(_sc, _r, _v) bus_rite_4((_sc)->mem_res, (_r), (_v))57#define RD4(_sc, _r) bus_read_4((_sc)->mem_res, (_r))5859#define LOCK(_sc) sx_xlock(&(_sc)->lock)60#define UNLOCK(_sc) sx_xunlock(&(_sc)->lock)61#define SLEEP(_sc, timeout) sx_sleep(sc, &sc->lock, 0, "host1x", timeout);62#define LOCK_INIT(_sc) sx_init(&_sc->lock, "host1x")63#define LOCK_DESTROY(_sc) sx_destroy(&_sc->lock)64#define ASSERT_LOCKED(_sc) sx_assert(&_sc->lock, SA_LOCKED)65#define ASSERT_UNLOCKED(_sc) sx_assert(&_sc->lock, SA_UNLOCKED)6667static struct ofw_compat_data compat_data[] = {68{"nvidia,tegra124-host1x", 1},69{NULL, 0}70};7172#define DRIVER_NAME "tegra"73#define DRIVER_DESC "NVIDIA Tegra TK1"74#define DRIVER_DATE "20151101"75#define DRIVER_MAJOR 076#define DRIVER_MINOR 077#define DRIVER_PATCHLEVEL 07879struct client_info;80TAILQ_HEAD(client_list, client_info);81typedef struct client_list client_list_t;8283struct client_info {84TAILQ_ENTRY(client_info) list_e;85device_t client;86int activated;87};8889struct host1x_softc {90struct simplebus_softc simplebus_sc; /* must be first */91device_t dev;92struct sx lock;93int attach_done;9495struct resource *mem_res;96struct resource *syncpt_irq_res;97void *syncpt_irq_h;98struct resource *gen_irq_res;99void *gen_irq_h;100101clk_t clk;102hwreset_t reset;103struct intr_config_hook irq_hook;104105int drm_inited;106client_list_t clients;107108struct tegra_drm *tegra_drm;109};110111static void112host1x_output_poll_changed(struct drm_device *drm_dev)113{114struct tegra_drm *drm;115116drm = container_of(drm_dev, struct tegra_drm, drm_dev);117if (drm->fb != NULL)118drm_fb_helper_hotplug_event(&drm->fb->fb_helper);119}120121static const struct drm_mode_config_funcs mode_config_funcs = {122.fb_create = tegra_drm_fb_create,123.output_poll_changed = host1x_output_poll_changed,124};125126static int127host1x_drm_init(struct host1x_softc *sc)128{129struct client_info *entry;130int rv;131132LOCK(sc);133134TAILQ_FOREACH(entry, &sc->clients, list_e) {135if (entry->activated)136continue;137rv = TEGRA_DRM_INIT_CLIENT(entry->client, sc->dev,138sc->tegra_drm);139if (rv != 0) {140device_printf(sc->dev,141"Cannot init DRM client %s: %d\n",142device_get_name(entry->client), rv);143return (rv);144}145entry->activated = 1;146}147UNLOCK(sc);148149return (0);150}151152static int153host1x_drm_exit(struct host1x_softc *sc)154{155struct client_info *entry;156int rv;157#ifdef FREEBSD_NOTYET158struct drm_device *dev, *tmp;159#endif160LOCK(sc);161if (!sc->drm_inited) {162UNLOCK(sc);163return (0);164}165TAILQ_FOREACH_REVERSE(entry, &sc->clients, client_list, list_e) {166if (!entry->activated)167continue;168rv = TEGRA_DRM_EXIT_CLIENT(entry->client, sc->dev,169sc->tegra_drm);170if (rv != 0) {171device_printf(sc->dev,172"Cannot exit DRM client %s: %d\n",173device_get_name(entry->client), rv);174}175entry->activated = 0;176}177178#ifdef FREEBSD_NOTYET179list_for_each_entry_safe(dev, tmp, &driver->device_list, driver_item)180drm_put_dev(dev);181#endif182sc->drm_inited = 0;183UNLOCK(sc);184185return (0);186}187188static int189host1x_drm_load(struct drm_device *drm_dev, unsigned long flags)190{191struct host1x_softc *sc;192int rv;193194sc = device_get_softc(drm_dev->dev);195196drm_mode_config_init(drm_dev);197drm_dev->mode_config.min_width = 32;198drm_dev->mode_config.min_height = 32;199drm_dev->mode_config.max_width = 4096;200drm_dev->mode_config.max_height = 4096;201drm_dev->mode_config.funcs = &mode_config_funcs;202203rv = host1x_drm_init(sc);204if (rv != 0)205goto fail_host1x;206207drm_dev->irq_enabled = true;208drm_dev->max_vblank_count = 0xffffffff;209drm_dev->vblank_disable_allowed = true;210211rv = drm_vblank_init(drm_dev, drm_dev->mode_config.num_crtc);212if (rv != 0)213goto fail_vblank;214215drm_mode_config_reset(drm_dev);216217rv = tegra_drm_fb_init(drm_dev);218if (rv != 0)219goto fail_fb;220drm_kms_helper_poll_init(drm_dev);221222return (0);223224fail_fb:225tegra_drm_fb_destroy(drm_dev);226drm_vblank_cleanup(drm_dev);227fail_vblank:228host1x_drm_exit(sc);229fail_host1x:230drm_mode_config_cleanup(drm_dev);231232return (rv);233}234235static int236host1x_drm_unload(struct drm_device *drm_dev)237{238struct host1x_softc *sc;239int rv;240241sc = device_get_softc(drm_dev->dev);242243drm_kms_helper_poll_fini(drm_dev);244tegra_drm_fb_destroy(drm_dev);245drm_mode_config_cleanup(drm_dev);246247rv = host1x_drm_exit(sc);248if (rv < 0)249return (rv);250return (0);251}252253static int254host1x_drm_open(struct drm_device *drm_dev, struct drm_file *filp)255{256257return (0);258}259260static void261tegra_drm_preclose(struct drm_device *drm, struct drm_file *file)262{263struct drm_crtc *crtc;264265list_for_each_entry(crtc, &drm->mode_config.crtc_list, head)266tegra_dc_cancel_page_flip(crtc, file);267}268269static void270host1x_drm_lastclose(struct drm_device *drm_dev)271{272273struct tegra_drm *drm;274275drm = container_of(drm_dev, struct tegra_drm, drm_dev);276if (drm->fb != NULL)277drm_fb_helper_restore_fbdev_mode(&drm->fb->fb_helper);278}279280static int281host1x_drm_enable_vblank(struct drm_device *drm_dev, int pipe)282{283struct drm_crtc *crtc;284285list_for_each_entry(crtc, &drm_dev->mode_config.crtc_list, head) {286if (pipe == tegra_dc_get_pipe(crtc)) {287tegra_dc_enable_vblank(crtc);288return (0);289}290}291return (-ENODEV);292}293294static void295host1x_drm_disable_vblank(struct drm_device *drm_dev, int pipe)296{297struct drm_crtc *crtc;298299list_for_each_entry(crtc, &drm_dev->mode_config.crtc_list, head) {300if (pipe == tegra_dc_get_pipe(crtc)) {301tegra_dc_disable_vblank(crtc);302return;303}304}305}306307static struct drm_ioctl_desc host1x_drm_ioctls[] = {308};309310struct drm_driver tegra_drm_driver = {311.driver_features = DRIVER_MODESET | DRIVER_GEM | DRIVER_PRIME,312.load = host1x_drm_load,313.unload = host1x_drm_unload,314.open = host1x_drm_open,315.preclose = tegra_drm_preclose,316.lastclose = host1x_drm_lastclose,317318.get_vblank_counter = drm_vblank_count,319.enable_vblank = host1x_drm_enable_vblank,320.disable_vblank = host1x_drm_disable_vblank,321322/* Fields filled by tegra_bo_driver_register()323.gem_free_object324.gem_pager_ops325.dumb_create326.dumb_map_offset327.dumb_destroy328*/329.ioctls = host1x_drm_ioctls,330.num_ioctls = nitems(host1x_drm_ioctls),331332.name = DRIVER_NAME,333.desc = DRIVER_DESC,334.date = DRIVER_DATE,335.major = DRIVER_MAJOR,336.minor = DRIVER_MINOR,337.patchlevel = DRIVER_PATCHLEVEL,338};339340/*341* ----------------- Device methods -------------------------342*/343static void344host1x_irq_hook(void *arg)345{346struct host1x_softc *sc;347int rv;348349sc = arg;350config_intrhook_disestablish(&sc->irq_hook);351352tegra_bo_driver_register(&tegra_drm_driver);353rv = drm_get_platform_dev(sc->dev, &sc->tegra_drm->drm_dev,354&tegra_drm_driver);355if (rv != 0) {356device_printf(sc->dev, "drm_get_platform_dev(): %d\n", rv);357return;358}359360sc->drm_inited = 1;361}362363static struct fb_info *364host1x_fb_helper_getinfo(device_t dev)365{366struct host1x_softc *sc;367368sc = device_get_softc(dev);369if (sc->tegra_drm == NULL)370return (NULL);371return (tegra_drm_fb_getinfo(&sc->tegra_drm->drm_dev));372}373374static int375host1x_register_client(device_t dev, device_t client)376{377struct host1x_softc *sc;378struct client_info *entry;379380sc = device_get_softc(dev);381382entry = malloc(sizeof(struct client_info), M_DEVBUF, M_WAITOK | M_ZERO);383entry->client = client;384entry->activated = 0;385386LOCK(sc);387TAILQ_INSERT_TAIL(&sc->clients, entry, list_e);388UNLOCK(sc);389390return (0);391}392393static int394host1x_deregister_client(device_t dev, device_t client)395{396struct host1x_softc *sc;397struct client_info *entry;398399sc = device_get_softc(dev);400401LOCK(sc);402TAILQ_FOREACH(entry, &sc->clients, list_e) {403if (entry->client == client) {404if (entry->activated)405panic("Tegra DRM: Attempt to deregister "406"activated client");407TAILQ_REMOVE(&sc->clients, entry, list_e);408free(entry, M_DEVBUF);409UNLOCK(sc);410return (0);411}412}413UNLOCK(sc);414415return (0);416}417418static void419host1x_gen_intr(void *arg)420{421struct host1x_softc *sc;422423sc = (struct host1x_softc *)arg;424LOCK(sc);425UNLOCK(sc);426}427428static void429host1x_syncpt_intr(void *arg)430{431struct host1x_softc *sc;432433sc = (struct host1x_softc *)arg;434LOCK(sc);435UNLOCK(sc);436}437438static void439host1x_new_pass(device_t dev)440{441struct host1x_softc *sc;442int rv, rid;443phandle_t node;444445/*446* We attach during BUS_PASS_BUS (because we must overcome simplebus),447* but some of our FDT resources are not ready until BUS_PASS_DEFAULT448*/449sc = device_get_softc(dev);450if (sc->attach_done || bus_get_pass() < BUS_PASS_DEFAULT) {451bus_generic_new_pass(dev);452return;453}454455sc->attach_done = 1;456node = ofw_bus_get_node(dev);457458/* Allocate our IRQ resource. */459rid = 0;460sc->syncpt_irq_res = bus_alloc_resource_any(dev, SYS_RES_IRQ, &rid,461RF_ACTIVE);462if (sc->syncpt_irq_res == NULL) {463device_printf(dev, "Cannot allocate interrupt.\n");464rv = ENXIO;465goto fail;466}467rid = 1;468sc->gen_irq_res = bus_alloc_resource_any(dev, SYS_RES_IRQ, &rid,469RF_ACTIVE);470if (sc->gen_irq_res == NULL) {471device_printf(dev, "Cannot allocate interrupt.\n");472rv = ENXIO;473goto fail;474}475476/* FDT resources */477rv = hwreset_get_by_ofw_name(sc->dev, 0, "host1x", &sc->reset);478if (rv != 0) {479device_printf(dev, "Cannot get fuse reset\n");480goto fail;481}482rv = clk_get_by_ofw_index(sc->dev, 0, 0, &sc->clk);483if (rv != 0) {484device_printf(dev, "Cannot get i2c clock: %d\n", rv);485goto fail;486}487488rv = clk_enable(sc->clk);489if (rv != 0) {490device_printf(dev, "Cannot enable clock: %d\n", rv);491goto fail;492}493rv = hwreset_deassert(sc->reset);494if (rv != 0) {495device_printf(sc->dev, "Cannot clear reset\n");496goto fail;497}498499/* Setup interrupts */500rv = bus_setup_intr(dev, sc->gen_irq_res,501INTR_TYPE_MISC | INTR_MPSAFE, NULL, host1x_gen_intr,502sc, &sc->gen_irq_h);503if (rv) {504device_printf(dev, "Cannot setup gen interrupt.\n");505goto fail;506}507508rv = bus_setup_intr(dev, sc->syncpt_irq_res,509INTR_TYPE_MISC | INTR_MPSAFE, NULL, host1x_syncpt_intr,510sc, &sc->syncpt_irq_h);511if (rv) {512device_printf(dev, "Cannot setup syncpt interrupt.\n");513goto fail;514}515516simplebus_init(dev, 0);517for (node = OF_child(node); node > 0; node = OF_peer(node))518simplebus_add_device(dev, node, 0, NULL, -1, NULL);519520sc->irq_hook.ich_func = host1x_irq_hook;521sc->irq_hook.ich_arg = sc;522config_intrhook_establish(&sc->irq_hook);523bus_generic_new_pass(dev);524return;525526fail:527device_detach(dev);528return;529}530531static int532host1x_probe(device_t dev)533{534535if (!ofw_bus_status_okay(dev))536return (ENXIO);537538if (ofw_bus_search_compatible(dev, compat_data)->ocd_data == 0)539return (ENXIO);540541return (BUS_PROBE_DEFAULT);542}543544static int545host1x_attach(device_t dev)546{547int rv, rid;548struct host1x_softc *sc;549550sc = device_get_softc(dev);551sc->tegra_drm = malloc(sizeof(struct tegra_drm), DRM_MEM_DRIVER,552M_WAITOK | M_ZERO);553554/* crosslink together all worlds */555sc->dev = dev;556sc->tegra_drm->drm_dev.dev_private = &sc->tegra_drm;557sc->tegra_drm->drm_dev.dev = dev;558559TAILQ_INIT(&sc->clients);560561LOCK_INIT(sc);562563/* Get the memory resource for the register mapping. */564rid = 0;565sc->mem_res = bus_alloc_resource_any(dev, SYS_RES_MEMORY, &rid,566RF_ACTIVE);567if (sc->mem_res == NULL) {568device_printf(dev, "Cannot map registers.\n");569rv = ENXIO;570goto fail;571}572573bus_attach_children(dev);574return (0);575576fail:577if (sc->tegra_drm != NULL)578free(sc->tegra_drm, DRM_MEM_DRIVER);579if (sc->mem_res != NULL)580bus_release_resource(dev, SYS_RES_MEMORY, 0, sc->mem_res);581LOCK_DESTROY(sc);582return (rv);583}584585static int586host1x_detach(device_t dev)587{588struct host1x_softc *sc;589int error;590591error = bus_generic_detach(dev);592if (error != 0)593return (error);594595sc = device_get_softc(dev);596597host1x_drm_exit(sc);598599if (sc->gen_irq_h != NULL)600bus_teardown_intr(dev, sc->gen_irq_res, sc->gen_irq_h);601if (sc->tegra_drm != NULL)602free(sc->tegra_drm, DRM_MEM_DRIVER);603if (sc->clk != NULL)604clk_release(sc->clk);605if (sc->reset != NULL)606hwreset_release(sc->reset);607if (sc->syncpt_irq_h != NULL)608bus_teardown_intr(dev, sc->syncpt_irq_res, sc->syncpt_irq_h);609if (sc->gen_irq_res != NULL)610bus_release_resource(dev, SYS_RES_IRQ, 1, sc->gen_irq_res);611if (sc->syncpt_irq_res != NULL)612bus_release_resource(dev, SYS_RES_IRQ, 0, sc->syncpt_irq_res);613if (sc->mem_res != NULL)614bus_release_resource(dev, SYS_RES_MEMORY, 0, sc->mem_res);615LOCK_DESTROY(sc);616return (0);617}618619static device_method_t host1x_methods[] = {620/* Device interface */621DEVMETHOD(device_probe, host1x_probe),622DEVMETHOD(device_attach, host1x_attach),623DEVMETHOD(device_detach, host1x_detach),624625/* Bus interface */626DEVMETHOD(bus_new_pass, host1x_new_pass),627628/* Framebuffer service methods */629DEVMETHOD(fb_getinfo, host1x_fb_helper_getinfo),630631/* tegra drm interface */632DEVMETHOD(tegra_drm_register_client, host1x_register_client),633DEVMETHOD(tegra_drm_deregister_client, host1x_deregister_client),634635DEVMETHOD_END636};637638DEFINE_CLASS_1(host1x, host1x_driver, host1x_methods,639sizeof(struct host1x_softc), simplebus_driver);640EARLY_DRIVER_MODULE(host1x, simplebus, host1x_driver, 0, 0, BUS_PASS_BUS);641642/* Bindings for fbd device. */643extern driver_t fbd_driver;644DRIVER_MODULE(fbd, host1x, fbd_driver, 0, 0);645646647