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