1662 lines
43 KiB
C
1662 lines
43 KiB
C
/* $NetBSD: vidcrender.c,v 1.18 2005/12/24 20:06:52 perry Exp $ */
|
|
|
|
/*
|
|
* Copyright (c) 1996 Mark Brinicombe
|
|
* Copyright (c) 1996 Robert Black
|
|
* Copyright (c) 1994-1995 Melvyn Tang-Richardson
|
|
* Copyright (c) 1994-1995 RiscBSD kernel team
|
|
* 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.
|
|
* 3. All advertising materials mentioning features or use of this software
|
|
* must display the following acknowledgement:
|
|
* This product includes software developed by the RiscBSD kernel team
|
|
* 4. The name of the company nor the name of the author may be used to
|
|
* endorse or promote products derived from this software without specific
|
|
* prior written permission.
|
|
*
|
|
* THIS SOFTWARE IS PROVIDED BY THE RISCBSD TEAM ``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 AUTHORS 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.
|
|
*
|
|
* RiscBSD kernel project
|
|
*
|
|
* vidcrender.c
|
|
*
|
|
* Console assembly functions
|
|
*
|
|
* Created : 17/09/94
|
|
* Last updated : 07/02/96
|
|
*/
|
|
|
|
#include <sys/cdefs.h>
|
|
__KERNEL_RCSID(0, "$NetBSD: vidcrender.c,v 1.18 2005/12/24 20:06:52 perry Exp $");
|
|
|
|
#include <sys/types.h>
|
|
#include <sys/param.h>
|
|
#include <sys/systm.h>
|
|
#include <sys/malloc.h>
|
|
#include <sys/ioctl.h>
|
|
#include <sys/conf.h>
|
|
#include <sys/tty.h>
|
|
#include <sys/device.h>
|
|
#include <sys/proc.h>
|
|
#include <sys/user.h>
|
|
#include <sys/syslog.h>
|
|
#include <sys/resourcevar.h>
|
|
|
|
#include <uvm/uvm_extern.h>
|
|
|
|
#include <machine/cpu.h>
|
|
#include <machine/param.h>
|
|
#include <arm/arm32/katelib.h>
|
|
#include <machine/bootconfig.h>
|
|
#include <machine/intr.h>
|
|
#include <machine/pmap.h>
|
|
#include <arm/iomd/iomdreg.h>
|
|
#include <arm/iomd/iomdvar.h>
|
|
#include <arm/iomd/vidc.h>
|
|
#include <arm/iomd/console/console.h>
|
|
#include <machine/vconsole.h>
|
|
|
|
#include <arm/iomd/console/fonts/font_normal.h>
|
|
#include <arm/iomd/console/fonts/font_bold.h>
|
|
#include <arm/iomd/console/fonts/font_italic.h>
|
|
|
|
#ifndef DEBUGTERM
|
|
#define dprintf(x) ;
|
|
#endif
|
|
|
|
/* Options */
|
|
#define ACTIVITY_WARNING
|
|
#define SCREENBLANKER
|
|
#undef INVERSE_CONSOLE
|
|
|
|
/* Internal defines only */
|
|
#define DEVLOPING
|
|
#undef SELFTEST
|
|
#undef SILLIES
|
|
|
|
#ifdef SILLIES
|
|
#define PRETTYCURSOR
|
|
#endif
|
|
|
|
extern struct vconsole *vconsole_default;
|
|
|
|
extern videomemory_t videomemory;
|
|
|
|
extern font_struct font_terminal_14normal;
|
|
extern font_struct font_terminal_14bold;
|
|
extern font_struct font_terminal_14italic;
|
|
|
|
#define font_normal font_terminal_14normal
|
|
#define font_bold font_terminal_14bold
|
|
#define font_italic font_terminal_14italic
|
|
|
|
#define VIDC_ENGINE_NAME "VIDC"
|
|
#define R_DATA ((struct vidc_info *)vc->r_data)
|
|
#define MODE (R_DATA->mode)
|
|
|
|
static int cold_init = 0;
|
|
|
|
extern struct vconsole *vconsole_master;
|
|
extern struct vconsole *vconsole_current;
|
|
static struct vidc_mode vidc_initialmode;
|
|
static struct vidc_mode *vidc_currentmode;
|
|
|
|
unsigned int dispstart;
|
|
unsigned int dispsize;
|
|
unsigned int dispbase;
|
|
unsigned int dispend;
|
|
unsigned int ptov;
|
|
unsigned int transfersize;
|
|
unsigned int vmem_base;
|
|
unsigned int phys_base;
|
|
int flash;
|
|
int cursor_flash;
|
|
char *cursor_normal;
|
|
char *cursor_transparent;
|
|
int p_cursor_normal;
|
|
int p_cursor_transparent;
|
|
|
|
/* Local function prototypes */
|
|
static int vidcrender_coldinit(struct vconsole *);
|
|
static void vidcrender_cls(struct vconsole *);
|
|
static int vidc_cursor_init(struct vconsole *);
|
|
int vidcrender_cursorintr(void *);
|
|
int vidcrender_flashintr(void *);
|
|
static int vidcrender_textpalette(struct vconsole *);
|
|
static void vidcrender_render(struct vconsole *, char);
|
|
static void vidcrender_mode(struct vconsole *, struct vidc_mode *);
|
|
int vidcrender_init(struct vconsole *);
|
|
int vidcrender_flash(struct vconsole *, int);
|
|
int vidcrender_cursorflash(struct vconsole *, int);
|
|
int vidcrender_flash_go(struct vconsole *);
|
|
int vidcrender_blank(struct vconsole *, int);
|
|
void vidcrender_reinit(void);
|
|
void vidcrender_putchar(dev_t, char, struct vconsole *);
|
|
int vidcrender_spawn(struct vconsole *);
|
|
int vidcrender_redraw(struct vconsole *, int, int, int, int);
|
|
int vidcrender_swapin(struct vconsole *);
|
|
paddr_t vidcrender_mmap(struct vconsole *, off_t, int);
|
|
void vidcrender_scrollup(struct vconsole *, int, int);
|
|
void vidcrender_scrolldown(struct vconsole *, int, int);
|
|
int vidcrender_scrollback(struct vconsole *);
|
|
void vidcrender_update(struct vconsole *);
|
|
int vidcrender_scrollforward(struct vconsole *);
|
|
int vidcrender_scrollbackend(struct vconsole *);
|
|
int vidcrender_clreos(struct vconsole *, int);
|
|
int vidcrender_debugprint(struct vconsole *);
|
|
static int vidcrender_cursorupdate(struct vconsole *);
|
|
static int vidcrender_cursorflashrate(struct vconsole *, int);
|
|
int vidcrender_setfgcol(struct vconsole *, int);
|
|
int vidcrender_setbgcol(struct vconsole *, int);
|
|
int vidcrender_sgr(struct vconsole *, int);
|
|
int vidcrender_scrollregion(struct vconsole *, int, int);
|
|
int vidcrender_ioctl(struct vconsole *, dev_t, int, caddr_t, int,
|
|
struct lwp *);
|
|
int vidcrender_attach(struct vconsole *, struct device *,
|
|
struct device *, void *);
|
|
|
|
extern int vidcrendermc_cls(unsigned char *, unsigned char *, int);
|
|
extern void vidcrendermc_render(unsigned char *, unsigned char *, int, int);
|
|
|
|
void physcon_display_base(u_int);
|
|
|
|
/*
|
|
* This will be called while still in the mode that we were left
|
|
* in after exiting from riscos
|
|
*/
|
|
|
|
static irqhandler_t cursor_ih;
|
|
irqhandler_t flash_ih;
|
|
|
|
#ifndef HARDCODEDMODES
|
|
/* The table of modes is separately compiled */
|
|
extern struct vidc_mode vidcmodes[];
|
|
#else /* HARDCODEDMODES */
|
|
|
|
/* We have a hard compiled table of modes and a list of definable modes */
|
|
static struct vidc_mode vidcmodes[] = {
|
|
{32500,/**/52, 64, 30, 640, 30, 14,/**/3, 28, 0, 480, 0, 9,/**/0,/**/3},
|
|
{65000,/**/128, 36, 60, 1024, 60 ,36,/**/6, 29, 0, 768, 0, 3,/**/0,/**/3},
|
|
};
|
|
#endif /* HARDCODEDMODES */
|
|
|
|
|
|
struct fsyn {
|
|
int r, v, f;
|
|
};
|
|
|
|
static struct fsyn fsyn_pref[] = {
|
|
{ 6, 2, 8000 },
|
|
{ 4, 2, 12000 },
|
|
{ 3, 2, 16000 },
|
|
{ 2, 2, 24000 },
|
|
{ 41, 43, 25171 },
|
|
{ 50, 59, 28320 },
|
|
{ 3, 4, 32000 },
|
|
{ 2, 3, 36000 },
|
|
{ 31, 58, 44903 },
|
|
{ 12, 35, 70000 },
|
|
{ 0, 0, 00000 }
|
|
};
|
|
|
|
|
|
/*#define mod(x) (((x) > 0) ? (x) : (-x))*/
|
|
|
|
static inline int
|
|
mod(int n)
|
|
{
|
|
if (n < 0)
|
|
return(-n);
|
|
else
|
|
return(n);
|
|
}
|
|
|
|
static int
|
|
vidcrender_coldinit(vc)
|
|
struct vconsole *vc;
|
|
{
|
|
int found;
|
|
int loop;
|
|
|
|
/* Blank out the cursor */
|
|
|
|
vidc_write(VIDC_CP1, 0x0);
|
|
vidc_write(VIDC_CP2, 0x0);
|
|
vidc_write(VIDC_CP3, 0x0);
|
|
|
|
/* the old vidc-console code can't handle all display depths */
|
|
bootconfig.log2_bpp = 3;
|
|
|
|
/* Try to determine the current mode */
|
|
vidc_initialmode.hder = bootconfig.width+1;
|
|
vidc_initialmode.vder = bootconfig.height+1;
|
|
vidc_initialmode.log2_bpp = bootconfig.log2_bpp;
|
|
|
|
dispbase = vmem_base = dispstart = videomemory.vidm_vbase;
|
|
phys_base = videomemory.vidm_pbase;
|
|
|
|
/* Nut - should be using videomemory.vidm_size - mark */
|
|
|
|
if (videomemory.vidm_type == VIDEOMEM_TYPE_DRAM) {
|
|
dispsize = videomemory.vidm_size;
|
|
transfersize = 16;
|
|
} else {
|
|
dispsize = bootconfig.vram[0].pages * PAGE_SIZE;
|
|
transfersize = dispsize >> 10;
|
|
}
|
|
|
|
ptov = dispbase - phys_base;
|
|
|
|
dispend = dispstart+dispsize;
|
|
|
|
vidc_currentmode = &vidcmodes[0];
|
|
loop = 0;
|
|
found = 0;
|
|
|
|
while (vidcmodes[loop].pixel_rate != 0) {
|
|
if (vidcmodes[loop].hder == (bootconfig.width + 1)
|
|
&& vidcmodes[loop].vder == (bootconfig.height + 1)
|
|
&& vidcmodes[loop].frame_rate == bootconfig.framerate) {
|
|
vidc_currentmode = &vidcmodes[loop];
|
|
found = 1;
|
|
}
|
|
++loop;
|
|
}
|
|
|
|
if (!found) {
|
|
vidc_currentmode = &vidcmodes[0];
|
|
loop = 0;
|
|
found = 0;
|
|
|
|
while (vidcmodes[loop].pixel_rate != 0) {
|
|
if (vidcmodes[loop].hder == (bootconfig.width + 1)
|
|
&& vidcmodes[loop].vder == (bootconfig.height + 1)) {
|
|
vidc_currentmode = &vidcmodes[loop];
|
|
found = 1;
|
|
}
|
|
++loop;
|
|
}
|
|
}
|
|
|
|
/* vidc_currentmode = &vidcmodes[0]; */
|
|
vidc_currentmode->log2_bpp = bootconfig.log2_bpp;
|
|
|
|
if (vc)
|
|
R_DATA->flash = R_DATA->cursor_flash = 0;
|
|
|
|
dispstart = dispbase;
|
|
dispend = dispstart+dispsize;
|
|
|
|
IOMD_WRITE_WORD(IOMD_VIDINIT, dispstart-ptov);
|
|
IOMD_WRITE_WORD(IOMD_VIDSTART, dispstart-ptov);
|
|
IOMD_WRITE_WORD(IOMD_VIDEND, (dispend-transfersize)-ptov);
|
|
return 0;
|
|
}
|
|
|
|
struct vidc_mode newmode;
|
|
|
|
static const int bpp_mask_table[] = {
|
|
0, /* 1bpp */
|
|
1, /* 2bpp */
|
|
2, /* 4bpp */
|
|
3, /* 8bpp */
|
|
4, /* 16bpp */
|
|
6 /* 32bpp */
|
|
};
|
|
|
|
void
|
|
vidcrender_mode(vc, mode)
|
|
struct vconsole *vc;
|
|
struct vidc_mode *mode;
|
|
{
|
|
register int acc;
|
|
int bpp_mask;
|
|
int ereg;
|
|
|
|
int best_r, best_v, best_match;
|
|
|
|
/*
|
|
* Find out what bit mask we need to or with the vidc20 control register
|
|
* in order to generate the desired number of bits per pixel.
|
|
* log2_bpp is log base 2 of the number of bits per pixel.
|
|
*/
|
|
|
|
bpp_mask = bpp_mask_table[mode->log2_bpp];
|
|
|
|
/*
|
|
printf ( "res = (%d, %d) rate = %d\n", mode->hder, mode->vder, mode->pixel_rate );
|
|
*/
|
|
|
|
newmode = *mode;
|
|
vidc_currentmode = &newmode;
|
|
|
|
/* Program the VCO Look up a preferred value before choosing one */
|
|
{
|
|
int least_error = mod(fsyn_pref[0].f - vidc_currentmode->pixel_rate);
|
|
int counter;
|
|
best_r = fsyn_pref[0].r;
|
|
best_match = fsyn_pref[0].f;
|
|
best_v = fsyn_pref[0].v;
|
|
|
|
/* Look up */
|
|
|
|
counter=0;
|
|
|
|
while (fsyn_pref[counter].r != 0) {
|
|
if (least_error > mod(fsyn_pref[counter].f - vidc_currentmode->pixel_rate)) {
|
|
best_match = fsyn_pref[counter].f;
|
|
least_error = mod(fsyn_pref[counter].f - vidc_currentmode->pixel_rate);
|
|
best_r = fsyn_pref[counter].r;
|
|
best_v = fsyn_pref[counter].v;
|
|
}
|
|
counter++;
|
|
}
|
|
|
|
if (least_error > 0) { /* Accuracy of 1000Hz */
|
|
int r, v, f;
|
|
for (v = 63; v > 0; v--)
|
|
for (r = 63; r > 0; r--) {
|
|
f = ((v * vidc_fref)/1000) / r;
|
|
if (least_error >= mod(f - vidc_currentmode->pixel_rate)) {
|
|
best_match = f;
|
|
least_error = mod(f - vidc_currentmode->pixel_rate);
|
|
best_r = r;
|
|
best_v = v;
|
|
}
|
|
}
|
|
}
|
|
|
|
if (best_r > 63) best_r=63;
|
|
if (best_v > 63) best_v=63;
|
|
if (best_r < 1) best_r= 1;
|
|
if (best_v < 1) best_v= 1;
|
|
|
|
}
|
|
/*
|
|
printf ( "best_v = %d best_r = %d best_f = %d\n", best_v, best_r, best_match );
|
|
*/
|
|
|
|
if (vc==vconsole_current) {
|
|
vidc_write(VIDC_FSYNREG, (best_v-1)<<8 | (best_r-1)<<0);
|
|
|
|
acc=0;
|
|
acc+=vidc_currentmode->hswr; vidc_write(VIDC_HSWR, (acc - 8 ) & (~1));
|
|
acc+=vidc_currentmode->hbsr; vidc_write(VIDC_HBSR, (acc - 12) & (~1));
|
|
acc+=vidc_currentmode->hdsr; vidc_write(VIDC_HDSR, (acc - 18) & (~1));
|
|
acc+=vidc_currentmode->hder; vidc_write(VIDC_HDER, (acc - 18) & (~1));
|
|
acc+=vidc_currentmode->hber; vidc_write(VIDC_HBER, (acc - 12) & (~1));
|
|
acc+=vidc_currentmode->hcr; vidc_write(VIDC_HCR, (acc - 8 ) & (~3));
|
|
|
|
acc=0;
|
|
acc+=vidc_currentmode->vswr; vidc_write(VIDC_VSWR, (acc - 1));
|
|
acc+=vidc_currentmode->vbsr; vidc_write(VIDC_VBSR, (acc - 1));
|
|
acc+=vidc_currentmode->vdsr; vidc_write(VIDC_VDSR, (acc - 1));
|
|
acc+=vidc_currentmode->vder; vidc_write(VIDC_VDER, (acc - 1));
|
|
acc+=vidc_currentmode->vber; vidc_write(VIDC_VBER, (acc - 1));
|
|
acc+=vidc_currentmode->vcr; vidc_write(VIDC_VCR, (acc - 1));
|
|
|
|
IOMD_WRITE_WORD(IOMD_FSIZE, vidc_currentmode->vcr
|
|
+ vidc_currentmode->vswr
|
|
+ vidc_currentmode->vber
|
|
+ vidc_currentmode->vbsr - 1);
|
|
|
|
if (dispsize <= 1024*1024)
|
|
vidc_write(VIDC_DCTL, vidc_currentmode->hder>>2 | 1<<16 | 1<<12);
|
|
else
|
|
vidc_write(VIDC_DCTL, vidc_currentmode->hder>>2 | 3<<16 | 1<<12);
|
|
|
|
ereg = 1<<12;
|
|
if (vidc_currentmode->sync_pol & 0x01)
|
|
ereg |= 1<<16;
|
|
if (vidc_currentmode->sync_pol & 0x02)
|
|
ereg |= 1<<18;
|
|
vidc_write(VIDC_EREG, ereg);
|
|
if (dispsize > 1024*1024) {
|
|
if (vidc_currentmode->hder >= 800)
|
|
vidc_write(VIDC_CONREG, 7<<8 | bpp_mask<<5);
|
|
else
|
|
vidc_write(VIDC_CONREG, 6<<8 | bpp_mask<<5);
|
|
} else {
|
|
vidc_write(VIDC_CONREG, 7<<8 | bpp_mask<<5);
|
|
}
|
|
}
|
|
|
|
R_DATA->mode = *vidc_currentmode;
|
|
R_DATA->screensize = R_DATA->XRES * R_DATA->YRES * (1 << R_DATA->mode.log2_bpp)/8;
|
|
R_DATA->pixelsperbyte = 8 / (1 << R_DATA->mode.log2_bpp);
|
|
R_DATA->frontporch = MODE.hswr + MODE.hbsr + MODE.hdsr;
|
|
R_DATA->topporch = MODE.vswr + MODE.vbsr + MODE.vdsr;
|
|
R_DATA->bytes_per_line = R_DATA->XRES *
|
|
R_DATA->font->y_spacing/R_DATA->pixelsperbyte;
|
|
R_DATA->bytes_per_scroll = (vc->ycur % R_DATA->font->y_spacing)
|
|
* vc->xcur / R_DATA->pixelsperbyte;
|
|
R_DATA->text_width = R_DATA->XRES / R_DATA->font->x_spacing;
|
|
R_DATA->text_height = R_DATA->YRES / R_DATA->font->y_spacing;
|
|
vc->xchars = R_DATA->text_width;
|
|
vc->ychars = R_DATA->text_height;
|
|
}
|
|
|
|
void
|
|
physcon_display_base(base)
|
|
u_int base;
|
|
{
|
|
dispstart = dispstart-dispbase + base;
|
|
dispbase = vmem_base = base;
|
|
dispend = base + dispsize;
|
|
ptov = dispbase - phys_base;
|
|
}
|
|
|
|
static struct vidc_info masterinfo;
|
|
static int cursor_init = 0;
|
|
|
|
int
|
|
vidcrender_init(vc)
|
|
struct vconsole *vc;
|
|
{
|
|
struct vidc_info *new;
|
|
int loop;
|
|
|
|
if (cold_init == 0) {
|
|
vidcrender_coldinit(vc);
|
|
} else {
|
|
if (cursor_init == 0)
|
|
vidcrender_flash_go(vc);
|
|
}
|
|
|
|
/*
|
|
* If vc->r_data is initialised then this means that the previous
|
|
* render engine on this vconsole was not freed properly. I should
|
|
* not try to clear it up, since I could panic the kernel. Instead
|
|
* I forget about its memory, which could cause a memory leak, but
|
|
* this would be easily detectable and fixable
|
|
*/
|
|
|
|
#ifdef SELFTEST
|
|
if (vc->r_data != 0) {
|
|
printf( "*********************************************************\n" );
|
|
printf( "You have configured SELFTEST mode in the console driver\n" );
|
|
printf( "vc->rdata non zero. This could mean a new console\n" );
|
|
printf( "render engine has not freed up its data structure when\n" );
|
|
printf( "exiting.\n" );
|
|
printf( "DO NOT COMPILE NON DEVELOPMENT KERNELS WITH SELFTEST\n" );
|
|
printf( "*********************************************************" );
|
|
}
|
|
#endif
|
|
|
|
if (vc == vconsole_master) {
|
|
vc->r_data = (char *)&masterinfo;
|
|
} else {
|
|
MALLOC((vc->r_data), char *, sizeof(struct vidc_info),
|
|
M_DEVBUF, M_NOWAIT);
|
|
}
|
|
|
|
if (vc->r_data==0)
|
|
panic("render engine initialisation failed. CLEAN THIS UP!");
|
|
|
|
R_DATA->normalfont = &font_normal;
|
|
R_DATA->italicfont = &font_italic;
|
|
R_DATA->boldfont = &font_bold;
|
|
R_DATA->font = R_DATA->normalfont;
|
|
|
|
vidcrender_mode(vc, vidc_currentmode);
|
|
R_DATA->scrollback_end = dispstart;
|
|
|
|
new = (struct vidc_info *)vc->r_data;
|
|
|
|
R_DATA->text_colours = 1 << (1 << R_DATA->mode.log2_bpp); /* real number of colours */
|
|
if (R_DATA->text_colours > 8) R_DATA->text_colours = 8;
|
|
|
|
#ifdef INVERSE_CONSOLE
|
|
R_DATA->n_backcolour = R_DATA->text_colours - 1;
|
|
R_DATA->n_forecolour = 0;
|
|
#else
|
|
R_DATA->n_backcolour = 0;
|
|
R_DATA->n_forecolour = R_DATA->text_colours - 1;
|
|
#endif
|
|
|
|
R_DATA->backcolour = R_DATA->n_backcolour;
|
|
R_DATA->forecolour = R_DATA->n_forecolour;
|
|
|
|
R_DATA->forefillcolour = 0;
|
|
R_DATA->backfillcolour = 0;
|
|
|
|
R_DATA->bold = 0;
|
|
R_DATA->reverse = 0;
|
|
|
|
for (loop = 0; loop < R_DATA->pixelsperbyte; ++loop) {
|
|
R_DATA->forefillcolour |= (R_DATA->forecolour <<
|
|
loop * (1 << R_DATA->mode.log2_bpp));
|
|
R_DATA->backfillcolour |= (R_DATA->backcolour <<
|
|
loop * (1 << R_DATA->mode.log2_bpp));
|
|
}
|
|
|
|
R_DATA->fast_render = R_DATA->forecolour | (R_DATA->backcolour<<8) | (R_DATA->font->pixel_height<<16);
|
|
R_DATA->blanked=0;
|
|
vc->BLANK(vc, BLANK_NONE);
|
|
|
|
if (vc == vconsole_current)
|
|
vidcrender_textpalette(vc);
|
|
|
|
|
|
if (cold_init == 0) {
|
|
vidc_write(VIDC_CP1, 0x0);
|
|
vidc_write(VIDC_CP2, 0x0);
|
|
vidc_write(VIDC_CP3, 0x0);
|
|
} else
|
|
vidc_cursor_init(vc) ;
|
|
|
|
cold_init=1;
|
|
return 0;
|
|
}
|
|
|
|
void
|
|
vidcrender_reinit(void)
|
|
{
|
|
vidcrender_coldinit(vconsole_current);
|
|
vidcrender_mode(vconsole_current, vidc_currentmode);
|
|
}
|
|
|
|
void
|
|
vidcrender_putchar(dev, c, vc)
|
|
dev_t dev;
|
|
char c;
|
|
struct vconsole *vc;
|
|
{
|
|
vc->PUTSTRING(&c, 1, vc);
|
|
}
|
|
|
|
int
|
|
vidcrender_spawn(vc)
|
|
struct vconsole *vc;
|
|
{
|
|
vc->xchars = R_DATA->text_width;
|
|
vc->ychars = R_DATA->text_height;
|
|
vc->xcur = 0;
|
|
vc->ycur = 0;
|
|
vc->flags = 0;
|
|
return 0;
|
|
}
|
|
|
|
int
|
|
vidcrender_redraw(vc, x, y, a, b)
|
|
struct vconsole *vc;
|
|
int x, y;
|
|
int a, b;
|
|
{
|
|
int xs, ys;
|
|
struct vidc_state vidc;
|
|
font_struct *p_font = R_DATA->font;
|
|
int p_forecol = R_DATA->forecolour;
|
|
int p_backcol = R_DATA->backcolour;
|
|
if (x<0) x=0;
|
|
if (y<0) y=0;
|
|
if (x>(vc->xchars-1)) x=vc->xchars-1;
|
|
if (y>(vc->ychars-1)) x=vc->ychars-1;
|
|
|
|
if (a>(vc->xchars-1)) a=vc->xchars-1;
|
|
if (b>(vc->ychars-1)) b=vc->ychars-1;
|
|
|
|
if (a<x) a=x;
|
|
if (b<y) b=y;
|
|
|
|
|
|
vidc = *vidc_current;
|
|
xs=vc->xcur;
|
|
ys=vc->ycur;
|
|
|
|
vc->xcur = 0;
|
|
vc->ycur = 0;
|
|
if ((vc->flags&LOSSY) == 0)
|
|
{
|
|
register int c;
|
|
/* This has *GOT* to be turboed */
|
|
for ( vc->ycur=y; vc->ycur<=b; vc->ycur++ )
|
|
{
|
|
for ( vc->xcur=x; vc->xcur<=a; vc->xcur++ )
|
|
{
|
|
c = (vc->charmap)[vc->xcur+vc->ycur*vc->xchars];
|
|
if ((c&BOLD)!=0)
|
|
R_DATA->font = R_DATA->boldfont;
|
|
else
|
|
R_DATA->font = R_DATA->normalfont;
|
|
R_DATA->fast_render = ((c>>8)&0x7)|(((c>>11)&0x7)<<8)| (R_DATA->font->pixel_height<<16);
|
|
if ( c & BLINKING )
|
|
c+=1<<8 | 1;
|
|
if ((c&BLINKING)!=0) {
|
|
R_DATA->forecolour+=16;
|
|
R_DATA->backcolour+=16;
|
|
}
|
|
vidcrender_render( vc, c&0xff );
|
|
}
|
|
}
|
|
}
|
|
vc->xcur = xs;
|
|
vc->ycur = ys;
|
|
R_DATA->forecolour = p_forecol;
|
|
R_DATA->backcolour = p_backcol;
|
|
R_DATA->font = p_font;
|
|
return 0;
|
|
}
|
|
|
|
|
|
int
|
|
vidcrender_swapin(vc)
|
|
struct vconsole *vc;
|
|
{
|
|
register int counter;
|
|
int xs, ys;
|
|
struct vidc_state vidc;
|
|
font_struct *p_font = R_DATA->font;
|
|
int p_forecol = R_DATA->forecolour;
|
|
int p_backcol = R_DATA->backcolour;
|
|
|
|
#ifdef ACTIVITY_WARNING
|
|
vconsole_pending = 0;
|
|
#endif
|
|
vidc_write ( VIDC_CP1, 0x0 );
|
|
|
|
vidc = *vidc_current;
|
|
vidc_write ( VIDC_PALREG, 0x00000000 );
|
|
for ( counter=0; counter<255; counter++ )
|
|
vidc_write ( VIDC_PALETTE, 0x00000000 );
|
|
xs=vc->xcur;
|
|
ys=vc->ycur;
|
|
/*TODO This needs to be vidc_restore (something) */
|
|
vidcrender_mode ( vc, &MODE );
|
|
|
|
vc->xcur = 0;
|
|
vc->ycur = 0;
|
|
if ( (vc->flags&LOSSY) == 0 )
|
|
{
|
|
register int c;
|
|
/* This has *GOT* to be turboed */
|
|
for ( vc->ycur=0; vc->ycur<vc->ychars; vc->ycur++ )
|
|
{
|
|
for ( vc->xcur=0; vc->xcur<vc->xchars; vc->xcur++ )
|
|
{
|
|
c = (vc->charmap)[vc->xcur+vc->ycur*vc->xchars];
|
|
if ((c&BOLD)!=0)
|
|
R_DATA->font = R_DATA->boldfont;
|
|
else
|
|
R_DATA->font = R_DATA->normalfont;
|
|
/*
|
|
R_DATA->forecolour = ((c>>8)&0x7);
|
|
R_DATA->backcolour = ((c>>11)&0x7);
|
|
*/
|
|
R_DATA->fast_render = ((c>>8)&0x7)|(((c>>11)&0x7)<<8)| (R_DATA->font->pixel_height<<16);
|
|
if ( c & BLINKING )
|
|
c+=1<<8 | 1;
|
|
if ((c&BLINKING)!=0)
|
|
{
|
|
R_DATA->forecolour+=16;
|
|
R_DATA->backcolour+=16;
|
|
}
|
|
vidcrender_render( vc, c&0xff );
|
|
}
|
|
}
|
|
}
|
|
else
|
|
{
|
|
vc->CLS ( vc );
|
|
}
|
|
|
|
if ( vc->vtty==1 )
|
|
{
|
|
vc->xcur = xs;
|
|
vc->ycur = ys;
|
|
vidcrender_textpalette ( vc );
|
|
vidc_write ( VIDC_CP1, 0xffffff );
|
|
R_DATA->forecolour = p_forecol;
|
|
R_DATA->backcolour = p_backcol;
|
|
R_DATA->font = p_font;
|
|
}
|
|
/* Make the cursor blank */
|
|
IOMD_WRITE_WORD(IOMD_CURSINIT,p_cursor_transparent);
|
|
return 0;
|
|
|
|
}
|
|
|
|
paddr_t
|
|
vidcrender_mmap(vc, offset, nprot)
|
|
struct vconsole *vc;
|
|
off_t offset;
|
|
int nprot;
|
|
{
|
|
if ((u_int)offset >= videomemory.vidm_size)
|
|
return (-1);
|
|
return(arm_btop(((videomemory.vidm_pbase) + (offset))));
|
|
}
|
|
|
|
void
|
|
vidcrender_render(vc, c)
|
|
struct vconsole *vc;
|
|
char c;
|
|
{
|
|
register unsigned char *fontaddr;
|
|
register unsigned char *addr;
|
|
|
|
/* Calculate the font's address */
|
|
|
|
fontaddr = R_DATA->font->data
|
|
+ ((c-(0x20)) * R_DATA->font->height
|
|
* R_DATA->font->width);
|
|
|
|
addr = (unsigned char *)dispstart
|
|
+ (vc->xcur * R_DATA->font->x_spacing)
|
|
+ (vc->ycur * R_DATA->bytes_per_line);
|
|
|
|
vidcrendermc_render ( addr, fontaddr, R_DATA->fast_render,
|
|
R_DATA->XRES );
|
|
}
|
|
|
|
/*
|
|
* Uugh. vidc graphics dont support scrolling regions so we have to emulate
|
|
* it here. This would normally require much software scrolling which is
|
|
* horriblly slow, so I'm going to try and do a composite scroll, which
|
|
* causes problems for scrollback but it's less speed critical
|
|
*/
|
|
|
|
void
|
|
vidcrender_scrollup(vc, low, high)
|
|
struct vconsole *vc;
|
|
int low;
|
|
int high;
|
|
{
|
|
unsigned char *start, *end;
|
|
|
|
if ( ( low==0 ) && ( high==vc->ychars-1 ))
|
|
{
|
|
/* All hardware scroll */
|
|
dispstart+=R_DATA->bytes_per_line;
|
|
if ( dispstart >= dispend )
|
|
dispstart -= dispsize;
|
|
|
|
high=high+1; /* Big hack */
|
|
|
|
IOMD_WRITE_WORD(IOMD_VIDINIT, dispstart - ptov );
|
|
}
|
|
else
|
|
{
|
|
char *oldstart=(char *)dispstart;
|
|
|
|
/* Composite scroll */
|
|
|
|
if ( (high-low) > (vc->ychars>>1) )
|
|
{
|
|
/* Scroll region greater than half the screen */
|
|
|
|
dispstart+=R_DATA->bytes_per_line;
|
|
if ( dispstart >= dispend ) dispstart -= dispsize;
|
|
|
|
IOMD_WRITE_WORD(IOMD_VIDINIT, dispstart - ptov );
|
|
|
|
if ( low!=0 )
|
|
{
|
|
start = (unsigned char *)oldstart;
|
|
end=(unsigned char*)oldstart+((low+1) * R_DATA->bytes_per_line);
|
|
memcpy(start+R_DATA->bytes_per_line, start,
|
|
end-start-R_DATA->bytes_per_line);
|
|
}
|
|
|
|
if ( high!=(vc->ychars-1) )
|
|
{
|
|
start =(unsigned char *)dispstart+(high)*R_DATA->bytes_per_line;
|
|
end=(unsigned char*)dispstart+((vc->ychars)*R_DATA->bytes_per_line);
|
|
memcpy(start+R_DATA->bytes_per_line, start,
|
|
end-start-R_DATA->bytes_per_line);
|
|
}
|
|
high++;
|
|
}
|
|
else
|
|
{
|
|
/* Scroll region less than half the screen */
|
|
|
|
/* NO COMPOSITE SCROLL YET */
|
|
|
|
high++;
|
|
if (low<0) low=0;
|
|
if (high>(vc->ychars)) high=vc->ychars;
|
|
if (low>high) return; /* yuck */
|
|
start = (unsigned char *)dispstart + ((low)*R_DATA->bytes_per_line);
|
|
end = (unsigned char *)dispstart + ((high)*R_DATA->bytes_per_line);
|
|
memcpy(start, start+R_DATA->bytes_per_line,
|
|
(end-start)-R_DATA->bytes_per_line );
|
|
R_DATA->scrollback_end = dispstart;
|
|
}
|
|
}
|
|
memset ( (char *) dispstart + ((high-1)*R_DATA->bytes_per_line) ,
|
|
R_DATA->backfillcolour,
|
|
R_DATA->bytes_per_line );
|
|
}
|
|
|
|
void
|
|
vidcrender_scrolldown(vc, low, high)
|
|
struct vconsole *vc;
|
|
int low;
|
|
int high;
|
|
{
|
|
unsigned char *start;
|
|
unsigned char *end;
|
|
|
|
if ( low<0 ) low = 0;
|
|
if ( high>(vc->ychars-1) ) high=vc->ychars-1;
|
|
|
|
if ( ( low==0 ) && ( high==vc->ychars-1 ))
|
|
{
|
|
dispstart-=R_DATA->bytes_per_line;
|
|
|
|
if ( dispstart < dispbase )
|
|
dispstart += dispsize;
|
|
|
|
IOMD_WRITE_WORD(IOMD_VIDINIT, dispstart - ptov );
|
|
}
|
|
else
|
|
{
|
|
if ( ((high-low) > (vc->ychars>>1)) )
|
|
{
|
|
high--;
|
|
if (high!=(vc->ychars-1))
|
|
{
|
|
start =(unsigned char*)dispstart+((high+1)*R_DATA->bytes_per_line);
|
|
end=(unsigned char*)dispstart+((vc->ychars)*R_DATA->bytes_per_line);
|
|
memcpy(start, start+R_DATA->bytes_per_line,
|
|
(end-start)-R_DATA->bytes_per_line );
|
|
}
|
|
|
|
dispstart-=R_DATA->bytes_per_line;
|
|
if ( dispstart < dispbase )
|
|
dispstart += dispsize;
|
|
IOMD_WRITE_WORD(IOMD_VIDINIT, dispstart - ptov);
|
|
start = (unsigned char *)dispstart + (low * R_DATA->bytes_per_line);
|
|
|
|
if (low!=0)
|
|
{
|
|
end = (unsigned char *)dispstart + ((low+1)*R_DATA->bytes_per_line);
|
|
memcpy((char *)dispstart,
|
|
(char *)(dispstart+R_DATA->bytes_per_line),
|
|
(int)((end-dispstart)-R_DATA->bytes_per_line ));
|
|
}
|
|
}
|
|
else
|
|
{
|
|
start = (unsigned char *)dispstart + (low * R_DATA->bytes_per_line);
|
|
end = (unsigned char *)dispstart + ((high+1) * R_DATA->bytes_per_line);
|
|
memcpy(start+R_DATA->bytes_per_line, start,
|
|
end-start-R_DATA->bytes_per_line);
|
|
}
|
|
|
|
}
|
|
memset ((char*) dispstart + (low*R_DATA->bytes_per_line) ,
|
|
R_DATA->backfillcolour, R_DATA->bytes_per_line );
|
|
}
|
|
|
|
void
|
|
vidcrender_cls(vc)
|
|
struct vconsole *vc;
|
|
{
|
|
vidcrendermc_cls ( (char *)dispstart, (char *)dispstart+R_DATA->screensize, R_DATA->backfillcolour );
|
|
/*
|
|
memset((char *)dispstart,
|
|
R_DATA->backfillcolour, R_DATA->screensize);
|
|
*/
|
|
vc->xcur = vc->ycur = 0;
|
|
}
|
|
|
|
void
|
|
vidcrender_update(vc)
|
|
struct vconsole *vc;
|
|
{
|
|
}
|
|
|
|
static char vidcrender_name[] = VIDC_ENGINE_NAME;
|
|
|
|
static int scrollback_ptr = 0;
|
|
|
|
int
|
|
vidcrender_scrollback(vc)
|
|
struct vconsole *vc;
|
|
{
|
|
int temp;
|
|
|
|
if (scrollback_ptr==0)
|
|
scrollback_ptr=dispstart;
|
|
|
|
temp = scrollback_ptr;
|
|
|
|
scrollback_ptr-=R_DATA->bytes_per_line * (vc->ychars-2);
|
|
|
|
if ( scrollback_ptr < dispbase )
|
|
scrollback_ptr += dispsize;
|
|
|
|
if ( (scrollback_ptr>dispstart)&&
|
|
(scrollback_ptr<(dispstart+R_DATA->screensize) ) )
|
|
{
|
|
scrollback_ptr=temp;
|
|
return 0;
|
|
}
|
|
|
|
vc->r_scrolledback = 1;
|
|
|
|
IOMD_WRITE_WORD(IOMD_VIDINIT, scrollback_ptr - ptov);
|
|
return 0;
|
|
}
|
|
|
|
int
|
|
vidcrender_scrollforward(vc)
|
|
struct vconsole *vc;
|
|
{
|
|
register int temp;
|
|
|
|
if (scrollback_ptr==0)
|
|
return 0;
|
|
|
|
temp = scrollback_ptr;
|
|
|
|
scrollback_ptr+=R_DATA->bytes_per_line * (vc->ychars - 2);
|
|
|
|
if ( scrollback_ptr >= dispend )
|
|
scrollback_ptr -= dispsize;
|
|
|
|
if ( scrollback_ptr == dispstart )
|
|
{
|
|
IOMD_WRITE_WORD(IOMD_VIDINIT, scrollback_ptr - ptov);
|
|
scrollback_ptr=0;
|
|
vc->r_scrolledback = 0;
|
|
return 0;
|
|
}
|
|
|
|
IOMD_WRITE_WORD(IOMD_VIDINIT, scrollback_ptr - ptov);
|
|
return 0;
|
|
}
|
|
|
|
int
|
|
vidcrender_scrollbackend(vc)
|
|
struct vconsole *vc;
|
|
{
|
|
scrollback_ptr = 0;
|
|
IOMD_WRITE_WORD(IOMD_VIDINIT, dispstart - ptov);
|
|
vc->r_scrolledback = 0;
|
|
return 0;
|
|
}
|
|
|
|
int
|
|
vidcrender_clreos(vc, code)
|
|
struct vconsole *vc;
|
|
int code;
|
|
{
|
|
char *addr;
|
|
char *endofscreen;
|
|
|
|
addr = (unsigned char *)dispstart
|
|
+ (vc->xcur * R_DATA->font->x_spacing)
|
|
+ (vc->ycur * R_DATA->bytes_per_line);
|
|
|
|
endofscreen = (unsigned char *)dispstart
|
|
+ (vc->xchars * R_DATA->font->x_spacing)
|
|
+ (vc->ychars * R_DATA->bytes_per_line);
|
|
|
|
|
|
switch (code)
|
|
{
|
|
case 0:
|
|
vidcrendermc_cls ( addr,
|
|
(unsigned char *)dispend,
|
|
R_DATA->backfillcolour );
|
|
if ((unsigned char *)endofscreen > (unsigned char *)dispend) {
|
|
char string[80];
|
|
sprintf(string, "(addr=%08x eos=%08x dispend=%08x dispstart=%08x base=%08x)",
|
|
(u_int)addr, (u_int)endofscreen, dispend, dispstart, dispbase);
|
|
dprintf(string);
|
|
vidcrendermc_cls((unsigned char *)dispbase, (unsigned char *)(dispbase + (endofscreen - dispend)), R_DATA->backfillcolour);
|
|
}
|
|
break;
|
|
|
|
case 1:
|
|
vidcrendermc_cls ( (unsigned char *)dispstart+R_DATA->screensize,
|
|
addr,
|
|
R_DATA->backfillcolour );
|
|
break;
|
|
|
|
case 2:
|
|
default:
|
|
vidcrender_cls ( vc );
|
|
break;
|
|
}
|
|
return 0;
|
|
}
|
|
|
|
#define VIDC R_DATA->vidc
|
|
|
|
int
|
|
vidcrender_debugprint(vc)
|
|
struct vconsole *vc;
|
|
{
|
|
#ifdef DEVLOPING
|
|
printf ( "VIDCCONSOLE DEBUG INFORMATION\n\n" );
|
|
printf ( "res (%d, %d) charsize (%d, %d) cursor (%d, %d)\n"
|
|
, R_DATA->XRES, R_DATA->YRES
|
|
, vc->xchars, vc->ychars, vc->xcur, vc->ycur );
|
|
printf ( "bytes_per_line %d\n" , R_DATA->bytes_per_line );
|
|
printf ( "pixelsperbyte %d\n" , R_DATA->pixelsperbyte );
|
|
printf ( "dispstart %08x\n" , dispstart );
|
|
printf ( "dispend %08x\n" , dispend );
|
|
printf ( "screensize %08x\n" , R_DATA->screensize );
|
|
|
|
printf ( "fontwidth %08x\n" , R_DATA->font->pixel_width );
|
|
printf ( "fontheight %08x\n" , R_DATA->font->pixel_height );
|
|
printf ( "\n" );
|
|
printf ( "palreg = %08x bcol = %08x\n" , VIDC.palreg, VIDC.bcol );
|
|
printf ( "cp1 = %08x cp2 = %08x cp3 = %08x\n" , VIDC.cp1, VIDC.cp2, VIDC.cp3 );
|
|
printf ( "hcr = %08x hswr = %08x hbsr = %08x\n" , VIDC.hcr, VIDC.hswr, VIDC.hbsr );
|
|
printf ( "hder = %08x hber = %08x hcsr = %08x\n" , VIDC.hder, VIDC.hber, VIDC.hcsr );
|
|
printf ( "hir = %08x\n" , VIDC.hir );
|
|
printf ( "vcr = %08x vswr = %08x vbsr = %08x\n" , VIDC.vcr, VIDC.vswr, VIDC.vbsr );
|
|
printf ( "vder = %08x vber = %08x vcsr = %08x\n" , VIDC.vder, VIDC.vber, VIDC.vcsr );
|
|
printf ( "vcer = %08x\n" , VIDC.vcer );
|
|
printf ( "ereg = %08x fsynreg = %08x conreg = %08x\n" , VIDC.ereg, VIDC.fsynreg, VIDC.conreg );
|
|
printf ( "\n" );
|
|
printf ( "flash %08x, cursor_flash %08x", R_DATA->flash, R_DATA->cursor_flash );
|
|
#else
|
|
printf ( "VIDCCONSOLE - NO DEBUG INFO\n" );
|
|
#endif
|
|
return 0;
|
|
}
|
|
|
|
#ifdef NICE_UPDATE
|
|
static int need_update = 0;
|
|
|
|
void
|
|
vidcrender_updatecursor(arg)
|
|
void *arg;
|
|
{
|
|
struct vconsole *vc = vconsole_current;
|
|
|
|
vidc_write(VIDC_HCSR, R_DATA->frontporch-17+ (vc->xcur)*R_DATA->font->pixel_width );
|
|
vidc_write(VIDC_VCSR, R_DATA->topporch-2+ (vc->ycur+1)*R_DATA->font->pixel_height-2 + 3
|
|
- R_DATA->font->pixel_height);
|
|
vidc_write(VIDC_VCER, R_DATA->topporch-2+ (vc->ycur+3)*R_DATA->font->pixel_height+2 + 3 );
|
|
return;
|
|
}
|
|
|
|
int
|
|
vidcrender_cursorupdate(vc)
|
|
struct vconsole *vc;
|
|
{
|
|
timeout ( vidcrender_updatecursor, NULL, 20 );
|
|
return 0;
|
|
}
|
|
|
|
#else
|
|
|
|
static int
|
|
vidcrender_cursorupdate(vc)
|
|
struct vconsole *vc;
|
|
{
|
|
vidc_write(VIDC_HCSR, R_DATA->frontporch-17+ (vc->xcur)*R_DATA->font->pixel_width );
|
|
vidc_write(VIDC_VCSR, R_DATA->topporch-2+ (vc->ycur+1)*R_DATA->font->pixel_height-2 + 3
|
|
- R_DATA->font->pixel_height);
|
|
vidc_write(VIDC_VCER, R_DATA->topporch-2+ (vc->ycur+3)*R_DATA->font->pixel_height+2 + 3 );
|
|
return (0);
|
|
}
|
|
|
|
#endif
|
|
|
|
#define DEFAULT_CURSORSPEED (25)
|
|
|
|
static int CURSORSPEED = DEFAULT_CURSORSPEED;
|
|
|
|
static int
|
|
vidcrender_cursorflashrate(vc, rate)
|
|
struct vconsole *vc;
|
|
int rate;
|
|
{
|
|
CURSORSPEED = 60/rate;
|
|
return(0);
|
|
}
|
|
|
|
static int cursorcounter=DEFAULT_CURSORSPEED;
|
|
static int flashcounter=DEFAULT_CURSORSPEED;
|
|
#ifdef PRETTYCURSOR
|
|
static int pretty=0xff;
|
|
#endif
|
|
|
|
static int cursor_col = 0x0;
|
|
|
|
int
|
|
vidcrender_cursorintr(arg)
|
|
void *arg;
|
|
{
|
|
struct vconsole *vc = arg;
|
|
if ( cursor_flash==0 )
|
|
return 0;
|
|
|
|
/*
|
|
* We don't need this.
|
|
*/
|
|
if (vconsole_blankcounter >= 0)
|
|
vconsole_blankcounter--;
|
|
|
|
if (vconsole_blankcounter == 0) {
|
|
vconsole_blankcounter=vconsole_blankinit;
|
|
vidcrender_blank ( vc, BLANK_OFF );
|
|
}
|
|
|
|
cursorcounter--;
|
|
if (cursorcounter<=0) {
|
|
cursorcounter=CURSORSPEED;
|
|
cursor_col = cursor_col ^ 0xffffff;
|
|
#ifdef ACTIVITY_WARNING
|
|
if (vconsole_pending) {
|
|
if ( cursor_col==0 )
|
|
IOMD_WRITE_WORD(IOMD_CURSINIT,p_cursor_transparent);
|
|
else
|
|
IOMD_WRITE_WORD(IOMD_CURSINIT,p_cursor_normal);
|
|
vidc_write ( VIDC_CP1, cursor_col&0xff );
|
|
} else
|
|
#endif
|
|
{
|
|
if ( cursor_col==0 )
|
|
IOMD_WRITE_WORD(IOMD_CURSINIT,p_cursor_transparent);
|
|
else
|
|
IOMD_WRITE_WORD(IOMD_CURSINIT,p_cursor_normal);
|
|
vidc_write ( VIDC_CP1, 0xffffff );
|
|
}
|
|
}
|
|
return(0); /* Pass interrupt on down the chain */
|
|
}
|
|
|
|
int
|
|
vidcrender_flashintr(arg)
|
|
void *arg;
|
|
{
|
|
/* struct vconsole *vc = arg;*/
|
|
if ( flash==0 )
|
|
return 0;
|
|
|
|
flashcounter--;
|
|
if (flashcounter<=0) {
|
|
flashcounter=CURSORSPEED;
|
|
if ( cursor_col == 0 ) {
|
|
|
|
vidc_write(VIDC_PALREG, 0x00000010);
|
|
vidc_write(VIDC_PALETTE, VIDC_COL( 0, 0, 0));
|
|
vidc_write(VIDC_PALETTE, VIDC_COL(255, 0, 0));
|
|
vidc_write(VIDC_PALETTE, VIDC_COL( 0, 255, 0));
|
|
vidc_write(VIDC_PALETTE, VIDC_COL(255, 255, 0));
|
|
vidc_write(VIDC_PALETTE, VIDC_COL( 0, 0, 255));
|
|
vidc_write(VIDC_PALETTE, VIDC_COL(255, 0, 255));
|
|
vidc_write(VIDC_PALETTE, VIDC_COL( 0, 255, 255));
|
|
vidc_write(VIDC_PALETTE, VIDC_COL(255, 255, 255));
|
|
vidc_write(VIDC_PALETTE, VIDC_COL(128, 128, 128));
|
|
vidc_write(VIDC_PALETTE, VIDC_COL(255, 128, 128));
|
|
vidc_write(VIDC_PALETTE, VIDC_COL(128, 255, 128));
|
|
vidc_write(VIDC_PALETTE, VIDC_COL(255, 255, 128));
|
|
vidc_write(VIDC_PALETTE, VIDC_COL(128, 128, 255));
|
|
vidc_write(VIDC_PALETTE, VIDC_COL(255, 128, 255));
|
|
vidc_write(VIDC_PALETTE, VIDC_COL(255, 255, 255));
|
|
} else {
|
|
vidc_write(VIDC_PALREG, 0x00000010);
|
|
vidc_write(VIDC_PALETTE, VIDC_COL( 0, 0, 0));
|
|
vidc_write(VIDC_PALETTE, VIDC_COL( 0, 0, 0));
|
|
vidc_write(VIDC_PALETTE, VIDC_COL( 0, 0, 0));
|
|
vidc_write(VIDC_PALETTE, VIDC_COL( 0, 0, 0));
|
|
vidc_write(VIDC_PALETTE, VIDC_COL( 0, 0, 0));
|
|
vidc_write(VIDC_PALETTE, VIDC_COL( 0, 0, 0));
|
|
vidc_write(VIDC_PALETTE, VIDC_COL( 0, 0, 0));
|
|
vidc_write(VIDC_PALETTE, VIDC_COL( 0, 0, 0));
|
|
vidc_write(VIDC_PALETTE, VIDC_COL( 0, 0, 0));
|
|
vidc_write(VIDC_PALETTE, VIDC_COL( 0, 0, 0));
|
|
vidc_write(VIDC_PALETTE, VIDC_COL( 0, 0, 0));
|
|
vidc_write(VIDC_PALETTE, VIDC_COL( 0, 0, 0));
|
|
vidc_write(VIDC_PALETTE, VIDC_COL( 0, 0, 0));
|
|
vidc_write(VIDC_PALETTE, VIDC_COL( 0, 0, 0));
|
|
vidc_write(VIDC_PALETTE, VIDC_COL( 0, 0, 0));
|
|
vidc_write(VIDC_PALETTE, VIDC_COL( 0, 0, 0));
|
|
}
|
|
}
|
|
return(0); /* Pass interrupt on down the chain */
|
|
}
|
|
|
|
static int
|
|
vidc_cursor_init(vc)
|
|
struct vconsole *vc;
|
|
{
|
|
static char *cursor_data = NULL;
|
|
int counter;
|
|
int line;
|
|
paddr_t pa;
|
|
|
|
if (!cursor_data) {
|
|
/* Allocate cursor memory first time round */
|
|
cursor_data = (char *)uvm_km_alloc(kernel_map, PAGE_SIZE, 0,
|
|
UVM_KMF_VAONLY);
|
|
if (!cursor_data)
|
|
panic("Cannot allocate memory for hardware cursor");
|
|
(void) pmap_extract(pmap_kernel(), (vaddr_t)cursor_data, &pa);
|
|
IOMD_WRITE_WORD(IOMD_CURSINIT, pa);
|
|
}
|
|
|
|
/* Blank the cursor while initialising it's sprite */
|
|
|
|
vidc_write ( VIDC_CP1, 0x0 );
|
|
vidc_write ( VIDC_CP2, 0x0 );
|
|
vidc_write ( VIDC_CP3, 0x0 );
|
|
|
|
cursor_normal = cursor_data;
|
|
cursor_transparent = cursor_data + (R_DATA->font->pixel_height *
|
|
R_DATA->font->pixel_width);
|
|
|
|
cursor_transparent += 32;
|
|
cursor_transparent = (char *)((int)cursor_transparent & (~31) );
|
|
|
|
for ( line = 0; line<R_DATA->font->pixel_height; ++line )
|
|
{
|
|
for ( counter=0; counter<R_DATA->font->pixel_width/4;counter++ )
|
|
cursor_normal[line*R_DATA->font->pixel_width + counter]=0x55;
|
|
for ( ; counter<8; counter++ )
|
|
cursor_normal[line*R_DATA->font->pixel_width + counter]=0;
|
|
}
|
|
|
|
for ( line = 0; line<R_DATA->font->pixel_height; ++line )
|
|
{
|
|
for ( counter=0; counter<R_DATA->font->pixel_width/4;counter++ )
|
|
cursor_transparent[line*R_DATA->font->pixel_width + counter]=0x00;
|
|
for ( ; counter<8; counter++ )
|
|
cursor_transparent[line*R_DATA->font->pixel_width + counter]=0;
|
|
}
|
|
|
|
|
|
(void) pmap_extract(pmap_kernel(), (vaddr_t)cursor_normal,
|
|
(void *)&p_cursor_normal);
|
|
(void) pmap_extract(pmap_kernel(), (vaddr_t)cursor_transparent,
|
|
(void *)&p_cursor_transparent);
|
|
|
|
/*
|
|
memset ( cursor_normal, 0x55,
|
|
R_DATA->font->pixel_width*R_DATA->font->pixel_height );
|
|
|
|
memset ( cursor_transparent, 0x55,
|
|
R_DATA->font->pixel_width*R_DATA->font->pixel_height );
|
|
*/
|
|
|
|
/* Ok, now see the cursor */
|
|
|
|
vidc_write ( VIDC_CP1, 0xffffff );
|
|
return 0;
|
|
}
|
|
|
|
int
|
|
vidcrender_setfgcol(vc, col)
|
|
struct vconsole *vc;
|
|
int col;
|
|
{
|
|
register int loop;
|
|
|
|
if ( R_DATA->forecolour >= 16 )
|
|
R_DATA->forecolour=16;
|
|
else
|
|
R_DATA->forecolour=0;
|
|
|
|
R_DATA->forefillcolour = 0;
|
|
|
|
R_DATA->forecolour += col;
|
|
|
|
/*TODO
|
|
if ( R_DATA->forecolour >> 1<<R_DATA->NUMCOLOURS )
|
|
R_DATA->forecolour>>1;
|
|
*/
|
|
|
|
for (loop = 0; loop < R_DATA->pixelsperbyte; ++loop) {
|
|
R_DATA->forefillcolour |= (R_DATA->forecolour <<
|
|
loop * (1 << R_DATA->mode.log2_bpp));
|
|
}
|
|
R_DATA->fast_render = R_DATA->forecolour | (R_DATA->backcolour<<8) | (R_DATA->font->pixel_height<<16);
|
|
return 0;
|
|
}
|
|
|
|
int
|
|
vidcrender_setbgcol(vc, col)
|
|
struct vconsole *vc;
|
|
int col;
|
|
{
|
|
register int loop;
|
|
|
|
if ( R_DATA->backcolour >= 16 )
|
|
R_DATA->backcolour=16;
|
|
else
|
|
R_DATA->backcolour=0;
|
|
|
|
R_DATA->backfillcolour = 0;
|
|
R_DATA->backcolour += col;
|
|
/*TODO
|
|
if ( R_DATA->backcolour >> 1<<R_DATA->NUMCOLOURS )
|
|
R_DATA->backcolour>>1;
|
|
*/
|
|
|
|
for (loop = 0; loop < R_DATA->pixelsperbyte; ++loop) {
|
|
R_DATA->backfillcolour |= (R_DATA->backcolour <<
|
|
loop * (1 << R_DATA->mode.log2_bpp));
|
|
}
|
|
return 0;
|
|
}
|
|
|
|
int
|
|
vidcrender_textpalette(vc)
|
|
struct vconsole *vc;
|
|
{
|
|
R_DATA->forecolour = COLOUR_WHITE_8;
|
|
R_DATA->backcolour = COLOUR_BLACK_8;
|
|
|
|
vidc_write(VIDC_PALREG, 0x00000000);
|
|
vidc_write(VIDC_PALETTE, VIDC_COL( 0, 0, 0));
|
|
vidc_write(VIDC_PALETTE, VIDC_COL(255, 0, 0));
|
|
vidc_write(VIDC_PALETTE, VIDC_COL( 0, 255, 0));
|
|
vidc_write(VIDC_PALETTE, VIDC_COL(255, 255, 0));
|
|
vidc_write(VIDC_PALETTE, VIDC_COL( 0, 0, 255));
|
|
vidc_write(VIDC_PALETTE, VIDC_COL(255, 0, 255));
|
|
vidc_write(VIDC_PALETTE, VIDC_COL( 0, 255, 255));
|
|
vidc_write(VIDC_PALETTE, VIDC_COL(255, 255, 255));
|
|
vidc_write(VIDC_PALETTE, VIDC_COL(128, 128, 128));
|
|
vidc_write(VIDC_PALETTE, VIDC_COL(255, 128, 128));
|
|
vidc_write(VIDC_PALETTE, VIDC_COL(128, 255, 128));
|
|
vidc_write(VIDC_PALETTE, VIDC_COL(255, 255, 128));
|
|
vidc_write(VIDC_PALETTE, VIDC_COL(128, 128, 255));
|
|
vidc_write(VIDC_PALETTE, VIDC_COL(255, 128, 255));
|
|
vidc_write(VIDC_PALETTE, VIDC_COL(255, 255, 255));
|
|
|
|
R_DATA->fast_render = R_DATA->forecolour | (R_DATA->backcolour<<8) | (R_DATA->font->pixel_height<<16);
|
|
return 0;
|
|
}
|
|
|
|
int
|
|
vidcrender_sgr(vc, type)
|
|
struct vconsole *vc;
|
|
int type;
|
|
{
|
|
switch(type)
|
|
{
|
|
case 0: /* Normal */
|
|
if ((1 << R_DATA->mode.log2_bpp) == 8)
|
|
{
|
|
R_DATA->n_forecolour = COLOUR_WHITE_8;
|
|
R_DATA->n_backcolour = COLOUR_BLACK_8;
|
|
}
|
|
|
|
R_DATA->forecolour = R_DATA->n_forecolour;
|
|
R_DATA->backcolour = R_DATA->n_backcolour;
|
|
R_DATA->font = R_DATA->normalfont;
|
|
break;
|
|
|
|
case 1: /* bold */
|
|
R_DATA->font = R_DATA->boldfont;
|
|
break;
|
|
|
|
case 22: /* not bold */
|
|
R_DATA->font = R_DATA->normalfont;
|
|
break;
|
|
|
|
case 5: /* blinking */
|
|
if ( R_DATA->forecolour < 16 )
|
|
{
|
|
R_DATA->forecolour+=16;
|
|
R_DATA->backcolour+=16;
|
|
R_DATA->n_forecolour+=16;
|
|
R_DATA->n_backcolour+=16;
|
|
}
|
|
break;
|
|
|
|
case 25: /* not blinking */
|
|
if ( R_DATA->forecolour >= 16 )
|
|
{
|
|
R_DATA->forecolour-=16;
|
|
R_DATA->backcolour-=16;
|
|
R_DATA->n_forecolour-=16;
|
|
R_DATA->n_backcolour-=16;
|
|
}
|
|
break;
|
|
|
|
case 7: /* reverse */
|
|
R_DATA->forecolour = R_DATA->n_backcolour;
|
|
R_DATA->backcolour = R_DATA->n_forecolour;
|
|
break;
|
|
|
|
case 27: /* not reverse */
|
|
R_DATA->forecolour = R_DATA->n_forecolour;
|
|
R_DATA->backcolour = R_DATA->n_backcolour;
|
|
break;
|
|
}
|
|
R_DATA->fast_render = R_DATA->forecolour | (R_DATA->backcolour<<8) | (R_DATA->font->pixel_height<<16);
|
|
return 0;
|
|
}
|
|
|
|
int
|
|
vidcrender_scrollregion(vc, low, high)
|
|
struct vconsole *vc;
|
|
int low;
|
|
int high;
|
|
{
|
|
return 0;
|
|
}
|
|
|
|
int
|
|
vidcrender_blank(vc, type)
|
|
struct vconsole *vc;
|
|
int type;
|
|
{
|
|
int ereg;
|
|
|
|
vc->blanked=type;
|
|
|
|
ereg = 1<<12;
|
|
if (vidc_currentmode->sync_pol & 0x01)
|
|
ereg |= 1<<16;
|
|
if (vidc_currentmode->sync_pol & 0x02)
|
|
ereg |= 1<<18;
|
|
|
|
switch (type) {
|
|
case 0:
|
|
vidc_write(VIDC_EREG, ereg);
|
|
break;
|
|
|
|
case 1: /* not implemented yet */
|
|
case 2:
|
|
case 3:
|
|
vidc_write(VIDC_EREG, 0);
|
|
break;
|
|
}
|
|
return 0;
|
|
}
|
|
|
|
int vidcrender_ioctl ( struct vconsole *vc, dev_t dev, int cmd, caddr_t data,
|
|
int flag, struct lwp *l )
|
|
{
|
|
int error;
|
|
int bpp, log2_bpp;
|
|
struct tty *tp;
|
|
struct winsize ws;
|
|
struct vidc_mode *vmode;
|
|
|
|
switch (cmd) {
|
|
case CONSOLE_MODE:
|
|
tp = find_tp(dev);
|
|
vmode = (struct vidc_mode *) data;
|
|
|
|
/* the interface specifies bpp instead of log2_bpp ... fix me! */
|
|
bpp = vmode->log2_bpp;
|
|
|
|
/* translate 24 to 32 ... since this is the correct log2 value */
|
|
if (bpp == 24) bpp=32;
|
|
|
|
/* translate the bpp to log2_bpp */
|
|
if (bpp < 1 || bpp > 32)
|
|
bpp = 8; /* Set 8 bpp if we get asked for something silly */
|
|
for (log2_bpp = 0; bpp != 1; bpp >>= 1)
|
|
log2_bpp++;
|
|
vmode->log2_bpp = log2_bpp;
|
|
|
|
/* set new mode and update structures */
|
|
vidcrender_mode ( vc, vmode );
|
|
vc->MODECHANGE ( vc );
|
|
ws.ws_row=vc->ychars;
|
|
ws.ws_col=vc->xchars;
|
|
error = (*tp->t_linesw->l_ioctl)(tp, TIOCSWINSZ, (char *)&ws, flag, l);
|
|
if (error != EPASSTHROUGH)
|
|
return (error);
|
|
return ttioctl(tp, TIOCSWINSZ, (char *)&ws, flag, l);
|
|
break;
|
|
|
|
case CONSOLE_RESETSCREEN:
|
|
{
|
|
extern unsigned int dispbase;
|
|
extern unsigned int dispsize;
|
|
extern unsigned int ptov;
|
|
extern unsigned int transfersize;
|
|
|
|
IOMD_WRITE_WORD(IOMD_VIDINIT, dispbase-ptov);
|
|
IOMD_WRITE_WORD(IOMD_VIDSTART, dispbase-ptov);
|
|
IOMD_WRITE_WORD(IOMD_VIDEND, (dispbase+dispsize-transfersize)-ptov);
|
|
return 0;
|
|
}
|
|
case CONSOLE_RESTORESCREEN:
|
|
{
|
|
extern unsigned int dispstart;
|
|
extern unsigned int dispsize;
|
|
extern unsigned int ptov;
|
|
extern unsigned int transfersize;
|
|
|
|
IOMD_WRITE_WORD(IOMD_VIDINIT, dispstart-ptov);
|
|
IOMD_WRITE_WORD(IOMD_VIDSTART, dispstart-ptov);
|
|
IOMD_WRITE_WORD(IOMD_VIDEND, (dispstart+dispsize-transfersize)-ptov);
|
|
vidc_stdpalette();
|
|
return 0;
|
|
}
|
|
case CONSOLE_GETINFO:
|
|
{
|
|
extern videomemory_t videomemory;
|
|
register struct console_info *inf = (void *)data;
|
|
|
|
|
|
inf->videomemory = videomemory;
|
|
inf->width = R_DATA->mode.hder;
|
|
inf->height = R_DATA->mode.vder;
|
|
|
|
inf->bpp = 1 << R_DATA->mode.log2_bpp;
|
|
|
|
return 0;
|
|
}
|
|
case CONSOLE_PALETTE:
|
|
{
|
|
register struct console_palette *pal = (void *)data;
|
|
vidc_write(VIDC_PALREG, pal->entry);
|
|
vidc_write(VIDC_PALETTE, VIDC_COL(pal->red, pal->green, pal->blue));
|
|
return 0;
|
|
}
|
|
}
|
|
return (EPASSTHROUGH);
|
|
}
|
|
|
|
int
|
|
vidcrender_attach(vc, parent, self, aux)
|
|
struct vconsole *vc;
|
|
struct device *parent;
|
|
struct device *self;
|
|
void *aux;
|
|
{
|
|
vidc_cursor_init ( vc );
|
|
vidcrender_flash_go ( vc );
|
|
return 0;
|
|
}
|
|
|
|
int
|
|
vidcrender_flash_go(vc)
|
|
struct vconsole *vc;
|
|
{
|
|
static int irqclaimed = 0;
|
|
static int lock=0;
|
|
if (lock==1)
|
|
return -1;
|
|
lock=0;
|
|
|
|
if (!irqclaimed) {
|
|
cursor_ih.ih_func = vidcrender_cursorintr;
|
|
cursor_ih.ih_arg = vc;
|
|
cursor_ih.ih_level = IPL_TTY;
|
|
cursor_ih.ih_name = "vsync";
|
|
irq_claim ( IRQ_FLYBACK, &cursor_ih );
|
|
irqclaimed = 1;
|
|
}
|
|
|
|
cursor_init = 0;
|
|
return 0;
|
|
}
|
|
|
|
/* What does this function do ? */
|
|
int
|
|
vidcrender_flash(vc, flashit)
|
|
struct vconsole *vc;
|
|
int flashit;
|
|
{
|
|
flashit = flashit;
|
|
return(0);
|
|
}
|
|
|
|
int
|
|
vidcrender_cursorflash(vc, flashit)
|
|
struct vconsole *vc;
|
|
int flashit;
|
|
{
|
|
cursor_flash = flashit;
|
|
return(0);
|
|
}
|
|
|
|
struct render_engine vidcrender = {
|
|
vidcrender_name,
|
|
vidcrender_init,
|
|
|
|
vidcrender_putchar,
|
|
|
|
vidcrender_spawn,
|
|
vidcrender_swapin,
|
|
vidcrender_mmap,
|
|
vidcrender_render,
|
|
vidcrender_scrollup,
|
|
vidcrender_scrolldown,
|
|
vidcrender_cls,
|
|
vidcrender_update,
|
|
vidcrender_scrollback,
|
|
vidcrender_scrollforward,
|
|
vidcrender_scrollbackend,
|
|
vidcrender_clreos,
|
|
vidcrender_debugprint,
|
|
vidcrender_cursorupdate,
|
|
vidcrender_cursorflashrate,
|
|
vidcrender_setfgcol,
|
|
vidcrender_setbgcol,
|
|
vidcrender_textpalette,
|
|
vidcrender_sgr,
|
|
vidcrender_blank,
|
|
vidcrender_ioctl,
|
|
vidcrender_redraw,
|
|
vidcrender_attach,
|
|
vidcrender_flash,
|
|
vidcrender_cursorflash
|
|
};
|