mirror of
https://github.com/google/pebble.git
synced 2025-05-10 19:41:41 -04:00
239 lines
6.9 KiB
C
239 lines
6.9 KiB
C
/*
|
|
* Copyright 2024 Google LLC
|
|
*
|
|
* Licensed under the Apache License, Version 2.0 (the "License");
|
|
* you may not use this file except in compliance with the License.
|
|
* You may obtain a copy of the License at
|
|
*
|
|
* http://www.apache.org/licenses/LICENSE-2.0
|
|
*
|
|
* Unless required by applicable law or agreed to in writing, software
|
|
* distributed under the License is distributed on an "AS IS" BASIS,
|
|
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
|
* See the License for the specific language governing permissions and
|
|
* limitations under the License.
|
|
*/
|
|
|
|
|
|
#include "applib/graphics/gtypes.h"
|
|
#include "applib/graphics/graphics.h"
|
|
#include "applib/graphics/framebuffer.h"
|
|
#include "applib/graphics/gdraw_command_image.h"
|
|
#include "applib/graphics/gdraw_command_sequence.h"
|
|
#include "applib/graphics/gdraw_command_private.h"
|
|
#include "applib/graphics/graphics_private_raw.h"
|
|
|
|
#include "stubs_applib_resource.h"
|
|
#include "stubs_app_state.h"
|
|
#include "stubs_heap.h"
|
|
#include "stubs_logging.h"
|
|
#include "stubs_memory_layout.h"
|
|
#include "stubs_pbl_malloc.h"
|
|
#include "stubs_resources.h"
|
|
#include "stubs_syscalls.h"
|
|
|
|
|
|
#include <stdio.h>
|
|
#include <sys/stat.h>
|
|
#include <unistd.h>
|
|
#include <string.h>
|
|
#include <errno.h>
|
|
#include <libgen.h>
|
|
|
|
#include "util_pbi.h"
|
|
|
|
static char *s_pbi2png_path;
|
|
|
|
// Stubs
|
|
void passert_failed(const char* filename, int line_number, const char* message, ...) {
|
|
exit(EXIT_FAILURE);
|
|
}
|
|
|
|
void passert_failed_no_message(const char* filename, int line_number) {
|
|
exit(EXIT_FAILURE);
|
|
}
|
|
|
|
void wtf(void) {
|
|
exit(EXIT_FAILURE);
|
|
}
|
|
|
|
bool process_manager_compiled_with_legacy2_sdk(void) {
|
|
return false;
|
|
}
|
|
|
|
const uint8_t *resource_get_builtin_bytes(ResAppNum app_num, uint32_t resource_id,
|
|
uint32_t *num_bytes_out) {
|
|
return NULL;
|
|
}
|
|
|
|
// Get the path of the filename before the extension
|
|
size_t prv_get_base_path(const char *filename, char *base) {
|
|
char *ext = strrchr(filename, '.');
|
|
|
|
size_t len;
|
|
if (ext == NULL) {
|
|
len = strlen(filename);
|
|
} else {
|
|
len = ext - filename;
|
|
}
|
|
strncpy(base, filename, len);
|
|
base[len] = '\0';
|
|
return len;
|
|
}
|
|
|
|
// Reset the graphics context, fill the background color and center the drawing box according to the
|
|
// bounds provided
|
|
static void prv_setup_context(GContext *ctx, GSize bounds) {
|
|
// just a fake FB so that we can successfully call _context_init()
|
|
// the actual pixel data we will draw into exists independently and will be allocated below
|
|
FrameBuffer fb;
|
|
framebuffer_init(&fb, &bounds);
|
|
framebuffer_clear(&fb);
|
|
// Reset graphics context
|
|
graphics_context_init(ctx, &fb, GContextInitializationMode_System);
|
|
|
|
const uint16_t row_size_bytes = bounds.w;
|
|
ctx->dest_bitmap = (GBitmap){
|
|
.addr = malloc(row_size_bytes * bounds.h),
|
|
.bounds = (GRect){.size = bounds},
|
|
.row_size_bytes = row_size_bytes,
|
|
.info.version = 1,
|
|
.info.format = GBitmapFormat8Bit,
|
|
};
|
|
|
|
ctx->draw_state.clip_box = ctx->dest_bitmap.bounds;
|
|
ctx->draw_state.drawing_box = ctx->draw_state.clip_box;
|
|
|
|
// Always use anti-aliased
|
|
graphics_context_set_antialiased(ctx, true);
|
|
|
|
// Setup the draw state
|
|
graphics_context_set_fill_color(ctx, GColorElectricBlue);
|
|
graphics_fill_rect(ctx, &(GRect){.size = bounds});
|
|
}
|
|
|
|
static void prv_teardown_context(GContext *ctx) {
|
|
free(ctx->dest_bitmap.addr);
|
|
ctx->dest_bitmap.addr = NULL;
|
|
}
|
|
|
|
// Read PDC sequence file an write to a sequence of PNGs
|
|
void prv_convert_sequence(const char *filename, void *data, size_t size) {
|
|
GDrawCommandSequence *sequence = data;
|
|
|
|
// Validate
|
|
if (!gdraw_command_sequence_validate(sequence, size)) {
|
|
printf("Invalid PDC sequence: %s\n", filename);
|
|
return;
|
|
}
|
|
|
|
// Create directory from base name (with extension stripped)
|
|
char output[strlen(filename) + 1 + strlen("_65535.png")]; // max size of a file path
|
|
size_t len = prv_get_base_path(filename, output);
|
|
|
|
// Write out each frame as a PNG
|
|
for (int i = 0; i < gdraw_command_sequence_get_num_frames(sequence); i++) {
|
|
// Set up the context for every frame
|
|
GContext ctx;
|
|
prv_setup_context(&ctx, gdraw_command_sequence_get_bounds_size(sequence));
|
|
|
|
GDrawCommandFrame *frame = gdraw_command_sequence_get_frame_by_index(sequence, i);
|
|
gdraw_command_frame_draw(&ctx, sequence, frame, GPoint(0, 0));
|
|
|
|
// Write frames into new directory as a numbered sequence
|
|
sprintf(&output[len], "_%d.png", i + 1);
|
|
write_gbitmap_to_pbi(&ctx.dest_bitmap, output, s_pbi2png_path);
|
|
prv_teardown_context(&ctx);
|
|
}
|
|
}
|
|
|
|
// Read PDC image file and convert to PNG
|
|
void prv_convert_image(const char *filename, void *data, size_t size) {
|
|
GDrawCommandImage *image = data;
|
|
|
|
// Validate
|
|
if (!gdraw_command_image_validate(image, size)) {
|
|
printf("Invalid PDC image: %s\n", filename);
|
|
return;
|
|
}
|
|
|
|
GContext ctx;
|
|
prv_setup_context(&ctx, gdraw_command_image_get_bounds_size(image));
|
|
|
|
gdraw_command_image_draw(&ctx, image, GPoint(0, 0));
|
|
|
|
// Write output to file of the same name with extension replaced with .png
|
|
char output[strlen(filename) + 5]; // add space for .pdc + '\0'
|
|
prv_get_base_path(filename, output);
|
|
strcat(output, ".png");
|
|
|
|
write_gbitmap_to_pbi(&ctx.dest_bitmap, output, s_pbi2png_path);
|
|
|
|
prv_teardown_context(&ctx);
|
|
}
|
|
|
|
static void prv_convert_pdc(const char *filename) {
|
|
|
|
FILE *f = fopen(filename, "rb");
|
|
if (f == NULL) {
|
|
printf("File not found: %s\n", filename);
|
|
return;
|
|
}
|
|
|
|
// Check that file is a PDC file
|
|
char magic[4];
|
|
if (fread(magic, sizeof(magic), 1, f) != 1) {
|
|
printf("Failed to read PDC magic word: %s\n", filename);
|
|
fclose(f);
|
|
return;
|
|
}
|
|
|
|
// Read size of data
|
|
size_t size;
|
|
if (fread(&size, sizeof(size), 1, f) != 1) {
|
|
printf("Failed to read PDC size: %s\n", filename);
|
|
fclose(f);
|
|
return;
|
|
}
|
|
|
|
// Read data into memory
|
|
void *data = malloc(size);
|
|
if (fread(data, size, 1, f) != 1) {
|
|
printf("Failed to read PDC content: %s\n", filename);
|
|
free(data);
|
|
fclose(f);
|
|
return;
|
|
}
|
|
fclose(f);
|
|
|
|
if (strncmp(magic, "PDCI", sizeof(magic)) == 0) {
|
|
prv_convert_image(filename, data, size);
|
|
} else if (strncmp(magic, "PDCS", sizeof(magic)) == 0) {
|
|
prv_convert_sequence(filename, data, size);
|
|
} else {
|
|
printf("Invalid file: %s\n", filename);
|
|
}
|
|
free(data);
|
|
}
|
|
|
|
int main(int argc, const char* argv[]) {
|
|
// pdc2png file path always passed in as first argument
|
|
if (argc == 1) {
|
|
printf("No files specified. Pass a list of file paths to convert "
|
|
"(e.g. pdc2png [path-to-file1] [path-to-file2] ...)\n");
|
|
}
|
|
|
|
char *dir = dirname((char *) argv[0]);
|
|
printf("%s\n", dir);
|
|
s_pbi2png_path = malloc(strlen(dir) + 1 + strlen(PBI2PNG_EXE) + 1);
|
|
sprintf(s_pbi2png_path, "%s/%s", dir, PBI2PNG_EXE);
|
|
|
|
// skip first argument (this file path)
|
|
for (int i = 1; i < argc; i++) {
|
|
// Treat all arguments as filenames and attempt to convert them as PDC into PNG
|
|
// This seems to handle invalid arguments ok (it just can't find the files to open, so it fails
|
|
// with a message saying so)
|
|
printf("Converting %s...\n", argv[i]);
|
|
prv_convert_pdc(argv[i]);
|
|
}
|
|
}
|