mirror of
https://github.com/google/pebble.git
synced 2025-06-23 01:26:17 +00:00
Import of the watch repository from Pebble
This commit is contained in:
commit
3b92768480
10334 changed files with 2564465 additions and 0 deletions
239
tools/pdc2png/src/pdc2png.c
Normal file
239
tools/pdc2png/src/pdc2png.c
Normal file
|
@ -0,0 +1,239 @@
|
|||
/*
|
||||
* 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]);
|
||||
}
|
||||
}
|
Loading…
Add table
Add a link
Reference in a new issue