2015-09-15 08:29:57 +02:00
|
|
|
#include <cstdio>
|
|
|
|
#include <cstdlib>
|
|
|
|
#include <cstring>
|
|
|
|
#include <cassert>
|
|
|
|
#include <new>
|
|
|
|
|
|
|
|
#include <rw.h>
|
2016-01-03 19:45:51 +01:00
|
|
|
#include <args.h>
|
2015-09-15 08:29:57 +02:00
|
|
|
#include <src/gtaplg.h>
|
|
|
|
|
2016-01-03 19:45:51 +01:00
|
|
|
char *argv0;
|
|
|
|
|
2015-09-15 08:29:57 +02:00
|
|
|
using namespace std;
|
|
|
|
using namespace rw;
|
|
|
|
|
2016-01-03 19:45:51 +01:00
|
|
|
struct {
|
|
|
|
char *str;
|
|
|
|
uint32 val;
|
|
|
|
} platforms[] = {
|
|
|
|
{ "mobile", PLATFORM_OGL },
|
|
|
|
{ "ps2", PLATFORM_PS2 },
|
|
|
|
{ "xbox", PLATFORM_XBOX },
|
|
|
|
{ "d3d8", PLATFORM_D3D8 },
|
|
|
|
{ "d3d9", PLATFORM_D3D9 },
|
|
|
|
{ NULL, 0 }
|
|
|
|
};
|
|
|
|
|
2015-09-19 19:28:23 +02:00
|
|
|
Raster*
|
|
|
|
xboxToD3d8(Raster *raster)
|
|
|
|
{
|
|
|
|
using namespace xbox;
|
|
|
|
|
|
|
|
Raster *newras;
|
|
|
|
if(raster->platform != PLATFORM_XBOX)
|
|
|
|
return raster;
|
|
|
|
XboxRaster *ras = PLUGINOFFSET(XboxRaster, raster, nativeRasterOffset);
|
|
|
|
|
|
|
|
int32 numLevels = raster->getNumLevels();
|
|
|
|
|
|
|
|
int32 format = raster->format;
|
|
|
|
// format &= ~Raster::MIPMAP;
|
|
|
|
if(ras->format){
|
2016-01-11 18:22:59 +01:00
|
|
|
newras = Raster::create(raster->width, raster->height, raster->depth,
|
|
|
|
format | raster->type | 0x80, PLATFORM_D3D8);
|
2015-09-19 19:28:23 +02:00
|
|
|
int32 dxt = 0;
|
|
|
|
switch(ras->format){
|
|
|
|
case D3DFMT_DXT1:
|
|
|
|
dxt = 1;
|
|
|
|
break;
|
|
|
|
case D3DFMT_DXT3:
|
|
|
|
dxt = 3;
|
|
|
|
break;
|
|
|
|
case D3DFMT_DXT5:
|
|
|
|
dxt = 5;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
d3d::allocateDXT(newras, dxt, numLevels, ras->hasAlpha);
|
|
|
|
}else{
|
|
|
|
printf("swizzled!\n");
|
2016-01-11 18:22:59 +01:00
|
|
|
newras = Raster::create(raster->width, raster->height, raster->depth,
|
|
|
|
format | raster->type, PLATFORM_D3D8);
|
2015-09-19 19:28:23 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
if(raster->format & Raster::PAL4)
|
|
|
|
d3d::setPalette(newras, ras->palette, 32);
|
|
|
|
else if(raster->format & Raster::PAL8)
|
|
|
|
d3d::setPalette(newras, ras->palette, 256);
|
|
|
|
|
|
|
|
uint8 *data;
|
|
|
|
for(int32 i = 0; i < numLevels; i++){
|
|
|
|
if(i >= newras->getNumLevels())
|
|
|
|
break;
|
|
|
|
data = raster->lock(i);
|
|
|
|
d3d::setTexels(newras, data, i);
|
|
|
|
raster->unlock(i);
|
|
|
|
}
|
|
|
|
|
|
|
|
delete raster;
|
|
|
|
return newras;
|
|
|
|
}
|
|
|
|
|
2016-01-03 19:45:51 +01:00
|
|
|
void
|
|
|
|
usage(void)
|
|
|
|
{
|
|
|
|
fprintf(stderr, "usage: %s [-v version] [-o platform] in.txd [out.txd]\n", argv0);
|
|
|
|
fprintf(stderr, "\t-v RW version, e.g. 33004 for 3.3.0.4\n");
|
|
|
|
fprintf(stderr, "\t-o output platform. ps2, xbox, mobile, d3d8, d3d9\n");
|
|
|
|
exit(1);
|
|
|
|
}
|
|
|
|
|
2015-09-15 08:29:57 +02:00
|
|
|
int
|
|
|
|
main(int argc, char *argv[])
|
|
|
|
{
|
|
|
|
gta::attachPlugins();
|
|
|
|
|
2016-01-03 19:45:51 +01:00
|
|
|
rw::version = 0;
|
|
|
|
rw::platform = rw::PLATFORM_PS2;
|
2015-09-15 08:29:57 +02:00
|
|
|
// rw::platform = rw::PLATFORM_OGL;
|
2015-09-17 10:44:07 +02:00
|
|
|
// rw::platform = rw::PLATFORM_XBOX;
|
2015-09-15 08:29:57 +02:00
|
|
|
// rw::platform = rw::PLATFORM_D3D8;
|
|
|
|
// rw::platform = rw::PLATFORM_D3D9;
|
2016-01-03 19:45:51 +01:00
|
|
|
int outplatform = rw::PLATFORM_PS2;
|
|
|
|
|
|
|
|
char *s;
|
|
|
|
ARGBEGIN{
|
|
|
|
case 'v':
|
|
|
|
sscanf(EARGF(usage()), "%x", &rw::version);
|
|
|
|
break;
|
|
|
|
case 'o':
|
|
|
|
s = EARGF(usage());
|
|
|
|
for(int i = 0; platforms[i].str; i++){
|
|
|
|
if(strcmp(platforms[i].str, s) == 0){
|
|
|
|
outplatform = platforms[i].val;
|
|
|
|
goto found;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
printf("unknown platform %s\n", s);
|
|
|
|
outplatform = PLATFORM_D3D8;
|
|
|
|
found:
|
|
|
|
break;
|
|
|
|
default:
|
|
|
|
usage();
|
|
|
|
}ARGEND;
|
2015-09-15 08:29:57 +02:00
|
|
|
|
2016-01-03 19:45:51 +01:00
|
|
|
if(argc < 1)
|
|
|
|
usage();
|
2015-09-15 08:29:57 +02:00
|
|
|
|
|
|
|
rw::StreamFile in;
|
2016-01-03 19:45:51 +01:00
|
|
|
if(in.open(argv[0], "rb") == NULL){
|
2015-09-19 19:28:23 +02:00
|
|
|
printf("couldn't open file %s\n", argv[1]);
|
2015-09-15 08:29:57 +02:00
|
|
|
return 1;
|
|
|
|
}
|
2016-01-03 19:45:51 +01:00
|
|
|
ChunkHeaderInfo header;
|
|
|
|
readChunkHeaderInfo(&in, &header);
|
|
|
|
assert(header.type == ID_TEXDICTIONARY);
|
2015-09-15 08:29:57 +02:00
|
|
|
rw::TexDictionary *txd;
|
|
|
|
txd = rw::TexDictionary::streamRead(&in);
|
|
|
|
assert(txd);
|
|
|
|
in.close();
|
|
|
|
rw::currentTexDictionary = txd;
|
|
|
|
|
2016-01-03 19:45:51 +01:00
|
|
|
if(rw::version == 0){
|
|
|
|
rw::version = header.version;
|
|
|
|
rw::build = header.build;
|
|
|
|
}
|
|
|
|
|
|
|
|
if(outplatform == PLATFORM_D3D8)
|
2016-01-11 18:22:59 +01:00
|
|
|
FORLIST(lnk, txd->textures){
|
|
|
|
Texture *tex = Texture::fromDict(lnk);
|
2016-01-03 19:45:51 +01:00
|
|
|
tex->raster = xboxToD3d8(tex->raster);
|
2016-01-11 18:22:59 +01:00
|
|
|
}
|
2015-12-14 18:52:50 +01:00
|
|
|
// for(Texture *tex = txd->first; tex; tex = tex->next)
|
|
|
|
// tex->filterAddressing = (tex->filterAddressing&~0xF) | 0x2;
|
2015-09-19 19:28:23 +02:00
|
|
|
|
2015-09-15 08:29:57 +02:00
|
|
|
rw::StreamFile out;
|
2016-01-03 19:45:51 +01:00
|
|
|
if(argc > 1)
|
|
|
|
out.open(argv[1], "wb");
|
2015-09-19 19:28:23 +02:00
|
|
|
else
|
|
|
|
out.open("out.txd", "wb");
|
2015-09-15 08:29:57 +02:00
|
|
|
txd->streamWrite(&out);
|
|
|
|
out.close();
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|