windows-nt/Source/XPSP1/NT/multimedia/media/avi/drawdib.16/dcilib/dva.c

467 lines
12 KiB
C
Raw Permalink Normal View History

2020-09-26 03:20:57 -05:00
#include <windows.h>
#undef WINAPI
#define WINAPI FAR PASCAL _loadds
#include "dva.h"
#include "dciman.h"
#include "lockbm.h"
/****************************************************************************
***************************************************************************/
extern BOOL vga_get_surface(HDC, int, DVASURFACEINFO FAR *);
extern BOOL ati_get_surface(HDC, int, DVASURFACEINFO FAR *);
#ifdef DEBUG
//extern BOOL dib_get_surface(HDC, int, DVASURFACEINFO FAR *);
extern BOOL thun_get_surface(HDC,int, DVASURFACEINFO FAR *);
extern BOOL vlb_get_surface(HDC, int, DVASURFACEINFO FAR *);
#endif
#define GetDS() SELECTOROF((LPVOID)&ScreenSel)
static short ScreenSel;
static BOOL InitSurface(DVASURFACEINFO FAR *pdva);
static BOOL TestSurface(DVASURFACEINFO FAR *pdva);
static void SetSelLimit(UINT sel, DWORD limit);
/****************************************************************************
***************************************************************************/
static void FAR PASCAL DVAInit()
{
}
/****************************************************************************
***************************************************************************/
static void FAR PASCAL DVATerm()
{
//
// free screen alias
//
if (ScreenSel)
{
SetSelLimit(ScreenSel, 0);
FreeSelector(ScreenSel);
ScreenSel = 0;
}
}
/****************************************************************************
***************************************************************************/
BOOL WINAPI DVAGetSurface(HDC hdc, int nSurface, DVASURFACEINFO FAR *pdva)
{
//
// should this be a function table? list?
//
if (!ati_get_surface(hdc, nSurface, pdva) &&
#ifdef DEBUG
// !dib_get_surface(hdc, nSurface, pdva) &&
!vlb_get_surface(hdc, nSurface, pdva) &&
!thun_get_surface(hdc, nSurface, pdva) &&
#endif
!vga_get_surface(hdc, nSurface, pdva))
return FALSE;
return InitSurface(pdva);
}
/****************************************************************************
***************************************************************************/
BOOL CALLBACK default_open_surface(LPVOID pv)
{
return TRUE;
}
/****************************************************************************
***************************************************************************/
void CALLBACK default_close_surface(LPVOID pv)
{
}
/****************************************************************************
***************************************************************************/
BOOL CALLBACK default_begin_access(LPVOID pv, int x, int y, int dx, int dy)
{
return TRUE;
}
/****************************************************************************
***************************************************************************/
void CALLBACK default_end_access(LPVOID pv)
{
}
/****************************************************************************
***************************************************************************/
UINT CALLBACK default_show_surface(LPVOID pv, HWND hwnd, LPRECT src, LPRECT dst)
{
return 1;
}
/****************************************************************************
***************************************************************************/
static BOOL InitSurface(DVASURFACEINFO FAR *pdva)
{
LPBITMAPINFOHEADER lpbi;
if (pdva->Version != 0x0100)
return FALSE;
lpbi = &pdva->BitmapInfo;
if (lpbi->biSize != sizeof(BITMAPINFOHEADER))
return FALSE;
if (lpbi->biPlanes != 1)
return FALSE;
//
// make the pointer a 16:16 pointer
//
if (pdva->offSurface >= 0x10000 &&
!(pdva->Flags & DVAF_1632_ACCESS))
{
if (ScreenSel == NULL)
ScreenSel = AllocSelector(GetDS());
if (pdva->selSurface != 0)
pdva->offSurface += GetSelectorBase(pdva->selSurface);
SetSelectorBase(ScreenSel,pdva->offSurface);
SetSelLimit(ScreenSel,lpbi->biSizeImage-1);
pdva->offSurface = 0;
pdva->selSurface = ScreenSel;
}
//
// fill in defaults.
//
if (pdva->OpenSurface == NULL)
pdva->OpenSurface = default_open_surface;
if (pdva->CloseSurface == NULL)
pdva->CloseSurface = default_close_surface;
if (pdva->ShowSurface == NULL)
pdva->ShowSurface = default_show_surface;
if (pdva->BeginAccess == NULL)
{
pdva->BeginAccess = default_begin_access;
pdva->EndAccess = default_end_access;
}
//
// only test RGB surfaces.
//
if (lpbi->biCompression == 0 ||
lpbi->biCompression == BI_BITFIELDS ||
lpbi->biCompression == BI_1632)
{
if (!TestSurface(pdva))
return FALSE;
}
//
// set BI_1632 if needed
//
if (pdva->Flags & DVAF_1632_ACCESS)
{
lpbi->biCompression = BI_1632;
}
return TRUE;
}
/****************************************************************************
***************************************************************************/
#pragma optimize("", off)
static void SetSelLimit(UINT sel, DWORD limit)
{
if (limit >= 1024*1024l)
limit = ((limit+4096) & ~4095) - 1;
_asm
{
mov ax,0008h ; DPMI set limit
mov bx,sel
mov dx,word ptr limit[0]
mov cx,word ptr limit[2]
int 31h
}
}
#pragma optimize("", on)
/****************************************************************************
***************************************************************************/
#define ASM66 _asm _emit 0x66 _asm
#define DB _asm _emit
#pragma optimize("", off)
static BYTE ReadByte(PDVA pdva, LPVOID lpBits, DWORD dw)
{
BYTE b=42;
DVABeginAccess(pdva, 0, 0, 1024, 1024);
_asm {
ASM66 xor bx,bx
les bx,lpBits
ASM66 add bx,word ptr dw
mov ax,es
ASM66 lsl ax,ax
ASM66 cmp bx,ax
ja exit
DB 26h ;mov al,es:[ebx]
DB 67h
DB 8Ah
DB 03h
mov b,al
exit:
}
DVAEndAccess(pdva);
return b;
}
#pragma optimize("", on)
/////////////////////////////////////////////////////////////////////////////
//
// SetPixel
//
// some cards cant't seam to do SetPixel right it is amazing they work at all
//
/////////////////////////////////////////////////////////////////////////////
static void SetPixelX(HDC hdc, int x, int y, COLORREF rgb)
{
RECT rc;
rc.left = x;
rc.top = y;
rc.right = x+1;
rc.bottom = y+1;
SetBkColor(hdc, rgb);
ExtTextOut(hdc, 0, 0, ETO_OPAQUE, &rc, NULL, 0, NULL);
}
#define SetPixel SetPixelX
/****************************************************************************
***************************************************************************/
static BOOL TestSurface(DVASURFACEINFO FAR *pdva)
{
HDC hdc;
int x,y,h,w,wb;
COLORREF rgb,rgb0,rgb1,rgb2,rgb3,rgb4;
DWORD dw;
BYTE b0,b1;
UINT uType=0;
LPBITMAPINFOHEADER lpbi;
LPVOID lpBits;
HCURSOR hcur;
if (!pdva->OpenSurface(pdva->lpSurface))
return FALSE;
lpbi = DVAGetSurfaceFmt(pdva);
lpBits = DVAGetSurfacePtr(pdva);
h = abs((int)lpbi->biHeight);
w = (int)lpbi->biWidth;
wb = (w * ((UINT)lpbi->biBitCount/8) + 3) & ~3;
dw = (DWORD)(UINT)(h-1) * (DWORD)(UINT)wb;
if ((int)lpbi->biHeight < 0)
y = 0;
else
y = h-1;
#ifdef XDEBUG
x = (int)lpbi->biWidth - 5;
((LPBYTE)lpBits) += x * (UINT)lpbi->biBitCount/8;
#else
x = 0;
#endif
hcur = SetCursor(NULL);
hdc = GetDC(NULL);
rgb = GetPixel(hdc, x, h-1-y);
SetPixel(hdc, x, h-1-y, RGB(0,0,0)); GetPixel(hdc, x, h-1-y); b0 = ReadByte(pdva, lpBits, dw);
SetPixel(hdc, x, h-1-y, RGB(255,255,255)); GetPixel(hdc, x, h-1-y); b1 = ReadByte(pdva, lpBits, dw);
SetPixel(hdc, x, h-1-y,rgb);
if (b0 != 0x00 || b1 == 0x00)
goto done;
rgb0 = GetPixel(hdc, x+0, y);
rgb1 = GetPixel(hdc, x+1, y);
rgb2 = GetPixel(hdc, x+2, y);
rgb3 = GetPixel(hdc, x+3, y);
rgb4 = GetPixel(hdc, x+4, y);
TestSurfaceType(hdc, x, y);
DVABeginAccess(pdva, x, y, 5, 1);
uType = GetSurfaceType(lpBits);
DVAEndAccess(pdva);
SetPixel(hdc, x+0, y,rgb0);
SetPixel(hdc, x+1, y,rgb1);
SetPixel(hdc, x+2, y,rgb2);
SetPixel(hdc, x+3, y,rgb3);
SetPixel(hdc, x+4, y,rgb4);
done:
ReleaseDC(NULL, hdc);
SetCursor(hcur);
pdva->CloseSurface(pdva->lpSurface);
switch (uType)
{
case BM_8BIT:
break;
case BM_16555:
((LPDWORD)(lpbi+1))[0] = 0x007C00;
((LPDWORD)(lpbi+1))[1] = 0x0003E0;
((LPDWORD)(lpbi+1))[2] = 0x00001F;
break;
case BM_24BGR:
case BM_32BGR:
((LPDWORD)(lpbi+1))[0] = 0xFF0000;
((LPDWORD)(lpbi+1))[1] = 0x00FF00;
((LPDWORD)(lpbi+1))[2] = 0x0000FF;
break;
case BM_16565:
lpbi->biCompression = BI_BITFIELDS;
((LPDWORD)(lpbi+1))[0] = 0x00F800;
((LPDWORD)(lpbi+1))[1] = 0x0007E0;
((LPDWORD)(lpbi+1))[2] = 0x00001F;
break;
case BM_24RGB:
case BM_32RGB:
lpbi->biCompression = BI_BITFIELDS;
((LPDWORD)(lpbi+1))[0] = 0x0000FF;
((LPDWORD)(lpbi+1))[1] = 0x00FF00;
((LPDWORD)(lpbi+1))[2] = 0xFF0000;
break;
}
return uType != 0;
}
/****************************************************************************
map DCI escapes to old DVA stuff.
***************************************************************************/
UINT CALLBACK dci_begin_access(DCISURFACEINFO FAR *pdci, RECT FAR *prc)
{
DVASURFACEINFO FAR *pdva = (DVASURFACEINFO FAR *)pdci->dwReserved1;
if (pdva->BeginAccess(pdva->lpSurface,prc->left,prc->top,prc->right-prc->left,prc->bottom-prc->top))
return 0;
else
return (UINT)-1;
}
void CALLBACK dci_end_access(DCISURFACEINFO FAR *pdci)
{
DVASURFACEINFO FAR *pdva = (DVASURFACEINFO FAR *)pdci->dwReserved1;
pdva->EndAccess(pdva->lpSurface);
}
void CALLBACK dci_destroy(DCISURFACEINFO FAR *pdci)
{
DVASURFACEINFO FAR *pdva = (DVASURFACEINFO FAR *)pdci->dwReserved1;
pdva->CloseSurface(pdva->lpSurface);
}
LONG DVAEscape(HDC hdc, UINT function, UINT size, LPVOID lp_in_data, LPVOID lp_out_data)
{
DCICMD FAR *pcmd;
static DVASURFACEINFO dva;
static DCISURFACEINFO dci;
switch (function)
{
case QUERYESCSUPPORT:
switch (*((UINT FAR *)lp_in_data))
{
case QUERYESCSUPPORT:
return TRUE;
case DCICOMMAND:
return TRUE;
default:
return FALSE;
}
case DCICOMMAND:
pcmd = (DCICMD FAR *)lp_in_data;
switch (pcmd->dwCommand)
{
case DCICREATEPRIMARYSURFACE:
if (!DVAGetSurface(hdc, 0, &dva))
return DCI_FAIL_UNSUPPORTED;
dci.dwSize = sizeof(dci);
dci.dwDCICaps = DCI_PRIMARY | DCI_VISIBLE | DCI_1632_ACCESS;
dci.dwCompression = dva.BitmapInfo.biCompression;
dci.dwMask[0] = dva.dwMask[0];
dci.dwMask[1] = dva.dwMask[1];
dci.dwMask[2] = dva.dwMask[2];
dci.dwWidth = dva.BitmapInfo.biWidth;
dci.dwHeight = abs(dva.BitmapInfo.biHeight);
dci.dwBitCount = dva.BitmapInfo.biBitCount;
dci.lStride = dci.dwWidth * dci.dwBitCount / 8;
dci.dwOffSurface = dva.offSurface;
dci.wSelSurface = dva.selSurface;
dci.wReserved = 0;
dci.dwReserved1 = (DWORD)(LPVOID)&dva;
dci.dwReserved2 = 0;
dci.dwReserved3 = 0;
dci.BeginAccess = dci_begin_access;
dci.EndAccess = dci_end_access;
dci.DestroySurface = dci_destroy;
*((DCISURFACEINFO FAR * FAR *)lp_out_data) = &dci;
return DCI_OK;
default:
return DCI_FAIL_UNSUPPORTED;
}
}
return -1;
}