610 lines
17 KiB
C
610 lines
17 KiB
C
/* $NetBSD: awin_debe.c,v 1.14 2014/12/21 18:36:05 jmcneill Exp $ */
|
|
|
|
/*-
|
|
* Copyright (c) 2014 Jared D. McNeill <jmcneill@invisible.ca>
|
|
* 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.
|
|
*/
|
|
|
|
#include "opt_allwinner.h"
|
|
#include "genfb.h"
|
|
#include "awin_mp.h"
|
|
|
|
#ifndef AWIN_DEBE_VIDEOMEM
|
|
#define AWIN_DEBE_VIDEOMEM (16 * 1024 * 1024)
|
|
#endif
|
|
|
|
#define AWIN_DEBE_CURMAX 64
|
|
|
|
#include <sys/cdefs.h>
|
|
__KERNEL_RCSID(0, "$NetBSD: awin_debe.c,v 1.14 2014/12/21 18:36:05 jmcneill Exp $");
|
|
|
|
#include <sys/param.h>
|
|
#include <sys/bus.h>
|
|
#include <sys/device.h>
|
|
#include <sys/intr.h>
|
|
#include <sys/systm.h>
|
|
#include <sys/kernel.h>
|
|
#include <sys/mutex.h>
|
|
#include <sys/condvar.h>
|
|
|
|
#include <arm/allwinner/awin_reg.h>
|
|
#include <arm/allwinner/awin_var.h>
|
|
|
|
#include <dev/videomode/videomode.h>
|
|
#include <dev/wscons/wsconsio.h>
|
|
|
|
struct awin_debe_softc {
|
|
device_t sc_dev;
|
|
device_t sc_fbdev;
|
|
bus_space_tag_t sc_bst;
|
|
bus_space_handle_t sc_bsh;
|
|
bus_space_handle_t sc_ccm_bsh;
|
|
bus_dma_tag_t sc_dmat;
|
|
unsigned int sc_port;
|
|
|
|
bus_dma_segment_t sc_dmasegs[1];
|
|
bus_size_t sc_dmasize;
|
|
bus_dmamap_t sc_dmamap;
|
|
void *sc_dmap;
|
|
|
|
uint16_t sc_margin;
|
|
|
|
bool sc_cursor_enable;
|
|
int sc_cursor_x, sc_cursor_y;
|
|
int sc_hot_x, sc_hot_y;
|
|
uint8_t sc_cursor_bitmap[8 * AWIN_DEBE_CURMAX];
|
|
uint8_t sc_cursor_mask[8 * AWIN_DEBE_CURMAX];
|
|
};
|
|
|
|
#define DEBE_READ(sc, reg) \
|
|
bus_space_read_4((sc)->sc_bst, (sc)->sc_bsh, (reg))
|
|
#define DEBE_WRITE(sc, reg, val) \
|
|
bus_space_write_4((sc)->sc_bst, (sc)->sc_bsh, (reg), (val))
|
|
|
|
static int awin_debe_match(device_t, cfdata_t, void *);
|
|
static void awin_debe_attach(device_t, device_t, void *);
|
|
|
|
static int awin_debe_alloc_videomem(struct awin_debe_softc *);
|
|
static void awin_debe_setup_fbdev(struct awin_debe_softc *,
|
|
const struct videomode *);
|
|
|
|
static int awin_debe_set_curpos(struct awin_debe_softc *, int, int);
|
|
static int awin_debe_set_cursor(struct awin_debe_softc *,
|
|
struct wsdisplay_cursor *);
|
|
|
|
CFATTACH_DECL_NEW(awin_debe, sizeof(struct awin_debe_softc),
|
|
awin_debe_match, awin_debe_attach, NULL, NULL);
|
|
|
|
static int
|
|
awin_debe_match(device_t parent, cfdata_t cf, void *aux)
|
|
{
|
|
struct awinio_attach_args * const aio = aux;
|
|
const struct awin_locators * const loc = &aio->aio_loc;
|
|
|
|
if (strcmp(cf->cf_name, loc->loc_name))
|
|
return 0;
|
|
|
|
return 1;
|
|
}
|
|
|
|
static void
|
|
awin_debe_attach(device_t parent, device_t self, void *aux)
|
|
{
|
|
struct awin_debe_softc *sc = device_private(self);
|
|
struct awinio_attach_args * const aio = aux;
|
|
const struct awin_locators * const loc = &aio->aio_loc;
|
|
prop_dictionary_t cfg = device_properties(self);
|
|
#if NAWIN_MP > 0
|
|
device_t mpdev;
|
|
#endif
|
|
#ifdef AWIN_DEBE_FWINIT
|
|
struct videomode mode;
|
|
#endif
|
|
int error;
|
|
|
|
sc->sc_dev = self;
|
|
sc->sc_bst = aio->aio_core_bst;
|
|
sc->sc_dmat = aio->aio_dmat;
|
|
sc->sc_port = loc->loc_port;
|
|
bus_space_subregion(sc->sc_bst, aio->aio_core_bsh,
|
|
loc->loc_offset, loc->loc_size, &sc->sc_bsh);
|
|
bus_space_subregion(sc->sc_bst, aio->aio_ccm_bsh, 0, 0x1000,
|
|
&sc->sc_ccm_bsh);
|
|
|
|
aprint_naive("\n");
|
|
aprint_normal(": Display Engine Backend (BE%d)\n", loc->loc_port);
|
|
|
|
prop_dictionary_get_uint16(cfg, "margin", &sc->sc_margin);
|
|
|
|
if (awin_chip_id() == AWIN_CHIP_ID_A31) {
|
|
awin_reg_set_clear(aio->aio_core_bst, aio->aio_ccm_bsh,
|
|
AWIN_A31_AHB_RESET1_REG,
|
|
AWIN_A31_AHB_RESET1_BE0_RST << loc->loc_port,
|
|
0);
|
|
} else if (awin_chip_id() == AWIN_CHIP_ID_A20) {
|
|
awin_reg_set_clear(aio->aio_core_bst, aio->aio_ccm_bsh,
|
|
AWIN_BE0_SCLK_CFG_REG + (loc->loc_port * 4),
|
|
AWIN_BEx_CLK_RST,
|
|
0);
|
|
}
|
|
|
|
if (awin_chip_id() == AWIN_CHIP_ID_A31) {
|
|
uint32_t pll6_freq = awin_pll6_get_rate() * 2;
|
|
unsigned int clk_div = (pll6_freq + 299999999) / 300000000;
|
|
|
|
#ifdef AWIN_DEBE_DEBUG
|
|
device_printf(sc->sc_dev, "PLL6 @ %u Hz\n", pll6_freq);
|
|
device_printf(sc->sc_dev, "div %d\n", clk_div);
|
|
#endif
|
|
awin_reg_set_clear(aio->aio_core_bst, aio->aio_ccm_bsh,
|
|
AWIN_BE0_SCLK_CFG_REG + (loc->loc_port * 4),
|
|
__SHIFTIN(AWIN_A31_BEx_CLK_SRC_SEL_PLL6_2X,
|
|
AWIN_A31_BEx_CLK_SRC_SEL) |
|
|
__SHIFTIN(clk_div - 1, AWIN_BEx_CLK_DIV_RATIO_M),
|
|
AWIN_A31_BEx_CLK_SRC_SEL | AWIN_BEx_CLK_DIV_RATIO_M);
|
|
} else if (awin_chip_id() == AWIN_CHIP_ID_A20) {
|
|
uint32_t pll5x_freq = awin_pll5x_get_rate();
|
|
unsigned int clk_div = (pll5x_freq + 299999999) / 300000000;
|
|
|
|
#ifdef AWIN_DEBE_DEBUG
|
|
device_printf(sc->sc_dev, "PLL5x @ %u Hz\n", pll5x_freq);
|
|
device_printf(sc->sc_dev, "div %d\n", clk_div);
|
|
#endif
|
|
|
|
awin_reg_set_clear(aio->aio_core_bst, aio->aio_ccm_bsh,
|
|
AWIN_BE0_SCLK_CFG_REG + (loc->loc_port * 4),
|
|
__SHIFTIN(AWIN_BEx_CLK_SRC_SEL_PLL5, AWIN_BEx_CLK_SRC_SEL) |
|
|
__SHIFTIN(clk_div - 1, AWIN_BEx_CLK_DIV_RATIO_M),
|
|
AWIN_BEx_CLK_SRC_SEL | AWIN_BEx_CLK_DIV_RATIO_M);
|
|
}
|
|
|
|
if (awin_chip_id() == AWIN_CHIP_ID_A20 ||
|
|
awin_chip_id() == AWIN_CHIP_ID_A31) {
|
|
awin_reg_set_clear(aio->aio_core_bst, aio->aio_ccm_bsh,
|
|
AWIN_AHB_GATING1_REG,
|
|
AWIN_AHB_GATING1_DE_BE0 << loc->loc_port, 0);
|
|
|
|
awin_reg_set_clear(aio->aio_core_bst, aio->aio_ccm_bsh,
|
|
AWIN_DRAM_CLK_REG,
|
|
AWIN_DRAM_CLK_BE0_DCLK_ENABLE << loc->loc_port, 0);
|
|
|
|
awin_reg_set_clear(aio->aio_core_bst, aio->aio_ccm_bsh,
|
|
AWIN_BE0_SCLK_CFG_REG + (loc->loc_port * 4),
|
|
AWIN_CLK_ENABLE, 0);
|
|
}
|
|
|
|
#ifdef AWIN_DEBE_FWINIT
|
|
const uint32_t modctl = DEBE_READ(sc, AWIN_DEBE_MODCTL_REG);
|
|
const uint32_t dissize = DEBE_READ(sc, AWIN_DEBE_DISSIZE_REG);
|
|
if ((modctl & AWIN_DEBE_MODCTL_EN) == 0) {
|
|
aprint_error_dev(sc->sc_dev, "disabled\n");
|
|
return;
|
|
}
|
|
if ((modctl & AWIN_DEBE_MODCTL_START_CTL) == 0) {
|
|
aprint_error_dev(sc->sc_dev, "stopped\n");
|
|
return;
|
|
}
|
|
memset(&mode, 0, sizeof(mode));
|
|
mode.hdisplay = (dissize & 0xffff) + 1;
|
|
mode.vdisplay = ((dissize >> 16) & 0xffff) + 1;
|
|
|
|
if (mode.hdisplay == 1 || mode.vdisplay == 1) {
|
|
aprint_error_dev(sc->sc_dev,
|
|
"couldn't determine video mode\n");
|
|
return;
|
|
}
|
|
|
|
aprint_verbose_dev(sc->sc_dev, "using %dx%d mode from firmware\n",
|
|
mode.hdisplay, mode.vdisplay);
|
|
|
|
sc->sc_dmasize = mode.hdisplay * mode.vdisplay * 4;
|
|
#else
|
|
for (unsigned int reg = 0x800; reg < 0x1000; reg += 4) {
|
|
DEBE_WRITE(sc, reg, 0);
|
|
}
|
|
|
|
DEBE_WRITE(sc, AWIN_DEBE_MODCTL_REG, AWIN_DEBE_MODCTL_EN);
|
|
|
|
sc->sc_dmasize = AWIN_DEBE_VIDEOMEM;
|
|
#endif
|
|
|
|
DEBE_WRITE(sc, AWIN_DEBE_HWC_PALETTE_TABLE, 0);
|
|
|
|
error = awin_debe_alloc_videomem(sc);
|
|
if (error) {
|
|
aprint_error_dev(sc->sc_dev,
|
|
"couldn't allocate video memory, error = %d\n", error);
|
|
return;
|
|
}
|
|
|
|
#if NAWIN_MP > 0
|
|
mpdev = device_find_by_driver_unit("awinmp", 0);
|
|
if (mpdev) {
|
|
paddr_t pa = sc->sc_dmamap->dm_segs[0].ds_addr;
|
|
if (pa >= AWIN_SDRAM_PBASE)
|
|
pa -= AWIN_SDRAM_PBASE;
|
|
awin_mp_setbase(mpdev, pa, sc->sc_dmasize);
|
|
}
|
|
#endif
|
|
|
|
#ifdef AWIN_DEBE_FWINIT
|
|
awin_debe_set_videomode(&mode);
|
|
awin_debe_enable(true);
|
|
#endif
|
|
}
|
|
|
|
static int
|
|
awin_debe_alloc_videomem(struct awin_debe_softc *sc)
|
|
{
|
|
int error, nsegs;
|
|
|
|
error = bus_dmamem_alloc(sc->sc_dmat, sc->sc_dmasize, 0x1000, 0,
|
|
sc->sc_dmasegs, 1, &nsegs, BUS_DMA_WAITOK);
|
|
if (error)
|
|
return error;
|
|
error = bus_dmamem_map(sc->sc_dmat, sc->sc_dmasegs, nsegs,
|
|
sc->sc_dmasize, &sc->sc_dmap, BUS_DMA_WAITOK | BUS_DMA_COHERENT);
|
|
if (error)
|
|
goto free;
|
|
error = bus_dmamap_create(sc->sc_dmat, sc->sc_dmasize, 1,
|
|
sc->sc_dmasize, 0, BUS_DMA_WAITOK, &sc->sc_dmamap);
|
|
if (error)
|
|
goto unmap;
|
|
error = bus_dmamap_load(sc->sc_dmat, sc->sc_dmamap, sc->sc_dmap,
|
|
sc->sc_dmasize, NULL, BUS_DMA_WAITOK);
|
|
if (error)
|
|
goto destroy;
|
|
|
|
memset(sc->sc_dmap, 0, sc->sc_dmasize);
|
|
|
|
return 0;
|
|
|
|
destroy:
|
|
bus_dmamap_destroy(sc->sc_dmat, sc->sc_dmamap);
|
|
unmap:
|
|
bus_dmamem_unmap(sc->sc_dmat, sc->sc_dmap, sc->sc_dmasize);
|
|
free:
|
|
bus_dmamem_free(sc->sc_dmat, sc->sc_dmasegs, nsegs);
|
|
|
|
sc->sc_dmasize = 0;
|
|
sc->sc_dmap = NULL;
|
|
|
|
return error;
|
|
}
|
|
|
|
static void
|
|
awin_debe_setup_fbdev(struct awin_debe_softc *sc, const struct videomode *mode)
|
|
{
|
|
if (mode == NULL)
|
|
return;
|
|
|
|
const u_int interlace_p = !!(mode->flags & VID_INTERLACE);
|
|
const u_int fb_width = mode->hdisplay - (sc->sc_margin * 2);
|
|
const u_int fb_height = (mode->vdisplay << interlace_p) -
|
|
(sc->sc_margin * 2);
|
|
|
|
if (mode && sc->sc_fbdev == NULL) {
|
|
struct awinfb_attach_args afb = {
|
|
.afb_fb = sc->sc_dmap,
|
|
.afb_width = fb_width,
|
|
.afb_height = fb_height,
|
|
.afb_dmat = sc->sc_dmat,
|
|
.afb_dmasegs = sc->sc_dmasegs,
|
|
.afb_ndmasegs = 1
|
|
};
|
|
sc->sc_fbdev = config_found_ia(sc->sc_dev, "awindebe",
|
|
&afb, NULL);
|
|
}
|
|
#if NGENFB > 0
|
|
else if (sc->sc_fbdev != NULL) {
|
|
awin_fb_set_videomode(sc->sc_fbdev, fb_width, fb_height);
|
|
}
|
|
#endif
|
|
}
|
|
|
|
static int
|
|
awin_debe_set_curpos(struct awin_debe_softc *sc, int x, int y)
|
|
{
|
|
int xx, yy;
|
|
u_int yoff, xoff;
|
|
|
|
xoff = yoff = 0;
|
|
xx = x - sc->sc_hot_x + sc->sc_margin;
|
|
yy = y - sc->sc_hot_y + sc->sc_margin;
|
|
if (xx < 0) {
|
|
xoff -= xx;
|
|
xx = 0;
|
|
}
|
|
if (yy < 0) {
|
|
yoff -= yy;
|
|
yy = 0;
|
|
}
|
|
|
|
DEBE_WRITE(sc, AWIN_DEBE_HWCCTL_REG,
|
|
__SHIFTIN(yy, AWIN_DEBE_HWCCTL_YCOOR) |
|
|
__SHIFTIN(xx, AWIN_DEBE_HWCCTL_XCOOR));
|
|
DEBE_WRITE(sc, AWIN_DEBE_HWCFBCTL_REG,
|
|
#if AWIN_DEBE_CURMAX == 32
|
|
__SHIFTIN(AWIN_DEBE_HWCFBCTL_YSIZE_32, AWIN_DEBE_HWCFBCTL_YSIZE) |
|
|
__SHIFTIN(AWIN_DEBE_HWCFBCTL_XSIZE_32, AWIN_DEBE_HWCFBCTL_XSIZE) |
|
|
#else
|
|
__SHIFTIN(AWIN_DEBE_HWCFBCTL_YSIZE_64, AWIN_DEBE_HWCFBCTL_YSIZE) |
|
|
__SHIFTIN(AWIN_DEBE_HWCFBCTL_XSIZE_64, AWIN_DEBE_HWCFBCTL_XSIZE) |
|
|
#endif
|
|
__SHIFTIN(AWIN_DEBE_HWCFBCTL_FBFMT_2BPP, AWIN_DEBE_HWCFBCTL_FBFMT) |
|
|
__SHIFTIN(yoff, AWIN_DEBE_HWCFBCTL_YCOOROFF) |
|
|
__SHIFTIN(xoff, AWIN_DEBE_HWCFBCTL_XCOOROFF));
|
|
|
|
return 0;
|
|
}
|
|
|
|
static int
|
|
awin_debe_set_cursor(struct awin_debe_softc *sc, struct wsdisplay_cursor *cur)
|
|
{
|
|
uint32_t val;
|
|
uint8_t r[4], g[4], b[4];
|
|
u_int index, count, shift, off, pcnt;
|
|
int i, j, idx, error;
|
|
uint8_t mask;
|
|
|
|
if (cur->which & WSDISPLAY_CURSOR_DOCUR) {
|
|
val = DEBE_READ(sc, AWIN_DEBE_MODCTL_REG);
|
|
if (cur->enable)
|
|
val |= AWIN_DEBE_MODCTL_HWC_EN;
|
|
else
|
|
val &= ~AWIN_DEBE_MODCTL_HWC_EN;
|
|
DEBE_WRITE(sc, AWIN_DEBE_MODCTL_REG, val);
|
|
|
|
sc->sc_cursor_enable = cur->enable;
|
|
}
|
|
|
|
if (cur->which & WSDISPLAY_CURSOR_DOHOT) {
|
|
sc->sc_hot_x = cur->hot.x;
|
|
sc->sc_hot_y = cur->hot.y;
|
|
cur->which |= WSDISPLAY_CURSOR_DOPOS;
|
|
}
|
|
|
|
if (cur->which & WSDISPLAY_CURSOR_DOPOS) {
|
|
awin_debe_set_curpos(sc, cur->pos.x, cur->pos.y);
|
|
}
|
|
|
|
if (cur->which & WSDISPLAY_CURSOR_DOCMAP) {
|
|
index = cur->cmap.index;
|
|
count = cur->cmap.count;
|
|
if (index >= 2 || (index + count) > 2)
|
|
return EINVAL;
|
|
error = copyin(cur->cmap.red, &r[index], count);
|
|
if (error)
|
|
return error;
|
|
error = copyin(cur->cmap.green, &g[index], count);
|
|
if (error)
|
|
return error;
|
|
error = copyin(cur->cmap.blue, &b[index], count);
|
|
if (error)
|
|
return error;
|
|
|
|
for (i = index; i < (index + count); i++) {
|
|
DEBE_WRITE(sc,
|
|
AWIN_DEBE_HWC_PALETTE_TABLE + (4 * (i + 2)),
|
|
(r[i] << 16) | (g[i] << 8) | b[i] | 0xff000000);
|
|
}
|
|
}
|
|
|
|
if (cur->which & WSDISPLAY_CURSOR_DOSHAPE) {
|
|
error = copyin(cur->mask, sc->sc_cursor_mask,
|
|
AWIN_DEBE_CURMAX * 8);
|
|
if (error)
|
|
return error;
|
|
error = copyin(cur->image, sc->sc_cursor_bitmap,
|
|
AWIN_DEBE_CURMAX * 8);
|
|
if (error)
|
|
return error;
|
|
}
|
|
|
|
if (cur->which & (WSDISPLAY_CURSOR_DOCMAP|WSDISPLAY_CURSOR_DOSHAPE)) {
|
|
for (i = 0, pcnt = 0; i < AWIN_DEBE_CURMAX * 8; i++) {
|
|
for (j = 0, mask = 1; j < 8; j++, mask <<= 1, pcnt++) {
|
|
idx = ((sc->sc_cursor_mask[i] & mask) ? 2 : 0) |
|
|
((sc->sc_cursor_bitmap[i] & mask) ? 1 : 0);
|
|
off = (pcnt >> 4) * 4;
|
|
shift = (pcnt & 0xf) * 2;
|
|
val = DEBE_READ(sc,
|
|
AWIN_DEBE_HWC_PATTERN_BLOCK + off);
|
|
val &= ~(3 << shift);
|
|
val |= (idx << shift);
|
|
DEBE_WRITE(sc,
|
|
AWIN_DEBE_HWC_PATTERN_BLOCK + off, val);
|
|
}
|
|
}
|
|
}
|
|
|
|
return 0;
|
|
}
|
|
|
|
void
|
|
awin_debe_enable(bool enable)
|
|
{
|
|
struct awin_debe_softc *sc;
|
|
device_t dev;
|
|
uint32_t val;
|
|
|
|
dev = device_find_by_driver_unit("awindebe", 0);
|
|
if (dev == NULL) {
|
|
printf("DEBE: no driver found\n");
|
|
return;
|
|
}
|
|
sc = device_private(dev);
|
|
|
|
if (enable) {
|
|
val = DEBE_READ(sc, AWIN_DEBE_REGBUFFCTL_REG);
|
|
val |= AWIN_DEBE_REGBUFFCTL_REGLOADCTL;
|
|
DEBE_WRITE(sc, AWIN_DEBE_REGBUFFCTL_REG, val);
|
|
|
|
val = DEBE_READ(sc, AWIN_DEBE_MODCTL_REG);
|
|
val |= AWIN_DEBE_MODCTL_START_CTL;
|
|
DEBE_WRITE(sc, AWIN_DEBE_MODCTL_REG, val);
|
|
} else {
|
|
val = DEBE_READ(sc, AWIN_DEBE_MODCTL_REG);
|
|
val &= ~AWIN_DEBE_MODCTL_START_CTL;
|
|
DEBE_WRITE(sc, AWIN_DEBE_MODCTL_REG, val);
|
|
}
|
|
}
|
|
|
|
void
|
|
awin_debe_set_videomode(const struct videomode *mode)
|
|
{
|
|
struct awin_debe_softc *sc;
|
|
device_t dev;
|
|
uint32_t val;
|
|
|
|
dev = device_find_by_driver_unit("awindebe", 0);
|
|
if (dev == NULL) {
|
|
printf("DEBE: no driver found\n");
|
|
return;
|
|
}
|
|
sc = device_private(dev);
|
|
|
|
if (mode) {
|
|
const u_int interlace_p = !!(mode->flags & VID_INTERLACE);
|
|
const u_int width = mode->hdisplay;
|
|
const u_int height = (mode->vdisplay << interlace_p);
|
|
const u_int fb_width = width - (sc->sc_margin * 2);
|
|
const u_int fb_height = height - (sc->sc_margin * 2);
|
|
uint32_t vmem = width * height * 4;
|
|
|
|
if (vmem > sc->sc_dmasize) {
|
|
device_printf(sc->sc_dev,
|
|
"not enough memory for %ux%u fb (req %u have %u)\n",
|
|
width, height, vmem, (unsigned int)sc->sc_dmasize);
|
|
return;
|
|
}
|
|
|
|
paddr_t pa = sc->sc_dmamap->dm_segs[0].ds_addr;
|
|
#if !defined(ALLWINNER_A80)
|
|
/*
|
|
* On 2GB systems, we need to subtract AWIN_SDRAM_PBASE from
|
|
* the phys addr.
|
|
*/
|
|
if (pa >= AWIN_SDRAM_PBASE)
|
|
pa -= AWIN_SDRAM_PBASE;
|
|
#endif
|
|
|
|
/* notify fb */
|
|
awin_debe_setup_fbdev(sc, mode);
|
|
|
|
DEBE_WRITE(sc, AWIN_DEBE_DISSIZE_REG,
|
|
((height - 1) << 16) | (width - 1));
|
|
DEBE_WRITE(sc, AWIN_DEBE_LAYSIZE_REG,
|
|
((fb_height - 1) << 16) | (fb_width - 1));
|
|
DEBE_WRITE(sc, AWIN_DEBE_LAYCOOR_REG,
|
|
(sc->sc_margin << 16) | sc->sc_margin);
|
|
DEBE_WRITE(sc, AWIN_DEBE_LAYLINEWIDTH_REG, (fb_width << 5));
|
|
DEBE_WRITE(sc, AWIN_DEBE_LAYFB_L32ADD_REG, pa << 3);
|
|
DEBE_WRITE(sc, AWIN_DEBE_LAYFB_H4ADD_REG, pa >> 29);
|
|
|
|
val = DEBE_READ(sc, AWIN_DEBE_ATTCTL1_REG);
|
|
val &= ~AWIN_DEBE_ATTCTL1_LAY_FBFMT;
|
|
val |= __SHIFTIN(AWIN_DEBE_ATTCTL1_LAY_FBFMT_XRGB8888,
|
|
AWIN_DEBE_ATTCTL1_LAY_FBFMT);
|
|
val &= ~AWIN_DEBE_ATTCTL1_LAY_BRSWAPEN;
|
|
val &= ~AWIN_DEBE_ATTCTL1_LAY_FBPS;
|
|
#if __ARMEB__
|
|
val |= __SHIFTIN(AWIN_DEBE_ATTCTL1_LAY_FBPS_32BPP_BGRA,
|
|
AWIN_DEBE_ATTCTL1_LAY_FBPS);
|
|
#else
|
|
val |= __SHIFTIN(AWIN_DEBE_ATTCTL1_LAY_FBPS_32BPP_ARGB,
|
|
AWIN_DEBE_ATTCTL1_LAY_FBPS);
|
|
#endif
|
|
DEBE_WRITE(sc, AWIN_DEBE_ATTCTL1_REG, val);
|
|
|
|
val = DEBE_READ(sc, AWIN_DEBE_MODCTL_REG);
|
|
val |= AWIN_DEBE_MODCTL_LAY0_EN;
|
|
if (interlace_p) {
|
|
val |= AWIN_DEBE_MODCTL_ITLMOD_EN;
|
|
} else {
|
|
val &= ~AWIN_DEBE_MODCTL_ITLMOD_EN;
|
|
}
|
|
DEBE_WRITE(sc, AWIN_DEBE_MODCTL_REG, val);
|
|
} else {
|
|
/* disable */
|
|
val = DEBE_READ(sc, AWIN_DEBE_MODCTL_REG);
|
|
val &= ~AWIN_DEBE_MODCTL_LAY0_EN;
|
|
val &= ~AWIN_DEBE_MODCTL_START_CTL;
|
|
DEBE_WRITE(sc, AWIN_DEBE_MODCTL_REG, val);
|
|
|
|
/* notify fb */
|
|
awin_debe_setup_fbdev(sc, mode);
|
|
}
|
|
}
|
|
|
|
int
|
|
awin_debe_ioctl(device_t self, u_long cmd, void *data)
|
|
{
|
|
struct awin_debe_softc *sc = device_private(self);
|
|
struct wsdisplay_curpos *cp;
|
|
uint32_t val;
|
|
int enable;
|
|
|
|
switch (cmd) {
|
|
case WSDISPLAYIO_SVIDEO:
|
|
enable = *(int *)data;
|
|
val = DEBE_READ(sc, AWIN_DEBE_MODCTL_REG);
|
|
if (enable) {
|
|
val |= AWIN_DEBE_MODCTL_LAY0_EN;
|
|
if (sc->sc_cursor_enable) {
|
|
val |= AWIN_DEBE_MODCTL_HWC_EN;
|
|
} else {
|
|
val &= ~AWIN_DEBE_MODCTL_HWC_EN;
|
|
}
|
|
} else {
|
|
val &= ~AWIN_DEBE_MODCTL_LAY0_EN;
|
|
val &= ~AWIN_DEBE_MODCTL_HWC_EN;
|
|
}
|
|
DEBE_WRITE(sc, AWIN_DEBE_MODCTL_REG, val);
|
|
return 0;
|
|
case WSDISPLAYIO_GVIDEO:
|
|
val = DEBE_READ(sc, AWIN_DEBE_MODCTL_REG);
|
|
*(int *)data = !!(val & AWIN_DEBE_MODCTL_LAY0_EN);
|
|
return 0;
|
|
case WSDISPLAYIO_GCURPOS:
|
|
cp = data;
|
|
cp->x = sc->sc_cursor_x;
|
|
cp->y = sc->sc_cursor_y;
|
|
return 0;
|
|
case WSDISPLAYIO_SCURPOS:
|
|
cp = data;
|
|
return awin_debe_set_curpos(sc, cp->x, cp->y);
|
|
case WSDISPLAYIO_GCURMAX:
|
|
cp = data;
|
|
cp->x = AWIN_DEBE_CURMAX;
|
|
cp->y = AWIN_DEBE_CURMAX;
|
|
return 0;
|
|
case WSDISPLAYIO_SCURSOR:
|
|
return awin_debe_set_cursor(sc, data);
|
|
}
|
|
|
|
return EPASSTHROUGH;
|
|
}
|
|
|