1cb93a386Sopenharmony_ci/* 2cb93a386Sopenharmony_ci * Copyright 2015 Google Inc. 3cb93a386Sopenharmony_ci * 4cb93a386Sopenharmony_ci * Use of this source code is governed by a BSD-style license that can be 5cb93a386Sopenharmony_ci * found in the LICENSE file. 6cb93a386Sopenharmony_ci */ 7cb93a386Sopenharmony_ci 8cb93a386Sopenharmony_ci#include "include/core/SkTypes.h" 9cb93a386Sopenharmony_ci#include "src/codec/SkCodecPriv.h" 10cb93a386Sopenharmony_ci#include "src/codec/SkMasks.h" 11cb93a386Sopenharmony_ci 12cb93a386Sopenharmony_ci/* 13cb93a386Sopenharmony_ci * 14cb93a386Sopenharmony_ci * Used to convert 1-7 bit color components into 8-bit color components 15cb93a386Sopenharmony_ci * 16cb93a386Sopenharmony_ci */ 17cb93a386Sopenharmony_cistatic constexpr uint8_t n_bit_to_8_bit_lookup_table[] = { 18cb93a386Sopenharmony_ci // 1 bit 19cb93a386Sopenharmony_ci 0, 255, 20cb93a386Sopenharmony_ci // 2 bits 21cb93a386Sopenharmony_ci 0, 85, 170, 255, 22cb93a386Sopenharmony_ci // 3 bits 23cb93a386Sopenharmony_ci 0, 36, 73, 109, 146, 182, 219, 255, 24cb93a386Sopenharmony_ci // 4 bits 25cb93a386Sopenharmony_ci 0, 17, 34, 51, 68, 85, 102, 119, 136, 153, 170, 187, 204, 221, 238, 255, 26cb93a386Sopenharmony_ci // 5 bits 27cb93a386Sopenharmony_ci 0, 8, 16, 25, 33, 41, 49, 58, 66, 74, 82, 90, 99, 107, 115, 123, 132, 140, 28cb93a386Sopenharmony_ci 148, 156, 165, 173, 181, 189, 197, 206, 214, 222, 230, 239, 247, 255, 29cb93a386Sopenharmony_ci // 6 bits 30cb93a386Sopenharmony_ci 0, 4, 8, 12, 16, 20, 24, 28, 32, 36, 40, 45, 49, 53, 57, 61, 65, 69, 73, 31cb93a386Sopenharmony_ci 77, 81, 85, 89, 93, 97, 101, 105, 109, 113, 117, 121, 125, 130, 134, 138, 32cb93a386Sopenharmony_ci 142, 146, 150, 154, 158, 162, 166, 170, 174, 178, 182, 186, 190, 194, 198, 33cb93a386Sopenharmony_ci 202, 206, 210, 215, 219, 223, 227, 231, 235, 239, 243, 247, 251, 255, 34cb93a386Sopenharmony_ci // 7 bits 35cb93a386Sopenharmony_ci 0, 2, 4, 6, 8, 10, 12, 14, 16, 18, 20, 22, 24, 26, 28, 30, 32, 34, 36, 38, 36cb93a386Sopenharmony_ci 40, 42, 44, 46, 48, 50, 52, 54, 56, 58, 60, 62, 64, 66, 68, 70, 72, 74, 76, 37cb93a386Sopenharmony_ci 78, 80, 82, 84, 86, 88, 90, 92, 94, 96, 98, 100, 102, 104, 106, 108, 110, 38cb93a386Sopenharmony_ci 112, 114, 116, 118, 120, 122, 124, 126, 129, 131, 133, 135, 137, 139, 141, 39cb93a386Sopenharmony_ci 143, 145, 147, 149, 151, 153, 155, 157, 159, 161, 163, 165, 167, 169, 171, 40cb93a386Sopenharmony_ci 173, 175, 177, 179, 181, 183, 185, 187, 189, 191, 193, 195, 197, 199, 201, 41cb93a386Sopenharmony_ci 203, 205, 207, 209, 211, 213, 215, 217, 219, 221, 223, 225, 227, 229, 231, 42cb93a386Sopenharmony_ci 233, 235, 237, 239, 241, 243, 245, 247, 249, 251, 253, 255 43cb93a386Sopenharmony_ci}; 44cb93a386Sopenharmony_ci 45cb93a386Sopenharmony_ci/* 46cb93a386Sopenharmony_ci * 47cb93a386Sopenharmony_ci * Convert an n bit component to an 8-bit component 48cb93a386Sopenharmony_ci * 49cb93a386Sopenharmony_ci */ 50cb93a386Sopenharmony_cistatic uint8_t convert_to_8(uint8_t component, uint32_t n) { 51cb93a386Sopenharmony_ci if (0 == n) { 52cb93a386Sopenharmony_ci return 0; 53cb93a386Sopenharmony_ci } else if (8 > n) { 54cb93a386Sopenharmony_ci return n_bit_to_8_bit_lookup_table[(1 << n) - 2 + component]; 55cb93a386Sopenharmony_ci } else { 56cb93a386Sopenharmony_ci SkASSERT(8 == n); 57cb93a386Sopenharmony_ci return component; 58cb93a386Sopenharmony_ci } 59cb93a386Sopenharmony_ci} 60cb93a386Sopenharmony_ci 61cb93a386Sopenharmony_cistatic uint8_t get_comp(uint32_t pixel, uint32_t mask, uint32_t shift, 62cb93a386Sopenharmony_ci uint32_t size) { 63cb93a386Sopenharmony_ci return convert_to_8((pixel & mask) >> shift, size); 64cb93a386Sopenharmony_ci} 65cb93a386Sopenharmony_ci 66cb93a386Sopenharmony_ci/* 67cb93a386Sopenharmony_ci * 68cb93a386Sopenharmony_ci * Get a color component 69cb93a386Sopenharmony_ci * 70cb93a386Sopenharmony_ci */ 71cb93a386Sopenharmony_ciuint8_t SkMasks::getRed(uint32_t pixel) const { 72cb93a386Sopenharmony_ci return get_comp(pixel, fRed.mask, fRed.shift, fRed.size); 73cb93a386Sopenharmony_ci} 74cb93a386Sopenharmony_ciuint8_t SkMasks::getGreen(uint32_t pixel) const { 75cb93a386Sopenharmony_ci return get_comp(pixel, fGreen.mask, fGreen.shift, fGreen.size); 76cb93a386Sopenharmony_ci} 77cb93a386Sopenharmony_ciuint8_t SkMasks::getBlue(uint32_t pixel) const { 78cb93a386Sopenharmony_ci return get_comp(pixel, fBlue.mask, fBlue.shift, fBlue.size); 79cb93a386Sopenharmony_ci} 80cb93a386Sopenharmony_ciuint8_t SkMasks::getAlpha(uint32_t pixel) const { 81cb93a386Sopenharmony_ci return get_comp(pixel, fAlpha.mask, fAlpha.shift, fAlpha.size); 82cb93a386Sopenharmony_ci} 83cb93a386Sopenharmony_ci 84cb93a386Sopenharmony_ci/* 85cb93a386Sopenharmony_ci * 86cb93a386Sopenharmony_ci * Process an input mask to obtain the necessary information 87cb93a386Sopenharmony_ci * 88cb93a386Sopenharmony_ci */ 89cb93a386Sopenharmony_cistatic SkMasks::MaskInfo process_mask(uint32_t mask) { 90cb93a386Sopenharmony_ci // Determine properties of the mask 91cb93a386Sopenharmony_ci uint32_t tempMask = mask; 92cb93a386Sopenharmony_ci uint32_t shift = 0; 93cb93a386Sopenharmony_ci uint32_t size = 0; 94cb93a386Sopenharmony_ci if (tempMask != 0) { 95cb93a386Sopenharmony_ci // Count trailing zeros on masks 96cb93a386Sopenharmony_ci for (; (tempMask & 1) == 0; tempMask >>= 1) { 97cb93a386Sopenharmony_ci shift++; 98cb93a386Sopenharmony_ci } 99cb93a386Sopenharmony_ci // Count the size of the mask 100cb93a386Sopenharmony_ci for (; tempMask & 1; tempMask >>= 1) { 101cb93a386Sopenharmony_ci size++; 102cb93a386Sopenharmony_ci } 103cb93a386Sopenharmony_ci // Verify that the mask is continuous 104cb93a386Sopenharmony_ci if (tempMask) { 105cb93a386Sopenharmony_ci SkCodecPrintf("Warning: Bit mask is not continuous.\n"); 106cb93a386Sopenharmony_ci // Finish processing the mask 107cb93a386Sopenharmony_ci for (; tempMask; tempMask >>= 1) { 108cb93a386Sopenharmony_ci size++; 109cb93a386Sopenharmony_ci } 110cb93a386Sopenharmony_ci } 111cb93a386Sopenharmony_ci // Truncate masks greater than 8 bits 112cb93a386Sopenharmony_ci if (size > 8) { 113cb93a386Sopenharmony_ci shift += size - 8; 114cb93a386Sopenharmony_ci size = 8; 115cb93a386Sopenharmony_ci mask &= 0xFF << shift; 116cb93a386Sopenharmony_ci } 117cb93a386Sopenharmony_ci } 118cb93a386Sopenharmony_ci 119cb93a386Sopenharmony_ci return { mask, shift, size }; 120cb93a386Sopenharmony_ci} 121cb93a386Sopenharmony_ci 122cb93a386Sopenharmony_ci/* 123cb93a386Sopenharmony_ci * 124cb93a386Sopenharmony_ci * Create the masks object 125cb93a386Sopenharmony_ci * 126cb93a386Sopenharmony_ci */ 127cb93a386Sopenharmony_ciSkMasks* SkMasks::CreateMasks(InputMasks masks, int bytesPerPixel) { 128cb93a386Sopenharmony_ci SkASSERT(0 < bytesPerPixel && bytesPerPixel <= 4); 129cb93a386Sopenharmony_ci 130cb93a386Sopenharmony_ci // Trim the input masks to match bytesPerPixel. 131cb93a386Sopenharmony_ci if (bytesPerPixel < 4) { 132cb93a386Sopenharmony_ci int bitsPerPixel = 8*bytesPerPixel; 133cb93a386Sopenharmony_ci masks.red &= (1 << bitsPerPixel) - 1; 134cb93a386Sopenharmony_ci masks.green &= (1 << bitsPerPixel) - 1; 135cb93a386Sopenharmony_ci masks.blue &= (1 << bitsPerPixel) - 1; 136cb93a386Sopenharmony_ci masks.alpha &= (1 << bitsPerPixel) - 1; 137cb93a386Sopenharmony_ci } 138cb93a386Sopenharmony_ci 139cb93a386Sopenharmony_ci // Check that masks do not overlap. 140cb93a386Sopenharmony_ci if (((masks.red & masks.green) | 141cb93a386Sopenharmony_ci (masks.red & masks.blue ) | 142cb93a386Sopenharmony_ci (masks.red & masks.alpha) | 143cb93a386Sopenharmony_ci (masks.green & masks.blue ) | 144cb93a386Sopenharmony_ci (masks.green & masks.alpha) | 145cb93a386Sopenharmony_ci (masks.blue & masks.alpha) ) != 0) { 146cb93a386Sopenharmony_ci return nullptr; 147cb93a386Sopenharmony_ci } 148cb93a386Sopenharmony_ci 149cb93a386Sopenharmony_ci return new SkMasks(process_mask(masks.red ), 150cb93a386Sopenharmony_ci process_mask(masks.green), 151cb93a386Sopenharmony_ci process_mask(masks.blue ), 152cb93a386Sopenharmony_ci process_mask(masks.alpha)); 153cb93a386Sopenharmony_ci} 154cb93a386Sopenharmony_ci 155