mirror of
https://git.planet-casio.com/Lephenixnoir/fxsdk.git
synced 2025-04-04 09:37:11 +02:00
implement fxgxa and use it in both build systems
This commit is contained in:
parent
ec39aa5cde
commit
2cc5d7ac5b
15 changed files with 991 additions and 565 deletions
|
@ -23,19 +23,15 @@ set(BIN "${CMAKE_CURRENT_BINARY_DIR}")
|
||||||
|
|
||||||
add_compile_options(-Wall -Wextra -std=c11 -Og -g -D_GNU_SOURCE)
|
add_compile_options(-Wall -Wextra -std=c11 -Og -g -D_GNU_SOURCE)
|
||||||
|
|
||||||
# fxg1a
|
# fxgxa
|
||||||
add_executable(fxg1a fxgxa/dump.c fxgxa/edit.c fxgxa/file.c fxgxa/icon.c
|
add_executable(fxgxa fxgxa/dump.c fxgxa/edit.c fxgxa/file.c fxgxa/icon.c
|
||||||
fxgxa/main.c fxgxa/util.c)
|
fxgxa/main.c fxgxa/util.c)
|
||||||
target_include_directories(fxg1a PUBLIC fxgxa/)
|
target_include_directories(fxgxa PUBLIC fxgxa/)
|
||||||
target_link_libraries(fxg1a PkgConfig::libpng)
|
target_link_libraries(fxgxa PkgConfig::libpng)
|
||||||
target_compile_definitions(fxg1a PRIVATE -DFXGXA_FORMAT_G1A)
|
|
||||||
|
|
||||||
# fxg3a
|
# fxg1a as a symlink (for compatibility=
|
||||||
add_executable(fxg3a fxgxa/dump.c fxgxa/edit.c fxgxa/file.c fxgxa/icon.c
|
add_custom_target(fxg1a ALL
|
||||||
fxgxa/main.c fxgxa/util.c)
|
COMMAND ${CMAKE_COMMAND} -E create_symlink "fxgxa" "fxg1a")
|
||||||
target_include_directories(fxg3a PUBLIC fxgxa/)
|
|
||||||
target_link_libraries(fxg3a PkgConfig::libpng)
|
|
||||||
target_compile_definitions(fxg3a PRIVATE -DFXGXA_FORMAT_G3A)
|
|
||||||
|
|
||||||
# fxsdk
|
# fxsdk
|
||||||
add_custom_command(OUTPUT "${BIN}/fxsdk.sh"
|
add_custom_command(OUTPUT "${BIN}/fxsdk.sh"
|
||||||
|
@ -64,10 +60,9 @@ endif()
|
||||||
install(PROGRAMS "${BIN}/fxsdk.sh" TYPE BIN RENAME fxsdk)
|
install(PROGRAMS "${BIN}/fxsdk.sh" TYPE BIN RENAME fxsdk)
|
||||||
install(DIRECTORY fxsdk/assets DESTINATION share/fxsdk)
|
install(DIRECTORY fxsdk/assets DESTINATION share/fxsdk)
|
||||||
install(DIRECTORY fxsdk/cmake/ DESTINATION lib/cmake/fxsdk)
|
install(DIRECTORY fxsdk/cmake/ DESTINATION lib/cmake/fxsdk)
|
||||||
# fxg1a
|
# fxgxa, fxg1a
|
||||||
install(TARGETS fxg1a)
|
install(TARGETS fxgxa)
|
||||||
# fxg3a
|
install(FILES "${BIN}/fxg1a" TYPE BIN)
|
||||||
install(TARGETS fxg3a)
|
|
||||||
# fxconv
|
# fxconv
|
||||||
install(PROGRAMS fxconv/fxconv-main.py TYPE BIN RENAME fxconv)
|
install(PROGRAMS fxconv/fxconv-main.py TYPE BIN RENAME fxconv)
|
||||||
install(FILES fxconv/fxconv.py TYPE BIN)
|
install(FILES fxconv/fxconv.py TYPE BIN)
|
||||||
|
|
32
README.md
32
README.md
|
@ -70,29 +70,27 @@ Summary of commands (`fxsdk --help` for details):
|
||||||
* `fxsdk build/build-fx/build-cg`: Configure and compile add-ins and libraries
|
* `fxsdk build/build-fx/build-cg`: Configure and compile add-ins and libraries
|
||||||
* `fxsdk send/send-fx/send-cg`: Install files to the calculator (WIP)
|
* `fxsdk send/send-fx/send-cg`: Install files to the calculator (WIP)
|
||||||
|
|
||||||
**G1A file generation** with `fxg1a`
|
**G1A/G3A file generation** with `fxgxa`
|
||||||
|
|
||||||
`fxg1a` is a versatile g1a file editor that creates, edits and dumps the header
|
`fxgxa` is a versatile g1a/g3a file editor that creates, edits and dumps the
|
||||||
of fx-9860G add-ins files. It is used to build a g1a file out of a binary
|
header of fx-9860G add-ins files. It is used to build g1a/g3a files out of a
|
||||||
program.
|
binary program.
|
||||||
|
|
||||||
It supports PNG icons, checking the validity and checksums of the header,
|
It supports PNG icons of any formats, checking the validity and checksums of
|
||||||
repairing broken headers and dumping both the application data and icon.
|
the header, repairing broken headers and dumping both the application data and
|
||||||
|
icons.
|
||||||
|
|
||||||
`fxg1a` is called automatically by the build system in your add-in, so you
|
`fxgxa` is called automatically by the build system in your add-in, so you
|
||||||
don't need to worry about it, but here are the main commands:
|
don't need to worry about it, but here are the main commands:
|
||||||
|
|
||||||
* `fxg1a -g`: Generate g1a files
|
* `fxgxa --g1a|--g3a`: Generate g1a/g3a files
|
||||||
* `fxg1a -e`: Edit g1a files
|
* `fxgxa -e`: Edit g1a/g3a files
|
||||||
* `fxg1a -d`: Dump metadata, checksum, and icon
|
* `fxgxa -d`: Dump metadata, checksum, and icon
|
||||||
* `fxg1a -r`: Repair control bytes and checksums for broken files
|
* `fxgxa -r`: Repair control bytes and checksums for broken files
|
||||||
* `fxg1a -x`: Extract icon into a PNG file
|
* `fxgxa -x`: Extract icons into PNG files
|
||||||
|
|
||||||
**G3A file generation** with `fxg3a`
|
`fxgxa` has an alias, `fxg1a`, for compatibility with fxSDK up to 2.7.0 for
|
||||||
|
which there was no g3a file editor in the fxSDK.
|
||||||
`fxg3a` is very similar to `fxg1a`, but generates and edits g3a files instead.
|
|
||||||
These files are used for fx-CG 10/20/50 programs. The main commands are
|
|
||||||
identical.
|
|
||||||
|
|
||||||
**Asset conversion** with `fxconv`
|
**Asset conversion** with `fxconv`
|
||||||
|
|
||||||
|
|
171
fxgxa/dump.c
171
fxgxa/dump.c
|
@ -2,10 +2,10 @@
|
||||||
#include <string.h>
|
#include <string.h>
|
||||||
#include <endianness.h>
|
#include <endianness.h>
|
||||||
|
|
||||||
#include <fxg1a.h>
|
#include <fxgxa.h>
|
||||||
#include <g1a.h>
|
#include <g1a.h>
|
||||||
|
|
||||||
/* check(): Check validity of a g1a control or fixed field
|
/* check_g1a(): Check validity of a g1a control or fixed field
|
||||||
|
|
||||||
This function checks a single field of a g1a header (depending on the value
|
This function checks a single field of a g1a header (depending on the value
|
||||||
of @test, from 0 up) and returns:
|
of @test, from 0 up) and returns:
|
||||||
|
@ -21,11 +21,11 @@
|
||||||
@g1a G1A file being manipulated
|
@g1a G1A file being manipulated
|
||||||
@size File size
|
@size File size
|
||||||
@status Array row, at least 81 bytes free */
|
@status Array row, at least 81 bytes free */
|
||||||
static int check(int test, struct g1a const *g1a, size_t size, char *status)
|
#define m(msg, ...) sprintf(status, msg, ##__VA_ARGS__)
|
||||||
|
static int check_g1a(int test, struct g1a const *g1a, size_t size,char *status)
|
||||||
{
|
{
|
||||||
#define m(msg, ...) sprintf(status, msg, ##__VA_ARGS__)
|
|
||||||
|
|
||||||
struct header const *h = &g1a->header;
|
struct g1a_header const *h = &g1a->header;
|
||||||
uint8_t const *raw = (void *)h;
|
uint8_t const *raw = (void *)h;
|
||||||
|
|
||||||
uint16_t sum;
|
uint16_t sum;
|
||||||
|
@ -43,7 +43,7 @@ static int check(int test, struct g1a const *g1a, size_t size, char *status)
|
||||||
case 2:
|
case 2:
|
||||||
m("Sequence 1 0x0010001000 0x%02x%02x%02x%02x%02x",
|
m("Sequence 1 0x0010001000 0x%02x%02x%02x%02x%02x",
|
||||||
h->seq1[0], h->seq1[1], h->seq1[2], h->seq1[3], h->seq1[4]);
|
h->seq1[0], h->seq1[1], h->seq1[2], h->seq1[3], h->seq1[4]);
|
||||||
return strncmp((const char *)h->seq1, "\x00\x01\x00\x01\x00",
|
return memcmp((const char *)h->seq1, "\x00\x10\x00\x10\x00",
|
||||||
5) ? 1:0;
|
5) ? 1:0;
|
||||||
case 3:
|
case 3:
|
||||||
ctrl = raw[0x13] + 0x41;
|
ctrl = raw[0x13] + 0x41;
|
||||||
|
@ -61,7 +61,7 @@ static int check(int test, struct g1a const *g1a, size_t size, char *status)
|
||||||
m("Control 2 0x%02x 0x%02x", ctrl, h->control2);
|
m("Control 2 0x%02x 0x%02x", ctrl, h->control2);
|
||||||
return (h->control2 != ctrl) ? 2:0;
|
return (h->control2 != ctrl) ? 2:0;
|
||||||
case 7:
|
case 7:
|
||||||
sum = checksum(g1a, size);
|
sum = checksum_g1a(g1a, size);
|
||||||
m("Checksum 0x%02x 0x%02x", sum,
|
m("Checksum 0x%02x 0x%02x", sum,
|
||||||
be16toh(h->checksum));
|
be16toh(h->checksum));
|
||||||
return (be16toh(h->checksum) != sum) ? 2:0;
|
return (be16toh(h->checksum) != sum) ? 2:0;
|
||||||
|
@ -74,10 +74,7 @@ static int check(int test, struct g1a const *g1a, size_t size, char *status)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
/* unknown(): Print an unknown field
|
/* unknown(): Dump the contents of an unknown field */
|
||||||
@data Address of field
|
|
||||||
@offset Offset of field in header
|
|
||||||
@size Number of consecutive unknown bytes */
|
|
||||||
static void unknown(uint8_t const *data, size_t offset, size_t size)
|
static void unknown(uint8_t const *data, size_t offset, size_t size)
|
||||||
{
|
{
|
||||||
printf(" 0x%03zx %-4zd 0x", offset, size);
|
printf(" 0x%03zx %-4zd 0x", offset, size);
|
||||||
|
@ -85,32 +82,28 @@ static void unknown(uint8_t const *data, size_t offset, size_t size)
|
||||||
printf("\n");
|
printf("\n");
|
||||||
}
|
}
|
||||||
|
|
||||||
/* field(): Print a text field with limited size
|
/* field(): Print a potentially not NUL-terminated text field */
|
||||||
@field Address of text field
|
|
||||||
@size Maximum number of bytes to print */
|
|
||||||
static void field(const char *field, size_t size)
|
static void field(const char *field, size_t size)
|
||||||
{
|
{
|
||||||
for(size_t i = 0; i < size && field[i]; i++) putchar(field[i]);
|
for(size_t i = 0; i < size && field[i]; i++) putchar(field[i]);
|
||||||
printf("\n");
|
printf("\n");
|
||||||
}
|
}
|
||||||
|
|
||||||
/* dump(): Print the detailed header fields of a g1a file */
|
void dump_g1a(struct g1a const *g1a, size_t size)
|
||||||
void dump(struct g1a const *g1a, size_t size)
|
|
||||||
{
|
{
|
||||||
struct header const *header = &g1a->header;
|
struct g1a_header const *header = &g1a->header;
|
||||||
uint8_t const *raw = (void *)header;
|
uint8_t const *raw = (void *)header;
|
||||||
|
|
||||||
/* Checks for g1a files */
|
/* Checks for g1a files */
|
||||||
char status[81];
|
char status[81];
|
||||||
int ret = 0;
|
int ret=0, passed=0;
|
||||||
int passed = 0;
|
|
||||||
|
|
||||||
printf("G1A signature checks:\n\n");
|
printf("G1A signature checks:\n\n");
|
||||||
printf(" Sta. Field Expected Value\n");
|
printf(" Sta. Field Expected Value\n");
|
||||||
|
|
||||||
for(int test = 0; ret >= 0; test++)
|
for(int test = 0; ret >= 0; test++)
|
||||||
{
|
{
|
||||||
ret = check(test, g1a, size, status);
|
ret = check_g1a(test, g1a, size, status);
|
||||||
passed += !ret;
|
passed += !ret;
|
||||||
if(ret < 0) break;
|
if(ret < 0) break;
|
||||||
|
|
||||||
|
@ -142,5 +135,141 @@ void dump(struct g1a const *g1a, size_t size)
|
||||||
field(header->date, 14);
|
field(header->date, 14);
|
||||||
|
|
||||||
printf("\nProgram icon:\n\n");
|
printf("\nProgram icon:\n\n");
|
||||||
icon_print(header->icon);
|
icon_print_1(header->icon, 30, 17);
|
||||||
|
}
|
||||||
|
|
||||||
|
/* See check_g3a() for a description */
|
||||||
|
static int check_g3a(int test, struct g3a const *g3a, size_t size,char *status)
|
||||||
|
{
|
||||||
|
struct g3a_header const *h = &g3a->header;
|
||||||
|
uint8_t const *raw = (void *)h;
|
||||||
|
|
||||||
|
uint16_t sum;
|
||||||
|
uint32_t sum2;
|
||||||
|
uint8_t ctrl;
|
||||||
|
|
||||||
|
switch(test)
|
||||||
|
{
|
||||||
|
case 0:
|
||||||
|
m("Signature \"USBPower\" \"########\"");
|
||||||
|
strncpy(status + 28, h->magic, 8);
|
||||||
|
return strncmp(h->magic, "USBPower", 8) ? 2:0;
|
||||||
|
case 1:
|
||||||
|
m("MCS Type 0x2c 0x%02x", h->mcs_type);
|
||||||
|
return (h->mcs_type != 0x2c) ? 2:0;
|
||||||
|
case 2:
|
||||||
|
m("Sequence 1 0x0010001000 0x%02x%02x%02x%02x%02x",
|
||||||
|
h->seq1[0], h->seq1[1], h->seq1[2], h->seq1[3], h->seq1[4]);
|
||||||
|
return memcmp((const char *)h->seq1, "\x00\x01\x00\x01\x00",
|
||||||
|
5) ? 1:0;
|
||||||
|
case 3:
|
||||||
|
ctrl = raw[0x13] + 0x41;
|
||||||
|
m("Control 1 0x%02x 0x%02x", ctrl, h->control1);
|
||||||
|
return (h->control1 != ctrl) ? 2:0;
|
||||||
|
case 4:
|
||||||
|
m("Sequence 2 0x01 0x%02x", h->seq2);
|
||||||
|
return (h->seq2 != 0x01) ? 1:0;
|
||||||
|
case 5:
|
||||||
|
m("File size 1 %-8zu %u", size,
|
||||||
|
be32toh(h->filesize_be1));
|
||||||
|
return (be32toh(h->filesize_be1) != size) ? 2:0;
|
||||||
|
case 6:
|
||||||
|
ctrl = raw[0x13] + 0xb8;
|
||||||
|
m("Control 2 0x%02x 0x%02x", ctrl, h->control2);
|
||||||
|
return (h->control2 != ctrl) ? 2:0;
|
||||||
|
case 7:
|
||||||
|
sum = checksum_g3a(g3a, size);
|
||||||
|
m("Checksum 0x%02x 0x%02x", sum,
|
||||||
|
be16toh(h->checksum));
|
||||||
|
return (be16toh(h->checksum) != sum) ? 2:0;
|
||||||
|
case 8:
|
||||||
|
m("File size 2 %-8zu %u", size - 0x7004,
|
||||||
|
be32toh(h->filesize_be2));
|
||||||
|
return (be32toh(h->filesize_be2) != size - 0x7004) ? 2:0;
|
||||||
|
case 9:
|
||||||
|
sum2 = checksum_g3a_2(g3a, size);
|
||||||
|
m("Checksum 2 0x%08x 0x%08x", sum2,
|
||||||
|
be32toh(h->checksum_2));
|
||||||
|
return (be32toh(h->checksum_2) != sum2) ? 2:0;
|
||||||
|
case 10:
|
||||||
|
m("File size 1 %-8zu %u", size,
|
||||||
|
be32toh(h->filesize_be3));
|
||||||
|
return (be32toh(h->filesize_be3) != size) ? 2:0;
|
||||||
|
case 11:
|
||||||
|
sum2 = checksum_g3a_2(g3a, size);
|
||||||
|
uint32_t footer = be32toh(*(uint32_t *)((void *)g3a+size-4));
|
||||||
|
m("Footer 0x%08x 0x%08x", sum2, footer);
|
||||||
|
return (footer != sum2) ? 2:0;
|
||||||
|
default:
|
||||||
|
return -1;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
void dump_g3a(struct g3a const *g3a, size_t size)
|
||||||
|
{
|
||||||
|
struct g3a_header const *header = &g3a->header;
|
||||||
|
uint8_t const *raw = (void *)header;
|
||||||
|
|
||||||
|
/* Checks for g3a files */
|
||||||
|
char status[81];
|
||||||
|
int ret=0, passed=0;
|
||||||
|
|
||||||
|
printf("G3A signature checks:\n\n");
|
||||||
|
printf(" Sta. Field Expected Value\n");
|
||||||
|
|
||||||
|
for(int test = 0; ret >= 0; test++)
|
||||||
|
{
|
||||||
|
ret = check_g3a(test, g3a, size, status);
|
||||||
|
passed += !ret;
|
||||||
|
if(ret < 0) break;
|
||||||
|
|
||||||
|
printf(" %s %s\n", ret ? "FAIL" : "OK ", status);
|
||||||
|
}
|
||||||
|
|
||||||
|
printf("\nFields with unknown meanings:\n\n");
|
||||||
|
printf(" Offset Size Value\n");
|
||||||
|
|
||||||
|
unknown(raw, 0x015, 1);
|
||||||
|
unknown(raw, 0x018, 6);
|
||||||
|
unknown(raw, 0x026, 8);
|
||||||
|
unknown(raw, 0x032, 14);
|
||||||
|
unknown(raw, 0x050, 12);
|
||||||
|
unknown(raw, 0x12c, 4);
|
||||||
|
unknown(raw, 0x13a, 2);
|
||||||
|
unknown(raw, 0x14a, 38);
|
||||||
|
|
||||||
|
printf(" 0x590 2348 ");
|
||||||
|
bool is_zeros = true;
|
||||||
|
for(int i = 0; i < 2348; i++)
|
||||||
|
if(raw[0x590+i] != 0) is_zeros =1;
|
||||||
|
printf(is_zeros ? "<All zeros>\n" : "<Not shown (non-zero)>\n");
|
||||||
|
|
||||||
|
printf("\nApplication metadata:\n\n");
|
||||||
|
|
||||||
|
printf(" Program name: ");
|
||||||
|
field(header->name, 16);
|
||||||
|
printf(" Internal name: ");
|
||||||
|
field(header->internal, 11);
|
||||||
|
printf(" Version: ");
|
||||||
|
field(header->version, 10);
|
||||||
|
printf(" Build date: ");
|
||||||
|
field(header->date, 14);
|
||||||
|
printf(" Filename: ");
|
||||||
|
field(header->filename, 324);
|
||||||
|
|
||||||
|
printf("\nUnselected program icon:\n\n");
|
||||||
|
icon_print_16(header->icon_uns, 92, 64);
|
||||||
|
|
||||||
|
printf("\nSelected program icon:\n\n");
|
||||||
|
icon_print_16(header->icon_sel, 92, 64);
|
||||||
|
|
||||||
|
printf("\n");
|
||||||
|
}
|
||||||
|
|
||||||
|
void dump(void *gxa, size_t size)
|
||||||
|
{
|
||||||
|
if(is_g1a(gxa))
|
||||||
|
return dump_g1a(gxa, size);
|
||||||
|
if(is_g3a(gxa))
|
||||||
|
return dump_g3a(gxa, size);
|
||||||
}
|
}
|
||||||
|
|
143
fxgxa/edit.c
143
fxgxa/edit.c
|
@ -1,72 +1,139 @@
|
||||||
#include <fxg1a.h>
|
#include <fxgxa.h>
|
||||||
#include <stdio.h>
|
#include <stdio.h>
|
||||||
#include <string.h>
|
#include <string.h>
|
||||||
#include <endianness.h>
|
#include <endianness.h>
|
||||||
|
|
||||||
/* sign(): Sign header by filling fixed fields and checksums */
|
void sign(void *gxa, size_t size)
|
||||||
void sign(struct g1a *g1a, size_t size)
|
|
||||||
{
|
{
|
||||||
struct header *header = &g1a->header;
|
if(is_g1a(gxa)) {
|
||||||
|
struct g1a_header *header = gxa;
|
||||||
|
|
||||||
/* Fixed elements */
|
/* Fixed elements */
|
||||||
|
memcpy(header->magic, "USBPower", 8);
|
||||||
|
header->mcs_type = 0xf3;
|
||||||
|
memcpy(header->seq1, "\x00\x10\x00\x10\x00", 5);
|
||||||
|
header->seq2 = 0x01;
|
||||||
|
header->filesize_be1 = htobe32(size);
|
||||||
|
header->filesize_be2 = htobe32(size);
|
||||||
|
|
||||||
memcpy(header->magic, "USBPower", 8);
|
/* Control bytes and checksums */
|
||||||
header->mcs_type = 0xf3;
|
header->control1 = size + 0x41;
|
||||||
memcpy(header->seq1, "\x00\x10\x00\x10\x00", 5);
|
header->control2 = size + 0xb8;
|
||||||
header->seq2 = 0x01;
|
header->checksum = htobe16(checksum_g1a(gxa, size));
|
||||||
|
}
|
||||||
|
else if(is_g3a(gxa)) {
|
||||||
|
struct g3a_header *header = gxa;
|
||||||
|
|
||||||
header->filesize_be1 = htobe32(size);
|
/* Fixed elements */
|
||||||
header->filesize_be2 = htobe32(size);
|
memcpy(header->magic, "USBPower", 8);
|
||||||
|
header->mcs_type = 0x2c;
|
||||||
|
memcpy(header->seq1, "\x00\x01\x00\x01\x00", 5);
|
||||||
|
header->seq2 = 0x01;
|
||||||
|
memcpy(header->seq3, "\x01\x01", 2);
|
||||||
|
header->filesize_be1 = htobe32(size);
|
||||||
|
header->filesize_be2 = htobe32(size - 0x7000 - 4);
|
||||||
|
header->filesize_be3 = htobe32(size);
|
||||||
|
|
||||||
/* Control bytes and checksums */
|
/* Control bytes and checksums */
|
||||||
|
header->control1 = size + 0x41;
|
||||||
|
header->control2 = size + 0xb8;
|
||||||
|
header->checksum = htobe16(checksum_g3a(gxa, size));
|
||||||
|
header->checksum_2 = htobe32(checksum_g3a_2(gxa, size));
|
||||||
|
|
||||||
header->control1 = size + 0x41;
|
/* Last 4 bytes */
|
||||||
header->control2 = size + 0xb8;
|
uint32_t *footer = gxa + size - 4;
|
||||||
header->checksum = htobe16(checksum(g1a, size));
|
*footer = header->checksum_2;
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
/* edit_name(): Set application name */
|
void edit_name(void *gxa, const char *name)
|
||||||
void edit_name(struct g1a *g1a, const char *name)
|
|
||||||
{
|
{
|
||||||
memset(g1a->header.name, 0, 8);
|
if(is_g1a(gxa)) {
|
||||||
if(!name) return;
|
memset(G1A(gxa)->header.name, 0, 8);
|
||||||
|
if(!name) return;
|
||||||
|
|
||||||
for(int i = 0; name[i] && i < 8; i++)
|
for(int i = 0; name[i] && i < 8; i++)
|
||||||
g1a->header.name[i] = name[i];
|
G1A(gxa)->header.name[i] = name[i];
|
||||||
|
}
|
||||||
|
else if(is_g3a(gxa)) {
|
||||||
|
memset(G3A(gxa)->header.name, 0, 16);
|
||||||
|
if(!name) return;
|
||||||
|
|
||||||
|
for(int i = 0; name[i] && i < 16; i++)
|
||||||
|
G3A(gxa)->header.name[i] = name[i];
|
||||||
|
|
||||||
|
for(int j = 0; j < 8; j++) {
|
||||||
|
memset(G3A(gxa)->header.label[j], 0, 24);
|
||||||
|
for(int i = 0; name[i] && i < 24; i++)
|
||||||
|
G3A(gxa)->header.label[j][i] = name[i];
|
||||||
|
}
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
/* edit_internal(): Set internal name */
|
void edit_internal(void *gxa, const char *internal)
|
||||||
void edit_internal(struct g1a *g1a, const char *internal)
|
|
||||||
{
|
{
|
||||||
memset(g1a->header.internal, 0, 8);
|
char *dst = NULL;
|
||||||
|
int size = 0;
|
||||||
|
|
||||||
|
if(is_g1a(gxa)) {
|
||||||
|
dst = G1A(gxa)->header.internal;
|
||||||
|
size = 8;
|
||||||
|
}
|
||||||
|
else if(is_g3a(gxa)) {
|
||||||
|
dst = G3A(gxa)->header.internal;
|
||||||
|
size = 11;
|
||||||
|
}
|
||||||
|
|
||||||
|
memset(dst, 0, size);
|
||||||
if(!internal) return;
|
if(!internal) return;
|
||||||
|
|
||||||
for(int i = 0; internal[i] && i < 8; i++)
|
for(int i = 0; internal[i] && i < size; i++)
|
||||||
g1a->header.internal[i] = internal[i];
|
dst[i] = internal[i];
|
||||||
}
|
}
|
||||||
|
|
||||||
/* edit_version(): Set version string */
|
void edit_version(void *gxa, const char *version)
|
||||||
void edit_version(struct g1a *g1a, const char *version)
|
|
||||||
{
|
{
|
||||||
memset(g1a->header.version, 0, 10);
|
char *dst = NULL;
|
||||||
|
int size = 10;
|
||||||
|
|
||||||
|
if(is_g1a(gxa))
|
||||||
|
dst = G1A(gxa)->header.version;
|
||||||
|
else if(is_g3a(gxa))
|
||||||
|
dst = G3A(gxa)->header.version;
|
||||||
|
|
||||||
|
memset(dst, 0, size);
|
||||||
if(!version) return;
|
if(!version) return;
|
||||||
|
|
||||||
for(int i = 0; version[i] && i < 10; i++)
|
for(int i = 0; version[i] && i < size; i++)
|
||||||
g1a->header.version[i] = version[i];
|
dst[i] = version[i];
|
||||||
}
|
}
|
||||||
|
|
||||||
/* edit_date(): Set build date */
|
void edit_date(void *gxa, const char *date)
|
||||||
void edit_date(struct g1a *g1a, const char *date)
|
|
||||||
{
|
{
|
||||||
memset(g1a->header.date, 0, 14);
|
char *dst = NULL;
|
||||||
|
int size = 14;
|
||||||
|
|
||||||
|
if(is_g1a(gxa))
|
||||||
|
dst = G1A(gxa)->header.date;
|
||||||
|
else if(is_g3a(gxa))
|
||||||
|
dst = G3A(gxa)->header.date;
|
||||||
|
|
||||||
|
memset(dst, 0, size);
|
||||||
if(!date) return;
|
if(!date) return;
|
||||||
|
|
||||||
for(int i = 0; date[i] && i < 14; i++)
|
for(int i = 0; date[i] && i < size; i++)
|
||||||
g1a->header.date[i] = date[i];
|
dst[i] = date[i];
|
||||||
}
|
}
|
||||||
|
|
||||||
/* edit_icon(): Set icon from monochrome bitmap */
|
void edit_g1a_icon(struct g1a *g1a, uint8_t const *mono)
|
||||||
void edit_icon(struct g1a *g1a, uint8_t const *mono)
|
|
||||||
{
|
{
|
||||||
memcpy(g1a->header.icon, mono, 68);
|
memcpy(g1a->header.icon, mono, 68);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
void edit_g3a_icon(struct g3a *g3a, uint16_t const *icon, bool selected)
|
||||||
|
{
|
||||||
|
if(selected)
|
||||||
|
memcpy(g3a->header.icon_sel, icon, 92*64*2);
|
||||||
|
else
|
||||||
|
memcpy(g3a->header.icon_uns, icon, 92*64*2);
|
||||||
|
}
|
||||||
|
|
51
fxgxa/file.c
51
fxgxa/file.c
|
@ -7,13 +7,13 @@
|
||||||
#include <stdlib.h>
|
#include <stdlib.h>
|
||||||
#include <string.h>
|
#include <string.h>
|
||||||
|
|
||||||
#include <fxg1a.h>
|
#include <fxgxa.h>
|
||||||
|
|
||||||
/* invert_header(): Bit-invert a standard header
|
/* invert_header(): Bit-invert a standard header
|
||||||
Part of the header is stored inverted in files for obfuscation purposes. */
|
Part of the header is stored inverted in files for obfuscation purposes. */
|
||||||
static void invert_header(struct g1a *g1a)
|
static void invert_header(void *gxa)
|
||||||
{
|
{
|
||||||
uint8_t *data = (void *)&g1a->header;
|
uint8_t *data = gxa;
|
||||||
for(size_t i = 0; i < 0x20; i++) data[i] = ~data[i];
|
for(size_t i = 0; i < 0x20; i++) data[i] = ~data[i];
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -26,7 +26,7 @@ static void invert_header(struct g1a *g1a)
|
||||||
|
|
||||||
/* load(): Fully load a file into memory
|
/* load(): Fully load a file into memory
|
||||||
Allocates a buffer with @prepend leading bytes initialized to zero. */
|
Allocates a buffer with @prepend leading bytes initialized to zero. */
|
||||||
static void *load(const char *filename, size_t *size, size_t prepend)
|
static void *load(const char *filename, size_t *size, int header, int footer)
|
||||||
{
|
{
|
||||||
int fd;
|
int fd;
|
||||||
struct stat statbuf;
|
struct stat statbuf;
|
||||||
|
@ -40,13 +40,13 @@ static void *load(const char *filename, size_t *size, size_t prepend)
|
||||||
if(x > 0) fail("cannot stat %s", filename);
|
if(x > 0) fail("cannot stat %s", filename);
|
||||||
|
|
||||||
filesize = statbuf.st_size;
|
filesize = statbuf.st_size;
|
||||||
data = malloc(prepend + filesize);
|
data = malloc(header + filesize + footer);
|
||||||
if(!data) fail("cannot load %s", filename);
|
if(!data) fail("cannot load %s", filename);
|
||||||
|
|
||||||
size_t remaining = filesize;
|
size_t remaining = filesize;
|
||||||
while(remaining > 0)
|
while(remaining > 0)
|
||||||
{
|
{
|
||||||
size_t offset = prepend + filesize - remaining;
|
size_t offset = header + filesize - remaining;
|
||||||
ssize_t y = read(fd, data + offset, remaining);
|
ssize_t y = read(fd, data + offset, remaining);
|
||||||
|
|
||||||
if(y < 0) fail("cannot read from %s", filename);
|
if(y < 0) fail("cannot read from %s", filename);
|
||||||
|
@ -54,58 +54,55 @@ static void *load(const char *filename, size_t *size, size_t prepend)
|
||||||
}
|
}
|
||||||
close(fd);
|
close(fd);
|
||||||
|
|
||||||
memset(data, 0, prepend);
|
memset(data, 0, header);
|
||||||
|
memset(data + header + filesize, 0, footer);
|
||||||
|
|
||||||
if(size) *size = prepend + filesize;
|
if(size) *size = header + filesize + footer;
|
||||||
return data;
|
return data;
|
||||||
}
|
}
|
||||||
|
|
||||||
/* load_g1a(): Load a g1a file into memory */
|
void *load_gxa(const char *filename, size_t *size)
|
||||||
struct g1a *load_g1a(const char *filename, size_t *size)
|
|
||||||
{
|
{
|
||||||
struct g1a *ret = load(filename, size, 0);
|
void *ret = load(filename, size, 0, 0);
|
||||||
if(ret) invert_header(ret);
|
if(ret) invert_header(ret);
|
||||||
return ret;
|
return ret;
|
||||||
}
|
}
|
||||||
|
|
||||||
/* load_binary(): Load a binary file into memory */
|
void *load_binary(const char *filename, size_t *size, int header, int footer)
|
||||||
struct g1a *load_binary(const char *filename, size_t *size)
|
|
||||||
{
|
{
|
||||||
struct g1a *ret = load(filename, size, 0x200);
|
void *ret = load(filename, size, header, footer);
|
||||||
if(ret) memset(ret, 0xff, 0x20);
|
if(ret) invert_header(ret);
|
||||||
return ret;
|
return ret;
|
||||||
}
|
}
|
||||||
|
|
||||||
#undef fail
|
#undef fail
|
||||||
#define fail(msg, ...) { \
|
#define fail(msg, ...) { \
|
||||||
fprintf(stderr, "error: " msg ": %m\n", ##__VA_ARGS__); \
|
fprintf(stderr, "error: " msg ": %m\n", ##__VA_ARGS__); \
|
||||||
close(fd); \
|
rc = 1; \
|
||||||
invert_header(g1a); \
|
goto end; \
|
||||||
return 1; \
|
|
||||||
}
|
}
|
||||||
|
|
||||||
/* save_g1a(): Save a g1a file to disk */
|
int save_gxa(const char *filename, void *gxa, size_t size)
|
||||||
int save_g1a(const char *filename, struct g1a *g1a, size_t size)
|
|
||||||
{
|
{
|
||||||
/* Invert header before saving */
|
/* Invert header before saving */
|
||||||
invert_header(g1a);
|
invert_header(gxa);
|
||||||
|
|
||||||
|
int rc = 0;
|
||||||
int fd = creat(filename, 0644);
|
int fd = creat(filename, 0644);
|
||||||
if(fd < 0) fail("cannot open %s", filename);
|
if(fd < 0) fail("cannot open %s", filename);
|
||||||
|
|
||||||
void const *raw = g1a;
|
|
||||||
ssize_t status;
|
ssize_t status;
|
||||||
|
|
||||||
size_t written = 0;
|
size_t written = 0;
|
||||||
while(written < size)
|
while(written < size)
|
||||||
{
|
{
|
||||||
status = write(fd, raw + written, size - written);
|
status = write(fd, gxa + written, size - written);
|
||||||
if(status < 0) fail("cannot write to %s", filename);
|
if(status < 0) fail("cannot write to %s", filename);
|
||||||
written += status;
|
written += status;
|
||||||
}
|
}
|
||||||
close(fd);
|
|
||||||
|
|
||||||
|
end:
|
||||||
/* Before returning, re-invert header for further use */
|
/* Before returning, re-invert header for further use */
|
||||||
invert_header(g1a);
|
if(fd >= 0) close(fd);
|
||||||
return 0;
|
invert_header(gxa);
|
||||||
|
return rc;
|
||||||
}
|
}
|
||||||
|
|
176
fxgxa/fxg1a.h
176
fxgxa/fxg1a.h
|
@ -1,176 +0,0 @@
|
||||||
//---
|
|
||||||
// fxg1a:fxg1a - Main interfaces
|
|
||||||
//---
|
|
||||||
|
|
||||||
#ifndef FX_FXG1A
|
|
||||||
#define FX_FXG1A
|
|
||||||
|
|
||||||
#include <stdlib.h>
|
|
||||||
#include <stdint.h>
|
|
||||||
#include <stddef.h>
|
|
||||||
#include <g1a.h>
|
|
||||||
|
|
||||||
/*
|
|
||||||
** Header dumping (dump.c)
|
|
||||||
*/
|
|
||||||
|
|
||||||
/* dump(): Print the detailed header fields of a g1a file
|
|
||||||
This function takes as argument the full file loaded into memory and the
|
|
||||||
size of the file. It does various printing to stdout as main job.
|
|
||||||
|
|
||||||
@g1a Full file data
|
|
||||||
@size Size of g1a file */
|
|
||||||
void dump(struct g1a const *g1a, size_t size);
|
|
||||||
|
|
||||||
|
|
||||||
/*
|
|
||||||
** Header manipulation (edit.c)
|
|
||||||
*/
|
|
||||||
|
|
||||||
/* sign(): Sign header by filling fixed fields and checksums
|
|
||||||
This function fills the fixed fields and various checksums of a g1a file. To
|
|
||||||
do this it accesses some of the binary data. To set the user-customizable
|
|
||||||
field, use the edit_*() functions. (The value of the customizable fields
|
|
||||||
does not influence the checksums so it's okay to not call this function
|
|
||||||
afterwards.)
|
|
||||||
|
|
||||||
@g1a Header to sign
|
|
||||||
@size Size of raw file data */
|
|
||||||
void sign(struct g1a *g1a, size_t size);
|
|
||||||
|
|
||||||
/* edit_*(): Set various fields of a g1a header */
|
|
||||||
|
|
||||||
void edit_name (struct g1a *g1a, const char *name);
|
|
||||||
void edit_internal (struct g1a *g1a, const char *internal);
|
|
||||||
void edit_version (struct g1a *g1a, const char *version);
|
|
||||||
void edit_date (struct g1a *g1a, const char *date);
|
|
||||||
|
|
||||||
/* edit_icon(): Set monochrome icon of a g1a header
|
|
||||||
The icon parameter must be loaded in 1-bit bitmap format. */
|
|
||||||
void edit_icon(struct g1a *header, uint8_t const *mono);
|
|
||||||
|
|
||||||
|
|
||||||
/*
|
|
||||||
** Utility functions (util.c)
|
|
||||||
*/
|
|
||||||
|
|
||||||
/* checksum(): Sum of 8 big-endian shorts at 0x300
|
|
||||||
Computes the third checksum by summing bytes from the data part of the file.
|
|
||||||
|
|
||||||
@g1a Add-in file whose checksum is requested
|
|
||||||
@size Size of file */
|
|
||||||
uint16_t checksum(struct g1a const *g1a, size_t size);
|
|
||||||
|
|
||||||
/* default_output(): Calculate default output file name
|
|
||||||
This function computes a default file name by replacing the extension of
|
|
||||||
@name (if it exists) or adding one. The extension is specified as a suffix,
|
|
||||||
usually in the form ".ext".
|
|
||||||
|
|
||||||
The resulting string might be as long as the length of @name plus that of
|
|
||||||
@suffix (plus one NUL byte); the provided buffer must point to a suitably-
|
|
||||||
large allocated space.
|
|
||||||
|
|
||||||
@name Input file name
|
|
||||||
@suffix Suffix to add or replace @name's extension with
|
|
||||||
@output Output file name */
|
|
||||||
void default_output(const char *name, const char *suffix, char *output);
|
|
||||||
|
|
||||||
/* default_internal(): Calculate default internal name
|
|
||||||
This function determines a default internal name, which is '@' followed by
|
|
||||||
at most 7 uppercase letters taken from the application name.
|
|
||||||
|
|
||||||
@name Application name
|
|
||||||
@output Internal name string (9 bytes) */
|
|
||||||
void default_internal(const char *name, char *output);
|
|
||||||
|
|
||||||
|
|
||||||
/*
|
|
||||||
** File manipulation (file.c)
|
|
||||||
*/
|
|
||||||
|
|
||||||
/* load_g1a(): Load a g1a file into memory
|
|
||||||
This function loads @filename into a dynamically-allocated buffer and
|
|
||||||
returns the address of that buffer; it must be free()'d after use. When
|
|
||||||
loading the file, if @size is not NULL, it receives the size of the file.
|
|
||||||
On error, load() prints a message an stderr and returns NULL. The header
|
|
||||||
is inverted before this function returns.
|
|
||||||
|
|
||||||
@filename File to load
|
|
||||||
@size If non-NULL, receives the file size
|
|
||||||
Returns a pointer to a buffer with loaded data, or NULL on error. */
|
|
||||||
struct g1a *load_g1a(const char *filename, size_t *size);
|
|
||||||
|
|
||||||
/* load_binary(): Load a binary file into memory
|
|
||||||
This function operates like load_g1a() but reserves space for an empty
|
|
||||||
header. The header is initialized with all zeros.
|
|
||||||
|
|
||||||
@filename File to load
|
|
||||||
@size If non-NULL, receives the file size
|
|
||||||
Returns a pointer to a buffer with loaded data, or NULL on error. */
|
|
||||||
struct g1a *load_binary(const char *filename, size_t *size);
|
|
||||||
|
|
||||||
/* save_g1a(): Save a g1a file to disk
|
|
||||||
This functions creates @filename, then writes a g1a header and a chunk of
|
|
||||||
raw data to it. Since it temporarily inverts the header to comply with
|
|
||||||
Casio's obfuscated format, it needs write access to @g1a. Returns non-zero
|
|
||||||
on error.
|
|
||||||
|
|
||||||
@filename File to write (it will be overridden if it exists)
|
|
||||||
@g1a G1A data to write
|
|
||||||
@size Size of data
|
|
||||||
Returns zero on success and a nonzero error code otherwise. */
|
|
||||||
int save_g1a(const char *filename, struct g1a *g1a, size_t size);
|
|
||||||
|
|
||||||
|
|
||||||
/*
|
|
||||||
** Icon management (icon.c)
|
|
||||||
*/
|
|
||||||
|
|
||||||
/* icon_print(): Show a monochrome 30*17 icon on stdout
|
|
||||||
The buffer should point to a 68-byte array. */
|
|
||||||
void icon_print(uint8_t const *icon);
|
|
||||||
|
|
||||||
/* icon_load(): Load a monochrome PNG image into an array
|
|
||||||
This function loads a PNG image into a 1-bit buffer; each row is represented
|
|
||||||
by a fixed number of bytes, each byte being 8 pixels. Rows are loaded from
|
|
||||||
top to bottom, and from left to right.
|
|
||||||
|
|
||||||
If the image is not a PNG image or a reading error occurs, this functions
|
|
||||||
prints an error message on stderr and returns NULL.
|
|
||||||
|
|
||||||
@filename PNG file to load
|
|
||||||
@width If non-NULL, receives image width
|
|
||||||
@height If non-NULL, receives image height
|
|
||||||
Returns a pointer to a free()able buffer with loaded data, NULL on error. */
|
|
||||||
uint8_t *icon_load(const char *filename, size_t *width, size_t *height);
|
|
||||||
|
|
||||||
/* icon_save(): Save an 8-bit array to a PNG image
|
|
||||||
Assumes 8-bit GRAY format.
|
|
||||||
|
|
||||||
@filename Target filename
|
|
||||||
@input An 8-bit GRAY image
|
|
||||||
@width Width of input, should be equal to stride
|
|
||||||
@height Height of input
|
|
||||||
Returns non-zero on error. */
|
|
||||||
int icon_save(const char *filename, uint8_t *input, size_t width,
|
|
||||||
size_t height);
|
|
||||||
|
|
||||||
/* icon_conv_8to1(): Convert an 8-bit icon to 1-bit
|
|
||||||
The returned 1-bit icon is always of size 30*17, if the input size does not
|
|
||||||
match it is adjusted.
|
|
||||||
|
|
||||||
@input 8-bi data
|
|
||||||
@width Width of input image
|
|
||||||
@height Height of input image
|
|
||||||
Returns a free()able buffer with a 1-bit icon on success, NULL on error. */
|
|
||||||
uint8_t *icon_conv_8to1(uint8_t const *input, size_t width, size_t height);
|
|
||||||
|
|
||||||
/* icon_conv_1to8(): Convert an 1-bit icon to 8-bit
|
|
||||||
Input 1-bit is assumed to be 30*17 in size, this function returns an 8-bit
|
|
||||||
buffer with the same dimensions.
|
|
||||||
|
|
||||||
@mono Input monochrome icon (from a g1a header, for instance)
|
|
||||||
Returns a free()able buffer, or NULL on error. */
|
|
||||||
uint8_t *icon_conv_1to8(uint8_t const *mono);
|
|
||||||
|
|
||||||
#endif /* FX_FXG1A */
|
|
198
fxgxa/fxgxa.h
Normal file
198
fxgxa/fxgxa.h
Normal file
|
@ -0,0 +1,198 @@
|
||||||
|
//---
|
||||||
|
// fxgxa:fxgxa - Main interfaces
|
||||||
|
//---
|
||||||
|
|
||||||
|
#ifndef FX_FXGXA
|
||||||
|
#define FX_FXGXA
|
||||||
|
|
||||||
|
#include <stdlib.h>
|
||||||
|
#include <stdint.h>
|
||||||
|
#include <stdbool.h>
|
||||||
|
#include <g1a.h>
|
||||||
|
#include <g3a.h>
|
||||||
|
|
||||||
|
/* In this file, many functions accept either a [struct g1a] or a [struct g3a]
|
||||||
|
as their argument. In this case, the argument is noted [void *gxa]. */
|
||||||
|
|
||||||
|
|
||||||
|
/*
|
||||||
|
** Header dumping (dump.c)
|
||||||
|
*/
|
||||||
|
|
||||||
|
/* dump(): Print the detailed header fields of a file
|
||||||
|
This function takes as argument the full file loaded into memory and the
|
||||||
|
size of the file. It does various printing to stdout as main job.
|
||||||
|
|
||||||
|
@gxa Full file data
|
||||||
|
@size Size of the file */
|
||||||
|
void dump(void *gxa, size_t size);
|
||||||
|
|
||||||
|
|
||||||
|
/*
|
||||||
|
** Header manipulation (edit.c)
|
||||||
|
*/
|
||||||
|
|
||||||
|
/* sign(): Sign header by filling fixed fields and checksums
|
||||||
|
This function fills the fixed fields and various checksums of a g1a file. To
|
||||||
|
do this it accesses some of the binary data. To set the user-customizable
|
||||||
|
field, use the edit_*() functions. (The value of the customizable fields
|
||||||
|
does not influence the checksums so it's okay to not call this function
|
||||||
|
afterwards.)
|
||||||
|
|
||||||
|
@g1a Header to sign
|
||||||
|
@size Size of raw file data */
|
||||||
|
void sign(void *gxa, size_t size);
|
||||||
|
|
||||||
|
/* edit_*(): Set various fields */
|
||||||
|
|
||||||
|
void edit_name (void *gxa, const char *name);
|
||||||
|
void edit_internal (void *gxa, const char *internal);
|
||||||
|
void edit_version (void *gxa, const char *version);
|
||||||
|
void edit_date (void *gxa, const char *date);
|
||||||
|
|
||||||
|
/* edit_g1a_icon(): Set monochrome icon of a g1a header
|
||||||
|
The icon parameter must be loaded in 1-bit bitmap format. */
|
||||||
|
void edit_g1a_icon(struct g1a *header, uint8_t const *mono);
|
||||||
|
|
||||||
|
/* edit_g3a_icon(): Set one of the color icons of a g3a header
|
||||||
|
The icon must be loaded in RGB565 16-bit format. */
|
||||||
|
void edit_g3a_icon(struct g3a *header, uint16_t const *icon, bool selected);
|
||||||
|
|
||||||
|
|
||||||
|
/*
|
||||||
|
** Utility functions (util.c)
|
||||||
|
*/
|
||||||
|
|
||||||
|
/* is_g1a(), is_g3a(): Check file type */
|
||||||
|
bool is_g1a(void *gxa);
|
||||||
|
bool is_g3a(void *gxa);
|
||||||
|
|
||||||
|
#define G1A(gxa) ((struct g1a *)(gxa))
|
||||||
|
#define G3A(gxa) ((struct g3a *)(gxa))
|
||||||
|
|
||||||
|
/* word_sum(): Sum of big-endian words at some offset in a file
|
||||||
|
This is used for the third checksum.
|
||||||
|
|
||||||
|
@gxa File whose checksum is requested
|
||||||
|
@offset File offset to start counting
|
||||||
|
@words Number of words to read
|
||||||
|
@size Size of file (in case the region overflows) */
|
||||||
|
uint16_t word_sum(void const *gxa, size_t offset, int words, size_t size);
|
||||||
|
|
||||||
|
/* checksum_g1a(): Word sum of 8 big-endian shorts at 0x300 */
|
||||||
|
uint16_t checksum_g1a(struct g1a const *g1a, size_t size);
|
||||||
|
|
||||||
|
/* checksum_g3a(): Word sum of 8 big-endian shorts at 0x7100 */
|
||||||
|
uint16_t checksum_g3a(struct g3a const *g3a, size_t size);
|
||||||
|
|
||||||
|
/* checksum_g3a_2(): Sum of ranges 0...0x20 + 0x24...EOF-4 (in-out) */
|
||||||
|
uint32_t checksum_g3a_2(struct g3a const *g3a, size_t size);
|
||||||
|
|
||||||
|
/* default_output(): Calculate default output file name
|
||||||
|
This function computes a default file name by replacing the extension of
|
||||||
|
@name (if it exists) or adding one. The extension is specified as a suffix,
|
||||||
|
usually in the form ".ext".
|
||||||
|
|
||||||
|
The resulting string might be as long as the length of @name plus that of
|
||||||
|
@suffix (plus one NUL byte); the provided buffer must point to a suitably-
|
||||||
|
large allocated space.
|
||||||
|
|
||||||
|
@name Input file name
|
||||||
|
@suffix Suffix to add or replace @name's extension with
|
||||||
|
@output Output file name */
|
||||||
|
void default_output(const char *name, const char *suffix, char *output);
|
||||||
|
|
||||||
|
/* default_internal(): Calculate default internal name
|
||||||
|
This function determines a default internal name, which is '@' followed by
|
||||||
|
at most size-1 uppercase letters taken from the application name. The buffer
|
||||||
|
must have [size+1] bytes reserved.
|
||||||
|
|
||||||
|
@name Application name
|
||||||
|
@output Internal name string (9 bytes) */
|
||||||
|
void default_internal(const char *name, char *output, size_t size);
|
||||||
|
|
||||||
|
|
||||||
|
/*
|
||||||
|
** File manipulation (file.c)
|
||||||
|
*/
|
||||||
|
|
||||||
|
/* load_gxa(): Load a g1a/g3a file into memory
|
||||||
|
This function loads @filename into a dynamically-allocated buffer and
|
||||||
|
returns the address of that buffer; it must be free()'d after use. When
|
||||||
|
loading the file, if @size is not NULL, it receives the size of the file.
|
||||||
|
On error, load() prints a message an stderr and returns NULL. The header
|
||||||
|
is inverted before this function returns.
|
||||||
|
|
||||||
|
@filename File to load
|
||||||
|
@size If non-NULL, receives the file size
|
||||||
|
Returns a pointer to a buffer with loaded data, or NULL on error. */
|
||||||
|
void *load_gxa(const char *filename, size_t *size);
|
||||||
|
|
||||||
|
/* load_binary(): Load a binary file into memory
|
||||||
|
This function operates like load_gxa() but reserves space for an empty
|
||||||
|
header. The header is initialized with all zeros.
|
||||||
|
|
||||||
|
@filename File to load
|
||||||
|
@size If non-NULL, receives the file size (with header/footer)
|
||||||
|
@header_size Extra room to add as header
|
||||||
|
@footer_size Extra room to add as footer
|
||||||
|
Returns a pointer to a buffer with loaded data, or NULL on error. */
|
||||||
|
void *load_binary(const char *filename, size_t *size, int header_size,
|
||||||
|
int footer_size);
|
||||||
|
|
||||||
|
/* save_gxa(): Save a g1a/g3a file to disk
|
||||||
|
This functions creates @filename, then writes a header and a chunk of raw
|
||||||
|
data to it. Since it temporarily inverts the header to comply with Casio's
|
||||||
|
obfuscated format, it needs write access to @g1a. Returns non-zero on error.
|
||||||
|
|
||||||
|
@filename File to write (it will be overridden if it exists)
|
||||||
|
@gxa G1A/G3A data to write
|
||||||
|
@size Size of data
|
||||||
|
Returns zero on success and a nonzero error code otherwise. */
|
||||||
|
int save_gxa(const char *filename, void *gxa, size_t size);
|
||||||
|
|
||||||
|
|
||||||
|
/*
|
||||||
|
** Icon management (icon.c)
|
||||||
|
*/
|
||||||
|
|
||||||
|
/* icon_load(): Load a PNG image into a RGB888 array
|
||||||
|
This function loads a PNG image into an RGB888 buffer. If the image is not a
|
||||||
|
PNG image or a reading error occurs, this functions prints an error message
|
||||||
|
on stderr and returns NULL.
|
||||||
|
|
||||||
|
@filename PNG file to load
|
||||||
|
@width If non-NULL, receives image width
|
||||||
|
@height If non-NULL, receives image height
|
||||||
|
Returns a pointer to a free()able buffer with loaded data, NULL on error. */
|
||||||
|
uint8_t *icon_load(const char *filename, int *width, int *height);
|
||||||
|
|
||||||
|
/* icon_save(): Save an RGB888 array to a PNG image.
|
||||||
|
@filename Target filename
|
||||||
|
@input A 24-bit RGB888 array
|
||||||
|
@width Width of input (should have no gaps)
|
||||||
|
@height Height of input
|
||||||
|
Returns non-zero on error. */
|
||||||
|
int icon_save(const char *filename, uint8_t *input, int width, int height);
|
||||||
|
|
||||||
|
/* icon_conv_24to1(): Convert RGB888 to 1-bit monochrome array
|
||||||
|
Rows are byte-padded (ie. width is rounded up to a multiple of 8). Returns
|
||||||
|
newly-allocated memory, NULL on error. */
|
||||||
|
uint8_t *icon_conv_24to1(uint8_t const *rgb24, int width, int height);
|
||||||
|
|
||||||
|
/* icon_conv_1to24(): Convert a 1-bit monochrome array to RGB888 */
|
||||||
|
uint8_t *icon_conv_1to24(uint8_t const *mono, int width, int height);
|
||||||
|
|
||||||
|
/* icon_conv_24to16(): Convert RGB888 to big-endian RGB565 */
|
||||||
|
uint16_t *icon_conv_24to16(uint8_t const *rgb24, int width, int height);
|
||||||
|
|
||||||
|
/* icon_conv_16to24(): Convert big-endian RGB565 to RGB888 */
|
||||||
|
uint8_t *icon_conv_16to24(uint16_t const *rgb16be, int width, int height);
|
||||||
|
|
||||||
|
/* icon_print_1(): Show a 1-bit image on stdout (ASCII art) */
|
||||||
|
void icon_print_1(uint8_t const *mono, int width, int height);
|
||||||
|
|
||||||
|
/* icon_print_16(): Show a 16-bit image on stdout (RGB ANSI escape codes) */
|
||||||
|
void icon_print_16(uint16_t const *rgb16be, int width, int height);
|
||||||
|
|
||||||
|
#endif /* FX_FXGXA */
|
73
fxgxa/g1a.h
73
fxgxa/g1a.h
|
@ -1,5 +1,5 @@
|
||||||
//---
|
//---
|
||||||
// fxg1a:g1a - Add-in header for Casio's G1A format
|
// fxgxa:g1a - Add-in header for Casio's G1A format
|
||||||
//---
|
//---
|
||||||
|
|
||||||
#ifndef FX_G1A
|
#ifndef FX_G1A
|
||||||
|
@ -8,52 +8,53 @@
|
||||||
#include <stdint.h>
|
#include <stdint.h>
|
||||||
|
|
||||||
/* TODO: eStrips are not supported yet */
|
/* TODO: eStrips are not supported yet */
|
||||||
struct estrip
|
struct g1a_estrip
|
||||||
{
|
{
|
||||||
uint8_t data[80];
|
uint8_t data[80];
|
||||||
};
|
};
|
||||||
|
|
||||||
/* G1A file header with 0x200 bytes. When output to a file the standard part
|
/* G1A file header with 0x200 bytes. When output to a file the standard part
|
||||||
(first 0x20 bytes) of this header is bit-inverted, but the non-inverted
|
(first 0x20 bytes) of this header is bit-inverted, but the non-inverted
|
||||||
version makes a lot more sens so we'll be using it. */
|
version makes a lot more sens so we'll be using it. */
|
||||||
struct header
|
struct g1a_header
|
||||||
{ /* Offset Size Value */
|
{ /* Offset Size Value */
|
||||||
char magic[8]; /* 0x000 8 "USBPower" */
|
char magic[8]; /* 0x000 8 "USBPower" */
|
||||||
uint8_t mcs_type; /* 0x008 1 0xf3 (AddIn) */
|
uint8_t mcs_type; /* 0x008 1 0xf3 (AddIn) */
|
||||||
uint8_t seq1[5]; /* 0x009 5 0x0010001000 */
|
uint8_t seq1[5]; /* 0x009 5 0x0010001000 */
|
||||||
uint8_t control1; /* 0x00e 1 *0x13 + 0x41 */
|
uint8_t control1; /* 0x00e 1 *0x13 + 0x41 */
|
||||||
uint8_t seq2; /* 0x00f 1 0x01 */
|
uint8_t seq2; /* 0x00f 1 0x01 */
|
||||||
uint32_t filesize_be1; /* 0x010 4 File size, big endian */
|
uint32_t filesize_be1; /* 0x010 4 File size, big endian */
|
||||||
uint8_t control2; /* 0x014 1 *0x13 + 0xb8 */
|
uint8_t control2; /* 0x014 1 *0x13 + 0xb8 */
|
||||||
uint8_t _1; /* 0x015 1 ??? */
|
uint8_t _1; /* 0x015 1 ??? */
|
||||||
uint16_t checksum; /* 0x016 2 BE sum of 8 shorts at 0x300 */
|
uint16_t checksum; /* 0x016 2 BE sum of 8 shorts at 0x300 */
|
||||||
uint8_t _2[6]; /* 0x018 6 ??? */
|
uint8_t _2[6]; /* 0x018 6 ??? */
|
||||||
uint16_t mcs_objects; /* 0x01e 2 MCS-only, unused */
|
uint16_t mcs_objects; /* 0x01e 2 MCS-only, unused */
|
||||||
char internal[8]; /* 0x020 8 Internal app name with '@' */
|
|
||||||
uint8_t _3[3]; /* 0x028 3 ??? */
|
char internal[8]; /* 0x020 8 Internal app name with '@' */
|
||||||
uint8_t estrips; /* 0x02b 1 Number of estrips (0..4) */
|
uint8_t _3[3]; /* 0x028 3 ??? */
|
||||||
uint8_t _4[4]; /* 0x02c 4 ??? */
|
uint8_t estrips; /* 0x02b 1 Number of estrips (0..4) */
|
||||||
char version[10]; /* 0x030 10 Version "MM.mm.pppp" */
|
uint8_t _4[4]; /* 0x02c 4 ??? */
|
||||||
uint8_t _5[2]; /* 0x03a 2 ??? */
|
char version[10]; /* 0x030 10 Version "MM.mm.pppp" */
|
||||||
char date[14]; /* 0x03c 14 Build date "yyyy.MMdd.hhmm" */
|
uint8_t _5[2]; /* 0x03a 2 ??? */
|
||||||
uint8_t _6[2]; /* 0x04a 2 ??? */
|
char date[14]; /* 0x03c 14 Build date "yyyy.MMdd.hhmm" */
|
||||||
uint8_t icon[68]; /* 0x04c 68 30*17 monochrome icon */
|
uint8_t _6[2]; /* 0x04a 2 ??? */
|
||||||
struct estrip estrip1; /* 0x090 80 eStrip 1 */
|
uint8_t icon[68]; /* 0x04c 68 30*17 monochrome icon */
|
||||||
struct estrip estrip2; /* 0x0e0 80 eStrip 2 */
|
struct g1a_estrip estrip1; /* 0x090 80 eStrip 1 */
|
||||||
struct estrip estrip3; /* 0x130 80 eStrip 3 */
|
struct g1a_estrip estrip2; /* 0x0e0 80 eStrip 2 */
|
||||||
struct estrip estrip4; /* 0x180 80 eStrip 4 */
|
struct g1a_estrip estrip3; /* 0x130 80 eStrip 3 */
|
||||||
uint8_t _7[4]; /* 0x1d0 4 ??? */
|
struct g1a_estrip estrip4; /* 0x180 80 eStrip 4 */
|
||||||
char name[8]; /* 0x1d4 8 Add-in name */
|
uint8_t _7[4]; /* 0x1d0 4 ??? */
|
||||||
uint8_t _8[20]; /* 0x1dc 20 ??? */
|
char name[8]; /* 0x1d4 8 Add-in name */
|
||||||
uint32_t filesize_be2; /* 0x1f0 4 File size, big endian */
|
uint8_t _8[20]; /* 0x1dc 20 ??? */
|
||||||
uint8_t _9[12]; /* 0x1f4 12 ??? */
|
uint32_t filesize_be2; /* 0x1f0 4 File size, big endian */
|
||||||
|
uint8_t _9[12]; /* 0x1f4 12 ??? */
|
||||||
};
|
};
|
||||||
|
|
||||||
/* A full g1a file, suitable for use with pointers */
|
/* A full g1a file, suitable for use with pointers */
|
||||||
struct g1a
|
struct g1a
|
||||||
{
|
{
|
||||||
struct header header;
|
struct g1a_header header;
|
||||||
uint8_t code[];
|
uint8_t code[];
|
||||||
};
|
};
|
||||||
|
|
||||||
#endif /* FX_G1A */
|
#endif /* FX_G1A */
|
||||||
|
|
58
fxgxa/g3a.h
Normal file
58
fxgxa/g3a.h
Normal file
|
@ -0,0 +1,58 @@
|
||||||
|
//---
|
||||||
|
// fxgxa:g3a - Add-in header for Casio's G3A format
|
||||||
|
//---
|
||||||
|
|
||||||
|
#ifndef FX_G3A
|
||||||
|
#define FX_G3A
|
||||||
|
|
||||||
|
#include <stdint.h>
|
||||||
|
|
||||||
|
/* G3A file header with 0x7000 bytes. When output to a file the standard part
|
||||||
|
(first 0x20 bytes) of this header is bit-inverted. */
|
||||||
|
struct g3a_header
|
||||||
|
{ /* Offset|Size|Value */
|
||||||
|
char magic[8]; /* 0x000 8 "USBPower" */
|
||||||
|
uint8_t mcs_type; /* 0x008 1 0x2c (AddIn) */
|
||||||
|
uint8_t seq1[5]; /* 0x009 5 0x0001000100 */
|
||||||
|
uint8_t control1; /* 0x00e 1 *0x13 + 0x41 */
|
||||||
|
uint8_t seq2; /* 0x00f 1 0x01 */
|
||||||
|
uint32_t filesize_be1; /* 0x010 4 File size, big endian */
|
||||||
|
uint8_t control2; /* 0x014 1 *0x13 + 0xb8 */
|
||||||
|
uint8_t _1; /* 0x015 1 ??? */
|
||||||
|
uint16_t checksum; /* 0x016 2 BE sum of 8 shorts at 7100 */
|
||||||
|
uint8_t _2[6]; /* 0x018 6 ??? */
|
||||||
|
uint16_t mcs_objects; /* 0x01e 2 MCS-only, unused */
|
||||||
|
|
||||||
|
uint32_t checksum_2; /* 0x020 4 Checksum: 0..1f+24..EOF-4 */
|
||||||
|
uint8_t seq3[2]; /* 0x024 2 0x0101 */
|
||||||
|
uint8_t _3[8]; /* 0x026 8 ??? */
|
||||||
|
uint32_t filesize_be2; /* 0x02e 4 Filesize - 0x7000 - 4 */
|
||||||
|
uint8_t _4[14]; /* 0x032 14 ??? */
|
||||||
|
char name[16]; /* 0x040 16 Add-in name + NUL */
|
||||||
|
uint8_t _5[12]; /* 0x050 12 ??? */
|
||||||
|
uint32_t filesize_be3; /* 0x05c 4 Filesize */
|
||||||
|
char internal[11]; /* 0x060 11 Internal name with '@' */
|
||||||
|
char label[8][24]; /* 0x06b 192 Language labels */
|
||||||
|
uint8_t allow_estrip; /* 0x12b 1 Allow use as eAct strip? */
|
||||||
|
uint8_t _6[4]; /* 0x12c 4 ??? */
|
||||||
|
char version[10]; /* 0x130 10 Version "MM.mm.pppp" */
|
||||||
|
uint8_t _7[2]; /* 0x13a 2 ??? */
|
||||||
|
char date[14]; /* 0x13c 14 Date "yyyy.MMdd.hhmm" */
|
||||||
|
uint8_t _8[38]; /* 0x14a 38 ??? */
|
||||||
|
char estrip_label[8][36]; /* 0x170 288 eStrip language labels */
|
||||||
|
uint8_t eact_icon[0x300]; /* 0x290 768 eAct icon (64x24) */
|
||||||
|
uint8_t _9[0x92c]; /* 0x590 2348 ??? */
|
||||||
|
char filename[324]; /* 0xebc 324 Filename "X.g3a" */
|
||||||
|
uint16_t icon_uns[0x1800]; /* 0x1000 5376 Unselected icon */
|
||||||
|
uint16_t icon_sel[0x1800]; /* 0x4000 5376 Selected icon */
|
||||||
|
|
||||||
|
} __attribute__((packed, aligned(4)));
|
||||||
|
|
||||||
|
/* A full g3a file, suitable for use with pointers */
|
||||||
|
struct g3a
|
||||||
|
{
|
||||||
|
struct g3a_header header;
|
||||||
|
uint8_t code[];
|
||||||
|
};
|
||||||
|
|
||||||
|
#endif /* FX_G3A */
|
247
fxgxa/icon.c
247
fxgxa/icon.c
|
@ -1,132 +1,173 @@
|
||||||
#include <stdio.h>
|
#include <stdio.h>
|
||||||
#include <string.h>
|
#include <string.h>
|
||||||
|
#include <fxgxa.h>
|
||||||
#include <fxg1a.h>
|
|
||||||
#include <png.h>
|
#include <png.h>
|
||||||
|
|
||||||
/* icon_print(): Show a monochrome 30*17 icon on stdout */
|
uint8_t *icon_load(const char *filename, int *width, int *height)
|
||||||
void icon_print(uint8_t const *icon)
|
|
||||||
{
|
{
|
||||||
for(int y = 0; y < 17; y++)
|
png_image img;
|
||||||
{
|
memset(&img, 0, sizeof img);
|
||||||
for(int x = 0; x < 30; x++)
|
img.opaque = NULL;
|
||||||
{
|
img.version = PNG_IMAGE_VERSION;
|
||||||
int v = icon[(y << 2) + (x >> 3)] & (0x80 >> (x & 7));
|
|
||||||
putchar(v ? '#' : ' ');
|
|
||||||
putchar(v ? '#' : ' ');
|
|
||||||
}
|
|
||||||
|
|
||||||
putchar('\n');
|
void *buffer = NULL;
|
||||||
}
|
|
||||||
|
png_image_begin_read_from_file(&img, filename);
|
||||||
|
if(img.warning_or_error) {
|
||||||
|
fprintf(stderr, "libpng %s: %s\n", img.warning_or_error == 1
|
||||||
|
? "warning": "error", img.message);
|
||||||
|
if(img.warning_or_error > 1) goto err;
|
||||||
|
}
|
||||||
|
|
||||||
|
img.format = PNG_FORMAT_RGB;
|
||||||
|
|
||||||
|
buffer = calloc(img.width * img.height, 3);
|
||||||
|
if(!buffer) {
|
||||||
|
fprintf(stderr, "error: cannot read %s: %m\n", filename);
|
||||||
|
goto err;
|
||||||
|
}
|
||||||
|
|
||||||
|
png_image_finish_read(&img, NULL, buffer, 0, NULL);
|
||||||
|
if(img.warning_or_error) {
|
||||||
|
fprintf(stderr, "libpng %s: %s\n", img.warning_or_error == 1
|
||||||
|
? "warning": "error", img.message);
|
||||||
|
if(img.warning_or_error > 1) goto err;
|
||||||
|
}
|
||||||
|
if(width) *width = img.width;
|
||||||
|
if(height) *height = img.height;
|
||||||
|
|
||||||
|
png_image_free(&img);
|
||||||
|
return buffer;
|
||||||
|
|
||||||
|
err:
|
||||||
|
png_image_free(&img);
|
||||||
|
free(buffer);
|
||||||
|
return NULL;
|
||||||
}
|
}
|
||||||
|
|
||||||
/* icon_load(): Load a monochrome PNG image into an array */
|
int icon_save(const char *filename, uint8_t *input, int width, int height)
|
||||||
uint8_t *icon_load(const char *filename, size_t *width, size_t *height)
|
|
||||||
{
|
{
|
||||||
png_image img;
|
png_image img;
|
||||||
memset(&img, 0, sizeof img);
|
memset(&img, 0, sizeof img);
|
||||||
img.opaque = NULL;
|
|
||||||
img.version = PNG_IMAGE_VERSION;
|
|
||||||
|
|
||||||
png_image_begin_read_from_file(&img, filename);
|
img.version = PNG_IMAGE_VERSION;
|
||||||
if(img.warning_or_error)
|
img.width = width;
|
||||||
{
|
img.height = height;
|
||||||
fprintf(stderr, "libpng %s: %s\n", img.warning_or_error == 1
|
img.format = PNG_FORMAT_RGB;
|
||||||
? "warning": "error", img.message);
|
|
||||||
if(img.warning_or_error > 1)
|
|
||||||
{
|
|
||||||
png_image_free(&img);
|
|
||||||
return NULL;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
img.format = PNG_FORMAT_GRAY;
|
png_image_write_to_file(&img, filename, 0, input, 0, NULL);
|
||||||
|
png_image_free(&img);
|
||||||
|
|
||||||
void *buffer = calloc(img.width * img.height, 1);
|
if(img.warning_or_error) {
|
||||||
if(!buffer)
|
fprintf(stderr, "libpng %s: %s\n", img.warning_or_error == 1
|
||||||
{
|
? "warning": "error", img.message);
|
||||||
fprintf(stderr, "error: cannot read %s: %m\n", filename);
|
if(img.warning_or_error > 1) return 1;
|
||||||
png_image_free(&img);
|
}
|
||||||
return NULL;
|
|
||||||
}
|
|
||||||
|
|
||||||
png_image_finish_read(&img, NULL, buffer, img.width, NULL);
|
return 0;
|
||||||
if(width) *width = img.width;
|
|
||||||
if(height) *height = img.height;
|
|
||||||
|
|
||||||
png_image_free(&img);
|
|
||||||
return buffer;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
/* icon_save(): Save an 8-bit array to a PNG image */
|
uint8_t *icon_conv_24to1(uint8_t const *rgb24, int width, int height)
|
||||||
int icon_save(const char *filename, uint8_t *input, size_t width,
|
|
||||||
size_t height)
|
|
||||||
{
|
{
|
||||||
png_image img;
|
int bytes_per_row = (width + 7) >> 3;
|
||||||
memset(&img, 0, sizeof img);
|
uint8_t *mono = calloc(bytes_per_row * height, 1);
|
||||||
|
if(!mono) return NULL;
|
||||||
|
|
||||||
img.version = PNG_IMAGE_VERSION;
|
for(int y = 0; y < height; y++)
|
||||||
img.width = width;
|
for(int x = 0; x < width; x++) {
|
||||||
img.height = height;
|
int in = 3 * (y * width + x);
|
||||||
img.format = PNG_FORMAT_GRAY;
|
int out = (y * bytes_per_row) + (x >> 3);
|
||||||
|
int color = (rgb24[in] + rgb24[in+1] + rgb24[in+2]) < 384;
|
||||||
|
mono[out] |= color << (~x & 7);
|
||||||
|
}
|
||||||
|
|
||||||
png_image_write_to_file(&img, filename, 0, input, 0, NULL);
|
return mono;
|
||||||
png_image_free(&img);
|
|
||||||
|
|
||||||
if(img.warning_or_error)
|
|
||||||
{
|
|
||||||
fprintf(stderr, "libpng %s: %s\n", img.warning_or_error == 1
|
|
||||||
? "warning": "error", img.message);
|
|
||||||
if(img.warning_or_error > 1) return 1;
|
|
||||||
}
|
|
||||||
|
|
||||||
return 0;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
/* icon_conv_8to1(): Convert an 8-bit icon to 1-bit */
|
uint8_t *icon_conv_1to24(uint8_t const *mono, int width, int height)
|
||||||
uint8_t *icon_conv_8to1(uint8_t const *input, size_t width, size_t height)
|
|
||||||
{
|
{
|
||||||
if(!input) return NULL;
|
int bytes_per_row = (width + 7) >> 3;
|
||||||
uint8_t *mono = calloc(68, 1);
|
uint8_t *rgb24 = calloc(width * height, 3);
|
||||||
if(!mono) return NULL;
|
if(!rgb24) return NULL;
|
||||||
size_t stride = width;
|
|
||||||
|
|
||||||
/* If the image is wider than 30 pixels, ignore columns at the right */
|
for(int y = 0; y < height; y++)
|
||||||
if(width > 30) width = 30;
|
for(int x = 0; x < width; x++) {
|
||||||
|
int in = (y * bytes_per_row) + (x >> 3);
|
||||||
|
int out = 3 * (y * width + x);
|
||||||
|
int color = (mono[in] & (0x80 >> (x & 7))) != 0 ? 0x00 : 0xff;
|
||||||
|
rgb24[out] = color;
|
||||||
|
rgb24[out+1] = color;
|
||||||
|
rgb24[out+2] = color;
|
||||||
|
}
|
||||||
|
|
||||||
/* Skip the first line if there is enough rows, because in standard
|
return rgb24;
|
||||||
30*19 icons, the first and last lines are skipped */
|
|
||||||
if(height > 17) input += stride, height--;
|
|
||||||
|
|
||||||
/* If height is still larger than 17, ignore rows at the bottom */
|
|
||||||
if(height > 17) height = 17;
|
|
||||||
|
|
||||||
/* Then copy individual pixels on the currently-blank image */
|
|
||||||
for(size_t y = 0; y < height; y++)
|
|
||||||
for(size_t x = 0; x < width; x++)
|
|
||||||
{
|
|
||||||
int offset = (y << 2) + (x >> 3);
|
|
||||||
int color = input[y * stride + x] < 128;
|
|
||||||
uint8_t mask = color << (~x & 7);
|
|
||||||
mono[offset] |= mask;
|
|
||||||
}
|
|
||||||
|
|
||||||
return mono;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
/* icon_conv_1to8(): Convert an 1-bit icon to 8-bit */
|
uint16_t *icon_conv_24to16(uint8_t const *rgb24, int width, int height)
|
||||||
uint8_t *icon_conv_1to8(uint8_t const *mono)
|
|
||||||
{
|
{
|
||||||
uint8_t *data = calloc(30 * 17, 1);
|
uint16_t *rgb16be = calloc(width * height, 2);
|
||||||
if(!data) return NULL;
|
if(!rgb16be) return NULL;
|
||||||
|
|
||||||
for(int y = 0; y < 17; y++)
|
for(int y = 0; y < height; y++)
|
||||||
for(int x = 0; x < 30; x++)
|
for(int x = 0; x < width; x++) {
|
||||||
{
|
int in = 3 * (y * width + x);
|
||||||
int offset = (y << 2) + (x >> 3);
|
int color = ((rgb24[in] & 0xf8) << 8)
|
||||||
int bit = mono[offset] & (0x80 >> (x & 7));
|
| ((rgb24[in+1] & 0xfc) << 3)
|
||||||
data[y * 30 + x] = (bit ? 0x00 : 0xff);
|
| ((rgb24[in+2] & 0xf8) >> 3);
|
||||||
}
|
rgb16be[y * width + x] = htobe16(color);
|
||||||
|
}
|
||||||
|
|
||||||
return data;
|
return rgb16be;
|
||||||
|
}
|
||||||
|
|
||||||
|
uint8_t *icon_conv_16to24(uint16_t const *rgb16be, int width, int height)
|
||||||
|
{
|
||||||
|
uint8_t *rgb24 = calloc(width * height, 3);
|
||||||
|
if(!rgb24) return NULL;
|
||||||
|
|
||||||
|
for(int y = 0; y < height; y++)
|
||||||
|
for(int x = 0; x < width; x++) {
|
||||||
|
int out = 3 * (y * width + x);
|
||||||
|
int color = be16toh(rgb16be[y * width + x]);
|
||||||
|
rgb24[out] = (color & 0xf800) >> 8;
|
||||||
|
rgb24[out+1] = (color & 0x07e0) >> 3;
|
||||||
|
rgb24[out+2] = (color & 0x001f) << 3;
|
||||||
|
}
|
||||||
|
|
||||||
|
return rgb24;
|
||||||
|
}
|
||||||
|
|
||||||
|
void icon_print_1(uint8_t const *mono, int width, int height)
|
||||||
|
{
|
||||||
|
int bytes_per_row = (width + 7) >> 3;
|
||||||
|
|
||||||
|
for(int y = 0; y < height; y++) {
|
||||||
|
for(int x = 0; x < width; x++) {
|
||||||
|
int in = (y * bytes_per_row) + (x >> 3);
|
||||||
|
int color = (mono[in] & (0x80 >> (x & 7))) != 0;
|
||||||
|
putchar(color ? '#' : ' ');
|
||||||
|
putchar(color ? '#' : ' ');
|
||||||
|
}
|
||||||
|
putchar('\n');
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
void icon_print_16(uint16_t const *rgb16be, int width, int height)
|
||||||
|
{
|
||||||
|
uint8_t *rgb24 = icon_conv_16to24(rgb16be, width, height);
|
||||||
|
if(!rgb24) {
|
||||||
|
fprintf(stderr, "error: %m\n");
|
||||||
|
return;
|
||||||
|
};
|
||||||
|
|
||||||
|
for(int y = 0; y < height; y++) {
|
||||||
|
for(int x = 0; x < width; x++) {
|
||||||
|
int in = 3 * (y * width + x);
|
||||||
|
int r=rgb24[in], g=rgb24[in+1], b=rgb24[in+2];
|
||||||
|
printf("\e[48;2;%d;%d;%dm ", r, g, b);
|
||||||
|
}
|
||||||
|
printf("\e[0m\n");
|
||||||
|
}
|
||||||
|
|
||||||
|
free(rgb24);
|
||||||
}
|
}
|
||||||
|
|
277
fxgxa/main.c
277
fxgxa/main.c
|
@ -3,33 +3,38 @@
|
||||||
#include <time.h>
|
#include <time.h>
|
||||||
#include <getopt.h>
|
#include <getopt.h>
|
||||||
|
|
||||||
#include <fxg1a.h>
|
#include <fxgxa.h>
|
||||||
#include <g1a.h>
|
#include <g1a.h>
|
||||||
|
#include <g3a.h>
|
||||||
|
|
||||||
static const char *help_string =
|
static const char *help_string =
|
||||||
"usage: %1$s [-g] <binary file> [options...]\n"
|
"usage: fxgxa [-g] <binary file> [options...]\n"
|
||||||
" %1$s -e <g1a file> [options...]\n"
|
" fxgxa -e <g1a/g3a file> [options...]\n"
|
||||||
" %1$s -d <g1a file>\n"
|
" fxgxa -d <g1a/g3a file>\n"
|
||||||
" %1$s -r <g1a file> [-o <g1a file>]\n"
|
" fxgxa -r <g1a/g3a file> [-o <g1a/g3a file>]\n"
|
||||||
" %1$s -x <g1a file> [-o <png file>]\n"
|
" fxgxa -x <g1a/g3a file> [-o <png file>]\n"
|
||||||
"\n"
|
"\n"
|
||||||
"fxg1a creates or edits g1a files (add-in applications for Casio fx9860g\n"
|
"fxgxa creates or edits g1a and g3a files (add-in applications for CASIO\n"
|
||||||
"calculator series) that consist of a g1a header followed by binary code.\n"
|
"fx-9860G and fx-CG series) that consist of a header followed by code.\n"
|
||||||
"\n"
|
"\n"
|
||||||
"Operating modes:\n"
|
"Operating modes:\n"
|
||||||
" -g, --g1a Generate a g1a file (default)\n"
|
" -g, --g1a, --g3a Generate a g1a/g3a file (default)\n"
|
||||||
" -e, --edit Edit header of an existing g1a file\n"
|
" -e, --edit Edit header of an existing g1a/g3a file\n"
|
||||||
" -d, --dump Dump header of an existing g1a file\n"
|
" -d, --dump Dump header of an existing g1a/g3a file\n"
|
||||||
" -r, --repair Recalculate control bytes and checksums\n"
|
" -r, --repair Recalculate control bytes and checksums\n"
|
||||||
" -x, --extract Extract icon into a PNG file\n"
|
" -x, --extract Extract icon into a PNG file\n"
|
||||||
"\n"
|
"\n"
|
||||||
"General options:\n"
|
"General options:\n"
|
||||||
" -o, --output=<file> Output file (default: input file with .g1a suffix\n"
|
" -o, --output=<file> Output file (default: input with .g1a/.g3a suffix\n"
|
||||||
" [-g]; with .png suffix [-x]; input file [-e, -r])\n"
|
" [-g]; with .png suffix [-x]; input file [-e, -r])\n"
|
||||||
|
" --output-uns=<file> Output for unselected icon with [-x] and g3a file\n"
|
||||||
|
" --output-sel=<file> Output for selected icon with [-x] and g3a file\n"
|
||||||
"\n"
|
"\n"
|
||||||
"Generation and edition options:\n"
|
"Generation and edition options:\n"
|
||||||
" -i, --icon=<png> Program icon, in PNG format (default: blank icon)\n"
|
" -i, --icon=<png> Program icon, in PNG format (default: blank) [g1a]\n"
|
||||||
" -n, --name=<name> Add-in name, 8 bytes (default: output file name)\n"
|
" --icon-uns=<png> Unselected program icon, in PNG format [g3a]\n"
|
||||||
|
" --icon-sel=<png> Selected program icon, in PNG format [g3a]\n"
|
||||||
|
" -n, --name=<name> Add-in name (default: output file name)\n"
|
||||||
" --version=<text> Program version, MM.mm.pppp format (default: empty)\n"
|
" --version=<text> Program version, MM.mm.pppp format (default: empty)\n"
|
||||||
" --internal=<name> Internal name, eg. '@NAME' (default: empty)\n"
|
" --internal=<name> Internal name, eg. '@NAME' (default: empty)\n"
|
||||||
" --date=<date> Date of build, yyyy.MMdd.hhmm (default: now)\n";
|
" --date=<date> Date of build, yyyy.MMdd.hhmm (default: now)\n";
|
||||||
|
@ -49,31 +54,58 @@ struct fields
|
||||||
const char *date;
|
const char *date;
|
||||||
/* Icon file name */
|
/* Icon file name */
|
||||||
const char *icon;
|
const char *icon;
|
||||||
|
const char *icon_uns, *icon_sel;
|
||||||
|
|
||||||
|
// TODO: G3A: Fill the filename field
|
||||||
};
|
};
|
||||||
|
|
||||||
/* fields_edit(): Set the value of some fields altogether
|
/* fields_edit(): Set the value of some fields altogether
|
||||||
@header Header to edit, is assumed checksumed and filled
|
@gxa Header to edit, is assumed checksumed and filled
|
||||||
@fields New values for fields, any members can be NULL */
|
@fields New values for fields, any members can be NULL */
|
||||||
void fields_edit(struct g1a *header, struct fields const *fields)
|
void fields_edit(void *gxa, struct fields const *fields)
|
||||||
{
|
{
|
||||||
/* For easy fields, just call the appropriate edition function */
|
/* For easy fields, just call the appropriate edition function */
|
||||||
if(fields->name) edit_name(header, fields->name);
|
if(fields->name) edit_name(gxa, fields->name);
|
||||||
if(fields->version) edit_version(header, fields->version);
|
if(fields->version) edit_version(gxa, fields->version);
|
||||||
if(fields->internal) edit_internal(header, fields->internal);
|
if(fields->internal) edit_internal(gxa, fields->internal);
|
||||||
if(fields->date) edit_date(header, fields->date);
|
if(fields->date) edit_date(gxa, fields->date);
|
||||||
|
|
||||||
/* Load icon from PNG file */
|
/* Load icon from PNG file */
|
||||||
if(fields->icon)
|
if(fields->icon && is_g1a(gxa)) {
|
||||||
{
|
int w, h;
|
||||||
size_t width, height;
|
uint8_t *rgb24 = icon_load(fields->icon, &w, &h);
|
||||||
uint8_t *data = icon_load(fields->icon, &width, &height);
|
|
||||||
if(!data) return;
|
|
||||||
|
|
||||||
uint8_t *mono = icon_conv_8to1(data, width, height);
|
if(rgb24) {
|
||||||
free(data);
|
/* Skip the first row if h > 17, since the usual
|
||||||
|
representation at 30x19 skips the first and last */
|
||||||
|
uint8_t *mono = icon_conv_24to1(
|
||||||
|
h > 17 ? rgb24 + 3*w : rgb24, w, h - (h > 17));
|
||||||
|
if(mono) edit_g1a_icon(gxa, mono);
|
||||||
|
free(mono);
|
||||||
|
}
|
||||||
|
free(rgb24);
|
||||||
|
}
|
||||||
|
if(fields->icon_uns && is_g3a(gxa)) {
|
||||||
|
int w, h;
|
||||||
|
uint8_t *rgb24 = icon_load(fields->icon_uns, &w, &h);
|
||||||
|
|
||||||
if(!mono) return;
|
if(rgb24) {
|
||||||
edit_icon(header, mono);
|
uint16_t *rgb16be = icon_conv_24to16(rgb24, w, h);
|
||||||
|
if(rgb16be) edit_g3a_icon(gxa, rgb16be, false);
|
||||||
|
free(rgb16be);
|
||||||
|
}
|
||||||
|
free(rgb24);
|
||||||
|
}
|
||||||
|
if(fields->icon_sel && is_g3a(gxa)) {
|
||||||
|
int w, h;
|
||||||
|
uint8_t *rgb24 = icon_load(fields->icon_sel, &w, &h);
|
||||||
|
|
||||||
|
if(rgb24) {
|
||||||
|
uint16_t *rgb16be = icon_conv_24to16(rgb24, w, h);
|
||||||
|
if(rgb16be) edit_g3a_icon(gxa, rgb16be, true);
|
||||||
|
free(rgb16be);
|
||||||
|
}
|
||||||
|
free(rgb24);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -87,21 +119,27 @@ int main(int argc, char **argv)
|
||||||
int mode = 'g', error = 0;
|
int mode = 'g', error = 0;
|
||||||
struct fields fields = { 0 };
|
struct fields fields = { 0 };
|
||||||
const char *output = NULL;
|
const char *output = NULL;
|
||||||
|
const char *output_uns=NULL, *output_sel=NULL;
|
||||||
|
|
||||||
const struct option longs[] = {
|
const struct option longs[] = {
|
||||||
{ "help", no_argument, NULL, 'h' },
|
{ "help", no_argument, NULL, 'h' },
|
||||||
{ "g1a", no_argument, NULL, 'g' },
|
{ "g1a", no_argument, NULL, '1' },
|
||||||
{ "edit", no_argument, NULL, 'e' },
|
{ "g3a", no_argument, NULL, '3' },
|
||||||
{ "dump", no_argument, NULL, 'd' },
|
{ "edit", no_argument, NULL, 'e' },
|
||||||
{ "repair", no_argument, NULL, 'r' },
|
{ "dump", no_argument, NULL, 'd' },
|
||||||
{ "extract", no_argument, NULL, 'x' },
|
{ "repair", no_argument, NULL, 'r' },
|
||||||
{ "output", required_argument, NULL, 'o' },
|
{ "extract", no_argument, NULL, 'x' },
|
||||||
{ "icon", required_argument, NULL, 'i' },
|
{ "output", required_argument, NULL, 'o' },
|
||||||
{ "name", required_argument, NULL, 'n' },
|
{ "output-uns", required_argument, NULL, 'O' },
|
||||||
{ "version", required_argument, NULL, 'v' },
|
{ "output-sel", required_argument, NULL, 'P' },
|
||||||
{ "internal", required_argument, NULL, 't' },
|
{ "icon", required_argument, NULL, 'i' },
|
||||||
{ "date", required_argument, NULL, 'a' },
|
{ "icon-uns", required_argument, NULL, 'I' },
|
||||||
{ NULL, 0, NULL, 0 },
|
{ "icon-sel", required_argument, NULL, 'J' },
|
||||||
|
{ "name", required_argument, NULL, 'n' },
|
||||||
|
{ "version", required_argument, NULL, 'v' },
|
||||||
|
{ "internal", required_argument, NULL, 't' },
|
||||||
|
{ "date", required_argument, NULL, 'a' },
|
||||||
|
{ NULL, 0, NULL, 0 },
|
||||||
};
|
};
|
||||||
|
|
||||||
int option = 0;
|
int option = 0;
|
||||||
|
@ -116,14 +154,28 @@ int main(int argc, char **argv)
|
||||||
case 'd':
|
case 'd':
|
||||||
case 'r':
|
case 'r':
|
||||||
case 'x':
|
case 'x':
|
||||||
|
case '1':
|
||||||
|
case '3':
|
||||||
mode = option;
|
mode = option;
|
||||||
break;
|
break;
|
||||||
case 'o':
|
case 'o':
|
||||||
output = optarg;
|
output = optarg;
|
||||||
break;
|
break;
|
||||||
|
case 'O':
|
||||||
|
output_uns = optarg;
|
||||||
|
break;
|
||||||
|
case 'P':
|
||||||
|
output_sel = optarg;
|
||||||
|
break;
|
||||||
case 'i':
|
case 'i':
|
||||||
fields.icon = optarg;
|
fields.icon = optarg;
|
||||||
break;
|
break;
|
||||||
|
case 'I':
|
||||||
|
fields.icon_uns = optarg;
|
||||||
|
break;
|
||||||
|
case 'J':
|
||||||
|
fields.icon_sel = optarg;
|
||||||
|
break;
|
||||||
case 'n':
|
case 'n':
|
||||||
fields.name = optarg;
|
fields.name = optarg;
|
||||||
break;
|
break;
|
||||||
|
@ -141,6 +193,10 @@ int main(int argc, char **argv)
|
||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
if(mode == 'g' && !strcmp(argv[0], "fxg1a"))
|
||||||
|
mode = '1';
|
||||||
|
if(mode == 'g' && !strcmp(argv[0], "fxg3a"))
|
||||||
|
mode = '3';
|
||||||
if(error) return 1;
|
if(error) return 1;
|
||||||
|
|
||||||
if(argv[optind] == NULL)
|
if(argv[optind] == NULL)
|
||||||
|
@ -148,12 +204,18 @@ int main(int argc, char **argv)
|
||||||
fprintf(stderr, help_string, argv[0]);
|
fprintf(stderr, help_string, argv[0]);
|
||||||
return 1;
|
return 1;
|
||||||
}
|
}
|
||||||
if(mode == 'g')
|
if(mode == 'g') {
|
||||||
|
fprintf(stderr, "cannot guess -g; use --g1a or --g3a\n");
|
||||||
|
return 1;
|
||||||
|
}
|
||||||
|
if(mode == '1' || mode == '3')
|
||||||
{
|
{
|
||||||
/* Load binary file into memory */
|
/* Load binary file into memory */
|
||||||
size_t size;
|
size_t size;
|
||||||
struct g1a *g1a = load_binary(argv[optind], &size);
|
int header = (mode == '1' ? 0x200 : 0x7000);
|
||||||
if(!g1a) return 1;
|
int footer = (mode == '1' ? 0 : 4);
|
||||||
|
void *gxa = load_binary(argv[optind], &size, header, footer);
|
||||||
|
if(!gxa) return 1;
|
||||||
|
|
||||||
/* If [output] is set, use it, otherwise compute a default */
|
/* If [output] is set, use it, otherwise compute a default */
|
||||||
char *alloc = NULL;
|
char *alloc = NULL;
|
||||||
|
@ -161,46 +223,52 @@ int main(int argc, char **argv)
|
||||||
{
|
{
|
||||||
alloc = malloc(strlen(argv[optind]) + 5);
|
alloc = malloc(strlen(argv[optind]) + 5);
|
||||||
if(!alloc) {fprintf(stderr, "error: %m\n"); return 1;}
|
if(!alloc) {fprintf(stderr, "error: %m\n"); return 1;}
|
||||||
default_output(argv[optind], ".g1a", alloc);
|
default_output(argv[optind],
|
||||||
|
(mode == '1' ? ".g1a" : ".g3a"), alloc);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/* First set the type so that is_g1a() and is_g3a() work */
|
||||||
|
((uint8_t *)gxa)[8] = (mode == '1' ? 0xf3 : 0x2c);
|
||||||
|
|
||||||
/* Start with output file name as application name */
|
/* Start with output file name as application name */
|
||||||
edit_name(g1a, output ? output : alloc);
|
edit_name(gxa, output ? output : alloc);
|
||||||
|
|
||||||
/* Start with "now" as build date */
|
/* Start with "now" as build date */
|
||||||
char date[15];
|
char date[15];
|
||||||
time_t t = time(NULL);
|
time_t t = time(NULL);
|
||||||
struct tm *now = localtime(&t);
|
struct tm *now = localtime(&t);
|
||||||
strftime(date, 15, "%Y.%m%d.%H%M", now);
|
strftime(date, 15, "%Y.%m%d.%H%M", now);
|
||||||
edit_date(g1a, date);
|
edit_date(gxa, date);
|
||||||
|
|
||||||
/* Start with an uppercase name as internal name */
|
/* Start with an uppercase name as internal name */
|
||||||
char internal[9];
|
char internal[12];
|
||||||
default_internal(fields.name ? fields.name : g1a->header.name,
|
if(fields.name)
|
||||||
internal);
|
default_internal(fields.name, internal, 11);
|
||||||
edit_internal(g1a, internal);
|
else if(is_g1a(gxa))
|
||||||
|
default_internal(G1A(gxa)->header.name, internal, 11);
|
||||||
|
else if(is_g3a(gxa))
|
||||||
|
default_internal(G3A(gxa)->header.name, internal, 11);
|
||||||
|
edit_internal(gxa, internal);
|
||||||
|
|
||||||
/* Edit the fields with user-customized values */
|
/* Edit the fields with user-customized values */
|
||||||
fields_edit(g1a, &fields);
|
fields_edit(gxa, &fields);
|
||||||
|
|
||||||
/* Set fixed fields and calculate checksums */
|
/* Set fixed fields and calculate checksums */
|
||||||
sign(g1a, size);
|
sign(gxa, size);
|
||||||
|
|
||||||
save_g1a(output ? output : alloc, g1a, size);
|
save_gxa(output ? output : alloc, gxa, size);
|
||||||
free(alloc);
|
free(alloc);
|
||||||
|
free(gxa);
|
||||||
/* Write output file */
|
|
||||||
free(g1a);
|
|
||||||
}
|
}
|
||||||
if(mode == 'e')
|
if(mode == 'e')
|
||||||
{
|
{
|
||||||
/* Load g1a file into memory */
|
/* Load file into memory */
|
||||||
size_t size;
|
size_t size;
|
||||||
struct g1a *g1a = load_g1a(argv[optind], &size);
|
void *gxa = load_gxa(argv[optind], &size);
|
||||||
if(!g1a) return 1;
|
if(!gxa) return 1;
|
||||||
|
|
||||||
/* Edit the fields with user-customized values */
|
/* Edit the fields with user-customized values */
|
||||||
fields_edit(g1a, &fields);
|
fields_edit(gxa, &fields);
|
||||||
|
|
||||||
/* We don't reset fixed fields or recalculate checksums because
|
/* We don't reset fixed fields or recalculate checksums because
|
||||||
we only want to edit what was requested by the user.
|
we only want to edit what was requested by the user.
|
||||||
|
@ -209,63 +277,78 @@ int main(int argc, char **argv)
|
||||||
|
|
||||||
/* Regenerate input file, or output somewhere else */
|
/* Regenerate input file, or output somewhere else */
|
||||||
if(!output) output = argv[optind];
|
if(!output) output = argv[optind];
|
||||||
save_g1a(output, g1a, size);
|
save_gxa(output, gxa, size);
|
||||||
free(g1a);
|
free(gxa);
|
||||||
}
|
}
|
||||||
if(mode == 'd')
|
if(mode == 'd')
|
||||||
{
|
{
|
||||||
/* Load and dump the g1a */
|
|
||||||
size_t size;
|
size_t size;
|
||||||
struct g1a *g1a = load_g1a(argv[optind], &size);
|
void *gxa = load_gxa(argv[optind], &size);
|
||||||
if(!g1a) return 1;
|
if(!gxa) return 1;
|
||||||
|
|
||||||
dump(g1a, size);
|
dump(gxa, size);
|
||||||
free(g1a);
|
free(gxa);
|
||||||
}
|
}
|
||||||
if(mode == 'r')
|
if(mode == 'r')
|
||||||
{
|
{
|
||||||
/* Load g1a file into memory */
|
|
||||||
size_t size;
|
size_t size;
|
||||||
struct g1a *g1a = load_g1a(argv[optind], &size);
|
void *gxa = load_gxa(argv[optind], &size);
|
||||||
if(!g1a) return 1;
|
if(!gxa) return 1;
|
||||||
|
|
||||||
/* Repair file by recalculating fixed fields and checksums */
|
/* Repair file by recalculating fixed fields and checksums */
|
||||||
sign(g1a, size);
|
sign(gxa, size);
|
||||||
|
|
||||||
/* Regenerate input file, or output somewhere else */
|
/* Regenerate input file, or output somewhere else */
|
||||||
if(!output) output = argv[optind];
|
if(!output) output = argv[optind];
|
||||||
save_g1a(output, g1a, size);
|
save_gxa(output, gxa, size);
|
||||||
free(g1a);
|
free(gxa);
|
||||||
}
|
}
|
||||||
if(mode == 'x')
|
if(mode == 'x')
|
||||||
{
|
{
|
||||||
/* Load g1a file into memory */
|
|
||||||
size_t size;
|
size_t size;
|
||||||
struct g1a *g1a = load_g1a(argv[optind], &size);
|
void *gxa = load_gxa(argv[optind], &size);
|
||||||
if(!g1a) return 1;
|
if(!gxa) return 1;
|
||||||
|
|
||||||
/* Generate 8-bit icon from g1a 1-bit */
|
if(is_g1a(gxa)) {
|
||||||
uint8_t *data = icon_conv_1to8(g1a->header.icon);
|
/* Add clean top/bottom rows */
|
||||||
if(!data)
|
uint8_t mono[76];
|
||||||
{
|
memcpy(mono, "\x00\x00\x00\x00", 4);
|
||||||
fprintf(stderr, "error: %m\n");
|
memcpy(mono+4, G1A(gxa)->header.icon, 68);
|
||||||
return 1;
|
memcpy(mono+72, "\x7f\xff\xff\xfc", 4);
|
||||||
}
|
uint8_t *rgb24 = icon_conv_1to24(mono, 30, 19);
|
||||||
|
|
||||||
/* Calculate a default output name if none is provided */
|
/* Calculate a default output name if none is given */
|
||||||
if(output)
|
char *alloc = NULL;
|
||||||
{
|
if(!output) {
|
||||||
icon_save(output, data, 30, 17);
|
alloc = malloc(strlen(argv[optind]) + 5);
|
||||||
}
|
if(!alloc) {
|
||||||
else
|
fprintf(stderr, "error: %m\n");
|
||||||
{
|
return 1;
|
||||||
char *alloc = malloc(strlen(argv[optind]) + 5);
|
}
|
||||||
if(!alloc) {fprintf(stderr, "error: %m\n"); return 1;}
|
default_output(argv[optind], ".png", alloc);
|
||||||
default_output(argv[optind], ".png", alloc);
|
}
|
||||||
|
|
||||||
icon_save(alloc, data, 30, 17);
|
icon_save(output ? output : alloc, rgb24, 30, 19);
|
||||||
free(alloc);
|
free(alloc);
|
||||||
|
free(rgb24);
|
||||||
}
|
}
|
||||||
|
else if(is_g3a(gxa)) {
|
||||||
|
uint8_t *rgb24_uns = icon_conv_16to24(
|
||||||
|
G3A(gxa)->header.icon_uns, 92, 64);
|
||||||
|
uint8_t *rgb24_sel = icon_conv_16to24(
|
||||||
|
G3A(gxa)->header.icon_sel, 92, 64);
|
||||||
|
|
||||||
|
if(output_uns)
|
||||||
|
icon_save(output_uns, rgb24_uns, 92, 64);
|
||||||
|
if(output_sel)
|
||||||
|
icon_save(output_sel, rgb24_sel, 92, 64);
|
||||||
|
if(!output_uns && !output_sel) fprintf(stderr, "Please"
|
||||||
|
" specify --output-uns or --output-sel.\n");
|
||||||
|
|
||||||
|
free(rgb24_uns);
|
||||||
|
free(rgb24_sel);
|
||||||
|
}
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|
||||||
return 0;
|
return 0;
|
||||||
|
|
68
fxgxa/util.c
68
fxgxa/util.c
|
@ -1,31 +1,64 @@
|
||||||
#include <string.h>
|
#include <string.h>
|
||||||
#include <ctype.h>
|
#include <ctype.h>
|
||||||
#include <endianness.h>
|
#include <endianness.h>
|
||||||
#include <fxg1a.h>
|
#include <fxgxa.h>
|
||||||
|
|
||||||
/*
|
bool is_g1a(void *gxa)
|
||||||
** Public API
|
|
||||||
*/
|
|
||||||
|
|
||||||
/* checksum(): Sum of 8 big-endian shorts at 0x300 */
|
|
||||||
uint16_t checksum(struct g1a const *g1a, size_t size)
|
|
||||||
{
|
{
|
||||||
uint16_t shorts[16] = { 0 };
|
/* Check the byte at offset 8 for file type 0xf3 */
|
||||||
|
uint8_t mcs_type = ((uint8_t *)gxa)[8];
|
||||||
|
return mcs_type == 0xf3;
|
||||||
|
}
|
||||||
|
|
||||||
/* Extract 16 bytes from the file (maybe less are available) */
|
bool is_g3a(void *gxa)
|
||||||
int available = size - 0x300;
|
{
|
||||||
|
/* Check the byte at offset 8 for file type 0x2c */
|
||||||
|
uint8_t mcs_type = ((uint8_t *)gxa)[8];
|
||||||
|
return mcs_type == 0x2c;
|
||||||
|
}
|
||||||
|
|
||||||
|
uint16_t word_sum(void const *gxa, size_t offset, int words, size_t size)
|
||||||
|
{
|
||||||
|
uint16_t shorts[words];
|
||||||
|
memset(shorts, 0, 2*words);
|
||||||
|
|
||||||
|
/* Extract up to [2*words] bytes from the file */
|
||||||
|
int available = (int)size - offset;
|
||||||
if(available < 0) available = 0;
|
if(available < 0) available = 0;
|
||||||
if(available > 16) available = 16;
|
if(available > 2*words) available = 2*words;
|
||||||
memcpy(shorts, g1a->code + 0x100, available);
|
memcpy(shorts, gxa+offset, available);
|
||||||
|
|
||||||
/* Do the big-endian sum */
|
/* Do the big-endian sum */
|
||||||
uint16_t sum = 0;
|
uint16_t sum = 0;
|
||||||
for(int i = 0; i < 8; i++) sum += htobe16(shorts[i]);
|
for(int i = 0; i < words; i++)
|
||||||
|
sum += htobe16(shorts[i]);
|
||||||
|
|
||||||
|
return sum;
|
||||||
|
}
|
||||||
|
|
||||||
|
uint16_t checksum_g1a(struct g1a const *g1a, size_t size)
|
||||||
|
{
|
||||||
|
return word_sum(g1a, 0x300, 8, size);
|
||||||
|
}
|
||||||
|
|
||||||
|
uint16_t checksum_g3a(struct g3a const *g3a, size_t size)
|
||||||
|
{
|
||||||
|
return word_sum(g3a, 0x7100, 8, size);
|
||||||
|
}
|
||||||
|
|
||||||
|
uint32_t checksum_g3a_2(struct g3a const *g3a, size_t size)
|
||||||
|
{
|
||||||
|
uint32_t sum = 0;
|
||||||
|
uint8_t *data = (void *)g3a;
|
||||||
|
|
||||||
|
for(size_t i = 0; i < 0x20; i++)
|
||||||
|
sum += (data[i] ^ 0xff);
|
||||||
|
for(size_t i = 0x24; i < size - 4; i++)
|
||||||
|
sum += data[i];
|
||||||
|
|
||||||
return sum;
|
return sum;
|
||||||
}
|
}
|
||||||
|
|
||||||
/* default_output(): Calculate default output file name */
|
|
||||||
void default_output(const char *name, const char *suffix, char *output)
|
void default_output(const char *name, const char *suffix, char *output)
|
||||||
{
|
{
|
||||||
/* Check if there is a dot at the end of @name, before the last '/'.
|
/* Check if there is a dot at the end of @name, before the last '/'.
|
||||||
|
@ -48,13 +81,12 @@ void default_output(const char *name, const char *suffix, char *output)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
/* default_internal(): Calculate default internal name */
|
void default_internal(const char *name, char *output, size_t size)
|
||||||
void default_internal(const char *name, char *output)
|
|
||||||
{
|
{
|
||||||
output[0] = '@';
|
output[0] = '@';
|
||||||
int i=1;
|
size_t i = 1;
|
||||||
|
|
||||||
for(int j=0; name[j] && i < 8; j++)
|
for(int j = 0; name[j] && i < size; j++)
|
||||||
{
|
{
|
||||||
if(isalpha(name[j])) output[i++] = toupper(name[j]);
|
if(isalpha(name[j])) output[i++] = toupper(name[j]);
|
||||||
}
|
}
|
||||||
|
|
|
@ -26,7 +26,7 @@ BINFLAGS := -R .bss -R .gint_bss
|
||||||
NAME_G1A ?= $(NAME)
|
NAME_G1A ?= $(NAME)
|
||||||
NAME_G3A ?= $(NAME)
|
NAME_G3A ?= $(NAME)
|
||||||
G1AF := -i "$(ICON_FX)" -n "$(NAME_G1A)" --internal="$(INTERNAL)"
|
G1AF := -i "$(ICON_FX)" -n "$(NAME_G1A)" --internal="$(INTERNAL)"
|
||||||
G3AF := -n basic:"$(NAME_G3A)" -i uns:"$(ICON_CG_UNS)" -i sel:"$(ICON_CG_SEL)"
|
G3AF := -n "$(NAME_G3A)" --icon-uns="$(ICON_CG_UNS)" --icon-sel="$(ICON_CG_SEL)"
|
||||||
|
|
||||||
ifeq "$(TOOLCHAIN_FX)" ""
|
ifeq "$(TOOLCHAIN_FX)" ""
|
||||||
TOOLCHAIN_FX := sh3eb-elf
|
TOOLCHAIN_FX := sh3eb-elf
|
||||||
|
@ -104,13 +104,13 @@ $(TARGET_FX): $(obj-fx) $(deps-fx)
|
||||||
@ mkdir -p $(dir $@)
|
@ mkdir -p $(dir $@)
|
||||||
$(TOOLCHAIN_FX)-gcc -o $(ELF_FX) $(obj-fx) $(CFLAGSFX) $(LDFLAGSFX)
|
$(TOOLCHAIN_FX)-gcc -o $(ELF_FX) $(obj-fx) $(CFLAGSFX) $(LDFLAGSFX)
|
||||||
$(TOOLCHAIN_FX)-objcopy -O binary $(BINFLAGS) $(ELF_FX) $(BIN_FX)
|
$(TOOLCHAIN_FX)-objcopy -O binary $(BINFLAGS) $(ELF_FX) $(BIN_FX)
|
||||||
fxg1a $(BIN_FX) -o $@ $(G1AF)
|
fxgxa --g1a $(BIN_FX) -o $@ $(G1AF)
|
||||||
|
|
||||||
$(TARGET_CG): $(obj-cg) $(deps-cg)
|
$(TARGET_CG): $(obj-cg) $(deps-cg)
|
||||||
@ mkdir -p $(dir $@)
|
@ mkdir -p $(dir $@)
|
||||||
$(TOOLCHAIN_CG)-gcc -o $(ELF_CG) $(obj-cg) $(CFLAGSCG) $(LDFLAGSCG)
|
$(TOOLCHAIN_CG)-gcc -o $(ELF_CG) $(obj-cg) $(CFLAGSCG) $(LDFLAGSCG)
|
||||||
$(TOOLCHAIN_CG)-objcopy -O binary $(BINFLAGS) $(ELF_CG) $(BIN_CG)
|
$(TOOLCHAIN_CG)-objcopy -O binary $(BINFLAGS) $(ELF_CG) $(BIN_CG)
|
||||||
mkg3a $(G3AF) $(BIN_CG) $@
|
fxgxa --g3a $(BIN_CG) -o $@ $(G3AF)
|
||||||
|
|
||||||
# C sources
|
# C sources
|
||||||
build-fx/%.c.o: %.c
|
build-fx/%.c.o: %.c
|
||||||
|
|
|
@ -20,27 +20,27 @@ function(generate_g1a)
|
||||||
set(G1A_OUTPUT "${G1A_TARGET}.g1a")
|
set(G1A_OUTPUT "${G1A_TARGET}.g1a")
|
||||||
endif()
|
endif()
|
||||||
|
|
||||||
# Compute the set of fxg1a arguments
|
# Compute the set of fxgxa arguments
|
||||||
|
|
||||||
set(FXG1A_ARGS "")
|
set(FXGXA_ARGS "")
|
||||||
|
|
||||||
# Support empty names enen though they're not normally used in g1a files
|
# Support empty names even though they're not normally used in g1a files
|
||||||
if(DEFINED G1A_NAME OR "NAME" IN_LIST G1A_KEYWORDS_MISSING_VALUES)
|
if(DEFINED G1A_NAME OR "NAME" IN_LIST G1A_KEYWORDS_MISSING_VALUES)
|
||||||
list(APPEND FXG1A_ARGS "-n" "${G1A_NAME}")
|
list(APPEND FXGXA_ARGS "-n" "${G1A_NAME}")
|
||||||
endif()
|
endif()
|
||||||
|
|
||||||
if(DEFINED G1A_ICON)
|
if(DEFINED G1A_ICON)
|
||||||
get_filename_component(G1A_ICON "${G1A_ICON}" ABSOLUTE
|
get_filename_component(G1A_ICON "${G1A_ICON}" ABSOLUTE
|
||||||
BASE_DIR "${CMAKE_CURRENT_SOURCE_DIR}")
|
BASE_DIR "${CMAKE_CURRENT_SOURCE_DIR}")
|
||||||
list(APPEND FXG1A_ARGS "-i" "${G1A_ICON}")
|
list(APPEND FXGXA_ARGS "-i" "${G1A_ICON}")
|
||||||
endif()
|
endif()
|
||||||
|
|
||||||
if(DEFINED G1A_INTERNAL)
|
if(DEFINED G1A_INTERNAL)
|
||||||
list(APPEND FXG1A_ARGS "--internal=${G1A_INTERNAL}")
|
list(APPEND FXGXA_ARGS "--internal=${G1A_INTERNAL}")
|
||||||
endif()
|
endif()
|
||||||
|
|
||||||
if(DEFINED G1A_VERSION)
|
if(DEFINED G1A_VERSION)
|
||||||
list(APPEND FXG1A_ARGS "--version=${G1A_VERSION}")
|
list(APPEND FXGXA_ARGS "--version=${G1A_VERSION}")
|
||||||
endif()
|
endif()
|
||||||
|
|
||||||
string(REGEX REPLACE "sh-elf-gcc$" "sh-elf-objcopy" OBJCOPY "${CMAKE_C_COMPILER}")
|
string(REGEX REPLACE "sh-elf-gcc$" "sh-elf-objcopy" OBJCOPY "${CMAKE_C_COMPILER}")
|
||||||
|
@ -48,7 +48,7 @@ function(generate_g1a)
|
||||||
add_custom_command(
|
add_custom_command(
|
||||||
TARGET "${G1A_TARGET}" POST_BUILD
|
TARGET "${G1A_TARGET}" POST_BUILD
|
||||||
COMMAND "${OBJCOPY}" -O binary -R .bss -R .gint_bss "${G1A_TARGET}" "${G1A_TARGET}.bin"
|
COMMAND "${OBJCOPY}" -O binary -R .bss -R .gint_bss "${G1A_TARGET}" "${G1A_TARGET}.bin"
|
||||||
COMMAND fxg1a ${FXG1A_ARGS} -o "${G1A_OUTPUT}" "${G1A_TARGET}.bin"
|
COMMAND fxgxa --g1a ${FXGXA_ARGS} -o "${G1A_OUTPUT}" "${G1A_TARGET}.bin"
|
||||||
WORKING_DIRECTORY "${CMAKE_CURRENT_BINARY_DIR}"
|
WORKING_DIRECTORY "${CMAKE_CURRENT_BINARY_DIR}"
|
||||||
)
|
)
|
||||||
if(DEFINED G1A_ICON)
|
if(DEFINED G1A_ICON)
|
||||||
|
|
|
@ -27,19 +27,22 @@ function(generate_g3a)
|
||||||
set(G3A_OUTPUT "${G3A_TARGET}.g3a")
|
set(G3A_OUTPUT "${G3A_TARGET}.g3a")
|
||||||
endif()
|
endif()
|
||||||
|
|
||||||
# Compute the set of mkg3a arguments
|
# Compute the set of fxgxa arguments
|
||||||
|
|
||||||
set(MKG3A_ARGS "")
|
set(FXGXA_ARGS "")
|
||||||
|
|
||||||
# Empty names are commonly used to avoid having the file name printed over
|
# Empty names are commonly used to avoid having the file name printed over
|
||||||
# the icon, but using ARGN in cmake_parse_arguments() drops the empty string.
|
# the icon, but using ARGN in cmake_parse_arguments() drops the empty string.
|
||||||
# Check in KEYWORDS_MISSING_VALUES as a complement.
|
# Check in KEYWORDS_MISSING_VALUES as a complement.
|
||||||
if(DEFINED G3A_NAME OR "NAME" IN_LIST G3A_KEYWORDS_MISSING_VALUES)
|
if(DEFINED G3A_NAME OR "NAME" IN_LIST G3A_KEYWORDS_MISSING_VALUES)
|
||||||
list(APPEND MKG3A_ARGS "-n" "basic:${G3A_NAME}")
|
if("${G3A_NAME}" STREQUAL "")
|
||||||
|
set(G3A_NAME "''")
|
||||||
|
endif()
|
||||||
|
list(APPEND FXGXA_ARGS "-n" "${G3A_NAME}")
|
||||||
endif()
|
endif()
|
||||||
|
|
||||||
if(DEFINED G3A_VERSION)
|
if(DEFINED G3A_VERSION)
|
||||||
list(APPEND MKG3A_ARGS "-V" "${G3A_VERSION}")
|
list(APPEND FXGXA_ARGS "--version=${G3A_VERSION}")
|
||||||
endif()
|
endif()
|
||||||
|
|
||||||
if(DEFINED G3A_ICONS)
|
if(DEFINED G3A_ICONS)
|
||||||
|
@ -52,7 +55,7 @@ function(generate_g3a)
|
||||||
# Who doesn't REALLY love to deal with escaping
|
# Who doesn't REALLY love to deal with escaping
|
||||||
string(REPLACE "'" "\\'" G3A_ICON1B "${G3A_ICON1}")
|
string(REPLACE "'" "\\'" G3A_ICON1B "${G3A_ICON1}")
|
||||||
string(REPLACE "'" "\\'" G3A_ICON2B "${G3A_ICON2}")
|
string(REPLACE "'" "\\'" G3A_ICON2B "${G3A_ICON2}")
|
||||||
list(APPEND MKG3A_ARGS "-i" "uns:${G3A_ICON1B}" "-i" "sel:${G3A_ICON2B}")
|
list(APPEND FXGXA_ARGS "--icon-uns=${G3A_ICON1B}" "--icon-sel=${G3A_ICON2B}")
|
||||||
endif()
|
endif()
|
||||||
|
|
||||||
string(REPLACE "gcc" "objcopy" OBJCOPY "${CMAKE_C_COMPILER}")
|
string(REPLACE "gcc" "objcopy" OBJCOPY "${CMAKE_C_COMPILER}")
|
||||||
|
@ -60,7 +63,7 @@ function(generate_g3a)
|
||||||
add_custom_command(
|
add_custom_command(
|
||||||
TARGET "${G3A_TARGET}" POST_BUILD
|
TARGET "${G3A_TARGET}" POST_BUILD
|
||||||
COMMAND ${OBJCOPY} -O binary -R .bss -R .gint_bss ${G3A_TARGET} ${G3A_TARGET}.bin
|
COMMAND ${OBJCOPY} -O binary -R .bss -R .gint_bss ${G3A_TARGET} ${G3A_TARGET}.bin
|
||||||
COMMAND mkg3a ${MKG3A_ARGS} "${G3A_TARGET}.bin" "${G3A_OUTPUT}"
|
COMMAND fxgxa --g3a ${FXGXA_ARGS} "${G3A_TARGET}.bin" -o "${G3A_OUTPUT}"
|
||||||
WORKING_DIRECTORY "${CMAKE_CURRENT_BINARY_DIR}"
|
WORKING_DIRECTORY "${CMAKE_CURRENT_BINARY_DIR}"
|
||||||
)
|
)
|
||||||
if(DEFINED G3A_ICONS)
|
if(DEFINED G3A_ICONS)
|
||||||
|
|
Loading…
Add table
Reference in a new issue