diff options
-rw-r--r-- | payloads/libpayload/drivers/video/graphics.c | 308 | ||||
-rw-r--r-- | payloads/libpayload/include/cbgfx.h | 7 |
2 files changed, 160 insertions, 155 deletions
diff --git a/payloads/libpayload/drivers/video/graphics.c b/payloads/libpayload/drivers/video/graphics.c index c1dc37486b..b4cdcdea69 100644 --- a/payloads/libpayload/drivers/video/graphics.c +++ b/payloads/libpayload/drivers/video/graphics.c @@ -45,18 +45,26 @@ static void add_vectors(struct vector *out, out->y = v1->y + v2->y; } +static int is_valid_fraction(const struct fraction *f) +{ + return f->d != 0; +} + /* * Transform a vector: * x' = x * a_x + offset_x * y' = y * a_y + offset_y */ -static void transform_vector(struct vector *out, - const struct vector *in, - const struct scale *a, - const struct vector *offset) +static int transform_vector(struct vector *out, + const struct vector *in, + const struct scale *a, + const struct vector *offset) { - out->x = a->x.nume * in->x / a->x.deno + offset->x; - out->y = a->y.nume * in->y / a->y.deno + offset->y; + if (!is_valid_fraction(&a->x) || !is_valid_fraction(&a->y)) + return CBGFX_ERROR_INVALID_PARAMETER; + out->x = a->x.n * in->x / a->x.d + offset->x; + out->y = a->y.n * in->y / a->y.d + offset->y; + return CBGFX_SUCCESS; } /* @@ -162,12 +170,12 @@ int draw_box(const struct rect *box, const struct rgb_color *rgb) struct vector p, t; const uint32_t color = calculate_color(rgb); const struct scale top_left_s = { - .x = { .nume = box->offset.x, .deno = CANVAS_SCALE, }, - .y = { .nume = box->offset.y, .deno = CANVAS_SCALE, } + .x = { .n = box->offset.x, .d = CANVAS_SCALE, }, + .y = { .n = box->offset.y, .d = CANVAS_SCALE, } }; const struct scale size_s = { - .x = { .nume = box->size.x, .deno = CANVAS_SCALE, }, - .y = { .nume = box->size.y, .deno = CANVAS_SCALE, } + .x = { .n = box->size.x, .d = CANVAS_SCALE, }, + .y = { .n = box->size.y, .d = CANVAS_SCALE, } }; if (cbgfx_init()) @@ -220,33 +228,25 @@ int clear_screen(const struct rgb_color *rgb) return CBGFX_SUCCESS; } -static int check_pixel_boundary(const struct vector *image, - const struct bitmap_header_v3 *header, - const struct scale *scale) -{ - struct vector p = { - .x = (image->width - 1) * scale->x.deno / scale->x.nume, - .y = (image->height -1) * scale->y.deno / scale->y.nume, - }; - struct rect bound = { - .offset = vzero, - .size = { .width = header->width, .height = header->height, }, - }; - return within_box(&p, &bound) < 0; -} - +/* + * Bi-linear Interpolation + * + * It estimates the value of a middle point (tx, ty) using the values from four + * adjacent points (q00, q01, q10, q11). + */ static uint32_t bli(uint32_t q00, uint32_t q10, uint32_t q01, uint32_t q11, struct fraction *tx, struct fraction *ty) { - uint32_t r0 = (tx->nume * q10 + (tx->deno - tx->nume) * q00) / tx->deno; - uint32_t r1 = (tx->nume * q11 + (tx->deno - tx->nume) * q01) / tx->deno; - uint32_t p = (ty->nume * r1 + (ty->deno - ty->nume) * r0) / ty->deno; + uint32_t r0 = (tx->n * q10 + (tx->d - tx->n) * q00) / tx->d; + uint32_t r1 = (tx->n * q11 + (tx->d - tx->n) * q01) / tx->d; + uint32_t p = (ty->n * r1 + (ty->d - ty->n) * r0) / ty->d; return p; } static int draw_bitmap_v3(const struct vector *top_left, const struct scale *scale, - const struct vector *image, + const struct vector *dim, + const struct vector *dim_org, const struct bitmap_header_v3 *header, const struct bitmap_palette_element_v3 *pal, const uint8_t *pixel_array) @@ -267,16 +267,12 @@ static int draw_bitmap_v3(const struct vector *top_left, LOG("Unsupported bits per pixel: %d\n", bpp); return CBGFX_ERROR_BITMAP_FORMAT; } - if (scale->x.nume == 0 || scale->y.nume == 0) { + if (scale->x.n == 0 || scale->y.n == 0) { LOG("Scaling out of range\n"); return CBGFX_ERROR_SCALE_OUT_OF_RANGE; } - /* This check guarantees we will not try to read outside pixel data. */ - if (check_pixel_boundary(image, header, scale)) - return CBGFX_ERROR_SCALE_OUT_OF_RANGE; - - const int32_t y_stride = ROUNDUP(header->width * bpp / 8, 4); + const int32_t y_stride = ROUNDUP(dim_org->width * bpp / 8, 4); /* * header->height can be positive or negative. * @@ -290,33 +286,39 @@ static int draw_bitmap_v3(const struct vector *top_left, if (header->height < 0) { dir = 1; } else { - p.y += image->height - 1; + p.y += dim->height - 1; dir = -1; } /* - * Plot pixels scaled by the bilinear interpolation. We scan - * over the image on canvas (using d) and find the corresponding pixel - * in the bitmap data (using s). + * Plot pixels scaled by the bilinear interpolation. We scan over the + * image on canvas (using d) and find the corresponding pixel in the + * bitmap data (using s0, s1). + * + * When d hits the right bottom corner, s0 also hits the right bottom + * corner of the pixel array because that's how scale->x and scale->y + * have been set. Since the pixel array size is already validated in + * parse_bitmap_header_v3, s0 is guranteed not to exceed pixel array + * boundary. */ struct vector s0, s1, d; struct fraction tx, ty; - for (d.y = 0; d.y < image->height; d.y++, p.y += dir) { - s0.y = d.y * scale->y.deno / scale->y.nume; + for (d.y = 0; d.y < dim->height; d.y++, p.y += dir) { + s0.y = d.y * scale->y.d / scale->y.n; s1.y = s0.y; - if (s0.y + 1 < ABS(header->height)) + if (s0.y + 1 < dim_org->height) s1.y++; - ty.deno = scale->y.nume; - ty.nume = (d.y * scale->y.deno) % scale->y.nume; + ty.d = scale->y.n; + ty.n = (d.y * scale->y.d) % scale->y.n; const uint8_t *data0 = pixel_array + s0.y * y_stride; const uint8_t *data1 = pixel_array + s1.y * y_stride; p.x = top_left->x; - for (d.x = 0; d.x < image->width; d.x++, p.x++) { - s0.x = d.x * scale->x.deno / scale->x.nume; + for (d.x = 0; d.x < dim->width; d.x++, p.x++) { + s0.x = d.x * scale->x.d / scale->x.n; s1.x = s0.x; - if (s1.x + 1 < header->width) + if (s1.x + 1 < dim_org->width) s1.x++; - tx.deno = scale->x.nume; - tx.nume = (d.x * scale->x.deno) % scale->x.nume; + tx.d = scale->x.n; + tx.n = (d.x * scale->x.d) % scale->x.n; uint8_t c00 = data0[s0.x]; uint8_t c10 = data0[s1.x]; uint8_t c01 = data1[s0.x]; @@ -370,18 +372,27 @@ static int get_bitmap_file_header(const void *bitmap, size_t size, return CBGFX_SUCCESS; } -static int parse_bitmap_header_v3(const uint8_t *bitmap, - const struct bitmap_file_header *file_header, +static int parse_bitmap_header_v3( + const uint8_t *bitmap, + size_t size, /* ^--- IN / OUT ---v */ struct bitmap_header_v3 *header, const struct bitmap_palette_element_v3 **palette, - const uint8_t **pixel_array) + const uint8_t **pixel_array, + struct vector *dim_org) { + struct bitmap_file_header file_header; struct bitmap_header_v3 *h; + int rv; + + rv = get_bitmap_file_header(bitmap, size, &file_header); + if (rv) + return rv; + size_t header_offset = sizeof(struct bitmap_file_header); size_t header_size = sizeof(struct bitmap_header_v3); size_t palette_offset = header_offset + header_size; - size_t file_size = file_header->file_size; + size_t file_size = file_header.file_size; h = (struct bitmap_header_v3 *)(bitmap + header_offset); header->header_size = le32toh(h->header_size); @@ -389,15 +400,23 @@ static int parse_bitmap_header_v3(const uint8_t *bitmap, LOG("Unsupported bitmap format\n"); return CBGFX_ERROR_BITMAP_FORMAT; } + header->width = le32toh(h->width); header->height = le32toh(h->height); + if (header->width == 0 || header->height == 0) { + LOG("Invalid image width or height\n"); + return CBGFX_ERROR_BITMAP_DATA; + } + dim_org->width = header->width; + dim_org->height = ABS(header->height); + header->bits_per_pixel = le16toh(h->bits_per_pixel); header->compression = le32toh(h->compression); header->size = le32toh(h->size); header->colors_used = le32toh(h->colors_used); size_t palette_size = header->colors_used * sizeof(struct bitmap_palette_element_v3); - size_t pixel_offset = file_header->bitmap_offset; + size_t pixel_offset = file_header.bitmap_offset; if (pixel_offset > file_size) { LOG("Bitmap pixel data exceeds buffer boundary\n"); return CBGFX_ERROR_BITMAP_DATA; @@ -410,8 +429,8 @@ static int parse_bitmap_header_v3(const uint8_t *bitmap, palette_offset); size_t pixel_size = header->size; - if (pixel_size != header->height * - ROUNDUP(header->width * header->bits_per_pixel / 8, 4)) { + if (pixel_size != dim_org->height * + ROUNDUP(dim_org->width * header->bits_per_pixel / 8, 4)) { LOG("Bitmap pixel array size does not match expected size\n"); return CBGFX_ERROR_BITMAP_DATA; } @@ -424,58 +443,51 @@ static int parse_bitmap_header_v3(const uint8_t *bitmap, return CBGFX_SUCCESS; } -static int calculate_scale(const struct bitmap_header_v3 *header, - const struct scale *dim_rel, - struct scale *scale) +/* + * This calculates the dimension of the image projected on the canvas from the + * dimension relative to the canvas size. If either width or height is zero, it + * is derived from the other (non-zero) value to keep the aspect ratio. + */ +static int calculate_dimension(const struct vector *dim_org, + const struct scale *dim_rel, + struct vector *dim) { - if (dim_rel->x.nume == 0 && dim_rel->y.nume == 0) + if (dim_rel->x.n == 0 && dim_rel->y.n == 0) return CBGFX_ERROR_INVALID_PARAMETER; - if (dim_rel->x.nume > dim_rel->x.deno - || dim_rel->y.nume > dim_rel->y.deno) + if (dim_rel->x.n > dim_rel->x.d || dim_rel->y.n > dim_rel->y.d) return CBGFX_ERROR_INVALID_PARAMETER; - if (dim_rel->x.nume > 0) { - scale->x.nume = dim_rel->x.nume * canvas.size.width; - scale->x.deno = header->width * dim_rel->x.deno; + if (dim_rel->x.n > 0) { + if (!is_valid_fraction(&dim_rel->x)) + return CBGFX_ERROR_INVALID_PARAMETER; + dim->width = canvas.size.width * dim_rel->x.n / dim_rel->x.d; } - if (dim_rel->y.nume > 0) { - scale->y.nume = dim_rel->y.nume * canvas.size.height; - scale->y.deno = header->height * dim_rel->y.deno; + if (dim_rel->y.n > 0) { + if (!is_valid_fraction(&dim_rel->y)) + return CBGFX_ERROR_INVALID_PARAMETER; + dim->height = canvas.size.height * dim_rel->y.n / dim_rel->y.d; } - if (dim_rel->y.nume == 0) { - scale->y.nume = scale->x.nume; - scale->y.deno = scale->x.deno; - } - if (dim_rel->x.nume == 0) { - scale->x.nume = scale->y.nume; - scale->x.deno = scale->y.deno; - } - - if (scale->x.deno == 0 || scale->y.deno == 0) - return CBGFX_ERROR_INVALID_PARAMETER; + /* Derive height from width using aspect ratio */ + if (dim_rel->y.n == 0) + dim->height = dim->width * dim_org->height / dim_org->width; + /* Derive width from height using aspect ratio */ + if (dim_rel->x.n == 0) + dim->width = dim->height * dim_org->width / dim_org->height; return CBGFX_SUCCESS; } -static void calculate_dimention(const struct bitmap_header_v3 *header, - const struct scale *scale, - struct vector *dim) -{ - dim->width = header->width; - dim->height = ABS(header->height); - transform_vector(dim, dim, scale, &vzero); -} - static int caclcuate_position(const struct vector *dim, const struct scale *pos_rel, uint8_t pivot, struct vector *top_left) { - if (pos_rel->x.deno == 0 || pos_rel->y.deno == 0) - return CBGFX_ERROR_INVALID_PARAMETER; + int rv; - transform_vector(top_left, &canvas.size, pos_rel, &canvas.offset); + rv = transform_vector(top_left, &canvas.size, pos_rel, &canvas.offset); + if (rv) + return rv; switch (pivot & PIVOT_H_MASK) { case PIVOT_H_LEFT: @@ -506,28 +518,6 @@ static int caclcuate_position(const struct vector *dim, return CBGFX_SUCCESS; } -static int setup_image(const struct bitmap_header_v3 *header, - const struct scale *dim_rel, - const struct scale *pos_rel, uint8_t pivot, - /* ^--- IN / OUT ---v */ - struct scale *scale, - struct vector *dim, - struct vector *top_left) -{ - int rv; - - /* Calculate self-scale (scale relative to image size) */ - rv = calculate_scale(header, dim_rel, scale); - if (rv) - return rv; - - /* Calculate height and width of the image */ - calculate_dimention(header, scale, dim); - - /* Calculate coordinate */ - return caclcuate_position(dim, pos_rel, pivot, top_left); -} - static int check_boundary(const struct vector *top_left, const struct vector *dim, const struct rect *bound) @@ -541,67 +531,83 @@ static int check_boundary(const struct vector *top_left, return CBGFX_SUCCESS; } -static int do_draw_bitmap(const void *bitmap, size_t size, - const struct scale *pos_rel, const uint8_t pivot, - const struct vector *position, - const struct scale *dim_rel) +int draw_bitmap(const void *bitmap, size_t size, + const struct scale *pos_rel, uint8_t pivot, + const struct scale *dim_rel) { - struct bitmap_file_header file_header; struct bitmap_header_v3 header; const struct bitmap_palette_element_v3 *palette; const uint8_t *pixel_array; - struct vector top_left, dim; + struct vector top_left, dim, dim_org; struct scale scale; int rv; if (cbgfx_init()) return CBGFX_ERROR_INIT; - rv = get_bitmap_file_header(bitmap, size, &file_header); + /* only v3 is supported now */ + rv = parse_bitmap_header_v3(bitmap, size, + &header, &palette, &pixel_array, &dim_org); if (rv) return rv; - /* only v3 is supported now */ - rv = parse_bitmap_header_v3(bitmap, &file_header, - &header, &palette, &pixel_array); + /* Calculate height and width of the image */ + rv = calculate_dimension(&dim_org, dim_rel, &dim); if (rv) return rv; - /* Calculate image scale, dimention, and position */ - if (position) { - scale.x.nume = 1; - scale.x.deno = 1; - scale.y.nume = 1; - scale.y.deno = 1; - calculate_dimention(&header, &scale, &dim); - add_vectors(&top_left, position, &vzero); - rv = check_boundary(&top_left, &dim, &screen); - } else { - rv = setup_image(&header, dim_rel, pos_rel, pivot, - &scale, &dim, &top_left); - if (rv) - return rv; - rv = check_boundary(&top_left, &dim, &canvas); - } + /* Calculate self scale */ + scale.x.n = dim.width; + scale.x.d = dim_org.width; + scale.y.n = dim.height; + scale.y.d = dim_org.height; + + /* Calculate coordinate */ + rv = caclcuate_position(&dim, pos_rel, pivot, &top_left); + if (rv) + return rv; + + rv = check_boundary(&top_left, &dim, &canvas); if (rv) { - LOG("Bitmap image exceeds screen or canvas boundary\n"); + LOG("Bitmap image exceeds canvas boundary\n"); return rv; } - return draw_bitmap_v3(&top_left, &scale, &dim, + return draw_bitmap_v3(&top_left, &scale, &dim, &dim_org, &header, palette, pixel_array); } -int draw_bitmap(const void *bitmap, size_t size, - const struct scale *pos_rel, uint8_t pivot, - const struct scale *dim_rel) -{ - return do_draw_bitmap(bitmap, size, pos_rel, pivot, NULL, dim_rel); -} - int draw_bitmap_direct(const void *bitmap, size_t size, - const struct vector *position) + const struct vector *top_left) { - return do_draw_bitmap(bitmap, size, NULL, 0, position, NULL); -} + struct bitmap_header_v3 header; + const struct bitmap_palette_element_v3 *palette; + const uint8_t *pixel_array; + struct vector dim; + struct scale scale; + int rv; + + if (cbgfx_init()) + return CBGFX_ERROR_INIT; + /* only v3 is supported now */ + rv = parse_bitmap_header_v3(bitmap, size, + &header, &palette, &pixel_array, &dim); + if (rv) + return rv; + + /* Calculate self scale */ + scale.x.n = 1; + scale.x.d = 1; + scale.y.n = 1; + scale.y.d = 1; + + rv = check_boundary(top_left, &dim, &screen); + if (rv) { + LOG("Bitmap image exceeds screen boundary\n"); + return rv; + } + + return draw_bitmap_v3(top_left, &scale, &dim, &dim, + &header, palette, pixel_array); +} diff --git a/payloads/libpayload/include/cbgfx.h b/payloads/libpayload/include/cbgfx.h index ffd8534d13..54d395395a 100644 --- a/payloads/libpayload/include/cbgfx.h +++ b/payloads/libpayload/include/cbgfx.h @@ -30,8 +30,8 @@ #define CBGFX_ERROR_SCALE_OUT_OF_RANGE 0x13 struct fraction { - int32_t nume; - int32_t deno; + int32_t n; + int32_t d; }; struct scale { @@ -147,5 +147,4 @@ int draw_bitmap(const void *bitmap, size_t size, * @return CBGFX_* error codes */ int draw_bitmap_direct(const void *bitmap, size_t size, - const struct vector *pos); - + const struct vector *top_left); |