pebble/src/fw/applib/graphics/graphics_mask.c
2025-01-27 11:38:16 -08:00

129 lines
4.3 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 "gcontext.h"
#include "graphics_private_raw.h"
#include "graphics_private_raw_mask.h"
#include "applib/applib_malloc.auto.h"
#include <string.h>
GDrawMask *graphics_context_mask_create(const GContext *ctx, bool transparent) {
#if CAPABILITY_HAS_MASKING
if (!ctx) {
return NULL;
}
const GBitmap *framebuffer_bitmap = &ctx->dest_bitmap;
const int framebuffer_bitmap_height = framebuffer_bitmap->bounds.size.h;
const size_t num_bytes_needed_for_mask_row_infos =
sizeof(GDrawMaskRowInfo) * framebuffer_bitmap_height;
// Iterate over framebuffer data row infos to calculate the Bytes needed for the pixel_mask_data
size_t num_pixels = 0;
for (int y = 0; y < framebuffer_bitmap_height; y++) {
const GBitmapDataRowInfo row_info = gbitmap_get_data_row_info(framebuffer_bitmap, (uint16_t)y);
const int row_width = row_info.max_x - row_info.min_x + 1;
num_pixels += row_width;
}
// Round up after dividing by the mask bits per pixel
const size_t num_bytes_needed_for_pixel_mask_data = DIVIDE_CEIL(num_pixels,
GDRAWMASK_BITS_PER_PIXEL);
GDrawMask *result = applib_zalloc(
sizeof(*result) + num_bytes_needed_for_mask_row_infos + num_bytes_needed_for_pixel_mask_data);
if (result) {
// We store the mask_row_infos first in the .data buffer, followed by the pixel_mask_data
*result = (GDrawMask) {
.mask_row_infos = (GDrawMaskRowInfo *)result->data,
.pixel_mask_data = ((uint8_t *)result->data) + num_bytes_needed_for_mask_row_infos,
};
// Initialize the mask according to the `transparent` argument
const uint8_t pixel_data_initial_byte_value = transparent ? (uint8_t)0b00000000 :
(uint8_t)0b11111111;
memset(result->pixel_mask_data, pixel_data_initial_byte_value,
num_bytes_needed_for_pixel_mask_data);
// Initialize the mask row infos
const uint16_t fixed_s16_s3_fraction_max_value = FIXED_S16_3_FACTOR - 1;
for (int y = 0; y < framebuffer_bitmap_height; y++) {
const GBitmapDataRowInfo row_info = gbitmap_get_data_row_info(framebuffer_bitmap,
(uint16_t)y);
result->mask_row_infos[y] = (GDrawMaskRowInfo) {
.type = transparent ? GDrawMaskRowInfoType_SemiTransparent : GDrawMaskRowInfoType_Opaque,
.min_x.integer = row_info.min_x,
.min_x.fraction = (uint16_t)(transparent ? fixed_s16_s3_fraction_max_value : 0),
.max_x.integer = row_info.max_x,
.max_x.fraction = (uint16_t)(transparent ? 0 : fixed_s16_s3_fraction_max_value),
};
}
}
return result;
#else
return NULL;
#endif
}
bool graphics_context_mask_record(GContext *ctx, GDrawMask *mask) {
#if CAPABILITY_HAS_MASKING
if (!ctx) {
return false;
}
if (ctx->draw_state.draw_mask && !mask) {
// TODO PBL-33766: Update the ctx->draw_state.draw_mask's .mask_row_infos
}
const GDrawRawImplementation *draw_implementation_to_set =
mask ? &g_mask_recording_draw_implementation : &g_default_draw_implementation;
ctx->draw_state.draw_implementation = draw_implementation_to_set;
ctx->draw_state.draw_mask = mask;
return true;
#else
return false;
#endif
}
bool graphics_context_mask_use(GContext *ctx, GDrawMask *mask) {
#if CAPABILITY_HAS_MASKING
if (!ctx) {
return false;
}
// Stop any recording
graphics_context_mask_record(ctx, NULL);
// If a valid mask is set, the default draw implementation routines will respect it
ctx->draw_state.draw_mask = mask;
return true;
#else
return false;
#endif
}
void graphics_context_mask_destroy(GContext *ctx, GDrawMask *mask) {
#if CAPABILITY_HAS_MASKING
graphics_context_mask_use(ctx, NULL);
applib_free(mask);
#endif
}