/******************************************************************************\ * * $Workfile: LAGUNA.H $ * * This file to be included by all host programs. * * Copyright (c) 1995,1997 Cirrus Logic, Inc. * * $Log: X:/log/laguna/nt35/displays/cl546x/LAGUNA.H $ * * Rev 1.38 Mar 04 1998 16:13:30 frido * Removed a warning message for the 5462/5464 chips in the REQUIRE * macro. * * Rev 1.37 Mar 04 1998 16:08:50 frido * Removed an invalid break. * * Rev 1.36 Mar 04 1998 14:54:48 frido * Added contional REQUIRE in the new shadow macros. * * Rev 1.35 Mar 04 1998 14:52:54 frido * Added new shadowing macros. * * Rev 1.34 Feb 27 1998 17:02:16 frido * Changed REQUIRE and WRITE_STRING macros for new * shadowQFREE register. * * Rev 1.33 Jan 20 1998 11:42:46 frido * Changed REQUIRE and WRITESTRING macros to support the new * scheme for GBP on. * Added shadowing of BGCOLOR and DRAWBLTDEF registers. * * Rev 1.32 Jan 16 1998 09:50:38 frido * Changed the way the GBP OFF handles WRITE_STRING macros. * * Rev 1.31 Dec 10 1997 13:24:58 frido * Merged from 1.62 branch. * * Rev 1.30.1.0 Nov 18 1997 18:09:00 frido * Changed WRITE_STRING macro so it will work when * DATASTREAMING is turned of. * Changed FUDGE to 0, we are not using DMA inside NT. * * Rev 1.30 Nov 04 1997 19:01:18 frido * Changed HOSTDATA size from 0x800 DWORDs into 0x800 BYTEs. Silly me! * * Rev 1.29 Nov 04 1997 09:17:30 frido * Added Data Streaming macros (REQUIRE and WRITE_STRING). * \******************************************************************************/ #ifndef _LAGUNA_H #define _LAGUNA_H #include "optimize.h" #include "config.h" #include "lgregs.h" // // PCI ID for Laguna chips. // #define CL_GD5462 0x00D0 // 5462 #define CL_GD5464 0x00D4 // 5464 #define CL_GD5464_BD 0x00D5 // 5464 BD #define CL_GD5465 0x00D6 // 5465 // // These chips don't exist yet, but we're FORWARD COMPATIBLE // so we'll define them anyway. I've been GUARENTEED that they // will look and feel just like 5465 chips. // #define CL_GD546x_F7 0x00D7 #define CL_GD546x_F8 0x00D8 #define CL_GD546x_F9 0x00D9 #define CL_GD546x_FA 0x00DA #define CL_GD546x_FB 0x00DB #define CL_GD546x_FC 0x00DC #define CL_GD546x_FD 0x00DD #define CL_GD546x_FE 0x00DE #define CL_GD546x_FF 0x00DF // // CHIP BUG: For certian values in PRESET register cursor enable/disable // causes scanlines to be duplicated at the cursor hot spot. (Seen // as screen jump.) There are lots of ways around this. The easiest // is to turn the cursor on and leave it on. Enable/Disable is handled by // moving the cursor on/off the visable screen. // #define HW_PRESET_BUG 1 // The 5465 (to at least AC) has a problem when PCI configuration space // is accessible in memory space. On 16-bit writes, a 32-bit write is // actually performed, so the next register has garbage written to it. // We get around this problem by clearing bit 0 of the Vendor Specific // Control register in PCI configuration space. When this bit is set // to 0, PCI configuration registers are not available through memory // mapped I/O. Since some functions, such as power management, require // access to PCI registers, the display driver must post a message to // the miniport to enable this bit when needed. // #define VS_CONTROL_HACK 1 #if ENABLE_LOG_FILE extern long lg_i; extern char lg_buf[256]; #endif #if POINTER_SWITCH_ENABLED extern int pointer_switch; #endif // The definitions are not portable. 486 / PC only !!! typedef struct { BYTE b; BYTE g; BYTE r; } pix24; typedef struct { BYTE u; BYTE y1; BYTE v; BYTE y2; } yuv_422; typedef struct { unsigned int v : 6; unsigned int u : 6; unsigned int y0: 5; unsigned int y1: 5; unsigned int y2: 5; unsigned int y3: 5; } yuv_411; typedef struct { unsigned int b : 5; unsigned int g : 5; unsigned int r : 5; } rgb_555; typedef struct { unsigned int b : 5; unsigned int g : 6; unsigned int r : 5; } rgb_565; typedef union { DWORD p32; yuv_422 yuv422; yuv_411 yuv411; pix24 p24; rgb_555 rgb555; rgb_565 rgb565; WORD p16[2]; BYTE p8[4]; } pixel; #define FALSE 0 #ifndef TRUE #define TRUE (~FALSE) #endif /* from struct.h */ #define fldoff(str, fld) ((int)&(((struct str *)0)->fld)) #define fldsiz(str, fld) (sizeof(((struct str *)0)->fld)) #define HPRR(pr_reg) (_AP_direct_read(PADDR(pr_reg),fldsiz(PLUTOREGS,pr_reg), (ul)0)) #define RPR(pr_reg) HPRR(pr_reg) #define EHIST (*(EXHIST*)excepttion) /* Exception History buffer */ #define STAMP (*(bytearray*)0x0) /* time date stamp */ #define HISTORYBUFFERADDR (ul)&history /* 34020 address of recording */ /* External functions the host program can call */ /*-------------------------------------------------------------------------*/ /* Function prototypes for emulator. Functions defined in host_if.c */ int _cdecl _AP_init(int mode, void * frame_buf); void _cdecl _AP_write(ul addr, int size, ul data); ul _cdecl _AP_read(ul addr,int size); void _cdecl _AP_run(void); boolean _cdecl _AP_busy(); boolean _cdecl _AP_done(); boolean _cdecl _AP_rfifo_empty(); boolean _cdecl _AP_require(int size); ul _cdecl _AP_direct_read(ul addr,int size); void _cdecl _AP_fb_write(ul offset, pixel data, ul size); pixel _cdecl _AP_fb_read(ul offset, ul size); #if LOG_QFREE #define START_OF_BLT() \ do{ \ CHECK_QFREE(); \ } while(0) #define END_OF_BLT() \ do{ \ } while(0) #else #define START_OF_BLT() #define END_OF_BLT() #endif // // This waits for the chip to go idle // #define WAIT_FOR_IDLE() \ do { \ while (LLDR_SZ (grSTATUS) != 0); \ } while (0) // // Macro to require a certian number of free queue entries. // #if DATASTREAMING #define FUDGE 2 #define REQUIRE(n) \ { \ if (ppdev->dwDataStreaming & 0x80000000) \ { \ if (ppdev->shadowQFREE < ((n) + FUDGE)) \ { \ while (ppdev->shadowQFREE < (n) + FUDGE) \ { \ ppdev->shadowQFREE = LLDR_SZ(grQFREE); \ } \ } \ ppdev->shadowQFREE -= (BYTE) n; \ } \ else if (ppdev->dwDataStreaming) \ { \ if (LLDR_SZ(grQFREE) < ((n) + FUDGE)) \ { \ while (LLDR_SZ(grSTATUS) & 0x8005) ; \ ppdev->dwDataStreaming = 0; \ } \ } \ } #define ENDREQUIRE() \ { \ ppdev->dwDataStreaming |= 1; \ } #define WRITE_STRING(src, dwords) \ { \ ULONG nDwords, nTotal = (ULONG) (dwords); \ PULONG data = (PULONG) (src); \ if (ppdev->dwDataStreaming & 0x80000000) \ { \ while (nTotal > 0) \ { \ nDwords = (ULONG) ppdev->shadowQFREE; \ if (nDwords > FUDGE) \ { \ nDwords = min(nDwords - FUDGE, nTotal); \ memcpy((LPVOID) ppdev->pLgREGS->grHOSTDATA, data, nDwords * 4); \ data += nDwords; \ nTotal -= nDwords; \ } \ ppdev->shadowQFREE = LLDR_SZ(grQFREE); \ } \ } \ else \ { \ if ( ppdev->dwDataStreaming && (LLDR_SZ(grQFREE) < nTotal) ) \ { \ while (LLDR_SZ(grSTATUS) & 0x8005) ; \ ppdev->dwDataStreaming = 0; \ } \ while (nTotal > 0) \ { \ nDwords = min(nTotal, 0x200); \ memcpy((LPVOID) ppdev->pLgREGS->grHOSTDATA, data, nDwords * 4); \ data += nDwords; \ nTotal -= nDwords; \ } \ } \ } #else #define REQUIRE(n) #define ENDREQUIRE() #define WRITE_STRING(src, dwords) \ { \ ULONG nDwords, nTotal = (ULONG) (dwords); \ PULONG data = (PULONG) (src); \ while (nTotal > 0) \ { \ nDwords = min(nTotal, 0x200); \ memcpy((LPVOID) ppdev->pLgREGS->grHOSTDATA, data, nDwords * 4); \ data += nDwords; \ nTotal -= nDwords; \ } \ } #endif // // Macros to read Laguna registers. // #define LADDR(pr_reg) fldoff(GAR,pr_reg) // #define LLDR(pr_reg,pr_siz) _AP_direct_read((ul)LADDR(pr_reg),pr_siz) #define LLDR(pr_reg,pr_siz) (ppdev->pLgREGS_real->pr_reg) #define LLDR_SZ(pr_reg) LLDR(pr_reg, fldsiz(GAR,pr_reg)) // #define LLR(pr_reg,pr_siz) _AP_read((ul)LADDR(pr_reg),pr_siz) #define LLR(pr_reg,pr_siz) LLDR(pr_reg,pr_siz) #define LLR_SZ(pr_reg) LLR(pr_reg, fldsiz(GAR,pr_reg)) #if LOG_WRITES #define LG_LOG(reg,val) \ do { \ lg_i = sprintf(lg_buf,"LL\t%4X\t%08X\r\n", \ ((DWORD)(&ppdev->pLgREGS->reg) - (DWORD)(&ppdev->pLgREGS->grCR0)), \ (val)); \ WriteLogFile(ppdev->pmfile, lg_buf, lg_i, ppdev->TxtBuff, &ppdev->TxtBuffIndex); \ } while(0) #else #define LG_LOG(reg,val) #endif // // Macros to write Laguna registers. // // This is an amazingly, incredibly hairy macro that, believe it or not, // will be greatly reduced by a good compiler. The "if" can be // pre-determined by the compiler. // The purpose of this is to ensure that exactly the right number of bytes // is written to the chip. If the programmer writes, say, a BYTE to a // DWORD sized register, we need to be sure that the byte is zero extended // and that a full DWORD gets written. // #define LRWRITE(pr_reg,pr_siz,value) \ do { \ LG_LOG(pr_reg,(value)); \ if (sizeof(ppdev->pLgREGS->pr_reg) == sizeof(BYTE)) \ { \ *(volatile BYTE *)(&ppdev->pLgREGS->pr_reg) = (BYTE)(value); \ } \ else if (sizeof(ppdev->pLgREGS->pr_reg) == sizeof(WORD)) \ { \ *(volatile WORD *)(&ppdev->pLgREGS->pr_reg) = (WORD)(value); \ *(volatile WORD *)(&ppdev->pLgREGS->grBOGUS) = (WORD)(value); \ LG_LOG(grBOGUS,(value)); \ } \ else \ { \ *(volatile DWORD *)(&ppdev->pLgREGS->pr_reg) = (DWORD)(value); \ } \ } while(0) #define LL(pr_reg,value) LRWRITE(pr_reg, fldsiz(GAR,pr_reg), value) // ---------------------------------------------------------------------------- // // Certian registers have been giving us problems. We provide special // write macros for them. // // // Writes any 8 bit register. // #define LL8(pr_reg,value) \ do { \ LG_LOG(pr_reg,(value)); \ (*(volatile BYTE *)(&ppdev->pLgREGS->pr_reg) = (BYTE)(value)); \ } while(0) // // Writes any 16 bit register. // #define LL16(pr_reg,value) \ do { \ LG_LOG(pr_reg,(value)); \ (*(volatile WORD *)(&ppdev->pLgREGS->pr_reg) = (WORD)(value)); \ } while(0) // // Double writes any 16 bit register. // #define LL16d(pr_reg,value) \ do { \ (*(volatile WORD *)(&ppdev->pLgREGS->pr_reg) = (WORD)(value)); \ LG_LOG(pr_reg,(value)); \ (*(volatile WORD *)(&ppdev->pLgREGS->grBOGUS) = (WORD)(value)); \ LG_LOG(grBOGUS,(value)); \ } while(0) // // Writes any 32 bit register. // #define LL32(pr_reg,value) \ { \ *(volatile DWORD *)(&ppdev->pLgREGS->pr_reg) = (DWORD)(value); \ } // // MACROS FOR BLTEXT REGISTER. // #define LL_BLTEXT(x,y) \ LL32 (grBLTEXT_EX.dw, (((DWORD)(y) << 16) | ((DWORD)(x)))) #if ! DRIVER_5465 #define LL_MBLTEXT(x,y) \ do { \ LL16 (grMBLTEXT_EX.pt.X, x); \ LL16 (grBLTEXT_EX.pt.Y, y); \ } while(0) #else #define LL_MBLTEXT(x,y) \ LL32 (grMBLTEXT_EX.dw, (((DWORD)(y) << 16) | ((DWORD)(x)))) #endif #define LL_BLTEXTR(x,y) \ LL32 (grBLTEXTR_EX.dw, (((DWORD)(y) << 16) | ((DWORD)(x)))) #define LL_BLTEXT_EXT(x,y) \ LL32 (grBLTEXT.dw, (((DWORD)(y) << 16) | ((DWORD)(x)))) #define LL_MBLTEXT_EXT(x,y) \ LL32 (grMBLTEXT.dw, (((DWORD)(y) << 16) | ((DWORD)(x)))) // Launch a BLT using the color translation features of the // resize engine. (1:1 resize) #define LL_BLTEXT_XLATE(src_bpp, x, y) \ do {\ LL16 (grMIN_X, (~((x)-1)));\ LL16 (grMAJ_X, (x));\ LL16 (grACCUM_X, ((x)-1));\ LL16 (grMIN_Y, (~((y)-1)));\ LL16 (grMAJ_Y, (y));\ LL16 (grACCUM_Y, ((y)-1));\ LL16 (grSRCX, (((x)*(src_bpp)) >> 3) );\ LL_BLTEXTR((x), (y));\ } while(0) // // MACROS FOR CLIPULE/CLIPLOR REGISTERS. // #define LL_CLIPULE(x,y) \ LL32 (grCLIPULE.dw, (((DWORD)(y) << 16) | ((DWORD)(x)))); #define LL_MCLIPULE(x,y) \ LL32 (grMCLIPULE.dw, (((DWORD)(y) << 16) | ((DWORD)(x)))); #define LL_CLIPLOR(x,y) \ LL32 (grCLIPLOR.dw, (((DWORD)(y) << 16) | ((DWORD)(x)))); #define LL_MCLIPLOR(x,y) \ LL32 (grMCLIPLOR.dw, (((DWORD)(y) << 16) | ((DWORD)(x)))); #define LL_CLIPULE_EX(x,y) \ LL32 (grCLIPULE_EX.dw, (((DWORD)(y) << 16) | ((DWORD)(x)))); #define LL_MCLIPULE_EX(x,y) \ LL32 (grMCLIPULE_EX.dw, (((DWORD)(y) << 16) | ((DWORD)(x)))); #define LL_CLIPLOR_EX(x,y) \ LL32 (grCLIPLOR_EX.dw, (((DWORD)(y) << 16) | ((DWORD)(x)))); #define LL_MCLIPLOR_EX(x,y) \ LL32 (grMCLIPLOR_EX.dw, (((DWORD)(y) << 16) | ((DWORD)(x)))); // // MACROS FOR OP0_opRDRAM REGISTER. // #define LL_OP0(x,y) \ LL32 (grOP0_opRDRAM.dw, (((DWORD)(y) << 16) | ((DWORD)(x)))) #define LL_OP0_MONO(x,y) \ LL32 (grOP0_opMRDRAM.dw, (((DWORD)(y) << 16) | ((DWORD)(x)))) // // MACROS FOR OP1_opRDRAM REGISTER. // #define LL_OP1(x,y) \ LL32 (grOP1_opRDRAM.dw, (((DWORD)(y) << 16) | ((DWORD)(x)))) #define LL_OP1_MONO(x,y) \ LL32 (grOP1_opMRDRAM.dw, (((DWORD)(y) << 16) | ((DWORD)(x)))) // // MACROS FOR OP2_opRDRAM REGISTER. // #define LL_OP2(x,y) \ LL32 (grOP2_opRDRAM.dw, (((DWORD)(y) << 16) | ((DWORD)(x)))) #define LL_OP2_MONO(x,y) \ LL32 (grOP2_opMRDRAM.dw, (((DWORD)(y) << 16) | ((DWORD)(x)))) // // -- End of special write macros -------------------------------------------- // /* HPR is the copy of REGISTER_STRUCTURE that the host reads and writes from. PR is the actual register state after the information has gone through the FIFO. Immediate registers are kept up-to-date in HPR. These structures are allocated in the link file. */ extern GAR PR; /* the emulator working copy */ extern GAR HPR[4]; /* the "host" copy */ #define LAGUNA_SRAM_SIZE 32 /* dwords */ #define IS_SRC 0x03 /* Source mask. */ struct _vid_mode { BYTE Bpp; // Bytes per pixel (8 / 16 / 24 / 32) WORD Xextent; // Display rsolution in pixels eg 1280 WORD Yextent; // Vertical display resolution WORD Xpitch; // Offset in bytes from line 0 to line 1 int Vesa_Mode; // Mode number for VESA ( if supported by S3 ) }; typedef struct _vid_mode vid_mode; typedef vid_mode *vid_ptr; /*************************************************************************** * * MACRO: SYNC_W_3D * * DESCRIPTION: If 3d context(s) active, wait until 3d engine idle * or until 1,000,000 checks have failed * ****************************************************************************/ #if WINNT_VER40 && DRIVER_5465 // WINNT_VER40 #define SYNC_3D_CONDITIONS (ST_POLY_ENG_BUSY|ST_EXEC_ENG_3D_BUSY|ST_XY_ENG_BUSY|/*ST_BLT_ENG_BUSY|*/ST_BLT_WF_EMPTY) #define ENSURE_3D_IDLE(ppdev) \ { \ int num_syncs=2; \ /* there is a slight chance of a window in which all bits go off while engine fetching */ \ /* next command - double read should catch that */ \ while (num_syncs--) \ { \ int status; \ volatile int wait_count=0; \ do \ { \ status = (*((volatile *)((DWORD *)(ppdev->pLgREGS) + PF_STATUS_3D)) & 0x3FF) ^ SYNC_3D_CONDITIONS; \ /* do something to give bus a breather, and to prevent eternal stall */ \ wait_count++; \ } while(((status & SYNC_3D_CONDITIONS) != SYNC_3D_CONDITIONS) && wait_count<1000000); \ } \ } #define SYNC_W_3D(ppdev) \ { \ if (ppdev->NumMCDContexts > 0) \ { \ ENSURE_3D_IDLE(ppdev); \ } \ } #else // WINNT_VER40 && DRIVER_5465 // no 3D on NT before NT4.0. No 3D on 62 and not used on 64. #define ENSURE_3D_IDLE(ppdev) {} #define SYNC_W_3D(ppdev) {} #endif // WINNT_VER40 // // New shadowing macros. // #define LL_FGCOLOR(color, r) \ { \ if ((DWORD) (color) != ppdev->shadowFGCOLOR) \ { \ if (r) REQUIRE(r); \ LL32(grOP_opFGCOLOR, ppdev->shadowFGCOLOR = (DWORD) (color)); \ } \ } #define LL_BGCOLOR(color, r) \ { \ if ((DWORD) (color) != ppdev->shadowBGCOLOR) \ { \ if (r) REQUIRE(r); \ LL32(grOP_opBGCOLOR, ppdev->shadowBGCOLOR = (DWORD) (color)); \ } \ } #define LL_DRAWBLTDEF(drawbltdef, r) \ { \ if ((DWORD) (drawbltdef) != ppdev->shadowDRAWBLTDEF) \ { \ if (r) REQUIRE(r); \ LL32(grDRAWBLTDEF, ppdev->shadowDRAWBLTDEF = (DWORD) (drawbltdef)); \ } \ } #endif /* ndef _LAGUNA_H */