Skip to content

Commit 9133b23

Browse files
committed
bitmaptools: Add readinto
When reading uncompressed bitmap data directly, readinto can work much more quickly than a Python-coded loop. On a Raspberry Pi Pico, I benchmarked a modified version of adafruit_bitmap_font's pcf reader which uses readinto instead of the existing code. My test font was a 72-point file created from Arial. This decreased the time to load all the ASCII glyphs from 4.9 seconds to just 0.44 seconds. While this attempts to support many pixel configurations (1/2/4/8/16/24/32 bpp; swapped words and pixels) only the single combination used by PCF fonts was tested.
1 parent d2563c5 commit 9133b23

File tree

3 files changed

+170
-0
lines changed

3 files changed

+170
-0
lines changed

shared-bindings/bitmaptools/__init__.c

Lines changed: 76 additions & 0 deletions
Original file line numberDiff line numberDiff line change
@@ -357,7 +357,83 @@ STATIC mp_obj_t bitmaptools_obj_draw_line(size_t n_args, const mp_obj_t *pos_arg
357357
MP_DEFINE_CONST_FUN_OBJ_KW(bitmaptools_draw_line_obj, 0, bitmaptools_obj_draw_line);
358358
// requires all 6 arguments
359359

360+
//| def readinto(bitmap: displayio.Bitmap, file: typing.BinaryIO, bits_per_pixel: int, element_size: int = 1, reverse_pixels_in_element: bool = False, swap_bytes_in_element: bool = False):
361+
//| """Read from a binary file into a bitmap
362+
//| The file must be positioned so that it consists of ``bitmap.height`` rows of pixel data, where each row is the smallest multiple of ``element_size`` bytes that can hold ``bitmap.width`` pixels.
363+
//|
364+
//| The bytes in an element can be optionally swapped, and the pixels in an element can be reversed.
365+
//|
366+
//| This function doesn't parse image headers, but is useful to speed up loading of uncompressed image formats such as PCF glyph data.
367+
//|
368+
//| :param displayio.Bitmap bitmap: A writable bitmap
369+
//| :param typing.BinaryIO file: A file opened in binary mode
370+
//| :param int bits_per_pixel: Number of bits per pixel. Values 1, 2, 4, 8, 16, 24, and 32 are supported;
371+
//| :param int element_size: Number of bytes per element. Values of 1, 2, and 4 are supported, except that 24 ``bits_per_pixel`` requires 1 byte per element.
372+
//| :param bool reverse_pixels_in_element: If set, the first pixel in a word is taken from the Most Signficant Bits; otherwise, it is taken from the Least Significant Bits.
373+
//| :param bool swap_bytes_in_element: If the ``element_size`` is not 1, then reverse the byte order of each element read.
374+
//| """
375+
//| ...
376+
//|
377+
378+
STATIC mp_obj_t bitmaptools_readinto(size_t n_args, const mp_obj_t *pos_args, mp_map_t *kw_args) {
379+
enum { ARG_bitmap, ARG_file, ARG_bits_per_pixel, ARG_element_size, ARG_reverse_pixels_in_element, ARG_swap_bytes_in_element };
380+
static const mp_arg_t allowed_args[] = {
381+
{ MP_QSTR_bitmap, MP_ARG_REQUIRED | MP_ARG_OBJ },
382+
{ MP_QSTR_file, MP_ARG_REQUIRED | MP_ARG_OBJ },
383+
{ MP_QSTR_bits_per_pixel, MP_ARG_REQUIRED | MP_ARG_INT },
384+
{ MP_QSTR_element_size, MP_ARG_INT, { .u_int = 1 } },
385+
{ MP_QSTR_reverse_pixels_in_element, MP_ARG_BOOL, { .u_bool = false } },
386+
{ MP_QSTR_swap_bytes_in_element, MP_ARG_BOOL, { .u_bool = false } },
387+
};
388+
389+
mp_arg_val_t args[MP_ARRAY_SIZE(allowed_args)];
390+
mp_arg_parse_all(n_args, pos_args, kw_args, MP_ARRAY_SIZE(allowed_args), allowed_args, args);
391+
392+
if (!MP_OBJ_IS_TYPE(args[ARG_bitmap].u_obj, &displayio_bitmap_type)) {
393+
mp_raise_TypeError(NULL);
394+
}
395+
displayio_bitmap_t *bitmap = MP_OBJ_TO_PTR(args[ARG_bitmap].u_obj);
396+
397+
if (!MP_OBJ_IS_TYPE(args[ARG_file].u_obj, &mp_type_fileio)) {
398+
mp_raise_TypeError(NULL);
399+
}
400+
pyb_file_obj_t* file = MP_OBJ_TO_PTR(args[ARG_file].u_obj);
401+
402+
int element_size = args[ARG_element_size].u_int;
403+
if (element_size != 1 && element_size != 2 && element_size != 4) {
404+
mp_raise_ValueError_varg(translate("invalid element_size %d, must be, 1, 2, or 4"), element_size);
405+
}
406+
407+
int bits_per_pixel = args[ARG_bits_per_pixel].u_int;
408+
switch (bits_per_pixel) {
409+
case 24:
410+
if (element_size != 1) {
411+
mp_raise_ValueError_varg(translate("invalid element size %d for bits_per_pixel %d\n"), element_size, bits_per_pixel);
412+
}
413+
break;
414+
case 1:
415+
case 2:
416+
case 4:
417+
case 8:
418+
case 16:
419+
case 32:
420+
break;
421+
default:
422+
mp_raise_ValueError_varg(translate("invalid bits_per_pixel %d, must be, 1, 4, 8, 16, 24, or 32"), bits_per_pixel);
423+
}
424+
425+
bool reverse_pixels_in_element = args[ARG_reverse_pixels_in_element].u_bool;
426+
bool swap_bytes_in_element = args[ARG_swap_bytes_in_element].u_bool;
427+
428+
common_hal_bitmaptools_readinto(bitmap, file, element_size, bits_per_pixel, reverse_pixels_in_element, swap_bytes_in_element);
429+
430+
return mp_const_none;
431+
}
432+
433+
MP_DEFINE_CONST_FUN_OBJ_KW(bitmaptools_readinto_obj, 0, bitmaptools_readinto);
434+
360435
STATIC const mp_rom_map_elem_t bitmaptools_module_globals_table[] = {
436+
{ MP_ROM_QSTR(MP_QSTR_readinto), MP_ROM_PTR(&bitmaptools_readinto_obj) },
361437
{ MP_ROM_QSTR(MP_QSTR_rotozoom), MP_ROM_PTR(&bitmaptools_rotozoom_obj) },
362438
{ MP_ROM_QSTR(MP_QSTR_fill_region), MP_ROM_PTR(&bitmaptools_fill_region_obj) },
363439
{ MP_ROM_QSTR(MP_QSTR_draw_line), MP_ROM_PTR(&bitmaptools_draw_line_obj) },

shared-bindings/bitmaptools/__init__.h

Lines changed: 4 additions & 0 deletions
Original file line numberDiff line numberDiff line change
@@ -27,7 +27,9 @@
2727
#ifndef MICROPY_INCLUDED_SHARED_BINDINGS_BITMAPTOOLS__INIT__H
2828
#define MICROPY_INCLUDED_SHARED_BINDINGS_BITMAPTOOLS__INIT__H
2929

30+
#include "shared-module/displayio/Bitmap.h"
3031
#include "py/obj.h"
32+
#include "extmod/vfs_fat.h"
3133

3234
void common_hal_bitmaptools_rotozoom(displayio_bitmap_t *self, int16_t ox, int16_t oy,
3335
int16_t dest_clip0_x, int16_t dest_clip0_y,
@@ -49,4 +51,6 @@ void common_hal_bitmaptools_draw_line(displayio_bitmap_t *destination,
4951
int16_t x1, int16_t y1,
5052
uint32_t value);
5153

54+
void common_hal_bitmaptools_readinto(displayio_bitmap_t *self, pyb_file_obj_t* file, int element_size, int bits_per_pixel, bool reverse_pixels_in_word, bool swap_bytes);
55+
5256
#endif // MICROPY_INCLUDED_SHARED_BINDINGS_BITMAPTOOLS__INIT__H

shared-module/bitmaptools/__init__.c

Lines changed: 90 additions & 0 deletions
Original file line numberDiff line numberDiff line change
@@ -25,10 +25,12 @@
2525
*/
2626

2727

28+
#include "shared-bindings/bitmaptools/__init__.h"
2829
#include "shared-bindings/displayio/Bitmap.h"
2930
#include "shared-module/displayio/Bitmap.h"
3031

3132
#include "py/runtime.h"
33+
#include "py/mperrno.h"
3234

3335
#include "math.h"
3436
#include "stdlib.h"
@@ -336,3 +338,91 @@ void common_hal_bitmaptools_draw_line(displayio_bitmap_t *destination,
336338
}
337339
}
338340
}
341+
342+
void common_hal_bitmaptools_readinto(displayio_bitmap_t *self, pyb_file_obj_t* file, int element_size, int bits_per_pixel, bool reverse_pixels_in_element, bool swap_bytes) {
343+
if (self->read_only) {
344+
mp_raise_RuntimeError(translate("Read-only object"));
345+
}
346+
347+
size_t elements_per_row = (self->width * bits_per_pixel + element_size * 8 - 1) / (element_size * 8);
348+
size_t rowsize = element_size * elements_per_row;
349+
size_t rowsize_in_u32 = (rowsize + sizeof(uint32_t) - 1) / sizeof(uint32_t);
350+
size_t rowsize_in_u16 = (rowsize + sizeof(uint16_t) - 1) / sizeof(uint16_t);
351+
for(int y=0; y<self->height; y++) {
352+
uint32_t rowdata32[rowsize_in_u32];
353+
uint16_t *rowdata16 = (uint16_t*)rowdata32;
354+
uint8_t *rowdata8 = (uint8_t*)rowdata32;
355+
356+
UINT bytes_read = 0;
357+
if (f_read(&file->fp, rowdata32, rowsize, &bytes_read) != FR_OK || bytes_read != rowsize) {
358+
mp_raise_OSError(MP_EIO);
359+
}
360+
361+
if (swap_bytes) {
362+
switch(element_size) {
363+
case 2:
364+
{
365+
for(int i=0; i< rowsize_in_u16; i++) {
366+
rowdata16[i] = __builtin_bswap16(rowdata16[i]);
367+
}
368+
}
369+
break;
370+
case 4:
371+
for(int i=0; i< rowsize_in_u32; i++) {
372+
rowdata32[i] = __builtin_bswap32(rowdata32[i]);
373+
}
374+
default:
375+
break;
376+
}
377+
}
378+
379+
for(int x=0; x<self->width; x++) {
380+
int value = 0;
381+
switch(bits_per_pixel) {
382+
case 1:
383+
{
384+
int byte_offset = x / 8;
385+
int bit_offset = reverse_pixels_in_element ? (7 - x % 8) : x % 8;
386+
387+
value = (rowdata8[byte_offset] >> bit_offset) & 1;
388+
break;
389+
}
390+
case 2:
391+
{
392+
int byte_offset = x / 4;
393+
int bit_offset = 2 * (reverse_pixels_in_element ? (3 - x % 4) : x % 4);
394+
395+
value = (rowdata8[byte_offset] >> bit_offset) & 3;
396+
break;
397+
}
398+
case 4:
399+
{
400+
int byte_offset = x / 2;
401+
int bit_offset = 4 * (reverse_pixels_in_element ? (1 - x % 2) : x % 2);
402+
403+
value = (rowdata8[byte_offset] >> bit_offset) & 7;
404+
break;
405+
}
406+
case 8:
407+
value = rowdata8[x];
408+
break;
409+
410+
case 16:
411+
value = rowdata16[x];
412+
break;
413+
414+
case 24:
415+
value = (rowdata8[x * 3] << 16) | (rowdata8[x * 3 + 1] << 8) | (rowdata8[x * 3 + 2] << 8);
416+
break;
417+
418+
case 32:
419+
value = rowdata32[x];
420+
break;
421+
}
422+
423+
displayio_bitmap_write_pixel(self, x, y, value);
424+
}
425+
}
426+
427+
displayio_bitmap_set_dirty_area(self, 0, 0, self->width, self->height);
428+
}

0 commit comments

Comments
 (0)