2 #include <snes/snes.hpp>
4 #include "core/misc.hpp"
5 #include "core/png.hpp"
6 #include "core/render.hpp"
16 #define TAG_ZEROWIDTH 0
19 #define TAG_TABULATION 3
20 #define TAG_WIDTH_MASK 3
21 #define TAG_LINECHANGE 4
23 extern const char* font_hex_data
;
27 std::vector
<uint32_t> font_glyph_data
;
28 std::map
<uint32_t, uint32_t> font_glyph_offsets
;
30 uint32_t parse_word(const char* x
)
35 unsigned long v
= strtoul(buf
, &end
, 16);
38 //std::cerr << "Parse word " << buf << std::endl;
44 static bool iflag
= false;
48 font_glyph_data
.resize(7);
50 font_glyph_data
[0] = TAG_NARROW
;
51 font_glyph_data
[1] = 0;
52 font_glyph_data
[2] = 0;
53 font_glyph_data
[3] = 0;
54 font_glyph_data
[4] = 0;
56 font_glyph_data
[5] = TAG_TABULATION
;
58 font_glyph_data
[6] = TAG_ZEROWIDTH
| TAG_LINECHANGE
;
62 for(size_t i
= 0;; i
++) {
64 switch(font_hex_data
[i
]) {
67 //Skip spaces at start of line.
77 if(lsptr
== i
|| font_hex_data
[lsptr
] == '#')
79 cp
= strtoul(font_hex_data
+ lsptr
, &end
, 16);
81 messages
<< "Malformed line " << lc
<< " in font data" << std::endl
;
84 fdatastart
= end
- font_hex_data
+ 1;
85 if(i
- fdatastart
== 32) {
87 font_glyph_offsets
[cp
] = font_glyph_data
.size();
88 font_glyph_data
.push_back(TAG_NARROW
);
89 for(uint32_t k
= 0; k
< 4; k
++)
90 font_glyph_data
.push_back(parse_word(end
+ 1 + 8 * k
));
91 } else if(i
- fdatastart
== 64) {
93 font_glyph_offsets
[cp
] = font_glyph_data
.size();
94 font_glyph_data
.push_back(TAG_WIDE
);
95 for(uint32_t k
= 0; k
< 8; k
++)
96 font_glyph_data
.push_back(parse_word(end
+ 1 + 8 * k
));
98 messages
<< "Malformed line " << lc
<< " in font data" << std::endl
;
102 if(font_hex_data
[i
] != '\r' || font_hex_data
[i
+ 1] != '\n')
107 if(!font_hex_data
[i
])
111 //Special characters.
112 font_glyph_offsets
[9] = 5;
113 font_glyph_offsets
[10] = 6;
114 font_glyph_offsets
[32] = 0;
117 uint32_t glyphs_narrow
= 0;
118 uint32_t glyphs_wide
= 0;
119 uint32_t glyphs_special
= 0;
120 for(auto i
: font_glyph_offsets
) {
121 if(font_glyph_data
[i
.second
] == TAG_NARROW
)
123 else if(font_glyph_data
[i
.second
] == TAG_WIDE
)
129 messages
<< "Loaded font data: " << glyphs
<< " glyphs (" << glyphs_narrow
<< " narrow, " <<
130 glyphs_wide
<< " wide, " << glyphs_special
<< " special)." << std::endl
;
134 inline uint32_t find_font_glyph_offset(uint32_t cp
)
136 return font_glyph_offsets
.count(cp
) ? font_glyph_offsets
[cp
] : 0;
139 inline uint32_t process_tag(uint32_t tag
, int32_t& x
, int32_t& y
, int32_t orig_x
)
142 switch(tag
& TAG_WIDTH_MASK
) {
153 dwidth
= 0x40 - (x
& 0x3F);
157 if(tag
& TAG_LINECHANGE
) {
164 inline bool is_visible(uint32_t tag
)
166 return ((tag
& TAG_WIDTH_MASK
) == TAG_NARROW
|| (tag
& TAG_WIDTH_MASK
) == TAG_WIDE
);
176 std::pair
<uint32_t, const uint32_t*> find_glyph(uint32_t codepoint
, int32_t x
, int32_t y
, int32_t orig_x
,
177 int32_t& next_x
, int32_t& next_y
) throw()
182 uint32_t offset
= find_font_glyph_offset(codepoint
);
183 uint32_t tag
= font_glyph_data
[offset
];
184 uint32_t dwidth
= process_tag(tag
, next_x
, next_y
, orig_x
);
185 bool visible
= is_visible(tag
);
186 return std::pair
<uint32_t, const uint32_t*>(dwidth
, visible
? &font_glyph_data
[offset
+ 1] : NULL
);
189 render_object::~render_object() throw()
193 void render_text(struct screen
& scr
, int32_t x
, int32_t y
, const std::string
& text
, premultiplied_color fg
,
194 premultiplied_color bg
) throw(std::bad_alloc
)
197 uint32_t unicode_code
= 0;
198 uint8_t unicode_left
= 0;
199 for(size_t i
= 0; i
< text
.length(); i
++) {
200 uint8_t ch
= text
[i
];
202 unicode_code
= text
[i
];
206 unicode_code
= 64 * unicode_code
+ ch
- 128;
209 } else if(ch
< 224) {
210 unicode_code
= ch
- 192;
213 } else if(ch
< 240) {
214 unicode_code
= ch
- 224;
217 } else if(ch
< 248) {
218 unicode_code
= ch
- 240;
223 int32_t next_x
, next_y
;
224 auto p
= find_glyph(unicode_code
, x
, y
, orig_x
, next_x
, next_y
);
226 uint32_t dw
= p
.first
;
229 uint32_t cx
= static_cast<uint32_t>(static_cast<int32_t>(scr
.originx
) + x
);
230 uint32_t cy
= static_cast<uint32_t>(static_cast<int32_t>(scr
.originy
) + y
);
231 while(cx
> scr
.width
&& dw
> 0) {
236 while(cy
> scr
.height
&& dh
> 0) {
241 while(cx
+ dw
> scr
.width
&& dw
> 0)
243 while(cy
+ dh
> scr
.height
&& dh
> 0)
246 continue; //Outside screen.
248 if(p
.second
== NULL
) {
250 for(uint32_t j
= 0; j
< dh
; j
++) {
251 uint32_t* base
= scr
.rowptr(cy
+ j
) + cx
;
252 for(uint32_t i
= 0; i
< dw
; i
++)
255 } else if(p
.first
== 16) {
257 for(uint32_t j
= 0; j
< dh
; j
++) {
258 uint32_t dataword
= p
.second
[(dy
+ j
) >> 1];
259 uint32_t* base
= scr
.rowptr(cy
+ j
) + cx
;
260 uint32_t rbit
= (~((dy
+ j
) << 4) & 0x1F) - dx
;
261 for(uint32_t i
= 0; i
< dw
; i
++)
262 if((dataword
>> (rbit
- i
)) & 1)
269 for(uint32_t j
= 0; j
< dh
; j
++) {
270 uint32_t dataword
= p
.second
[(dy
+ j
) >> 2];
271 uint32_t* base
= scr
.rowptr(cy
+ j
) + cx
;
272 uint32_t rbit
= (~((dy
+ j
) << 3) & 0x1F) - dx
;
273 for(uint32_t i
= 0; i
< dw
; i
++)
274 if((dataword
>> (rbit
- i
)) & 1)
285 void render_queue::add(struct render_object
& obj
) throw(std::bad_alloc
)
290 void render_queue::run(struct screen
& scr
) throw()
302 void render_queue::clear() throw()
309 render_queue::~render_queue() throw()
314 uint32_t screen::make_color(uint8_t r
, uint8_t g
, uint8_t b
) throw()
319 return (_r
<< 16) + (_g
<< 8) + _b
;
322 lcscreen::lcscreen(const uint32_t* mem
, bool hires
, bool interlace
, bool overscan
, bool region
) throw()
324 uint32_t dataoffset
= 0;
325 width
= hires
? 512 : 256;
330 dataoffset
= overscan
? 9 : 1;
334 dataoffset
= overscan
? 16 : 9;
338 memory
= mem
+ dataoffset
* 1024;
339 pitch
= interlace
? 512 : 1024;
343 lcscreen::lcscreen(const uint32_t* mem
, uint32_t _width
, uint32_t _height
) throw()
352 lcscreen::lcscreen() throw()
362 lcscreen::lcscreen(const lcscreen
& ls
) throw(std::bad_alloc
)
368 allocated
= static_cast<size_t>(width
) * height
;
369 memory
= new uint32_t[allocated
];
370 for(size_t l
= 0; l
< height
; l
++)
371 memcpy(const_cast<uint32_t*>(memory
+ l
* width
), ls
.memory
+ l
* ls
.pitch
, 4 * width
);
374 lcscreen
& lcscreen::operator=(const lcscreen
& ls
) throw(std::bad_alloc
, std::runtime_error
)
377 throw std::runtime_error("Can't copy to non-user memory");
380 if(allocated
< static_cast<size_t>(ls
.width
) * ls
.height
) {
381 size_t p_allocated
= static_cast<size_t>(ls
.width
) * ls
.height
;
382 memory
= new uint32_t[p_allocated
];
383 allocated
= p_allocated
;
388 for(size_t l
= 0; l
< height
; l
++)
389 memcpy(const_cast<uint32_t*>(memory
+ l
* width
), ls
.memory
+ l
* ls
.pitch
, 4 * width
);
393 lcscreen::~lcscreen()
396 delete[] const_cast<uint32_t*>(memory
);
399 void lcscreen::load(const std::vector
<char>& data
) throw(std::bad_alloc
, std::runtime_error
)
402 throw std::runtime_error("Can't load to non-user memory");
403 const uint8_t* data2
= reinterpret_cast<const uint8_t*>(&data
[0]);
405 throw std::runtime_error("Corrupt saved screenshot data");
406 uint32_t _width
= static_cast<uint32_t>(data2
[0]) * 256 + static_cast<uint32_t>(data2
[1]);
407 if(_width
> 1 && data
.size() % (3 * _width
) != 2)
408 throw std::runtime_error("Corrupt saved screenshot data");
409 uint32_t _height
= (data
.size() - 2) / (3 * _width
);
410 if(allocated
< static_cast<size_t>(_width
) * _height
) {
411 size_t p_allocated
= static_cast<size_t>(_width
) * _height
;
412 memory
= new uint32_t[p_allocated
];
413 allocated
= p_allocated
;
415 uint32_t* mem
= const_cast<uint32_t*>(memory
);
419 for(size_t i
= 0; i
< (data
.size() - 2) / 3; i
++)
420 mem
[i
] = static_cast<uint32_t>(data2
[2 + 3 * i
]) * 65536 +
421 static_cast<uint32_t>(data2
[2 + 3 * i
+ 1]) * 256 +
422 static_cast<uint32_t>(data2
[2 + 3 * i
+ 2]);
425 void lcscreen::save(std::vector
<char>& data
) throw(std::bad_alloc
)
427 data
.resize(2 + 3 * static_cast<size_t>(width
) * height
);
428 uint8_t* data2
= reinterpret_cast<uint8_t*>(&data
[0]);
429 data2
[0] = (width
>> 8);
431 for(size_t i
= 0; i
< (data
.size() - 2) / 3; i
++) {
432 data
[2 + 3 * i
] = memory
[(i
/ width
) * pitch
+ (i
% width
)] >> 16;
433 data
[2 + 3 * i
+ 1] = memory
[(i
/ width
) * pitch
+ (i
% width
)] >> 8;
434 data
[2 + 3 * i
+ 2] = memory
[(i
/ width
) * pitch
+ (i
% width
)];
438 void lcscreen::save_png(const std::string
& file
) throw(std::bad_alloc
, std::runtime_error
)
440 uint8_t* buffer
= new uint8_t[3 * static_cast<size_t>(width
) * height
];
441 for(uint32_t j
= 0; j
< height
; j
++)
442 for(uint32_t i
= 0; i
< width
; i
++) {
443 uint32_t word
= memory
[pitch
* j
+ i
];
444 uint32_t l
= 1 + ((word
>> 15) & 0xF);
445 uint32_t r
= l
* ((word
>> 0) & 0x1F);
446 uint32_t g
= l
* ((word
>> 5) & 0x1F);
447 uint32_t b
= l
* ((word
>> 10) & 0x1F);
448 buffer
[3 * static_cast<size_t>(width
) * j
+ 3 * i
+ 0] = r
* 255 / 496;
449 buffer
[3 * static_cast<size_t>(width
) * j
+ 3 * i
+ 1] = g
* 255 / 496;
450 buffer
[3 * static_cast<size_t>(width
) * j
+ 3 * i
+ 2] = b
* 255 / 496;
453 save_png_data(file
, buffer
, width
, height
);
461 void screen::copy_from(lcscreen
& scr
, uint32_t hscale
, uint32_t vscale
) throw()
463 if(width
< originx
|| height
< originy
) {
464 //Just clear the screen.
465 for(uint32_t y
= 0; y
< height
; y
++)
466 memset(rowptr(y
), 0, 4 * width
);
469 uint32_t copyable_width
= (width
- originx
) / hscale
;
470 uint32_t copyable_height
= (height
- originy
) / vscale
;
471 copyable_width
= (copyable_width
> scr
.width
) ? scr
.width
: copyable_width
;
472 copyable_height
= (copyable_height
> scr
.height
) ? scr
.height
: copyable_height
;
473 for(uint32_t y
= 0; y
< height
; y
++)
474 memset(rowptr(y
), 0, 4 * width
);
475 for(uint32_t y
= 0; y
< copyable_height
; y
++) {
476 uint32_t line
= y
* vscale
+ originy
;
477 uint32_t* ptr
= rowptr(line
) + originx
;
478 const uint32_t* sbase
= scr
.memory
+ y
* scr
.pitch
;
479 for(uint32_t x
= 0; x
< copyable_width
; x
++) {
480 uint32_t c
= palette
[sbase
[x
] & 0x7FFFF];
481 for(uint32_t i
= 0; i
< hscale
; i
++)
484 for(uint32_t j
= 1; j
< vscale
; j
++)
485 memcpy(rowptr(line
+ j
) + originx
, rowptr(line
) + originx
, 4 * hscale
* copyable_width
);
489 void screen::reallocate(uint32_t _width
, uint32_t _height
, bool upside_down
) throw(std::bad_alloc
)
491 if(_width
== width
&& _height
== height
)
493 if(!_width
|| !_height
) {
494 width
= height
= originx
= originy
= pitch
= 0;
495 if(memory
&& !user_memory
)
499 flipped
= upside_down
;
502 uint32_t* newmem
= new uint32_t[_width
* _height
];
506 if(memory
&& !user_memory
)
510 flipped
= upside_down
;
513 void screen::set(uint32_t* _memory
, uint32_t _width
, uint32_t _height
, uint32_t _pitch
) throw()
515 if(memory
&& !user_memory
)
525 void screen::set_origin(uint32_t _originx
, uint32_t _originy
) throw()
532 uint32_t* screen::rowptr(uint32_t row
) throw()
535 row
= height
- row
- 1;
536 return reinterpret_cast<uint32_t*>(reinterpret_cast<uint8_t*>(memory
) + row
* pitch
);
539 screen::screen() throw()
542 width
= height
= originx
= originy
= pitch
= 0;
546 set_palette(16, 8, 0);
549 screen::~screen() throw()
551 if(memory
&& !user_memory
)
555 void clip_range(uint32_t origin
, uint32_t size
, int32_t base
, int32_t& minc
, int32_t& maxc
) throw()
557 int64_t _origin
= origin
;
558 int64_t _size
= size
;
559 int64_t _base
= base
;
560 int64_t _minc
= minc
;
561 int64_t _maxc
= maxc
;
562 int64_t mincoordinate
= _base
+ _origin
+ _minc
;
563 int64_t maxcoordinate
= _base
+ _origin
+ _maxc
;
564 if(mincoordinate
< 0)
565 _minc
= _minc
- mincoordinate
;
566 if(maxcoordinate
> _size
)
567 _maxc
= _maxc
- (maxcoordinate
- _size
);
577 void screen::set_palette(uint32_t r
, uint32_t g
, uint32_t b
)
580 palette
= new uint32_t[0x80000];
581 else if(r
== palette_r
&& g
== palette_g
&& b
== palette_b
)
583 for(size_t i
= 0; i
< static_cast<size_t>(width
) * height
; i
++) {
584 uint32_t word
= memory
[i
];
585 uint32_t R
= (word
>> palette_r
) & 0xFF;
586 uint32_t G
= (word
>> palette_g
) & 0xFF;
587 uint32_t B
= (word
>> palette_b
) & 0xFF;
588 memory
[i
] = (R
<< r
) | (G
<< g
) | (B
<< b
);
590 for(unsigned i
= 0; i
< 0x80000; i
++) {
591 unsigned l
= 1 + ((i
>> 15) & 0xF);
592 unsigned R
= (i
>> 0) & 0x1F;
593 unsigned G
= (i
>> 5) & 0x1F;
594 unsigned B
= (i
>> 10) & 0x1F;
595 double _l
= static_cast<double>(l
);
596 double m
= 255.0 / 496.0;
597 R
= floor(m
* R
* _l
+ 0.5);
598 G
= floor(m
* G
* _l
+ 0.5);
599 B
= floor(m
* B
* _l
+ 0.5);
600 palette
[i
] = (R
<< r
) | (G
<< g
) | (B
<< b
);