/* * Copyright 2015 Google Inc. * * Use of this source code is governed by a BSD-style license that can be * found in the LICENSE file. */ #include "SkCodec.h" #include "SkCodecPriv.h" #include "SkSampler.h" #include "SkUtils.h" void SkSampler::Fill(const SkImageInfo& info, void* dst, size_t rowBytes, uint32_t colorOrIndex, SkCodec::ZeroInitialized zeroInit) { SkASSERT(dst != nullptr); // Calculate bytes to fill. We use getSafeSize since the last row may not be padded. const size_t bytesToFill = info.getSafeSize(rowBytes); const int width = info.width(); const int numRows = info.height(); // Use the proper memset routine to fill the remaining bytes switch (info.colorType()) { case kN32_SkColorType: { // If memory is zero initialized, we may not need to fill uint32_t color = colorOrIndex; if (SkCodec::kYes_ZeroInitialized == zeroInit && 0 == color) { return; } // We must fill row by row in the case of unaligned row bytes if (SkIsAlign4((size_t) dst) && SkIsAlign4(rowBytes)) { sk_memset32((uint32_t*) dst, color, (uint32_t) bytesToFill / sizeof(SkPMColor)); } else { // We must fill row by row in the case of unaligned row bytes. This is an // unlikely, slow case. SkCodecPrintf("Warning: Strange number of row bytes, fill will be slow.\n"); uint32_t* dstRow = (uint32_t*) dst; for (int row = 0; row < numRows; row++) { for (int col = 0; col < width; col++) { dstRow[col] = color; } dstRow = SkTAddOffset<uint32_t>(dstRow, rowBytes); } } break; } case kRGB_565_SkColorType: { // If the destination is k565, the caller passes in a 16-bit color. // We will not assert that the high bits of colorOrIndex must be zeroed. // This allows us to take advantage of the fact that the low 16 bits of an // SKPMColor may be a valid a 565 color. For example, the low 16 // bits of SK_ColorBLACK are identical to the 565 representation // for black. // If memory is zero initialized, we may not need to fill uint16_t color = (uint16_t) colorOrIndex; if (SkCodec::kYes_ZeroInitialized == zeroInit && 0 == color) { return; } if (SkIsAlign2((size_t) dst) && SkIsAlign2(rowBytes)) { sk_memset16((uint16_t*) dst, color, (uint32_t) bytesToFill / sizeof(uint16_t)); } else { // We must fill row by row in the case of unaligned row bytes. This is an // unlikely, slow case. SkCodecPrintf("Warning: Strange number of row bytes, fill will be slow.\n"); uint16_t* dstRow = (uint16_t*) dst; for (int row = 0; row < numRows; row++) { for (int col = 0; col < width; col++) { dstRow[col] = color; } dstRow = SkTAddOffset<uint16_t>(dstRow, rowBytes); } } break; } case kIndex_8_SkColorType: // On an index destination color type, always assume the input is an index. // Fall through case kGray_8_SkColorType: // If the destination is kGray, the caller passes in an 8-bit color. // We will not assert that the high bits of colorOrIndex must be zeroed. // This allows us to take advantage of the fact that the low 8 bits of an // SKPMColor may be a valid a grayscale color. For example, the low 8 // bits of SK_ColorBLACK are identical to the grayscale representation // for black. // If memory is zero initialized, we may not need to fill if (SkCodec::kYes_ZeroInitialized == zeroInit && 0 == (uint8_t) colorOrIndex) { return; } memset(dst, (uint8_t) colorOrIndex, bytesToFill); break; default: SkCodecPrintf("Error: Unsupported dst color type for fill(). Doing nothing.\n"); SkASSERT(false); break; } }