JunkBox_Lib++ (for Windows) 1.10.1
Loading...
Searching...
No Matches
TgaTool.cpp
Go to the documentation of this file.
1
9#include "TgaTool.h"
10
11
12using namespace jbxl;
13
14
20void TGAImage::init(void)
21{
22 xs = 0;
23 ys = 0;
24 col = 0;
25 length = 0;
26 state = 0;
27 gp = NULL;
28
29 memset(hd, 0, TGA_HEADER_SIZE);
30 memset(ft, 0, TGA_FOOTER_SIZE);
31
32 int len = (int)strlen(TGA_FOOTER_STR) + 1;
33 int pos = TGA_FOOTER_SIZE - len;
34 memcpy(ft + pos, TGA_FOOTER_STR, len);
35
36 return;
37}
38
39
46{
47 if (gp==NULL) return true;
48 return false;
49}
50
51
55void TGAImage::clear(void)
56{
57 if (gp!=NULL) memset(gp, 0, xs*ys*col);
58 return;
59}
60
61
66{
67 if (gp!=NULL) memset(gp, v, xs*ys*col);
68 return;
69}
70
71
77void TGAImage::free(void)
78{
79 if (gp!=NULL) ::free(gp);
80 init();
81
82 return;
83}
84
85
89void TGAImage::setzero(int x, int y, int c)
90{
91 getm(x, y, c);
92 if (gp==NULL) return;
93
94 xs = x;
95 ys = y;
96 col = c;
97 state = 0;
98
99 memset(gp, 0, xs*ys*col);
100
101 return;
102}
103
104
108void TGAImage::getm(int x, int y, int c)
109{
110 gp = (uByte*)malloc(x*y*c);
111 if (gp==NULL) {
113 return;
114 }
115 memset(gp, 0, x*y*c);
116
117 return;
118}
119
120
121
123
136TGAImage jbxl::readTGAFile(const char* fname)
137{
138 TGAImage tga;
139 FILE* fp;
140
141 fp = fopen(fname, "rb");
142 if (fp==NULL) {
143 tga.gp = NULL;
145 return tga;
146 }
147
148 tga = readTGAData(fp);
149 fclose(fp);
150
151 return tga;
152}
153
154
170{
171 TGAImage tga;
172 int rle = FALSE;
173
174 fseek(fp, 0, 0);
175 tga.free();
176
177 size_t ret = fread(&tga.hd, TGA_HEADER_SIZE, 1, fp);
178 if (ret<=0) {
179 DEBUG_MODE PRINT_MESG("JBXL::readTGAData: ERROR: File read Error\n");
181 return tga;
182 }
183
184 int kind = (int)tga.hd[2];
185 if (kind == 2) {
186 tga.col = 3;
187 }
188 else if (kind == 3) {
189 tga.col = 1;
190 }
191 else if (kind == 10) {
192 tga.col = 3;
193 rle = TRUE;
194 }
195 else if (kind == 11) {
196 tga.col = 1;
197 rle = TRUE;
198 }
199 else {
200 DEBUG_MODE PRINT_MESG("JBXL::readTGAData: ERROR: no supported File Format (%d)\n", kind);
202 return tga;
203 }
204
205 if (is_little_endian()) {
206 tga.xs = tga.hd[12] + tga.hd[13]*256;
207 tga.ys = tga.hd[14] + tga.hd[15]*256;
208 }
209 else {
210 tga.xs = tga.hd[12]*256 + tga.hd[13];
211 tga.ys = tga.hd[14]*256 + tga.hd[15];
212 }
213
214 uByte alpha = tga.hd[17] & 0x0f; // αチャンネル
215 if (alpha != 0x08 && alpha != 0x00) {
216 DEBUG_MODE PRINT_MESG("JBXL::readTGAData: ERROR: unkonwn Alpha Channel (0x%02x)\n", alpha);
218 return tga;
219 }
220 else if (alpha == 0x08) {
221 tga.col++;
222 }
223 if (tga.col*8 != (int)tga.hd[16]) {
224 tga.col = (int)tga.hd[16]/8;
225 PRINT_MESG("JBXL::readTGAData: Warning: Not Match Color Num! set color num = %d\n", tga.col);
226 }
227 PRINT_MESG("JBXL::readTGAData: TGA File (%d, %d, %d)\n", tga.xs, tga.ys, tga.col);
228
229 int datasize = tga.xs*tga.ys*tga.col;
230 tga.length = datasize;
231
232 tga.gp = (uByte*)malloc(datasize);
233 if (tga.gp==NULL) {
234 PRINT_MESG("JBXL::readTGAData: ERROR: out of Memory!\n");
236 return tga;
237 }
238 memset(tga.gp, 0, datasize);
239
240 fseek(fp, (int)tga.hd[0], SEEK_CUR);
241 if (rle) {
242 // RLE
243 int size = 0;
244 uByte chunk;
245 uByte buf[LBUF];
246
247 while (size<datasize && !feof(fp)) {
248 ret = fread(&chunk, 1, 1, fp);
249 if (ret<=0) {
250 DEBUG_MODE PRINT_MESG("JBXL::readTGAData: ERROR: File read Error\n");
252 return tga;
253 }
254 int rep = (int)(chunk & 0x7f) + 1;
255 if (chunk & 0x80) {
256 ret = fread(buf, tga.col, 1, fp);
257 if (ret<=0) {
259 return tga;
260 }
261 for (int j=0; j<rep; j++) {
262 for (int i=0; i<tga.col; i++) tga.gp[size++] = buf[i];
263 }
264 }
265 else {
266 if (tga.col*rep<LBUF) {
267 ret = fread(buf, tga.col*rep, 1, fp);
268 if (ret<=0) {
270 return tga;
271 }
272 for (int i=0; i<tga.col*rep; i++) tga.gp[size++] = buf[i];
273 }
274 else {
275 for (int j=0; j<rep; j++) {
276 ret = fread(buf, tga.col, 1, fp);
277 if (ret<=0) {
279 return tga;
280 }
281 for (int i=0; i<tga.col; i++) tga.gp[size++] = buf[i];
282 }
283 }
284 }
285 }
286 if (size!=datasize) {
287 DEBUG_MODE PRINT_MESG("JBXL::readTGAData: ERROR: unpack RLE failed. (%d != %d), format = %d\n", size, datasize, kind);
289 return tga;
290 }
291 tga.hd[2] &= ~0x08; // 非圧縮状態
292 }
293 else {
294 ret = fread(tga.gp, datasize, 1, fp);
295 if (ret<=0) {
297 return tga;
298 }
299 }
300 if (!feof(fp)) {
301 ret = fread(tga.ft, TGA_FOOTER_SIZE, 1, fp);
302 if (ret<=0) {
304 return tga;
305 }
306 }
307 tga.state = 0;
308 return tga;
309}
310
311
328int jbxl::writeTGAFile(const char* fname, TGAImage* tga)
329{
330 FILE* fp;
331 int ret;
332
333 if (fname==NULL) return JBXL_GRAPH_IVDARG_ERROR;
334 if (tga->col<=0 || tga->col>4) return JBXL_GRAPH_IVDCOLOR_ERROR;
335 if (tga->gp==NULL) return JBXL_GRAPH_NODATA_ERROR;
336
337 fp = fopen(fname, "wb");
338 if (fp==NULL) {
340 }
341
342 ret = writeTGAData(fp, tga);
343 fclose(fp);
344
345 return ret;
346}
347
348
365int jbxl::writeTGAData(FILE* fp, TGAImage* tga)
366{
367 if (fp==NULL) return JBXL_GRAPH_OPFILE_ERROR;
368 if (tga->col<=0 || tga->col>4) return JBXL_GRAPH_IVDCOLOR_ERROR;
369 if (tga->gp==NULL) return JBXL_GRAPH_NODATA_ERROR;
370
371 fwrite(tga->hd, TGA_HEADER_SIZE, 1, fp);
372 fwrite(tga->gp, tga->length, 1, fp);
373 fwrite(tga->ft, TGA_FOOTER_SIZE, 1, fp);
374
375 return 0;
376}
377
378
390int jbxl::setupTGAData(TGAImage* tga, bool rle)
391{
392 if (tga->col<=0 || tga->col>4) return JBXL_GRAPH_IVDCOLOR_ERROR;
393
394 memset(tga->hd, 0, TGA_HEADER_SIZE);
395 if (tga->col==3 || tga->col==4) tga->hd[2] = 2; // Full Color
396 else if (tga->col==1 || tga->col==2) tga->hd[2] = 3; // Gray Scale
397 else return JBXL_GRAPH_IVDFMT_ERROR;
398
399 if (is_little_endian()) {
400 tga->hd[12] = (uByte)(tga->xs%256);
401 tga->hd[13] = (uByte)(tga->xs/256);
402 tga->hd[14] = (uByte)(tga->ys%256);
403 tga->hd[15] = (uByte)(tga->ys/256);
404 }
405 else {
406 tga->hd[12] = (uByte)(tga->xs/256);
407 tga->hd[13] = (uByte)(tga->xs%256);
408 tga->hd[14] = (uByte)(tga->ys/256);
409 tga->hd[15] = (uByte)(tga->ys%256);
410 }
411
412 tga->hd[16] = tga->col*8; // depth
413 tga->hd[17] = 0x20; // 0x20: Y方向:Top->Down
414 tga->hd[10] = tga->hd[14]; // Y posion for 0x20
415 tga->hd[11] = tga->hd[15]; //
416 if (tga->col==2 || tga->col==4) {
417 tga->hd[17] |= 0x08; // 0x08: αチャンネル深度
418 }
419 tga->length = tga->xs*tga->ys*tga->col; // データ(gp)長
420
421 if (rle && tga->gp!=NULL) {
422 int len = tga->xs*tga->ys;
423 uByte* index = (uByte*)malloc(len);
424 if (index==NULL) return JBXL_GRAPH_MEMORY_ERROR;
425 memset(index, 0, len);
426
427 int idx = 0;
428 int p = 0;
429 while (p<len-1) {
430 if (!memcmp(tga->gp + p*tga->col, tga->gp + (p+1)*tga->col, tga->col)) {
431 index[idx]++;
432 if (index[idx]==127) {
433 idx = p + 2;
434 p++;
435 }
436 }
437 else {
438 idx = p + 1;
439 }
440 p++;
441 }
442
443 // 変換後のサイズを計算
444 int n = 0;
445 p = 0;
446 while (p < len-1) {
447 n += tga->col + 1;
448 if (index[p]!=0x00) p += index[p];
449 p++;
450 }
451
452 if ((float)n < len*tga->col*0.8f) {
453 DEBUG_MODE PRINT_MESG("JBXL::writeTGAData: exec RLE (%d -> %d)\n", len*tga->col, n);
454 uByte* buf = (uByte*)malloc(len*tga->col);
455 if (buf==NULL) {
456 ::free(index);
458 }
459 memset(buf, 0, len*tga->col);
460 //
461 int i = 0;
462 int j = 0;
463 while (i < len-1) {
464 if (index[i]!=0x00) {
465 buf[j++] = 0x80 + index[i];
466 memcpy(buf+j, tga->gp + i*tga->col, tga->col);
467 j += tga->col;
468 i += index[i];
469 }
470 else {
471 buf[j++] = 0x00;
472 memcpy(buf+j, tga->gp + i*tga->col, tga->col);
473 j += tga->col;
474 }
475 i++;
476 }
477 //
478 if (j==n) {
479 tga->hd[2] |= 0x08;
480 tga->length = n;
481 ::free(tga->gp);
482 tga->gp = buf;
483 }
484 else {
485 DEBUG_MODE PRINT_MESG("JBXL::writeTGAData: Warning: missmatch RLE size (%d != %d)\n", j, n);
486 DEBUG_MODE PRINT_MESG("JBXL::writeTGAData: Warning: stop RLE!\n");
487 ::free(buf);
488 }
489 ::free(index);
490 }
491 }
492 return 0;
493}
494
TGAグラフィックデータ定義用ヘッダ
#define TGA_HEADER_SIZE
Definition TgaTool.h:16
#define TGA_FOOTER_STR
Definition TgaTool.h:18
#define TGA_FOOTER_SIZE
Definition TgaTool.h:17
void fill(uByte v=(uByte) 0)
全空間を画素値 v にする
Definition TgaTool.cpp:65
void init(void)
グラフィックデータは解放しない
Definition TgaTool.cpp:20
void setzero(int x, int y, int c)
Definition TgaTool.cpp:89
uByte * gp
Definition TgaTool.h:38
uByte ft[TGA_FOOTER_SIZE]
Definition TgaTool.h:37
void free(void)
グラフィックデータを開放する
Definition TgaTool.cpp:77
bool isNull(void)
グラフィックデータを持っていないか?
Definition TgaTool.cpp:45
void clear(void)
全空間を画素値 0 にする
Definition TgaTool.cpp:55
void getm(int x, int y, int c)
Definition TgaTool.cpp:108
uByte hd[TGA_HEADER_SIZE]
Definition TgaTool.h:36
#define LBUF
Definition common.h:146
#define TRUE
Definition common.h:226
#define FALSE
Definition common.h:223
unsigned char uByte
1Byte
Definition common.h:332
#define JBXL_GRAPH_IVDARG_ERROR
無効な引数
Definition jbxl_state.h:178
#define JBXL_GRAPH_NODATA_ERROR
データが無い
Definition jbxl_state.h:171
#define JBXL_FILE_READ_ERROR
ファイル読み込みエラー
Definition jbxl_state.h:45
#define JBXL_GRAPH_OPFILE_ERROR
ファイルのオープンエラー
Definition jbxl_state.h:173
#define JBXL_GRAPH_IVDCOLOR_ERROR
無効なカラー指定
Definition jbxl_state.h:183
#define JBXL_GRAPH_IVDFMT_ERROR
無効なデータ形式
Definition jbxl_state.h:182
#define JBXL_GRAPH_FILESZ_ERROR
ファイルサイズのエラー
Definition jbxl_state.h:176
#define JBXL_GRAPH_MEMORY_ERROR
メモリエラー
Definition jbxl_state.h:170
Definition Brep.h:29
int writeTGAData(FILE *fp, TGAImage *tga)
Definition TgaTool.cpp:365
TGAImage readTGAFile(const char *fname)
Definition TgaTool.cpp:136
TGAImage readTGAData(FILE *fp)
Definition TgaTool.cpp:169
int setupTGAData(TGAImage *tga, bool rle)
Definition TgaTool.cpp:390
int writeTGAFile(const char *fname, TGAImage *tga)
Definition TgaTool.cpp:328
int is_little_endian(void)
エンディアンの動的チェック
Definition tools.cpp:80
#define PRINT_MESG(...)
環境依存用の出力関数.MS Windows用は未実装
Definition tools.h:469
#define DEBUG_MODE
Definition tools.h:502