pbsplash/src/pbsplash.c

320 lines
8.9 KiB
C
Raw Normal View History

#include <stdio.h>
#include <stdlib.h>
2022-02-22 16:10:24 -09:00
#include <signal.h>
#include <unistd.h>
#include <sys/stat.h>
#include <fcntl.h>
#include <tfblib/tfblib.h>
#include <tfblib/tfb_colors.h>
2022-01-22 13:27:56 -09:00
#include <string.h>
#include <math.h>
#define NANOSVG_ALL_COLOR_KEYWORDS // Include full list of color keywords.
#define NANOSVG_IMPLEMENTATION // Expands implementation
#include "nanosvg.h"
#define NANOSVGRAST_IMPLEMENTATION
#include "nanosvgrast.h"
#define MSG_MAX_LEN 4096
2022-01-23 14:54:01 -09:00
#define DEFAULT_FONT_PATH "/usr/share/pbsplash/OpenSans-Regular.svg"
2022-02-22 16:10:24 -09:00
#define LOGO_SIZE_MAX_MM 90
2022-02-27 13:50:20 -09:00
#define FONT_SIZE_PT 13
#define PT_TO_MM 0.38f
2022-02-22 16:10:24 -09:00
2022-02-27 15:49:26 -09:00
#define TTY_PATH_LEN 11
2022-02-22 16:10:24 -09:00
volatile sig_atomic_t terminate = 0;
2022-02-27 15:49:26 -09:00
bool debug = true;
2022-02-22 16:10:24 -09:00
#define LOG(fmt, ...) do { if (debug) fprintf(stdout, fmt, ##__VA_ARGS__); } while (0)
2022-01-22 13:27:56 -09:00
int usage()
{
2022-02-27 13:50:20 -09:00
fprintf(stderr, "pbsplash: postmarketOS bootsplash generator\n");
fprintf(stderr, "-------------------------------------------\n");
2022-02-22 16:10:24 -09:00
fprintf(stderr, "pbsplash [-h] [-d] [-f font] [-s splash image] [-m message]\n\n");
2022-02-27 13:50:20 -09:00
fprintf(stderr, " -v|--verbose enable verbose logging\n");
fprintf(stderr, " -h|--help show this help\n");
fprintf(stderr, " -f|--font path to SVG font file (default: %s)\n", DEFAULT_FONT_PATH);
fprintf(stderr, " -s|--splash-image path to splash image to display\n");
fprintf(stderr, " -m|--message message to show under the splash image\n");
fprintf(stderr, " -p|--font-size font size in pt (default: %d)\n", FONT_SIZE_PT);
fprintf(stderr, " -d|--dpi custom DPI (for testing)\n");
return 1;
}
2022-02-22 16:10:24 -09:00
void term(int signum)
2022-01-22 13:27:56 -09:00
{
2022-02-22 16:10:24 -09:00
printf("Caught\n");
terminate = 1;
2022-01-22 13:27:56 -09:00
}
// Blit an SVG to the framebuffer using tfblib. The image is
// scaled based on the target width and height.
2022-02-22 16:10:24 -09:00
static void draw_svg(NSVGimage *image, int x, int y, int largest_bound)
2022-01-22 13:27:56 -09:00
{
//fprintf(stdout, "size: %f x %f\n", image->width, image->height);
2022-02-27 15:49:26 -09:00
// float sz = (float)largest_bound / (image->width > image->height ? image->width : image->height);
// int w = image->width * sz + 0.5;
// int h = image->height * sz + 0.5;
//fprintf(stdout, "scale: %f\n", sz);
2022-02-27 15:49:26 -09:00
int w = image->width;
int h = image->height;
NSVGrasterizer *rast = nsvgCreateRasterizer();
unsigned char *img = malloc(w * h * 4);
2022-02-27 15:49:26 -09:00
nsvgRasterize(rast, image, 0, 0, 1, img, w, h, w * 4);
for (size_t i = 0; i < w; i++)
{
for (size_t j = 0; j < h; j++)
{
2022-02-27 15:49:26 -09:00
if (i == 0 || i == w - 1 || j == 0 || j == h-1) {
tfb_draw_pixel(x + i, y + h - j, tfb_red);
continue;
}
unsigned int col = tfb_make_color(img[(j * w + i) * 4 + 0], img[(j * w + i) * 4 + 1], img[(j * w + i) * 4 + 2]);
tfb_draw_pixel(x + i, y + j, col);
}
}
free(img);
}
2022-02-22 16:10:24 -09:00
/**
* Get the dimensions of a string in pixels.
* based on the font size and the font SVG file.
*/
static void getTextDimensions(NSVGimage *font, char *text, float scale, int *width, int *height)
{
int i = 0;
*width = 0;
2022-02-22 16:10:24 -09:00
// The height is simply the height of the font * the scale factor
*height = (font->fontAscent - font->fontDescent) * scale;
2022-02-22 16:10:24 -09:00
if (text == NULL)
return;
NSVGshape **shapes = nsvgGetTextShapes(font, text, strlen(text));
2022-02-22 16:10:24 -09:00
// Iterate over every glyph in the string to get the total width
for (i = 0; i < strlen(text); i++)
{
NSVGshape *shape = shapes[i];
if (shape) {
*width += shapes[i]->horizAdvX * scale + 1;
} else {
*width += font->defaultHorizAdv * scale;
}
}
}
2022-02-22 16:10:24 -09:00
static void draw_text(NSVGimage *font, char *text, int x, int y, int width, int height, float scale, unsigned int tfb_col)
{
2022-02-22 16:10:24 -09:00
LOG("text '%s'\nfontsz=%f, x=%d, y=%d, dimensions: %d x %d\n", text,
scale, x, y, width, height);
NSVGshape **shapes = nsvgGetTextShapes(font, text, strlen(text));
unsigned char *img = malloc(width * height * 4);
NSVGrasterizer *rast = nsvgCreateRasterizer();
nsvgRasterizeText(rast, font, 0, 0, scale, img, width, height, width * 4, text);
2022-01-22 13:27:56 -09:00
for (size_t i = 0; i < width; i++)
2022-01-22 13:27:56 -09:00
{
for (size_t j = 0; j < height; j++)
2022-01-22 13:27:56 -09:00
{
2022-02-27 15:49:26 -09:00
if (i == 0 || i == width - 1 || j == 0 || j == height-1) {
tfb_draw_pixel(x + i, y + height - j, tfb_red);
continue;
}
unsigned int col = tfb_make_color(img[(j * width + i) * 4 + 0], img[(j * width + i) * 4 + 1], img[(j * width + i) * 4 + 2]);
2022-02-22 16:10:24 -09:00
if (col != tfb_black)
tfb_draw_pixel(x + i, y + height - j, tfb_col);
2022-01-22 13:27:56 -09:00
}
}
free(img);
free(shapes);
2022-01-22 13:27:56 -09:00
}
int main(int argc, char **argv)
{
int rc = 0;
2022-02-22 16:10:24 -09:00
char *message = NULL;
char *splash_image = NULL;
2022-02-27 13:50:20 -09:00
char *font_path = DEFAULT_FONT_PATH;
2022-02-27 15:49:26 -09:00
char active_tty[TTY_PATH_LEN];
NSVGimage *image;
NSVGimage *font;
2022-02-22 16:10:24 -09:00
struct sigaction action;
2022-02-27 13:50:20 -09:00
float font_size = FONT_SIZE_PT;
int c;
long dpi = 0;
2022-02-22 16:10:24 -09:00
2022-02-27 15:49:26 -09:00
memset(active_tty, '\0', TTY_PATH_LEN);
strcat(active_tty, "/dev/");
2022-02-22 16:10:24 -09:00
memset(&action, 0, sizeof(action));
action.sa_handler = term;
sigaction(SIGTERM, &action, NULL);
sigaction(SIGINT, &action, NULL);
2022-02-27 13:50:20 -09:00
while ((c = getopt(argc, argv, "hvdf:s:m:p:d:")) != -1)
{
2022-02-27 13:50:20 -09:00
char *end = NULL;
switch (c)
2022-02-22 16:10:24 -09:00
{
2022-02-27 13:50:20 -09:00
case 'h':
return usage();
case 'v':
2022-02-27 15:49:26 -09:00
printf("DEBUGGING");
2022-02-22 16:10:24 -09:00
debug = true;
2022-02-27 13:50:20 -09:00
break;
case 'f':
font_path = optarg;
break;
case 's':
splash_image = optarg;
break;
case 'm':
message = optarg;
break;
case 'p':
font_size = strtof(optarg, &end);
if (end == optarg)
{
fprintf(stderr, "Invalid font size: %s\n", optarg);
return usage();
}
break;
case 'd':
2022-02-27 15:49:26 -09:00
if (!optarg)
{
fprintf(stderr, "--dpi requires an argument\n");
return usage();
}
dpi = strtol(optarg, &end, 10);
2022-02-27 13:50:20 -09:00
if (end == optarg)
{
fprintf(stderr, "Invalid font size: %s\n", optarg);
return usage();
}
break;
default:
return usage();
}
}
2022-02-27 15:49:26 -09:00
{
FILE *fp = fopen("/sys/devices/virtual/tty/tty0/active", "r");
char c;
if(fp != NULL)
{
while((c = getc(fp)) != EOF && strlen(active_tty) < TTY_PATH_LEN)
{
if (c == '\n')
break;
strcat(active_tty, &c);
}
fclose(fp);
}
}
LOG("active tty: '%s'\n", active_tty);
if ((rc = tfb_acquire_fb(0, "/dev/fb0", active_tty)) != TFB_SUCCESS)
2022-01-22 13:27:56 -09:00
{
fprintf(stderr, "tfb_acquire_fb() failed with error code: %d\n", rc);
rc = 1;
return rc;
}
2022-01-22 13:27:56 -09:00
int w = (int)tfb_screen_width();
int h = (int)tfb_screen_height();
2022-02-22 16:10:24 -09:00
int w_mm = tfb_screen_width_mm();
int h_mm = tfb_screen_height_mm();
2022-02-27 13:50:20 -09:00
// If DPI is specified on cmdline then calculate display size from it
// otherwise calculate the dpi based on the display size.
if (dpi > 0)
{
w_mm = w / (float)dpi * 25.4;
h_mm = h / (float)dpi * 25.4;
} else {
dpi = (float)w / (float)w_mm * 25.4;
}
2022-02-22 16:10:24 -09:00
int pixels_per_milli = (float)w / (float)w_mm;
float logo_size_px = (float)(w < h ? w : h) * 0.75f;
if (w_mm > 0 && h_mm > 0) {
if (w_mm < h_mm) {
if (w_mm > (float)LOGO_SIZE_MAX_MM * 1.2f)
logo_size_px = (float)LOGO_SIZE_MAX_MM * pixels_per_milli;
} else {
if (h_mm > (float)LOGO_SIZE_MAX_MM * 1.2f)
logo_size_px = (float)LOGO_SIZE_MAX_MM * pixels_per_milli;
}
}
2022-02-27 13:50:20 -09:00
LOG("%dx%d @ %dx%dmm, spi=%ld, logo_size_px=%f\n", w, h, w_mm, h_mm, dpi, logo_size_px);
2022-02-22 16:10:24 -09:00
float x = (float)w / 2 - logo_size_px / 2;
float y = (float)h / 2 - logo_size_px / 2;
2022-01-22 13:27:56 -09:00
2022-02-27 15:49:26 -09:00
image = nsvgParseFromFile(splash_image, "px", logo_size_px);
if (!image)
{
fprintf(stderr, "failed to load SVG image\n");
rc = 1;
2022-01-23 14:54:01 -09:00
goto release_fb;
}
tfb_clear_screen(tfb_black);
2022-02-22 16:10:24 -09:00
draw_svg(image, x, y, logo_size_px);
2022-02-22 16:10:24 -09:00
if (message) {
int textWidth, textHeight;
2022-02-27 15:49:26 -09:00
// float fontsz = pixels_per_milli / (float)(font->fontAscent - font->fontDescent)
// * (font_size * PT_TO_MM);
font = nsvgParseFromFile(font_path, "mm", (font_size * PT_TO_MM));
if (!font || !font->shapes)
{
fprintf(stderr, "failed to load SVG font\n");
rc = 1;
goto free_image;
}
2022-02-27 15:49:26 -09:00
getTextDimensions(font, message, 1 /*fontsz*/, &textWidth, &textHeight);
2022-02-22 16:10:24 -09:00
int tx = w / 2.f - textWidth / 2.f;
int ty = y + logo_size_px * 0.5f + textHeight * 0.5f;
2022-02-27 15:49:26 -09:00
draw_text(font, message, tx, ty, textWidth, textHeight, 1/*fontsz*/, tfb_gray);
2022-02-22 16:10:24 -09:00
}
2022-01-22 13:27:56 -09:00
tfb_flush_window();
tfb_flush_fb();
2022-02-27 15:49:26 -09:00
// FIXME::: THERE IS A SEGFAULT IN CLEANUP
tfb_release_fb();
return 0;
2022-02-22 16:10:24 -09:00
while (!terminate)
{
sleep(1);
}
2022-02-27 15:49:26 -09:00
// beweare a segfault lies here (don't run on PC!!!!!!!!)
if (font)
nsvgDelete(font);
free_image:
2022-01-22 13:27:56 -09:00
nsvgDelete(image);
2022-01-23 14:54:01 -09:00
release_fb:
2022-02-22 16:10:24 -09:00
// The TTY might end up in a weird state if this
// is not called!
2022-01-23 14:54:01 -09:00
tfb_release_fb();
return rc;
}