gdrv, render v1.

This commit is contained in:
oz 2020-11-08 18:37:59 +03:00
parent a925b6137f
commit 989ba6cdb2
25 changed files with 598 additions and 135 deletions

BIN
Doc/FuncStats.xlsx Normal file

Binary file not shown.

Binary file not shown.

View file

@ -7,6 +7,7 @@
#include "objlist_class.h" #include "objlist_class.h"
#include "partman.h" #include "partman.h"
#include "DatParser.h" #include "DatParser.h"
#include "gdrv.h"
#include "loader.h" #include "loader.h"
#include "pinball.h" #include "pinball.h"
#include "score.h" #include "score.h"
@ -18,8 +19,11 @@ int main()
std::cout << "Hello World!\n"; std::cout << "Hello World!\n";
pinball::hinst = GetModuleHandleA(nullptr); pinball::hinst = GetModuleHandleA(nullptr);
char cmdLine[1]; char cmdLine[1]{};
WinMain(pinball::hinst, 0, cmdLine, 10); //WinMain(pinball::hinst, 0, cmdLine, 10);
auto dib = gdrv::DibCreate(8, 1, 1);
gdrv::DibSetUsage(dib, 0, 1);
objlist_class d = objlist_class(2, 4); objlist_class d = objlist_class(2, 4);
for (int i = 0; i < 100; i++) for (int i = 0; i < 100; i++)
@ -35,17 +39,17 @@ int main()
auto datFile = partman::load_records(dataFileName); auto datFile = partman::load_records(dataFileName);
assert(datFile); assert(datFile);
assert(partman::field_size_nth(datFile, 0, String, 0) == 43); assert(partman::field_size_nth(datFile, 0, datFieldTypes::String, 0) == 43);
assert(partman::field_size_nth(datFile, 2, Palette, 0) == 1024); assert(partman::field_size_nth(datFile, 2, datFieldTypes::Palette, 0) == 1024);
assert(partman::field_size_nth(datFile, 101, FloatArray, 4) == 32); assert(partman::field_size_nth(datFile, 101, datFieldTypes::FloatArray, 4) == 32);
assert(strcmp(partman::field(datFile, 0, String), "3D-Pinball: Copyright 1994, Cinematronics") == 0); assert(strcmp(partman::field(datFile, 0, datFieldTypes::String), "3D-Pinball: Copyright 1994, Cinematronics") == 0);
assert(strcmp(partman::field(datFile, 540, GroupName), "table_objects") == 0); assert(strcmp(partman::field(datFile, 540, datFieldTypes::GroupName), "table_objects") == 0);
assert(partman::record_labeled(datFile, "background") == 2); assert(partman::record_labeled(datFile, "background") == 2);
assert(partman::record_labeled(datFile, "a_bump1") == 372); assert(partman::record_labeled(datFile, "a_bump1") == 372);
assert(memcmp(partman::field_labeled(datFile, "table_size", ShortArray), new short[2]{ 600, 416 }, 2 * 2) == 0); assert(memcmp(partman::field_labeled(datFile, "table_size", datFieldTypes::ShortArray), new short[2]{ 600, 416 }, 2 * 2) == 0);
//loader::error(25, 26); //loader::error(25, 26);
loader::loadfrom(datFile); loader::loadfrom(datFile);

View file

@ -157,6 +157,7 @@
<ClInclude Include="fullscrn.h" /> <ClInclude Include="fullscrn.h" />
<ClInclude Include="gdrv.h" /> <ClInclude Include="gdrv.h" />
<ClInclude Include="loader.h" /> <ClInclude Include="loader.h" />
<ClInclude Include="maths.h" />
<ClInclude Include="memory.h" /> <ClInclude Include="memory.h" />
<ClInclude Include="objlist_class.h" /> <ClInclude Include="objlist_class.h" />
<ClInclude Include="options.h" /> <ClInclude Include="options.h" />
@ -200,12 +201,14 @@
<ClInclude Include="TTripwire.h" /> <ClInclude Include="TTripwire.h" />
<ClInclude Include="TZmapList.h" /> <ClInclude Include="TZmapList.h" />
<ClInclude Include="winmain.h" /> <ClInclude Include="winmain.h" />
<ClInclude Include="zdrv.h" />
</ItemGroup> </ItemGroup>
<ItemGroup> <ItemGroup>
<ClCompile Include="DatParser.cpp" /> <ClCompile Include="DatParser.cpp" />
<ClCompile Include="fullscrn.cpp" /> <ClCompile Include="fullscrn.cpp" />
<ClCompile Include="gdrv.cpp" /> <ClCompile Include="gdrv.cpp" />
<ClCompile Include="loader.cpp" /> <ClCompile Include="loader.cpp" />
<ClCompile Include="maths.cpp" />
<ClCompile Include="memory.cpp" /> <ClCompile Include="memory.cpp" />
<ClCompile Include="objlist_class.cpp" /> <ClCompile Include="objlist_class.cpp" />
<ClCompile Include="options.cpp" /> <ClCompile Include="options.cpp" />
@ -253,6 +256,7 @@
<ClCompile Include="TTimer.cpp" /> <ClCompile Include="TTimer.cpp" />
<ClCompile Include="TTripwire.cpp" /> <ClCompile Include="TTripwire.cpp" />
<ClCompile Include="winmain.cpp" /> <ClCompile Include="winmain.cpp" />
<ClCompile Include="zdrv.cpp" />
</ItemGroup> </ItemGroup>
<ItemGroup> <ItemGroup>
<Natvis Include="NatvisFile.natvis" /> <Natvis Include="NatvisFile.natvis" />

View file

@ -162,6 +162,12 @@
<ClInclude Include="gdrv.h"> <ClInclude Include="gdrv.h">
<Filter>Header Files</Filter> <Filter>Header Files</Filter>
</ClInclude> </ClInclude>
<ClInclude Include="maths.h">
<Filter>Header Files</Filter>
</ClInclude>
<ClInclude Include="zdrv.h">
<Filter>Header Files</Filter>
</ClInclude>
</ItemGroup> </ItemGroup>
<ItemGroup> <ItemGroup>
<ClCompile Include="pch.cpp"> <ClCompile Include="pch.cpp">
@ -302,6 +308,12 @@
<ClCompile Include="gdrv.cpp"> <ClCompile Include="gdrv.cpp">
<Filter>Source Files</Filter> <Filter>Source Files</Filter>
</ClCompile> </ClCompile>
<ClCompile Include="maths.cpp">
<Filter>Source Files</Filter>
</ClCompile>
<ClCompile Include="zdrv.cpp">
<Filter>Source Files</Filter>
</ClCompile>
</ItemGroup> </ItemGroup>
<ItemGroup> <ItemGroup>
<Natvis Include="NatvisFile.natvis" /> <Natvis Include="NatvisFile.natvis" />

View file

@ -4,7 +4,7 @@ class TLight :
public TPinballComponent public TPinballComponent
{ {
public: public:
TLight(TPinballTable* table, int groupIndex) : TPinballComponent(table, groupIndex, false) TLight(TPinballTable* table, int groupIndex) : TPinballComponent(table, groupIndex, true)
{ {
} }
}; };

View file

@ -2,6 +2,7 @@
#include "TPinballComponent.h" #include "TPinballComponent.h"
#include "loader.h" #include "loader.h"
#include "objlist_class.h" #include "objlist_class.h"
#include "render.h"
#include "TZmapList.h" #include "TZmapList.h"
#include "TPinballTable.h" #include "TPinballTable.h"
@ -9,75 +10,69 @@ TPinballComponent::TPinballComponent(TPinballTable* table, int groupIndex, bool
{ {
visualStruct visual{}; // [esp+Ch] [ebp-6Ch] visualStruct visual{}; // [esp+Ch] [ebp-6Ch]
//this->VfTable = (int)&TPinballComponent::`vftable'; MessageField = 0;
this->MessageField = 0; UnknownBaseFlag1 = 0;
this->UnknownBaseFlag1 = 0; UnknownBaseFlag2 = 0;
this->UnknownBaseFlag2 = 0; PinballTable = table;
this->PinballTable = table; RenderSprite = nullptr;
this->Unknown7 = 0; ListBitmap = nullptr;
this->List1Bitmap8 = nullptr; ListZMap = nullptr;
this->List2Bitmap16 = nullptr;
if (table) if (table)
table->ListP1->Add(this); table->ListP1->Add(this);
if (groupIndex >= 0) if (groupIndex >= 0)
this->GroupName = loader::query_name(groupIndex); GroupName = loader::query_name(groupIndex);
if (loadVisuals && groupIndex >= 0) if (loadVisuals && groupIndex >= 0)
{ {
int visualCount = loader::query_visual_states(groupIndex); int visualCount = loader::query_visual_states(groupIndex);
for (int index = 0; index < visualCount; ++index) for (int index = 0; index < visualCount; ++index)
{ {
loader::query_visual(groupIndex, index, &visual); loader::query_visual(groupIndex, index, &visual);
if (visual.Bitmap8) if (visual.Bitmap)
{ {
if (!this->List1Bitmap8) if (!ListBitmap)
this->List1Bitmap8 = new TZmapList(visualCount, 4); ListBitmap = new TZmapList(visualCount, 4);
if (this->List1Bitmap8) if (ListBitmap)
this->List1Bitmap8->Add(visual.Bitmap8); ListBitmap->Add(visual.Bitmap);
} }
if (visual.Bitmap16) if (visual.ZMap)
{ {
if (!this->List2Bitmap16) if (!ListZMap)
this->List2Bitmap16 = new TZmapList(visualCount, 4); ListZMap = new TZmapList(visualCount, 4);
if (this->List2Bitmap16) if (ListZMap)
this->List2Bitmap16->Add(visual.Bitmap16); ListZMap->Add(visual.ZMap);
} }
} }
if (this->List2Bitmap16) zmap_header_type* zMap = nullptr;
int listVal0 = (int)this->List2Bitmap16->Get(0); if (ListZMap)
if (this->List1Bitmap8) zMap = static_cast<zmap_header_type*>(ListZMap->Get(0));
if (ListBitmap)
{ {
/*listVal0_2 = (int*)this->List1Bitmap8->Get(0); visual_rect bmp1Rect{}, tmpRect{};
v24 = *(int*)((char*)listVal0_2 + 29) - table->UnknownP49; auto rootBmp = static_cast<gdrv_bitmap8*>(ListBitmap->Get(0));
v15 = 1; bmp1Rect.XPosition = rootBmp->XPosition - table->XOffset;
v25 = *(int*)((char*)listVal0_2 + 33) - table->UnknownP50; bmp1Rect.YPosition = rootBmp->YPosition - table->YOffset;
v26 = listVal0_2[3]; bmp1Rect.Width = rootBmp->Width;
v27 = listVal0_2[4]; bmp1Rect.Height = rootBmp->Height;
if (List1Bitmap8->Count() > 1) for (int index = 1; index < ListBitmap->Count(); index++)
{ {
index = 12; auto bmp = static_cast<gdrv_bitmap8*>(ListBitmap->Get(index));
do tmpRect.XPosition = bmp->XPosition - table->XOffset;
{ tmpRect.YPosition = bmp->YPosition - table->YOffset;
v16 = *(int**)((char*)&this->List1Bitmap8->ListPtr->Size + index); tmpRect.Width = bmp->Width;
v20 = *(int*)((char*)v16 + 29) - table->UnknownP49; tmpRect.Height = bmp->Height;
v21 = *(int*)((char*)v16 + 33) - table->UnknownP50; maths::enclosing_box(&bmp1Rect, &tmpRect, &bmp1Rect);
v22 = v16[3];
v23 = v16[4];
enclosing_box(&v24, &v20, &v24);
index += 4;
++v15;
} while (v15 < this->List1Bitmap8->ListPtr->Count);
} }
v17 = this->List1Bitmap8->ListPtr->Array[0];
this->Unknown7 = (int)render_create_sprite( RenderSprite = render::create_sprite(
visualCount > 0, visualCount > 0 ? VisualType::Sprite :VisualType::None,
this->List1Bitmap8->ListPtr->Array[0], rootBmp,
listVal0, zMap,
*(int*)(v17 + 29) - table->UnknownP49, rootBmp->XPosition - table->XOffset,
*(int*)(v17 + 33) - table->UnknownP50, rootBmp->YPosition - table->YOffset,
&v24);*/ &bmp1Rect);
} }
} }
this->GroupIndex = groupIndex; GroupIndex = groupIndex;
} }
@ -87,25 +82,25 @@ TPinballComponent::~TPinballComponent()
if (table) if (table)
table->ListP1->Delete(this); table->ListP1->Delete(this);
delete List1Bitmap8; delete ListBitmap;
delete List2Bitmap16; delete ListZMap;
} }
int TPinballComponent::Message(int message1, float message2) int TPinballComponent::Message(int message1, float message2)
{ {
this->MessageField = message1; MessageField = message1;
if (message1 == 1024) if (message1 == 1024)
this->MessageField = 0; MessageField = 0;
return 0; return 0;
} }
void TPinballComponent::put_scoring( int score1, int score2) void TPinballComponent::put_scoring(int score1, int score2)
{ {
} }
int TPinballComponent::get_scoring(int score1) int TPinballComponent::get_scoring(int score1)
{ {
return 0; return 0;
} }

View file

@ -1,4 +1,5 @@
#pragma once #pragma once
#include "render.h"
#include "TZmapList.h" #include "TZmapList.h"
@ -19,8 +20,8 @@ public:
int Unknown4; int Unknown4;
int Unknown5; int Unknown5;
int GroupIndex; int GroupIndex;
int Unknown7; render_sprite_type_struct* RenderSprite;
TPinballTable* PinballTable; TPinballTable* PinballTable;
TZmapList* List1Bitmap8; TZmapList* ListBitmap;
TZmapList* List2Bitmap16; TZmapList* ListZMap;
}; };

View file

@ -63,8 +63,8 @@ public:
TPlunger* Plunger; TPlunger* Plunger;
TDrain* Drain; TDrain* Drain;
int UnknownP48; int UnknownP48;
int UnknownP49; int XOffset;
int UnknownP50; int YOffset;
int UnknownP51; int UnknownP51;
int UnknownP52; int UnknownP52;
objlist_class* ListP1; objlist_class* ListP1;

View file

@ -4,7 +4,7 @@ class TSound :
public TPinballComponent public TPinballComponent
{ {
public: public:
TSound(TPinballTable* table, int groupIndex) : TPinballComponent(table, groupIndex, false) TSound(TPinballTable* table, int groupIndex) : TPinballComponent(table, groupIndex, true)
{ {
} }
}; };

View file

@ -5,7 +5,7 @@ class TTextBox :
public TPinballComponent public TPinballComponent
{ {
public: public:
TTextBox(TPinballTable* table, int groupIndex) : TPinballComponent(table, groupIndex, false) TTextBox(TPinballTable* table, int groupIndex) : TPinballComponent(table, groupIndex, true)
{ {
} }

View file

@ -4,7 +4,7 @@ class TTimer :
public TPinballComponent public TPinballComponent
{ {
public: public:
TTimer(TPinballTable* table, int groupIndex) : TPinballComponent(table, groupIndex, false) TTimer(TPinballTable* table, int groupIndex) : TPinballComponent(table, groupIndex, true)
{ {
} }
}; };

View file

@ -1,6 +1,160 @@
#include "pch.h" #include "pch.h"
#include "gdrv.h" #include "gdrv.h"
#include "memory.h"
HPALETTE gdrv::palette_handle=0;
void gdrv::get_focus() void gdrv::get_focus()
{ {
} }
gdrv_dib* gdrv::DibCreate(__int16 bpp, int width, int height)
{
auto sizeBytes = height * ((width * bpp / 8 + 3) & 0xFFFFFFFC);
auto buf = GlobalAlloc(0x42u, sizeBytes + 1064);
auto dib = static_cast<gdrv_dib*>(GlobalLock(buf));
if (!dib)
return nullptr;
dib->BufferSize = sizeBytes;
dib->Width = width;
dib->PaletteOffset = 40;
dib->Height = height;
dib->Unknown3_1 = 1;
dib->Bpp = bpp;
dib->Unknown4 = 0;
dib->Unknown6 = 0;
dib->Unknown7 = 0;
dib->NumberOfColors = 0;
dib->Unknown9 = 0;
if (bpp == 4)
{
dib->NumberOfColors = 16;
}
else if (bpp == 8)
{
dib->NumberOfColors = 256;
}
auto pltPtr = &dib->Palette0;
for (auto index = 0; index < dib->NumberOfColors / 16; ++index, pltPtr++)
{
*pltPtr = gdrv_dib_palette{
{0},
{0x800000},
{0x8000},
{8421376},
{128},
{8388736},
{32896},
{12632256},
{8421504},
{16711680},
{65280},
{16776960},
{255},
{16711935},
{0xFFFF},
{0xFFFFFF},
};
}
return dib;
}
void gdrv::DibSetUsage(gdrv_dib* dib, HPALETTE hpal, int someFlag)
{
tagPALETTEENTRY pPalEntries[256]; // [esp+4h] [ebp-400h]
if (!hpal)
hpal = static_cast<HPALETTE>(GetStockObject(DEFAULT_PALETTE));
if (!dib)
return;
int numOfColors = dib->NumberOfColors;
if (!numOfColors)
{
auto bpp = dib->Bpp;
if (bpp <= 8u)
numOfColors = 1 << bpp;
}
if (numOfColors > 0 && (dib->Unknown4 != 3 || numOfColors == 3))
{
if (someFlag && someFlag <= 2)
{
auto pltPtr = (short*)((char*)dib + dib->PaletteOffset);
for (int i = 0; i < numOfColors; ++i)
{
*pltPtr++ = i;
}
}
else
{
assertm(false, "Entered bad code");
char* dibPtr = (char*)dib + dib->PaletteOffset;
if (numOfColors >= 256)
numOfColors = 256;
GetPaletteEntries(hpal, 0, numOfColors, pPalEntries);
int index = 0;
char* dibPtr2 = dibPtr + 1;
do
{
char v9 = pPalEntries[index++].peRed;
dibPtr2[1] = v9;
*dibPtr2 = dibPtr2[(char*)pPalEntries - dibPtr];
*(dibPtr2 - 1) = dibPtr2[&pPalEntries[0].peGreen - (unsigned char*)dibPtr];
dibPtr2[2] = 0;
dibPtr2 += 4;
}
while (index < numOfColors);
}
}
}
int gdrv::create_bitmap_dib(gdrv_bitmap8* bmp, int width, int height)
{
char* bmpBufPtr; // ecx
gdrv_dib* dib = DibCreate(8, width, height);
DibSetUsage(dib, palette_handle, 1);
bmp->Dib = dib;
bmp->Width = width;
bmp->Stride = width;
if (width % 4)
bmp->Stride = 4 - width % 4 + width;
gdrv_dib* dib2 = bmp->Dib;
bmp->Height = height;
bmp->SomeByte = 2;
if (dib2->Unknown4 == 3)
bmpBufPtr = (char*)&dib2->Unknown3_1 + dib2->PaletteOffset;
else
bmpBufPtr = (char*)&dib2->PaletteOffset + 4 * dib2->NumberOfColors + dib2->PaletteOffset;
bmp->BmpBufPtr1 = bmpBufPtr;
bmp->BmpBufPtr2 = bmpBufPtr;
return 0;
}
int gdrv::create_bitmap(gdrv_bitmap8* bmp, int width, int height)
{
return create_bitmap_dib(bmp, width, height);
}
int gdrv::create_raw_bitmap(gdrv_bitmap8* bmp, int width, int height, int flag)
{
bmp->Dib = nullptr;
bmp->Width = width;
bmp->Stride = width;
if (flag && width % 4)
bmp->Stride = width - width % 4 + 4;
unsigned int sizeInBytes = height * bmp->Stride;
bmp->Height = height;
bmp->SomeByte = 1;
char* buf = memory::allocate(sizeInBytes);
bmp->BmpBufPtr1 = buf;
if (!buf)
return -1;
bmp->BmpBufPtr2 = buf;
return 0;
}

View file

@ -1,7 +1,93 @@
#pragma once #pragma once
union tagPALETTEENTRY2
{
unsigned __int32 PltInt;
tagPALETTEENTRY Plt;
};
struct gdrv_dib_palette
{
tagPALETTEENTRY2 Color0;
tagPALETTEENTRY2 Color1;
tagPALETTEENTRY2 Color2;
tagPALETTEENTRY2 Color3;
tagPALETTEENTRY2 Color4;
tagPALETTEENTRY2 Color5;
tagPALETTEENTRY2 Color6;
tagPALETTEENTRY2 Color7;
tagPALETTEENTRY2 Color8;
tagPALETTEENTRY2 Color9;
tagPALETTEENTRY2 Color10;
tagPALETTEENTRY2 Color11;
tagPALETTEENTRY2 Color12;
tagPALETTEENTRY2 Color13;
tagPALETTEENTRY2 Color14;
tagPALETTEENTRY2 Color15;
};
struct __declspec(align(4)) gdrv_dib
{
int PaletteOffset;
int Width;
int Height;
__int16 Unknown3_1;
unsigned __int16 Bpp;
int Unknown4;
int BufferSize;
int Unknown6;
int Unknown7;
int NumberOfColors;
int Unknown9;
gdrv_dib_palette Palette0;
gdrv_dib_palette Palette1;
gdrv_dib_palette Palette2;
gdrv_dib_palette Palette3;
gdrv_dib_palette Palette4;
gdrv_dib_palette Palette5;
gdrv_dib_palette Palette6;
gdrv_dib_palette Palette7;
gdrv_dib_palette Palette8;
gdrv_dib_palette Palette9;
gdrv_dib_palette Palette10;
gdrv_dib_palette Palette11;
gdrv_dib_palette Palette12;
gdrv_dib_palette Palette13;
gdrv_dib_palette Palette14;
gdrv_dib_palette Palette15;
char BmpBuffer[1];
};
#pragma pack(push, 1)
struct __declspec(align(1)) gdrv_bitmap8
{
gdrv_dib* Dib;
char* BmpBufPtr2;
char* BmpBufPtr1;
int Width;
int Height;
int Stride;
char SomeByte;
int Color6;
int XPosition;
int YPosition;
};
#pragma pack(pop)
static_assert(sizeof(tagPALETTEENTRY2) == 4, "Wrong size of tagPALETTEENTRY2");
static_assert(sizeof(gdrv_dib_palette) == 4 * 16, "Wrong size of gdrv_dib_palette");
static_assert(sizeof(gdrv_dib) == (10 * 4) + sizeof(gdrv_dib_palette) * 16 + 4, "Wrong size of gdrv_dib");
static_assert(sizeof(gdrv_bitmap8) == 37, "Wrong size of gdrv_bitmap8");
class gdrv class gdrv
{ {
public: public:
static void get_focus(); static HPALETTE palette_handle;
static void get_focus();
static gdrv_dib* DibCreate(__int16 bpp, int width, int height);
static void DibSetUsage(gdrv_dib* dib, HPALETTE hpal, int someFlag);
static int create_bitmap_dib(gdrv_bitmap8* bmp, int width, int height);
static int create_bitmap(gdrv_bitmap8* bmp, int width, int height);
static int create_raw_bitmap(gdrv_bitmap8* bmp, int width, int height, int flag);
private:
}; };

View file

@ -1,10 +1,9 @@
#include "pch.h" #include "pch.h"
#include "loader.h" #include "loader.h"
#include "memory.h" #include "memory.h"
#include "partman.h" #include "partman.h"
#include "pinball.h" #include "pinball.h"
#include "zdrv.h"
/*_loader_errors dd 0, offset aBadHandle, 1, offset aNoTypeField, 2, offset aNoAttributesFi /*_loader_errors dd 0, offset aBadHandle, 1, offset aNoTypeField, 2, offset aNoAttributesFi
@ -91,8 +90,8 @@ void loader::default_vsi(visualStruct* visual)
visual->Unknown2F = 0.60000002f; visual->Unknown2F = 0.60000002f;
visual->FloatArrSizeDiv8Sub2 = 0; visual->FloatArrSizeDiv8Sub2 = 0;
visual->SoundIndex2 = 0; visual->SoundIndex2 = 0;
visual->Bitmap8 = 0; visual->Bitmap = 0;
visual->Bitmap16 = 0; visual->ZMap = 0;
visual->SoundIndex3 = 0; visual->SoundIndex3 = 0;
visual->SoundIndex4 = 0; visual->SoundIndex4 = 0;
} }
@ -107,7 +106,7 @@ void loader::loadfrom(datFileStruct* datFile)
{ {
do do
{ {
__int16* value = (__int16*)partman::field(datFile, groupIndex, ShortValue); __int16* value = (__int16*)partman::field(datFile, groupIndex, datFieldTypes::ShortValue);
if (value && *value == 202) if (value && *value == 202)
{ {
soundIndex = sound_count; soundIndex = sound_count;
@ -170,10 +169,10 @@ int loader::get_sound_id(int groupIndex)
sound_list[soundIndex].Volume = 0.0; sound_list[soundIndex].Volume = 0.0;
if (soundGroupId > 0 && !pinball::quickFlag) if (soundGroupId > 0 && !pinball::quickFlag)
{ {
__int16* value = (__int16*)partman::field(loader_table, soundGroupId, ShortValue); __int16* value = (__int16*)partman::field(loader_table, soundGroupId, datFieldTypes::ShortValue);
if (value && *value == 202) if (value && *value == 202)
{ {
const CHAR* fileName = partman::field(loader_table, soundGroupId, String); const CHAR* fileName = partman::field(loader_table, soundGroupId, datFieldTypes::String);
HFILE hFile = _lopen(fileName, 0); HFILE hFile = _lopen(fileName, 0);
sound_list[soundIndex].Volume = (float)((double)(_llseek(hFile, 0, 2)) * 0.0000909090909090909); sound_list[soundIndex].Volume = (float)((double)(_llseek(hFile, 0, 2)) * 0.0000909090909090909);
_lclose(hFile); _lclose(hFile);
@ -198,7 +197,7 @@ short loader::query_visual_states(int groupIndex)
short result; short result;
if (groupIndex < 0) if (groupIndex < 0)
return error(0, 17); return error(0, 17);
__int16* shortArr = (__int16*)partman::field(loader_table, groupIndex, ShortArray); __int16* shortArr = (__int16*)partman::field(loader_table, groupIndex, datFieldTypes::ShortArray);
if (shortArr && *shortArr == 100) if (shortArr && *shortArr == 100)
result = shortArr[1]; result = shortArr[1];
else else
@ -209,7 +208,7 @@ short loader::query_visual_states(int groupIndex)
char* loader::query_name(int groupIndex) char* loader::query_name(int groupIndex)
{ {
if (groupIndex >= 0) if (groupIndex >= 0)
return partman::field(loader_table, groupIndex, GroupName); return partman::field(loader_table, groupIndex, datFieldTypes::GroupName);
error(0, 19); error(0, 19);
return nullptr; return nullptr;
} }
@ -222,12 +221,12 @@ __int16* loader::query_iattribute(int groupIndex, int firstValue, int* arraySize
{ {
while (true) while (true)
{ {
__int16* shortArr = (__int16*)partman::field_nth(loader_table, groupIndex, ShortArray, skipIndex); __int16* shortArr = (__int16*)partman::field_nth(loader_table, groupIndex, datFieldTypes::ShortArray, skipIndex);
if (!shortArr) if (!shortArr)
break; break;
if (*shortArr == firstValue) if (*shortArr == firstValue)
{ {
*arraySize = partman::field_size(loader_table, groupIndex, ShortArray) / 2 - 1; *arraySize = partman::field_size(loader_table, groupIndex, datFieldTypes::ShortArray) / 2 - 1;
return shortArr + 1; return shortArr + 1;
} }
++skipIndex; ++skipIndex;
@ -255,7 +254,7 @@ float* loader::query_float_attribute(int groupIndex, int groupIndexOffset, int f
{ {
while (true) while (true)
{ {
float* floatArr = (float*)partman::field_nth(loader_table, groupIndexSum, FloatArray, skipIndex); float* floatArr = (float*)partman::field_nth(loader_table, groupIndexSum, datFieldTypes::FloatArray, skipIndex);
if (!floatArr) if (!floatArr)
break; break;
if (static_cast<__int16>(static_cast<__int64>(floor(*floatArr))) == firstValue) if (static_cast<__int16>(static_cast<__int64>(floor(*floatArr))) == firstValue)
@ -283,16 +282,16 @@ int loader::material(int groupIndex, visualStruct* visual)
{ {
if (groupIndex < 0) if (groupIndex < 0)
return error(0, 21); return error(0, 21);
__int16* shortArr = (__int16*)partman::field(loader_table, groupIndex, ShortValue); __int16* shortArr = (__int16*)partman::field(loader_table, groupIndex, datFieldTypes::ShortValue);
if (!shortArr) if (!shortArr)
return error(1, 21); return error(1, 21);
if (*shortArr != 300) if (*shortArr != 300)
return error(3, 21); return error(3, 21);
float* floatArr = (float*)partman::field(loader_table, groupIndex, FloatArray); float* floatArr = (float*)partman::field(loader_table, groupIndex, datFieldTypes::FloatArray);
if (!floatArr) if (!floatArr)
return error(11, 21); return error(11, 21);
int index = 0; int index = 0;
int floatArrLength = partman::field_size(loader_table, groupIndex, FloatArray) >> 2; int floatArrLength = partman::field_size(loader_table, groupIndex, datFieldTypes::FloatArray) >> 2;
if (floatArrLength > 0) if (floatArrLength > 0)
{ {
do do
@ -336,7 +335,7 @@ int loader::state_id(int groupIndex, int groupIndexOffset)
__int16 visualState = query_visual_states(groupIndex); __int16 visualState = query_visual_states(groupIndex);
if (visualState <= 0) if (visualState <= 0)
return error(12, 24); return error(12, 24);
__int16* shortArr = (__int16*)partman::field(loader_table, groupIndex, ShortValue); __int16* shortArr = (__int16*)partman::field(loader_table, groupIndex, datFieldTypes::ShortValue);
if (!shortArr) if (!shortArr)
return error(1, 24); return error(1, 24);
if (*shortArr != 200) if (*shortArr != 200)
@ -347,7 +346,7 @@ int loader::state_id(int groupIndex, int groupIndexOffset)
return groupIndex2; return groupIndex2;
groupIndex2 = groupIndexOffset + groupIndex; groupIndex2 = groupIndexOffset + groupIndex;
shortArr = (__int16*)partman::field(loader_table, groupIndexOffset + groupIndex, ShortValue); shortArr = (__int16*)partman::field(loader_table, groupIndexOffset + groupIndex, datFieldTypes::ShortValue);
if (!shortArr) if (!shortArr)
return error(1, 24); return error(1, 24);
if (*shortArr != 201) if (*shortArr != 201)
@ -361,15 +360,15 @@ int loader::kicker(int groupIndex, visualKickerStruct* kicker)
{ {
if (groupIndex < 0) if (groupIndex < 0)
return error(0, 20); return error(0, 20);
__int16* shortArr = (__int16*)partman::field(loader_table, groupIndex, ShortValue); __int16* shortArr = (__int16*)partman::field(loader_table, groupIndex, datFieldTypes::ShortValue);
if (!shortArr) if (!shortArr)
return error(1, 20); return error(1, 20);
if (*shortArr != 400) if (*shortArr != 400)
return error(4, 20); return error(4, 20);
float* floatArr = (float*)partman::field(loader_table, groupIndex, FloatArray); float* floatArr = (float*)partman::field(loader_table, groupIndex, datFieldTypes::FloatArray);
if (!floatArr) if (!floatArr)
return error(11, 20); return error(11, 20);
int floatArrLength = partman::field_size(loader_table, groupIndex, FloatArray) >> 2; int floatArrLength = partman::field_size(loader_table, groupIndex, datFieldTypes::FloatArray) >> 2;
int index = 0; int index = 0;
if (floatArrLength <= 0) if (floatArrLength <= 0)
return 0; return 0;
@ -418,7 +417,7 @@ int loader::query_visual(int groupIndex, int groupIndexOffset, visualStruct* vi
visualStruct* visual2; // edi visualStruct* visual2; // edi
int groupIndexSum; // eax int groupIndexSum; // eax
int groupIndexSum2; // ebx int groupIndexSum2; // ebx
char* bitmap16; // eax zmap_header_type* bitmap16; // eax
__int16* shortArr; // esi __int16* shortArr; // esi
unsigned int shortArrSize; // eax unsigned int shortArrSize; // eax
int index; // ebx int index; // ebx
@ -447,18 +446,18 @@ int loader::query_visual(int groupIndex, int groupIndexOffset, visualStruct* vi
groupIndexSum3 = groupIndexSum; groupIndexSum3 = groupIndexSum;
if (groupIndexSum < 0) if (groupIndexSum < 0)
return error(16, 18); return error(16, 18);
visual->Bitmap8 = partman::field(loader_table, groupIndexSum, Bitmap8bit); visual->Bitmap = (gdrv_bitmap8*)partman::field(loader_table, groupIndexSum, datFieldTypes::Bitmap8bit);
bitmap16 = partman::field(loader_table, groupIndexSum2, Bitmap16bit); bitmap16 = (zmap_header_type*)partman::field(loader_table, groupIndexSum2, datFieldTypes::Bitmap16bit);
visual->Bitmap16 = bitmap16; visual->ZMap = bitmap16;
if (bitmap16) if (bitmap16)
{ {
//*(int*)(bitmap16 + 6) = bitmap16 + 14; bitmap16->BmpBufPtr1 = bitmap16->BmpBuffer;
//*(int*)(visual->Bitmap16 + 10) = *(int*)(visual->Bitmap16 + 6); visual->ZMap->bmpBufPtr2 = visual->ZMap->BmpBufPtr1;
} }
shortArr = (__int16*)partman::field(loader_table, groupIndexSum2, ShortArray); shortArr = (__int16*)partman::field(loader_table, groupIndexSum2, datFieldTypes::ShortArray);
if (shortArr) if (shortArr)
{ {
shortArrSize = partman::field_size(loader_table, groupIndexSum2, ShortArray); shortArrSize = partman::field_size(loader_table, groupIndexSum2, datFieldTypes::ShortArray);
index = 0; index = 0;
shortArrLength = shortArrSize >> 1; shortArrLength = shortArrSize >> 1;
if ((__int16)(shortArrSize >> 1) > 0) if ((__int16)(shortArrSize >> 1) > 0)
@ -540,13 +539,13 @@ int loader::query_visual(int groupIndex, int groupIndexOffset, visualStruct* vi
LABEL_33: LABEL_33:
if (!visual2->Unknown14Flag) if (!visual2->Unknown14Flag)
visual2->Unknown14Flag = 1; visual2->Unknown14Flag = 1;
floatArr = (float*)partman::field(loader_table, groupIndexSum3, FloatArray); floatArr = (float*)partman::field(loader_table, groupIndexSum3, datFieldTypes::FloatArray);
if (!floatArr) if (!floatArr)
return 0; return 0;
nextFloatVal = floatArr + 1; nextFloatVal = floatArr + 1;
if (*floatArr != 600.0) if (*floatArr != 600.0)
return 0; return 0;
visual2->FloatArrSizeDiv8Sub2 = (partman::field_size(loader_table, groupIndexSum3, FloatArray) >> 2)/ 2- 2; visual2->FloatArrSizeDiv8Sub2 = (partman::field_size(loader_table, groupIndexSum3, datFieldTypes::FloatArray) >> 2)/ 2- 2;
floatVal = (__int64)(floor(*nextFloatVal) - 1.0); floatVal = (__int64)(floor(*nextFloatVal) - 1.0);
floatArrPtr = nextFloatVal + 1; floatArrPtr = nextFloatVal + 1;
if ((int)floatVal) if ((int)floatVal)

View file

@ -1,4 +1,6 @@
#pragma once #pragma once
#include "gdrv.h"
#include "zdrv.h"
struct datFileStruct; struct datFileStruct;
@ -42,8 +44,8 @@ struct __declspec(align(4)) visualStruct
int Unknown14Flag; int Unknown14Flag;
int SoundIndex4; int SoundIndex4;
int SoundIndex3; int SoundIndex3;
char* Bitmap8; gdrv_bitmap8* Bitmap;
char* Bitmap16; zmap_header_type* ZMap;
}; };

View file

@ -0,0 +1,36 @@
#include "pch.h"
#include "maths.h"
void maths::enclosing_box(visual_rect* rect1, visual_rect* rect2, visual_rect* dstRect)
{
int xPos1 = rect1->XPosition;
int yPos1 = rect1->YPosition;
int width1 = rect1->Width;
int height1 = rect1->Height;
int xPos2 = rect2->XPosition;
bool rect2XPosLessRect1 = rect2->XPosition < rect1->XPosition;
int yPos2 = rect2->YPosition;
int width2 = rect2->Width;
int height2 = rect2->Height;
int xPos2_2 = rect2->XPosition;
if (rect2XPosLessRect1)
{
width1 += xPos1 - xPos2;
xPos1 = xPos2;
}
if (yPos2 < yPos1)
{
height1 += yPos1 - yPos2;
yPos1 = yPos2;
}
if (width2 + xPos2 > xPos1 + width1)
width1 = xPos2_2 + width2 - xPos1;
int height1_2 = height1;
if (height2 + yPos2 > height1 + yPos1)
height1_2 = yPos2 + height2 - yPos1;
dstRect->YPosition = yPos1;
dstRect->Height = height1_2;
dstRect->XPosition = xPos1;
dstRect->Width = width1;
}

18
SpaceCadetPinball/maths.h Normal file
View file

@ -0,0 +1,18 @@
#pragma once
struct __declspec(align(4)) visual_rect
{
int XPosition;
int YPosition;
int Width;
int Height;
};
class maths
{
public:
static void enclosing_box(visual_rect* rect1, visual_rect* rect2, visual_rect* dstRect);
};

View file

@ -1,6 +1,6 @@
#include "pch.h" #include "pch.h"
#include "partman.h" #include "partman.h"
#include "gdrv.h"
#include "memory.h" #include "memory.h"
short partman::_field_size[] = { short partman::_field_size[] = {
@ -39,7 +39,7 @@ datFileStruct* partman::load_records(LPCSTR lpFileName)
else else
{ {
int lenOfStr = lstrlenA(Buffer.Description); int lenOfStr = lstrlenA(Buffer.Description);
auto descriptionBuf = (char*)memory::allocate(lenOfStr + 1); auto descriptionBuf = static_cast<char*>(memory::allocate(lenOfStr + 1));
datFile->Description = descriptionBuf; datFile->Description = descriptionBuf;
if (!descriptionBuf) if (!descriptionBuf)
{ {
@ -52,7 +52,7 @@ datFileStruct* partman::load_records(LPCSTR lpFileName)
if (Buffer.Unknown) if (Buffer.Unknown)
{ {
auto unknownBuf = (char*)memory::allocate(Buffer.Unknown); auto unknownBuf = static_cast<char*>(memory::allocate(Buffer.Unknown));
if (!unknownBuf) if (!unknownBuf)
{ {
_lclose(fileHandle); _lclose(fileHandle);
@ -99,30 +99,30 @@ datFileStruct* partman::load_records(LPCSTR lpFileName)
{ {
auto entryType = static_cast<datFieldTypes>(_lread_char(fileHandle)); auto entryType = static_cast<datFieldTypes>(_lread_char(fileHandle));
entryData->EntryType = entryType; entryData->EntryType = entryType;
int fieldSize = _field_size[entryType]; int fieldSize = _field_size[(int)entryType];
if (fieldSize < 0) if (fieldSize < 0)
{ {
fieldSize = _lread_long(fileHandle); fieldSize = _lread_long(fileHandle);
} }
if (entryType == Bitmap8bit) if (entryType == datFieldTypes::Bitmap8bit)
{ {
_hread(fileHandle, &bmpHeader, 14); _hread(fileHandle, &bmpHeader, 14);
char* bmpBuffer = (char*)memory::allocate(0x25u); auto bmp = (gdrv_bitmap8*)memory::allocate(sizeof(gdrv_bitmap8));
entryData->Buffer = bmpBuffer; entryData->Buffer = (char*)bmp;
if (!bmpBuffer) if (!bmp)
goto LABEL_41; goto LABEL_41;
/*if (bmpHeader.Unknown2 & 2 ? gdrv_create_bitmap((int)bmpBuffer, bmpHeader.Width, bmpHeader.Height) : gdrv_create_raw_bitmap((int)bmpBuffer, bmpHeader.Width, bmpHeader.Height, bmpHeader.Unknown2 & 1)) if (bmpHeader.Unknown2 & 2
goto LABEL_41;*/ ? gdrv::create_bitmap(bmp, bmpHeader.Width, bmpHeader.Height)
//_hread(fileHandle, *(LPVOID*)(entryData->Buffer + 8), bmpHeader.Size); : gdrv::create_raw_bitmap(bmp, bmpHeader.Width, bmpHeader.Height,
char* tempBuff = (char*)memory::allocate(bmpHeader.Size); bmpHeader.Unknown2 & 1))
_hread(fileHandle, tempBuff, bmpHeader.Size); goto LABEL_41;
memory::free(tempBuff); _hread(fileHandle, bmp->BmpBufPtr1, bmpHeader.Size);
//*((int*)entryData->Buffer + 29) = bmpHeader.XPosition; bmp->XPosition = bmpHeader.XPosition;
//*((int*)entryData->Buffer + 33) = bmpHeader.YPosition; bmp->YPosition = bmpHeader.YPosition;
} }
else else
{ {
char* entryBuffer = (char*)memory::allocate(fieldSize); char* entryBuffer = static_cast<char*>(memory::allocate(fieldSize));
entryData->Buffer = entryBuffer; entryData->Buffer = entryBuffer;
if (!entryBuffer) if (!entryBuffer)
goto LABEL_41; goto LABEL_41;
@ -193,7 +193,7 @@ char* partman::field(datFileStruct* datFile, int groupIndex, datFieldTypes targe
datEntryData* entry = groupData->Entries; datEntryData* entry = groupData->Entries;
while (true) while (true)
{ {
int entryType = entry->EntryType; auto entryType = entry->EntryType;
if (entryType == targetEntryType) if (entryType == targetEntryType)
break; break;
if (entryType > targetEntryType) if (entryType > targetEntryType)
@ -217,7 +217,7 @@ char* partman::field_nth(datFileStruct* datFile, int groupIndex, datFieldTypes t
datEntryData* entry = groupData->Entries; datEntryData* entry = groupData->Entries;
do do
{ {
int entryType = entry->EntryType; auto entryType = entry->EntryType;
if (entryType == targetEntryType) if (entryType == targetEntryType)
{ {
if (skipCount == skipFirstN) if (skipCount == skipFirstN)
@ -250,7 +250,7 @@ int partman::field_size_nth(datFileStruct* datFile, int groupIndex, datFieldType
datEntryData* entry = groupData->Entries; datEntryData* entry = groupData->Entries;
do do
{ {
int entryType = entry->EntryType; auto entryType = entry->EntryType;
if (entryType == targetEntryType) if (entryType == targetEntryType)
{ {
if (skipCount == skipFirstN) if (skipCount == skipFirstN)
@ -287,7 +287,7 @@ int partman::record_labeled(datFileStruct* datFile, LPCSTR targetGroupName)
{ {
if (--groupIndex < 0) if (--groupIndex < 0)
return -1; return -1;
char* groupName = field(datFile, groupIndex, GroupName); char* groupName = field(datFile, groupIndex, datFieldTypes::GroupName);
if (groupName) if (groupName)
{ {
int index = 0; int index = 0;

View file

@ -1,6 +1,6 @@
#pragma once #pragma once
enum datFieldTypes : __int16 enum class datFieldTypes : __int16
{ {
ShortValue = 0, ShortValue = 0,
//, does not have the 32bits size value, but a 16bits value(see above). //, does not have the 32bits size value, but a 16bits value(see above).

View file

@ -1,7 +1,15 @@
#include "pch.h" #include "pch.h"
#include "render.h" #include "render.h"
#include "memory.h"
int render::blit = 0; int render::blit = 0;
int render::many_dirty, render::many_sprites, render::many_balls;
render_sprite_type_struct **render::dirty_list = new render_sprite_type_struct*[1000], **render::sprite_list = new
render_sprite_type_struct* [1000], **render::ball_list = new render_sprite_type_struct* [
1000];
zmap_header_type* render::background_zmap;
int render::zmap_offset, render::zmap_offsetY;
void render::update() void render::update()
{ {
@ -13,4 +21,78 @@ void render::paint()
/*render_paint_balls(); /*render_paint_balls();
gdrv_blat((int)&vscreen, xDest, yDest); gdrv_blat((int)&vscreen, xDest, yDest);
render_unpaint_balls();*/ render_unpaint_balls();*/
} }
int render::sprite_modified(render_sprite_type_struct* sprite)
{
int result = 0; // eax
if (sprite->VisualType == VisualType::Ball)
return result;
result = many_dirty;
if (many_dirty < 999)
dirty_list[many_dirty++] = sprite;
return result;
}
render_sprite_type_struct* render::create_sprite(VisualType visualType, gdrv_bitmap8* rootBmp8, zmap_header_type* zMap,
int xPosition, int yPosition, visual_rect* rect)
{
render_sprite_type_struct* sprite = (render_sprite_type_struct*)memory::allocate(0x5Cu);
render_sprite_type_struct* result = nullptr;
if (!sprite)
return result;
sprite->YPosition = yPosition;
sprite->RootBmp8 = rootBmp8;
sprite->XPosition = xPosition;
sprite->VisualType = visualType;
sprite->Unknown6_0 = 0;
sprite->Unknown17 = 0;
sprite->Unknown18 = 0;
if (rect)
{
sprite->Rect = *rect;
}
else
{
sprite->Rect.Width = -1;
sprite->Rect.Height = -1;
sprite->Rect.XPosition = 0;
sprite->Rect.YPosition = 0;
}
if (rootBmp8)
{
sprite->Bmp8Width = rootBmp8->Width;
sprite->Bmp8Height = rootBmp8->Height;
}
else
{
sprite->Bmp8Width = 0;
sprite->Bmp8Height = 0;
}
sprite->ZMap = zMap;
sprite->ZMapOffestX = 0;
sprite->ZMapOffestY = 0;
if (!zMap && visualType != VisualType::Ball)
{
sprite->ZMap = background_zmap;
sprite->ZMapOffestY = xPosition - zmap_offset;
sprite->ZMapOffestX = yPosition - zmap_offsetY;
}
sprite->XPosition2 = sprite->XPosition;
sprite->YPosition2 = sprite->YPosition;
sprite->Bmp8Width2 = sprite->Bmp8Width;
sprite->Bmp8Height2 = sprite->Bmp8Height;
if (visualType == VisualType::Ball)
{
ball_list[many_balls++] = sprite;
}
else
{
sprite_list[many_sprites++] = sprite;
sprite_modified(sprite);
}
result = sprite;
return result;
}

View file

@ -1,9 +1,55 @@
#pragma once #pragma once
#include "gdrv.h"
#include "maths.h"
#include "zdrv.h"
enum class VisualType : char
{
None = 0,
Sprite = 1,
Ball = 2
};
struct __declspec(align(4)) render_sprite_type_struct
{
int XPosition;
int YPosition;
int Bmp8Width;
int Bmp8Height;
gdrv_bitmap8* RootBmp8;
zmap_header_type* ZMap;
char Unknown6_0;
VisualType VisualType;
char Unknown6_2;
char Unknown6_3;
int XPosition2;
int YPosition2;
int Bmp8Width2;
int Bmp8Height2;
int ZMapOffestY;
int ZMapOffestX;
int Unknown13;
int Unknown14;
int Unknown15;
int Unknown16;
int Unknown17;
int Unknown18;
visual_rect Rect;
};
class render class render
{ {
public: public:
static int blit; static int blit;
static int many_dirty, many_sprites, many_balls;
static render_sprite_type_struct **dirty_list, **sprite_list, **ball_list;
static zmap_header_type* background_zmap;
static int zmap_offset, zmap_offsetY;
static void update(); static void update();
static void paint(); static void paint();
static int sprite_modified(render_sprite_type_struct* sprite);
static render_sprite_type_struct* create_sprite(VisualType visualType, gdrv_bitmap8* rootBmp8,
zmap_header_type* zMap,
int xPosition, int yPosition, visual_rect* rect);
}; };

View file

@ -17,7 +17,7 @@ scoreStruct* score::create(LPCSTR fieldName, int renderBgBmp)
return nullptr; return nullptr;
score->Unknown1 = -9999; score->Unknown1 = -9999;
score->RenderBgBmp = renderBgBmp; score->RenderBgBmp = renderBgBmp;
__int16* shortArr = (__int16*)partman::field_labeled(loader::loader_table, fieldName, ShortArray); __int16* shortArr = (__int16*)partman::field_labeled(loader::loader_table, fieldName, datFieldTypes::ShortArray);
if (!shortArr) if (!shortArr)
{ {
memory::free(score); memory::free(score);
@ -32,7 +32,7 @@ scoreStruct* score::create(LPCSTR fieldName, int renderBgBmp)
int index = 10; int index = 10;
do do
{ {
*bmpPtr = partman::field(loader::loader_table, groupIndex, Bitmap8bit); *bmpPtr = partman::field(loader::loader_table, groupIndex, datFieldTypes::Bitmap8bit);
++bmpPtr; ++bmpPtr;
++groupIndex; ++groupIndex;
--index; --index;

View file

@ -0,0 +1,2 @@
#include "pch.h"
#include "zdrv.h"

22
SpaceCadetPinball/zdrv.h Normal file
View file

@ -0,0 +1,22 @@
#pragma once
#pragma pack(push, 1)
struct __declspec(align(1)) zmap_header_type
{
__int16 Width;
__int16 Height;
__int16 Stride;
char* BmpBufPtr1;
char* bmpBufPtr2;
char BmpBuffer[1];
};
#pragma pack(pop)
static_assert(sizeof(zmap_header_type) == 15, "Wrong size of zmap_header_type");
class zdrv
{
public:
};