Module Name:    src
Committed By:   bouyer
Date:           Wed Jul 16 18:30:43 UTC 2014

Added Files:
        src/sys/arch/arm/omap: tifb.c tifbreg.h

Log Message:
Work in progress on a driver for the graphic controller found in TI am355x SoCs,
based on the FreeBSD driver.
Only dumb framebuffer mode supported.
At this time, the video timings are hardcoded in the driver's source.
The timings inclued here are for the 12" 1280x800 LCD+LVDS cape from
Chalk Elec.
A better way to define video modes is clearly needed.


To generate a diff of this commit:
cvs rdiff -u -r0 -r1.1 src/sys/arch/arm/omap/tifb.c \
    src/sys/arch/arm/omap/tifbreg.h

Please note that diffs are not public domain; they are subject to the
copyright notices on the relevant files.

Added files:

Index: src/sys/arch/arm/omap/tifb.c
diff -u /dev/null src/sys/arch/arm/omap/tifb.c:1.1
--- /dev/null	Wed Jul 16 18:30:43 2014
+++ src/sys/arch/arm/omap/tifb.c	Wed Jul 16 18:30:43 2014
@@ -0,0 +1,1068 @@
+/*	$NetBSD: tifb.c,v 1.1 2014/07/16 18:30:43 bouyer Exp $	*/
+
+/*
+ * Copyright (c) 2010 Michael Lorenz
+ * All rights reserved.
+ *
+ * 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 AUTHOR ``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 AUTHOR 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.
+ */
+
+/*
+ * Copyright 2013 Oleksandr Tymoshenko <go...@freebsd.org>
+ * All rights reserved.
+ *
+ * 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 AUTHOR 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 AUTHOR 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.
+ */
+
+/*
+ * A framebuffer driver for TI 35xx built-in video controller
+ * tested on beaglebone
+ */
+
+#include <sys/cdefs.h>
+__KERNEL_RCSID(0, "$NetBSD: tifb.c,v 1.1 2014/07/16 18:30:43 bouyer Exp $");
+
+#include "opt_omap.h"
+
+#include <sys/param.h>
+#include <sys/systm.h>
+#include <sys/kernel.h>
+#include <sys/device.h>
+#include <sys/malloc.h>
+#include <sys/lwp.h>
+#include <sys/kauth.h>
+
+#include <uvm/uvm_extern.h>
+
+#include <dev/videomode/videomode.h>
+
+#include <sys/bus.h>
+#include <arm/omap/tifbreg.h>
+#include <arm/omap/omap_var.h>
+#include <arm/omap/omap2_obiovar.h>
+#include <arm/omap/omap2_obioreg.h>
+#ifdef TI_AM335X
+#  include <arm/omap/am335x_prcm.h>
+#  include <arm/omap/omap2_prcm.h>
+#  include <arm/omap/sitara_cm.h>
+#  include <arm/omap/sitara_cmreg.h>
+#endif
+
+#include <dev/wscons/wsdisplayvar.h>
+#include <dev/wscons/wsconsio.h>
+#include <dev/wsfont/wsfont.h>
+#include <dev/rasops/rasops.h>
+#include <dev/wscons/wsdisplay_vconsvar.h>
+
+#include <dev/videomode/edidvar.h>
+
+#include "locators.h"
+
+struct panel_info {
+	uint32_t panel_width;
+	uint32_t panel_height;
+	uint32_t panel_hfp;
+	uint32_t panel_hbp;
+	uint32_t panel_hsw;
+	uint32_t panel_vfp;
+	uint32_t panel_vbp;
+	uint32_t panel_vsw;
+	uint32_t ac_bias;
+	uint32_t ac_bias_intrpt;
+	uint32_t dma_burst_sz;
+	uint32_t bpp;
+	uint32_t fdd;
+	uint32_t invert_line_clock;
+	uint32_t invert_frm_clock;
+	uint32_t sync_edge;
+	uint32_t sync_ctrl;
+	uint32_t panel_pxl_clk;
+	uint32_t panel_invert_pxl_clk;
+};
+
+/* for chalk elec cape with 12" panel */
+struct panel_info default_panel_info = {
+	.panel_width = 1280,
+	.panel_height = 800,
+	.panel_hfp = 48,
+	.panel_hbp = 80,
+	.panel_hsw = 32,
+	.panel_vfp = 2,
+	.panel_vbp = 15,
+	.panel_vsw = 6,
+	.ac_bias = 255,
+	.ac_bias_intrpt = 0,
+	.dma_burst_sz = 16,
+	.bpp = 16,
+	.fdd = 16,
+	.invert_line_clock = 0,
+	.invert_frm_clock = 0,
+	.sync_edge = 0,
+	.sync_ctrl = 1,
+	.panel_pxl_clk = 69300000,
+	.panel_invert_pxl_clk = 0,
+};
+
+struct tifb_softc {
+	device_t sc_dev;
+
+	void *sc_ih;
+	bus_space_tag_t sc_iot;
+	bus_dma_tag_t sc_dmat;
+	bus_space_handle_t sc_regh;
+	bus_dmamap_t sc_dmamap;
+	bus_dma_segment_t sc_dmamem[1];
+	size_t sc_vramsize;
+	size_t sc_palettesize;
+
+	int sc_stride;
+	int sc_locked;
+	void *sc_fbaddr, *sc_vramaddr;
+
+	bus_addr_t sc_fbhwaddr;
+	uint16_t *sc_palette;
+	uint32_t sc_dispc_config;
+	struct vcons_screen sc_console_screen;
+	struct wsscreen_descr sc_defaultscreen_descr;
+	const struct wsscreen_descr *sc_screens[1];
+	struct wsscreen_list sc_screenlist;
+	struct vcons_data vd;
+	int sc_mode;
+	uint8_t sc_cmap_red[256], sc_cmap_green[256], sc_cmap_blue[256];
+	void (*sc_putchar)(void *, int, int, u_int, long);
+
+	uint8_t sc_edid_data[1024];
+	size_t sc_edid_size;
+
+	struct panel_info *sc_panel;
+};
+
+#define TIFB_READ(sc, reg) bus_space_read_4(sc->sc_iot, sc->sc_regh, reg)
+#define TIFB_WRITE(sc, reg, val) bus_space_write_4(sc->sc_iot, sc->sc_regh, reg, val)
+
+static int	tifb_match(device_t, cfdata_t, void *);
+static void	tifb_attach(device_t, device_t, void *);
+static int	tifb_intr(void *);
+
+
+#ifdef TI_AM335X
+static void am335x_clk_lcdc_activate(void);
+static int  am335x_clk_get_arm_disp_freq(unsigned int *);
+#endif
+
+
+CFATTACH_DECL_NEW(tifb, sizeof(struct tifb_softc),
+    tifb_match, tifb_attach, NULL, NULL);
+
+static int	tifb_ioctl(void *, void *, u_long, void *, int,
+			     struct lwp *);
+static paddr_t	tifb_mmap(void *, void *, off_t, int);
+static void	tifb_init_screen(void *, struct vcons_screen *, int, long *);
+
+static int	tifb_putcmap(struct tifb_softc *, struct wsdisplay_cmap *);
+static int 	tifb_getcmap(struct tifb_softc *, struct wsdisplay_cmap *);
+#if 0
+static void	tifb_restore_palette(struct tifb_softc *);
+static void 	tifb_putpalreg(struct tifb_softc *, int, uint8_t,
+			    uint8_t, uint8_t);
+
+static int	tifb_set_depth(struct tifb_softc *, int);
+#endif
+static void	tifb_set_video(struct tifb_softc *, int);
+
+struct wsdisplay_accessops tifb_accessops = {
+	tifb_ioctl,
+	tifb_mmap,
+	NULL,	/* alloc_screen */
+	NULL,	/* free_screen */
+	NULL,	/* show_screen */
+	NULL, 	/* load_font */
+	NULL,	/* pollc */
+	NULL	/* scroll */
+};
+
+extern const u_char rasops_cmap[768];
+
+static struct evcnt ev_sync_lost;
+static struct evcnt ev_palette;
+static struct evcnt ev_eof0;
+static struct evcnt ev_eof1;
+static struct evcnt ev_fifo_underflow;
+static struct evcnt ev_ac_bias;
+static struct evcnt ev_others;
+
+
+static uint32_t
+am335x_lcd_calc_divisor(uint32_t reference, uint32_t freq)
+{
+	uint32_t div;
+	/* Raster mode case: divisors are in range from 2 to 255 */
+	for (div = 2; div < 255; div++)
+		if (reference/div <= freq)
+			return (div);
+
+	return (255);
+}
+
+static int
+tifb_match(device_t parent, cfdata_t match, void *aux)
+{
+	struct obio_attach_args *obio = aux;
+
+	if ((obio->obio_addr == -1) || (obio->obio_size == 0))
+		return 0;
+	return 1;
+}
+
+static void
+tifb_attach(device_t parent, device_t self, void *aux)
+{
+	struct tifb_softc	*sc = device_private(self);
+	struct obio_attach_args *obio = aux;
+	struct rasops_info	*ri;
+	struct wsemuldisplaydev_attach_args aa;
+	prop_dictionary_t	dict;
+	/* prop_data_t		edid_data; XXX */
+	unsigned long		defattr;
+	bool			is_console = false;
+	uint32_t		reg, timing0, timing1, timing2, burst_log;
+	int			segs, i, div, ref_freq;
+
+#ifdef TI_AM335X
+	int ret;
+	const char *mode;
+	u_int state;
+	struct tifb_padconf {
+		const char *padname;
+		const char *padmode;
+	};
+	const struct tifb_padconf tifb_padconf_data[] = {
+		{"LCD_DATA0",  "lcd_data0"},
+		{"LCD_DATA1",  "lcd_data1"},
+		{"LCD_DATA2",  "lcd_data2"},
+		{"LCD_DATA3",  "lcd_data3"},
+		{"LCD_DATA4",  "lcd_data4"},
+		{"LCD_DATA5",  "lcd_data5"},
+		{"LCD_DATA6",  "lcd_data6"},
+		{"LCD_DATA7",  "lcd_data7"},
+		{"LCD_DATA8",  "lcd_data8"},
+		{"LCD_DATA9",  "lcd_data9"},
+		{"LCD_DATA10", "lcd_data10"},
+		{"LCD_DATA11", "lcd_data11"},
+		{"LCD_DATA12", "lcd_data12"},
+		{"LCD_DATA13", "lcd_data13"},
+		{"LCD_DATA14", "lcd_data14"},
+		{"LCD_DATA15", "lcd_data15"},
+		{"GPMC_AD15",  "lcd_data16"},
+		{"GPMC_AD14",  "lcd_data17"},
+		{"GPMC_AD13",  "lcd_data18"},
+		{"GPMC_AD12",  "lcd_data19"},
+		{"GPMC_AD11",  "lcd_data20"},
+		{"GPMC_AD10",  "lcd_data21"},
+		{"GPMC_AD9",   "lcd_data22"},
+		{"GPMC_AD8",   "lcd_data23"},
+	};
+	const struct tifb_padconf tifb_padconf_ctrl[] = {
+		{"LCD_VSYNC", "lcd_vsync"},
+		{"LCD_HSYNC", "lcd_hsync"},
+		{"LCD_PCLK",  "lcd_pclk"},
+		{"LCD_AC_BIAS_EN", "lcd_ac_bias_en"},
+	};
+#endif
+
+	evcnt_attach_dynamic(&ev_sync_lost, EVCNT_TYPE_MISC, NULL,
+	    "lcd", "sync lost");
+	evcnt_attach_dynamic(&ev_palette, EVCNT_TYPE_MISC, NULL,
+	    "lcd", "palette loaded");
+	evcnt_attach_dynamic(&ev_eof0, EVCNT_TYPE_MISC, NULL,
+	    "lcd", "eof0");
+	evcnt_attach_dynamic(&ev_eof1, EVCNT_TYPE_MISC, NULL,
+	    "lcd", "eof1");
+	evcnt_attach_dynamic(&ev_fifo_underflow, EVCNT_TYPE_MISC, NULL,
+	    "lcd", "fifo underflow");
+	evcnt_attach_dynamic(&ev_ac_bias, EVCNT_TYPE_MISC, NULL,
+	    "lcd", "ac bias");
+	evcnt_attach_dynamic(&ev_others, EVCNT_TYPE_MISC, NULL,
+	    "lcd", "others");
+
+	sc->sc_iot = obio->obio_iot;
+	sc->sc_dev = self;
+	sc->sc_dmat = obio->obio_dmat;
+
+	if (bus_space_map(obio->obio_iot, obio->obio_addr, obio->obio_size, 0,
+	    &sc->sc_regh)) {
+		aprint_error(": couldn't map register space\n");
+		return;
+	}
+
+	sc->sc_panel = &default_panel_info;/* XXX */
+
+	switch(sc->sc_panel->bpp) {
+	case 16:
+	case 24:
+	case 32:
+		break;
+	default:
+		aprint_error_dev(self, "bogus display bpp, not attaching\n");
+		return;
+	}
+
+	sc->sc_stride = sc->sc_panel->panel_width * sc->sc_panel->bpp / 8;
+
+	if (sc->sc_panel->panel_width == 0 ||
+	    sc->sc_panel->panel_height == 0) {
+		aprint_error_dev(self, "bogus display size, not attaching\n");
+		return;
+	}
+
+	if (obio->obio_intr == OBIOCF_INTR_DEFAULT) {
+		aprint_error(": no interrupt\n");
+		bus_space_unmap(obio->obio_iot, sc->sc_regh,
+		    obio->obio_size);
+		return;
+	}
+	sc->sc_ih = intr_establish(obio->obio_intr, IPL_NET, IST_LEVEL,
+	    tifb_intr, sc);
+	KASSERT(sc->sc_ih != NULL);
+
+	dict = device_properties(self);
+	prop_dictionary_get_bool(dict, "is_console", &is_console);
+#if 0
+	edid_data = prop_dictionary_get(dict, "EDID");
+	if (edid_data != NULL) {
+		struct edid_info ei;
+
+		sc->sc_edid_size = min(prop_data_size(edid_data), 1024);
+		memset(sc->sc_edid_data, 0, sizeof(sc->sc_edid_data));
+		memcpy(sc->sc_edid_data, prop_data_data_nocopy(edid_data),
+		    sc->sc_edid_size);
+
+		edid_parse(sc->sc_edid_data, &ei);
+		edid_print(&ei);
+	}
+#endif
+
+	/* setup video DMA */
+	switch(sc->sc_panel->bpp) {
+	case 8:
+		sc->sc_palettesize = 512;
+		break;
+	default:
+		sc->sc_palettesize = 32;
+		break;
+	}
+
+	sc->sc_vramsize = sc->sc_palettesize +
+	    sc->sc_stride * sc->sc_panel->panel_height;
+
+	if (bus_dmamem_alloc(sc->sc_dmat, sc->sc_vramsize, 0, 0,
+	    sc->sc_dmamem, 1, &segs, BUS_DMA_NOWAIT) != 0) {
+		aprint_error_dev(sc->sc_dev,
+		    "failed to allocate video memory\n");
+		return;
+	}
+
+	if (bus_dmamem_map(sc->sc_dmat, sc->sc_dmamem, 1, sc->sc_vramsize,
+	    &sc->sc_vramaddr, BUS_DMA_NOWAIT | BUS_DMA_COHERENT) != 0) {
+		aprint_error_dev(sc->sc_dev, "failed to map video RAM\n");
+		return;
+	}
+	sc->sc_fbaddr = (uint8_t *)sc->sc_vramaddr + sc->sc_palettesize;
+	sc->sc_palette = sc->sc_vramaddr;
+
+	if (bus_dmamap_create(sc->sc_dmat, sc->sc_vramsize, 1, sc->sc_vramsize,
+	    0, BUS_DMA_NOWAIT, &sc->sc_dmamap) != 0) {
+		aprint_error_dev(sc->sc_dev, "failed to create DMA map\n");
+		return;
+	}
+
+	if (bus_dmamap_load(sc->sc_dmat, sc->sc_dmamap, sc->sc_vramaddr,
+	    sc->sc_vramsize, NULL, BUS_DMA_NOWAIT) != 0) {
+		aprint_error_dev(sc->sc_dev, "failed to load DMA map\n");
+		return;
+	}
+
+	memset((void *)sc->sc_vramaddr, 0, sc->sc_vramsize);
+
+	sc->sc_palette[0] = 0x4000;
+#if 0 /* XXX */
+	if (sc->sc_panel->bpp == 8) {
+		j = 0;
+		for (i = 0; i < 256; i++) {
+			sc->sc_cmap_red[i] = rasops_cmap[j];
+			sc->sc_cmap_green[i] = rasops_cmap[j + 1];
+			sc->sc_cmap_blue[i] = rasops_cmap[j + 2];
+			j += 3;
+		}
+	} else {
+		for (i = 0; i < 256; i++) {
+			sc->sc_cmap_red[i] = i;
+			sc->sc_cmap_green[i] = i;
+			sc->sc_cmap_blue[i] = i;
+		}
+	}
+	tifb_restore_palette(sc);
+#endif
+
+#ifdef TI_AM335X
+	/* configure output pins */
+	for (i = 0; i < ((sc->sc_panel->bpp == 16) ? 16 : 23); i++) {
+		if (sitara_cm_padconf_get(tifb_padconf_data[i].padname,
+		    &mode, &state) == 0) {
+			aprint_debug(": %s mode %s state %d ",
+			    tifb_padconf_data[i].padname, mode, state);
+		}
+		if (sitara_cm_padconf_set(tifb_padconf_data[i].padname,
+		    tifb_padconf_data[i].padmode, 0) != 0) {
+			aprint_error(": can't switch %s pad from %s to %s\n", tifb_padconf_data[i].padname, mode, tifb_padconf_data[i].padmode);
+			return;
+		}
+	}
+	for (i = 0; i < __arraycount(tifb_padconf_ctrl); i++) {
+		if (sitara_cm_padconf_get(tifb_padconf_ctrl[i].padname,
+		    &mode, &state) == 0) {
+			aprint_debug(": %s mode %s state %d ",
+			    tifb_padconf_ctrl[i].padname, mode, state);
+		}
+		if (sitara_cm_padconf_set(tifb_padconf_ctrl[i].padname,
+		    tifb_padconf_ctrl[i].padmode, 0) != 0) {
+			aprint_error(": can't switch %s pad from %s to %s\n", tifb_padconf_ctrl[i].padname, mode, tifb_padconf_ctrl[i].padmode);
+			return;
+		}
+	}
+	am335x_clk_lcdc_activate();
+	/* get reference clk freq */
+	ret = am335x_clk_get_arm_disp_freq(&ref_freq);
+	if (ret != 0) {
+		aprint_error_dev(self, "display clock not enabled\n");
+		return;
+	}
+#endif
+	aprint_normal(": TI LCD controller\n");
+	aprint_debug_dev(self, ": fb@%p, palette@%p\n", sc->sc_fbaddr,
+	    sc->sc_palette);
+	/* Only raster mode is supported */
+	reg = CTRL_RASTER_MODE;
+	div = am335x_lcd_calc_divisor(ref_freq, sc->sc_panel->panel_pxl_clk);
+	reg |= (div << CTRL_DIV_SHIFT);
+	TIFB_WRITE(sc, LCD_CTRL, reg);
+	aprint_debug_dev(self, ": ref_freq %d div %d\n", ref_freq, div);
+
+	/* Set timing */
+	timing0 = timing1 = timing2 = 0;
+
+	/* Horizontal back porch */
+	timing0 |= (sc->sc_panel->panel_hbp & 0xff) << RASTER_TIMING_0_HBP_SHIFT;
+	timing2 |= ((sc->sc_panel->panel_hbp >> 8) & 3) << RASTER_TIMING_2_HBPHI_SHIFT;
+	/* Horizontal front porch */
+	timing0 |= (sc->sc_panel->panel_hfp & 0xff) << RASTER_TIMING_0_HFP_SHIFT;
+	timing2 |= ((sc->sc_panel->panel_hfp >> 8) & 3) << RASTER_TIMING_2_HFPHI_SHIFT;
+	/* Horizontal sync width */   
+	timing0 |= (sc->sc_panel->panel_hsw & 0x3f) << RASTER_TIMING_0_HSW_SHIFT;
+	timing2 |= ((sc->sc_panel->panel_hsw >> 6) & 0xf) << RASTER_TIMING_2_HSWHI_SHIFT
+;
+
+	/* Vertical back porch, front porch, sync width */
+	timing1 |= (sc->sc_panel->panel_vbp & 0xff) << RASTER_TIMING_1_VBP_SHIFT;
+	timing1 |= (sc->sc_panel->panel_vfp & 0xff) << RASTER_TIMING_1_VFP_SHIFT;
+	timing1 |= (sc->sc_panel->panel_vsw & 0x3f) << RASTER_TIMING_1_VSW_SHIFT;
+	/* Pixels per line */
+	timing0 |= (((sc->sc_panel->panel_width - 1) >> 10) & 1)
+	    << RASTER_TIMING_0_PPLMSB_SHIFT;
+	timing0 |= (((sc->sc_panel->panel_width - 1) >> 4) & 0x3f)
+	    << RASTER_TIMING_0_PPLLSB_SHIFT;
+
+	/* Lines per panel */
+	timing1 |= ((sc->sc_panel->panel_height - 1) & 0x3ff)
+	    << RASTER_TIMING_1_LPP_SHIFT;
+	timing2 |= (((sc->sc_panel->panel_height - 1) >> 10 ) & 1)
+	    << RASTER_TIMING_2_LPP_B10_SHIFT;
+
+	/* clock signal settings */   
+	if (sc->sc_panel->sync_ctrl)
+		timing2 |= RASTER_TIMING_2_PHSVS;
+	if (sc->sc_panel->sync_edge)
+		timing2 |= RASTER_TIMING_2_PHSVS_RISE;
+	else
+		timing2 |= RASTER_TIMING_2_PHSVS_FALL;
+	if (sc->sc_panel->invert_line_clock)  
+		timing2 |= RASTER_TIMING_2_IHS;
+	if (sc->sc_panel->invert_frm_clock)   
+		timing2 |= RASTER_TIMING_2_IVS;
+	if (sc->sc_panel->panel_invert_pxl_clk)
+		timing2 |= RASTER_TIMING_2_IPC;
+
+	/* AC bias */
+	timing2 |= (sc->sc_panel->ac_bias << RASTER_TIMING_2_ACB_SHIFT);
+	timing2 |= (sc->sc_panel->ac_bias_intrpt << RASTER_TIMING_2_ACBI_SHIFT);
+
+	TIFB_WRITE(sc, LCD_RASTER_TIMING_0, timing0);
+	TIFB_WRITE(sc, LCD_RASTER_TIMING_1, timing1);
+	TIFB_WRITE(sc, LCD_RASTER_TIMING_2, timing2);
+	aprint_debug_dev(self, ": timings 0x%x 0x%x 0x%x\n",
+	    timing0, timing1, timing2);
+
+	/* DMA settings */
+	reg = LCDDMA_CTRL_FB0_FB1;    
+	/* Find power of 2 for current burst size */
+	switch (sc->sc_panel->dma_burst_sz) { 
+	case 1:
+		burst_log = 0;
+		break;
+	case 2:
+		burst_log = 1;
+		break;
+	case 4:
+		burst_log = 2;
+		break;
+	case 8:
+		burst_log = 3;
+		break;
+	case 16:
+	default:
+		burst_log = 4;
+		break;
+	}
+	reg |= (burst_log << LCDDMA_CTRL_BURST_SIZE_SHIFT);
+	/* XXX: FIFO TH */
+	reg |= (0 << LCDDMA_CTRL_TH_FIFO_RDY_SHIFT);
+	TIFB_WRITE(sc, LCD_LCDDMA_CTRL, reg);
+	aprint_debug_dev(self, ": LCD_LCDDMA_CTRL 0x%x\n", reg);
+
+	TIFB_WRITE(sc, LCD_LCDDMA_FB0_BASE, sc->sc_dmamem->ds_addr);
+	TIFB_WRITE(sc, LCD_LCDDMA_FB0_CEILING,
+	    sc->sc_dmamem->ds_addr + sc->sc_vramsize - 1);
+	TIFB_WRITE(sc, LCD_LCDDMA_FB1_BASE, sc->sc_dmamem->ds_addr);
+	TIFB_WRITE(sc, LCD_LCDDMA_FB1_CEILING,
+	    sc->sc_dmamem->ds_addr + sc->sc_vramsize - 1);
+	aprint_debug_dev(self, ": LCD_LCDDMA 0x%x 0x%x\n",
+	    (u_int)sc->sc_dmamem->ds_addr, (u_int)(sc->sc_dmamem->ds_addr + sc->sc_vramsize - 1));
+
+	/* Enable LCD */
+	reg = RASTER_CTRL_LCDTFT;     
+	reg |= (sc->sc_panel->fdd << RASTER_CTRL_REQDLY_SHIFT);
+	reg |= (PALETTE_DATA_ONLY << RASTER_CTRL_PALMODE_SHIFT);
+	if (sc->sc_panel->bpp >= 24)
+		reg |= RASTER_CTRL_TFT24;
+	if (sc->sc_panel->bpp == 32)
+		reg |= RASTER_CTRL_TFT24_UNPACKED;
+	TIFB_WRITE(sc, LCD_RASTER_CTRL, reg);
+	aprint_debug_dev(self, ": LCD_RASTER_CTRL 0x%x\n", reg);
+
+	TIFB_WRITE(sc, LCD_CLKC_ENABLE,
+	    CLKC_ENABLE_DMA | CLKC_ENABLE_LDID | CLKC_ENABLE_CORE);     
+
+	TIFB_WRITE(sc, LCD_CLKC_RESET, CLKC_RESET_MAIN);
+	DELAY(100);
+	TIFB_WRITE(sc, LCD_CLKC_RESET, 0);
+	aprint_debug_dev(self, ": LCD_CLKC_ENABLE 0x%x\n", TIFB_READ(sc, LCD_CLKC_ENABLE));
+
+	reg = IRQ_EOF1 | IRQ_EOF0 | IRQ_FUF | IRQ_PL |
+	    IRQ_ACB | IRQ_SYNC_LOST |  IRQ_RASTER_DONE |
+	    IRQ_FRAME_DONE;
+	TIFB_WRITE(sc, LCD_IRQENABLE_SET, reg);
+
+	reg = TIFB_READ(sc, LCD_RASTER_CTRL);
+	reg |= RASTER_CTRL_LCDEN;     
+	TIFB_WRITE(sc, LCD_RASTER_CTRL, reg);
+	aprint_debug_dev(self, ": LCD_RASTER_CTRL 0x%x\n", TIFB_READ(sc, LCD_RASTER_CTRL));
+
+	TIFB_WRITE(sc, LCD_SYSCONFIG, 
+	    SYSCONFIG_STANDBY_SMART | SYSCONFIG_IDLE_SMART);
+	aprint_debug_dev(self, ": LCD_SYSCONFIG 0x%x\n", TIFB_READ(sc, LCD_SYSCONFIG));
+
+	/* attach wscons */
+	sc->sc_defaultscreen_descr = (struct wsscreen_descr){
+		"default",
+		0, 0,
+		NULL,
+		8, 16,
+		WSSCREEN_WSCOLORS | WSSCREEN_HILIT,
+		NULL
+	};
+	sc->sc_screens[0] = &sc->sc_defaultscreen_descr;
+	sc->sc_screenlist = (struct wsscreen_list){1, sc->sc_screens};
+	sc->sc_mode = WSDISPLAYIO_MODE_EMUL;
+	sc->sc_locked = 0;
+
+	vcons_init(&sc->vd, sc, &sc->sc_defaultscreen_descr,
+	    &tifb_accessops);
+	sc->vd.init_screen = tifb_init_screen;
+
+	/* init engine here */
+
+	ri = &sc->sc_console_screen.scr_ri;
+
+	vcons_init_screen(&sc->vd, &sc->sc_console_screen, 1, &defattr);
+	sc->sc_console_screen.scr_flags |= VCONS_SCREEN_IS_STATIC;
+	sc->sc_defaultscreen_descr.textops = &ri->ri_ops;
+	sc->sc_defaultscreen_descr.capabilities = ri->ri_caps;
+	sc->sc_defaultscreen_descr.nrows = ri->ri_rows;
+	sc->sc_defaultscreen_descr.ncols = ri->ri_cols;
+	if (is_console) {
+		wsdisplay_cnattach(&sc->sc_defaultscreen_descr, ri, 0, 0,
+		    defattr);
+		vcons_replay_msgbuf(&sc->sc_console_screen);
+	}
+
+	aa.console = is_console;
+	aa.scrdata = &sc->sc_screenlist;
+	aa.accessops = &tifb_accessops;
+	aa.accesscookie = &sc->vd;
+
+	config_found(sc->sc_dev, &aa, wsemuldisplaydevprint);
+}
+
+static int
+tifb_intr(void *v)
+{
+	struct tifb_softc *sc = v;
+	uint32_t reg;
+
+	reg = TIFB_READ(sc, LCD_IRQSTATUS);
+	TIFB_WRITE(sc, LCD_IRQSTATUS, reg);
+
+	if (reg & IRQ_SYNC_LOST) {
+		ev_sync_lost.ev_count ++;
+		reg = TIFB_READ(sc, LCD_RASTER_CTRL);
+		reg &= ~RASTER_CTRL_LCDEN;
+		TIFB_WRITE(sc, LCD_RASTER_CTRL, reg);
+
+		reg = TIFB_READ(sc, LCD_RASTER_CTRL);
+		reg |= RASTER_CTRL_LCDEN;
+		TIFB_WRITE(sc, LCD_RASTER_CTRL, reg);
+		return 0;
+	}
+
+	if (reg & IRQ_PL) {
+		ev_palette.ev_count ++;
+		reg = TIFB_READ(sc, LCD_RASTER_CTRL);
+		reg &= ~RASTER_CTRL_LCDEN;
+		TIFB_WRITE(sc, LCD_RASTER_CTRL, reg);
+
+		reg = TIFB_READ(sc, LCD_RASTER_CTRL);
+		reg |= RASTER_CTRL_LCDEN;
+		TIFB_WRITE(sc, LCD_RASTER_CTRL, reg);
+		return 0;
+	}
+
+	if (reg & IRQ_EOF0) {
+		ev_eof0.ev_count ++;
+		TIFB_WRITE(sc, LCD_LCDDMA_FB0_BASE, sc->sc_dmamem->ds_addr);
+		TIFB_WRITE(sc, LCD_LCDDMA_FB0_CEILING,
+		    sc->sc_dmamem->ds_addr + sc->sc_vramsize - 1);
+		reg &= ~IRQ_EOF0;
+	}
+
+	if (reg & IRQ_EOF1) {
+		ev_eof1.ev_count ++;
+		TIFB_WRITE(sc, LCD_LCDDMA_FB1_BASE, sc->sc_dmamem->ds_addr);
+		TIFB_WRITE(sc, LCD_LCDDMA_FB1_CEILING,
+		    sc->sc_dmamem->ds_addr + sc->sc_vramsize - 1);
+		reg &= ~IRQ_EOF1;     
+	}
+
+	if (reg & IRQ_FUF) {
+		ev_fifo_underflow.ev_count ++;
+		/* TODO: Handle FUF */
+		reg =~ IRQ_FUF;
+	}
+
+	if (reg & IRQ_ACB) {
+		ev_ac_bias.ev_count ++;
+		/* TODO: Handle ACB */
+		reg =~ IRQ_ACB;
+	}
+	if (reg)
+		ev_others.ev_count ++;
+	return 0;
+}
+
+static int
+tifb_ioctl(void *v, void *vs, u_long cmd, void *data, int flag,
+	struct lwp *l)
+{
+	struct vcons_data *vd = v;
+	struct tifb_softc *sc = vd->cookie;
+	struct wsdisplay_fbinfo *wdf;
+	struct vcons_screen *ms = vd->active;
+
+	switch (cmd) {
+
+		case WSDISPLAYIO_GTYPE:
+			*(u_int *)data = WSDISPLAY_TYPE_GENFB;
+			return 0;
+
+		case WSDISPLAYIO_GET_BUSID:
+			{
+				struct wsdisplayio_bus_id *busid;
+
+				busid = data;
+				busid->bus_type = WSDISPLAYIO_BUS_SOC;
+				return 0;
+			}
+
+		case WSDISPLAYIO_GINFO:
+			if (ms == NULL)
+				return ENODEV;
+			wdf = (void *)data;
+			wdf->height = ms->scr_ri.ri_height;
+			wdf->width = ms->scr_ri.ri_width;
+			wdf->depth = ms->scr_ri.ri_depth;
+			wdf->cmsize = 256;
+			return 0;
+
+		case WSDISPLAYIO_GETCMAP:
+			return tifb_getcmap(sc,
+			    (struct wsdisplay_cmap *)data);
+
+		case WSDISPLAYIO_PUTCMAP:
+			return tifb_putcmap(sc,
+			    (struct wsdisplay_cmap *)data);
+
+		case WSDISPLAYIO_LINEBYTES:
+			*(u_int *)data = sc->sc_stride;
+			return 0;
+
+		case WSDISPLAYIO_SMODE:
+			{
+				int new_mode = *(int*)data;
+
+				if (new_mode != sc->sc_mode) {
+					sc->sc_mode = new_mode;
+#if 0
+					if (new_mode == WSDISPLAYIO_MODE_EMUL) {
+						tifb_set_depth(sc, 16);
+						vcons_redraw_screen(ms);
+					} else {
+						tifb_set_depth(sc, 32);
+					}
+#endif
+				}
+			}
+			return 0;
+
+		case WSDISPLAYIO_GVIDEO:
+			{
+				int *on = data;
+				*on = 1; /* XXX sc->sc_video_is_on; */
+			}
+			return 0;
+
+		case WSDISPLAYIO_SVIDEO:
+			{
+				int *on = data;
+				tifb_set_video(sc, *on);
+			}
+			return 0;
+
+	}
+	return EPASSTHROUGH;
+}
+
+static paddr_t
+tifb_mmap(void *v, void *vs, off_t offset, int prot)
+{
+	paddr_t pa = -1;
+	struct vcons_data *vd = v;
+	struct tifb_softc *sc = vd->cookie;
+
+	/* 'regular' framebuffer mmap()ing */
+	if (offset < sc->sc_stride * sc->sc_panel->panel_height) {
+		pa = bus_dmamem_mmap(sc->sc_dmat, sc->sc_dmamem, 1,
+		    offset, prot, BUS_DMA_PREFETCHABLE);
+		return pa;
+	}
+	return pa;
+}
+
+static void
+tifb_init_screen(void *cookie, struct vcons_screen *scr,
+    int existing, long *defattr)
+{
+	struct tifb_softc *sc = cookie;
+	struct rasops_info *ri = &scr->scr_ri;
+
+	ri->ri_depth = sc->sc_panel->bpp;
+	ri->ri_width = sc->sc_panel->panel_width;
+	ri->ri_height = sc->sc_panel->panel_height;
+	ri->ri_stride = sc->sc_stride;
+	ri->ri_flg = RI_CENTER | RI_FULLCLEAR;
+
+	ri->ri_bits = (char *)sc->sc_fbaddr;
+
+	if (existing) {
+		ri->ri_flg |= RI_CLEAR;
+	}
+
+	rasops_init(ri, sc->sc_panel->panel_height / 8, sc->sc_panel->panel_width / 8);
+	ri->ri_caps = WSSCREEN_WSCOLORS;
+
+	rasops_reconfig(ri, sc->sc_panel->panel_height / ri->ri_font->fontheight,
+		    sc->sc_panel->panel_width / ri->ri_font->fontwidth);
+
+	ri->ri_hw = scr;
+}
+
+static int
+tifb_putcmap(struct tifb_softc *sc, struct wsdisplay_cmap *cm)
+{
+	return EINVAL; /* XXX */
+#if 0
+	u_char *r, *g, *b;
+	u_int index = cm->index;
+	u_int count = cm->count;
+	int i, error;
+	u_char rbuf[256], gbuf[256], bbuf[256];
+
+	if (cm->index >= 256 || cm->count > 256 ||
+	    (cm->index + cm->count) > 256)
+		return EINVAL;
+	error = copyin(cm->red, &rbuf[index], count);
+	if (error)
+		return error;
+	error = copyin(cm->green, &gbuf[index], count);
+	if (error)
+		return error;
+	error = copyin(cm->blue, &bbuf[index], count);
+	if (error)
+		return error;
+
+	memcpy(&sc->sc_cmap_red[index], &rbuf[index], count);
+	memcpy(&sc->sc_cmap_green[index], &gbuf[index], count);
+	memcpy(&sc->sc_cmap_blue[index], &bbuf[index], count);
+
+	r = &sc->sc_cmap_red[index];
+	g = &sc->sc_cmap_green[index];
+	b = &sc->sc_cmap_blue[index];
+
+	for (i = 0; i < count; i++) {
+		tifb_putpalreg(sc, index, *r, *g, *b);
+		index++;
+		r++, g++, b++;
+	}
+	return 0;
+#endif
+}
+
+static int
+tifb_getcmap(struct tifb_softc *sc, struct wsdisplay_cmap *cm)
+{
+	return EINVAL; /* XXX */
+#if 0
+	u_int index = cm->index;
+	u_int count = cm->count;
+	int error;
+
+	if (index >= 255 || count > 256 || index + count > 256)
+		return EINVAL;
+
+	error = copyout(&sc->sc_cmap_red[index],   cm->red,   count);
+	if (error)
+		return error;
+	error = copyout(&sc->sc_cmap_green[index], cm->green, count);
+	if (error)
+		return error;
+	error = copyout(&sc->sc_cmap_blue[index],  cm->blue,  count);
+	if (error)
+		return error;
+
+	return 0;
+#endif
+}
+
+#if 0
+static void
+tifb_restore_palette(struct tifb_softc *sc)
+{
+	int i;
+
+	for (i = 0; i < 256; i++) {
+		tifb_putpalreg(sc, i, sc->sc_cmap_red[i],
+		    sc->sc_cmap_green[i], sc->sc_cmap_blue[i]);
+	}
+}
+
+static void
+tifb_putpalreg(struct tifb_softc *sc, int idx, uint8_t r, uint8_t g,
+    uint8_t b)
+{
+	uint32_t reg;
+
+	if ((idx < 0) || (idx > 255))
+		return;
+	/* whack the DAC */
+	reg = (r << 16) | (g << 8) | b;
+	sc->sc_clut[idx] = reg;
+}
+
+static int
+tifb_set_depth(struct tifb_softc *sc, int d)
+{
+	uint32_t reg;
+
+	reg = OMAP_VID_ATTR_ENABLE |
+	      OMAP_VID_ATTR_BURST_16x32 |
+	      OMAP_VID_ATTR_REPLICATION;
+	switch (d) {
+		case 16:
+			reg |= OMAP_VID_ATTR_RGB16;
+			break;
+		case 32:
+			reg |= OMAP_VID_ATTR_RGB24;
+			break;
+		default:
+			aprint_error_dev(sc->sc_dev,
+			    "unsupported depth (%d)\n", d);
+			return EINVAL;
+	}
+
+	bus_space_write_4(sc->sc_iot, sc->sc_regh,
+	    OMAPFB_DISPC_VID1_ATTRIBUTES, reg);
+
+	/*
+	 * now tell the video controller that we're done mucking around and
+	 * actually update its settings
+	 */
+	reg = bus_space_read_4(sc->sc_iot, sc->sc_regh, OMAPFB_DISPC_CONTROL);
+	bus_space_write_4(sc->sc_iot, sc->sc_regh, OMAPFB_DISPC_CONTROL,
+	    reg | OMAP_DISPC_CTRL_GO_LCD | OMAP_DISPC_CTRL_GO_DIGITAL);
+
+	sc->sc_panel->bpp = d;
+	sc->sc_stride = sc->sc_panel->panel_width * (sc->sc_panel->bpp >> 3);
+
+	/* clear the screen here */
+	memset(sc->sc_fbaddr, 0, sc->sc_stride * sc->sc_panel->panel_height);
+	return 0;
+}
+#endif /* 0 */
+
+static void
+tifb_set_video(struct tifb_softc *sc, int on)
+{
+#if 0
+	uint32_t reg;
+
+	if (on == sc->sc_video_is_on)
+		return;
+	if (on) {
+		bus_space_write_4(sc->sc_iot, sc->sc_regh,
+		    OMAPFB_DISPC_CONFIG, sc->sc_dispc_config);
+		on = 1;
+	} else {
+		bus_space_write_4(sc->sc_iot, sc->sc_regh,
+		    OMAPFB_DISPC_CONFIG, sc->sc_dispc_config |
+		    OMAP_DISPC_CFG_VSYNC_GATED | OMAP_DISPC_CFG_HSYNC_GATED |
+		    OMAP_DISPC_CFG_PIXELCLK_GATED |
+		    OMAP_DISPC_CFG_PIXELDATA_GATED);
+	}
+
+	/*
+	 * now tell the video controller that we're done mucking around and
+	 * actually update its settings
+	 */
+	reg = bus_space_read_4(sc->sc_iot, sc->sc_regh, OMAPFB_DISPC_CONTROL);
+	bus_space_write_4(sc->sc_iot, sc->sc_regh, OMAPFB_DISPC_CONTROL,
+	    reg | OMAP_DISPC_CTRL_GO_LCD | OMAP_DISPC_CTRL_GO_DIGITAL);
+
+	aprint_debug_dev(sc->sc_dev, "%s %08x\n", __func__,
+	    bus_space_read_4(sc->sc_iot, sc->sc_regh, OMAPFB_DISPC_CONFIG));
+	sc->sc_video_is_on = on;
+#endif
+}
+
+#ifdef TI_AM335X
+static void
+am335x_clk_lcdc_activate(void)
+{
+	/* Bypass mode */
+	prcm_write_4(AM335X_PRCM_CM_WKUP, CM_WKUP_CM_CLKMODE_DPLL_DISP,
+	    AM335X_PRCM_CM_CLKMODE_DPLL_MN_BYP_MODE);
+
+	/* Make sure it's in bypass mode */
+	while (!(prcm_read_4(AM335X_PRCM_CM_WKUP, CM_WKUP_CM_IDLEST_DPLL_DISP)
+	    & (1 << 8)))
+		DELAY(10);
+
+	/*
+	 * For now set frequency to  5xSYSFREQ
+	 * More flexible control might be required
+	 */
+	prcm_write_4(AM335X_PRCM_CM_WKUP, CM_WKUP_CM_CLKSEL_DPLL_DISP,
+	    (5 << 8) | 0);
+
+	/* Locked mode */
+	prcm_write_4(AM335X_PRCM_CM_WKUP, CM_WKUP_CM_CLKMODE_DPLL_DISP, 0x7);
+
+	int timeout = 10000;
+	while ((!(prcm_read_4(AM335X_PRCM_CM_WKUP, CM_WKUP_CM_IDLEST_DPLL_DISP)
+	    & (1 << 0))) && timeout--)
+		DELAY(10);
+
+	/*set MODULEMODE to ENABLE(2) */
+	prcm_write_4(AM335X_PRCM_CM_PER, CM_PER_LCDC_CLKCTRL, 2);
+
+	/* wait for MODULEMODE to become ENABLE(2) */
+	while ((prcm_read_4(AM335X_PRCM_CM_PER, CM_PER_LCDC_CLKCTRL) & 0x3) != 2)
+		DELAY(10);
+
+	/* wait for IDLEST to become Func(0) */
+	while(prcm_read_4(AM335X_PRCM_CM_PER, CM_PER_LCDC_CLKCTRL) & (3<<16))
+		DELAY(10);
+}
+
+static int
+am335x_clk_get_arm_disp_freq(unsigned int *freq)
+{
+	uint32_t reg;
+#define DPLL_BYP_CLKSEL(reg)    ((reg>>23) & 1)
+#define DPLL_DIV(reg)	   ((reg & 0x7f)+1)
+#define DPLL_MULT(reg)	  ((reg>>8) & 0x7FF)
+	
+	reg = prcm_read_4(AM335X_PRCM_CM_WKUP, CM_WKUP_CM_CLKSEL_DPLL_DISP);
+	
+	/*Check if we are running in bypass */
+	if (DPLL_BYP_CLKSEL(reg))
+		return ENXIO;
+		
+	*freq = DPLL_MULT(reg) * (omap_sys_clk / DPLL_DIV(reg));
+	return(0);
+}
+#endif
Index: src/sys/arch/arm/omap/tifbreg.h
diff -u /dev/null src/sys/arch/arm/omap/tifbreg.h:1.1
--- /dev/null	Wed Jul 16 18:30:43 2014
+++ src/sys/arch/arm/omap/tifbreg.h	Wed Jul 16 18:30:43 2014
@@ -0,0 +1,124 @@
+/* $NetBSD: tifbreg.h,v 1.1 2014/07/16 18:30:43 bouyer Exp $ */
+/*-
+ * Copyright 2013 Oleksandr Tymoshenko <go...@freebsd.org>
+ * All rights reserved.
+ *
+ * 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 AUTHOR 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 AUTHOR 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.
+ */
+
+#define	LCD_PID			0x00
+#define	LCD_CTRL		0x04
+#define		CTRL_DIV_MASK		0xff
+#define		CTRL_DIV_SHIFT		8
+#define		CTRL_AUTO_UFLOW_RESTART	(1 << 1)
+#define		CTRL_RASTER_MODE	1
+#define		CTRL_LIDD_MODE		0
+#define	LCD_LIDD_CTRL		0x0C
+#define	LCD_LIDD_CS0_CONF	0x10
+#define	LCD_LIDD_CS0_ADDR	0x14
+#define	LCD_LIDD_CS0_DATA	0x18
+#define	LCD_LIDD_CS1_CONF	0x1C
+#define	LCD_LIDD_CS1_ADDR	0x20
+#define	LCD_LIDD_CS1_DATA	0x24
+#define	LCD_RASTER_CTRL		0x28
+#define		RASTER_CTRL_TFT24_UNPACKED	(1 << 26)
+#define		RASTER_CTRL_TFT24		(1 << 25)
+#define		RASTER_CTRL_STN565		(1 << 24)
+#define		RASTER_CTRL_TFTPMAP		(1 << 23)
+#define		RASTER_CTRL_NIBMODE		(1 << 22)
+#define		RASTER_CTRL_PALMODE_SHIFT	20
+#define		PALETTE_PALETTE_AND_DATA	0x00
+#define		PALETTE_PALETTE_ONLY		0x01
+#define		PALETTE_DATA_ONLY		0x02
+#define		RASTER_CTRL_REQDLY_SHIFT	12
+#define		RASTER_CTRL_MONO8B		(1 << 9)
+#define		RASTER_CTRL_RBORDER		(1 << 8)
+#define		RASTER_CTRL_LCDTFT		(1 << 7)
+#define		RASTER_CTRL_LCDBW		(1 << 1)
+#define		RASTER_CTRL_LCDEN		(1 << 0)
+#define	LCD_RASTER_TIMING_0	0x2C
+#define		RASTER_TIMING_0_HBP_SHIFT	24
+#define		RASTER_TIMING_0_HFP_SHIFT	16
+#define		RASTER_TIMING_0_HSW_SHIFT	10
+#define		RASTER_TIMING_0_PPLLSB_SHIFT	4
+#define		RASTER_TIMING_0_PPLMSB_SHIFT	3
+#define	LCD_RASTER_TIMING_1	0x30
+#define		RASTER_TIMING_1_VBP_SHIFT	24
+#define		RASTER_TIMING_1_VFP_SHIFT	16
+#define		RASTER_TIMING_1_VSW_SHIFT	10
+#define		RASTER_TIMING_1_LPP_SHIFT	0
+#define	LCD_RASTER_TIMING_2	0x34
+#define		RASTER_TIMING_2_HSWHI_SHIFT	27
+#define		RASTER_TIMING_2_LPP_B10_SHIFT	26
+#define		RASTER_TIMING_2_PHSVS		(1 << 25)
+#define		RASTER_TIMING_2_PHSVS_RISE	(1 << 24)
+#define		RASTER_TIMING_2_PHSVS_FALL	(0 << 24)
+#define		RASTER_TIMING_2_IOE		(1 << 23)
+#define		RASTER_TIMING_2_IPC		(1 << 22)
+#define		RASTER_TIMING_2_IHS		(1 << 21)
+#define		RASTER_TIMING_2_IVS		(1 << 20)
+#define		RASTER_TIMING_2_ACBI_SHIFT	16
+#define		RASTER_TIMING_2_ACB_SHIFT	8
+#define		RASTER_TIMING_2_HBPHI_SHIFT	4
+#define		RASTER_TIMING_2_HFPHI_SHIFT	0
+#define	LCD_RASTER_SUBPANEL	0x38
+#define	LCD_RASTER_SUBPANEL2	0x3C
+#define	LCD_LCDDMA_CTRL		0x40
+#define		LCDDMA_CTRL_DMA_MASTER_PRIO_SHIFT		16
+#define		LCDDMA_CTRL_TH_FIFO_RDY_SHIFT	8
+#define		LCDDMA_CTRL_BURST_SIZE_SHIFT	4
+#define		LCDDMA_CTRL_BYTES_SWAP		(1 << 3)
+#define		LCDDMA_CTRL_BE			(1 << 1)
+#define		LCDDMA_CTRL_FB0_ONLY		0
+#define		LCDDMA_CTRL_FB0_FB1		(1 << 0)
+#define	LCD_LCDDMA_FB0_BASE	0x44
+#define	LCD_LCDDMA_FB0_CEILING	0x48
+#define	LCD_LCDDMA_FB1_BASE	0x4C
+#define	LCD_LCDDMA_FB1_CEILING	0x50
+#define	LCD_SYSCONFIG		0x54
+#define		SYSCONFIG_STANDBY_FORCE		(0 << 4)
+#define		SYSCONFIG_STANDBY_NONE		(1 << 4)
+#define		SYSCONFIG_STANDBY_SMART		(2 << 4)
+#define		SYSCONFIG_IDLE_FORCE		(0 << 2)
+#define		SYSCONFIG_IDLE_NONE		(1 << 2)
+#define		SYSCONFIG_IDLE_SMART		(2 << 2)
+#define	LCD_IRQSTATUS_RAW	0x58
+#define	LCD_IRQSTATUS		0x5C
+#define	LCD_IRQENABLE_SET	0x60
+#define	LCD_IRQENABLE_CLEAR	0x64
+#define		IRQ_EOF1		(1 << 9)
+#define		IRQ_EOF0		(1 << 8)
+#define		IRQ_PL			(1 << 6)
+#define		IRQ_FUF			(1 << 5)
+#define		IRQ_ACB			(1 << 3)
+#define		IRQ_SYNC_LOST		(1 << 2)
+#define		IRQ_RASTER_DONE		(1 << 1)
+#define		IRQ_FRAME_DONE		(1 << 0)
+#define	LCD_CLKC_ENABLE		0x6C
+#define		CLKC_ENABLE_DMA		(1 << 2)
+#define		CLKC_ENABLE_LDID	(1 << 1)
+#define		CLKC_ENABLE_CORE	(1 << 0)
+#define	LCD_CLKC_RESET		0x70
+#define		CLKC_RESET_MAIN		(1 << 3)
+#define		CLKC_RESET_DMA		(1 << 2)
+#define		CLKC_RESET_LDID		(1 << 1)
+#define		CLKC_RESET_CORE		(1 << 0)

Reply via email to