refactor: spanify image utils (#44127)

* refactor: electron::util::AddImageSkiaRepFromJPEG() takes a span arg

* refactor: electron::util::AddImageSkiaRepFromPNG() takes a span arg

* refactor: electron::util::AddImageSkiaRepFromBuffer() takes a span arg

* feat: add Node-buffer-to-base-span helper function

* refactor: electron::api::NativeImage::CreateFromPNG() now takes a span param

* refactor: electron::api::NativeImage::CreateFromJPEG() now takes a span param

* refactor: use base::as_byte_span()

* fix: -Wunsafe-buffer-usage warning in NativeImage::CreateFromNamedImage()

Warning fixed by this commit:

../../electron/shell/common/api/electron_api_native_image_mac.mm:131:11: error: function introduces unsafe buffer manipulation [-Werror,-Wunsafe-buffer-usage]
  131 |           {reinterpret_cast<const uint8_t*>((char*)[png_data bytes]),
      |           ^~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~
  132 |            [png_data length]});
      |            ~~~~~~~~~~~~~~~~~~
../../electron/shell/common/api/electron_api_native_image_mac.mm:131:11: note: See //docs/unsafe_buffers.md for help.

* chore: add // SAFETY comment for Node-buffer-to-span func

* chore: add // SAFETY comment for NSData-to-span func
This commit is contained in:
Charles Kerr 2024-10-10 08:34:55 -05:00 committed by GitHub
parent d51ef7d794
commit 3d2f68a9df
No known key found for this signature in database
GPG key ID: B5690EEEBB952194
7 changed files with 77 additions and 57 deletions

View file

@ -30,6 +30,7 @@
#include "shell/common/gin_helper/function_template_extensions.h" #include "shell/common/gin_helper/function_template_extensions.h"
#include "shell/common/gin_helper/object_template_builder.h" #include "shell/common/gin_helper/object_template_builder.h"
#include "shell/common/node_includes.h" #include "shell/common/node_includes.h"
#include "shell/common/node_util.h"
#include "shell/common/process_util.h" #include "shell/common/process_util.h"
#include "shell/common/skia_util.h" #include "shell/common/skia_util.h"
#include "shell/common/thread_restrictions.h" #include "shell/common/thread_restrictions.h"
@ -397,20 +398,18 @@ void NativeImage::AddRepresentation(const gin_helper::Dictionary& options) {
v8::Local<v8::Value> buffer; v8::Local<v8::Value> buffer;
GURL url; GURL url;
if (options.Get("buffer", &buffer) && node::Buffer::HasInstance(buffer)) { if (options.Get("buffer", &buffer) && node::Buffer::HasInstance(buffer)) {
auto* data = reinterpret_cast<unsigned char*>(node::Buffer::Data(buffer));
auto size = node::Buffer::Length(buffer);
skia_rep_added = electron::util::AddImageSkiaRepFromBuffer( skia_rep_added = electron::util::AddImageSkiaRepFromBuffer(
&image_skia, data, size, width, height, scale_factor); &image_skia, electron::util::as_byte_span(buffer), width, height,
scale_factor);
} else if (options.Get("dataURL", &url)) { } else if (options.Get("dataURL", &url)) {
std::string mime_type, charset, data; std::string mime_type, charset, data;
if (net::DataURL::Parse(url, &mime_type, &charset, &data)) { if (net::DataURL::Parse(url, &mime_type, &charset, &data)) {
auto* data_ptr = reinterpret_cast<const unsigned char*>(data.c_str());
if (mime_type == "image/png") { if (mime_type == "image/png") {
skia_rep_added = electron::util::AddImageSkiaRepFromPNG( skia_rep_added = electron::util::AddImageSkiaRepFromPNG(
&image_skia, data_ptr, data.size(), scale_factor); &image_skia, base::as_byte_span(data), scale_factor);
} else if (mime_type == "image/jpeg") { } else if (mime_type == "image/jpeg") {
skia_rep_added = electron::util::AddImageSkiaRepFromJPEG( skia_rep_added = electron::util::AddImageSkiaRepFromJPEG(
&image_skia, data_ptr, data.size(), scale_factor); &image_skia, base::as_byte_span(data), scale_factor);
} }
} }
} }
@ -442,22 +441,20 @@ gin::Handle<NativeImage> NativeImage::Create(v8::Isolate* isolate,
} }
// static // static
gin::Handle<NativeImage> NativeImage::CreateFromPNG(v8::Isolate* isolate, gin::Handle<NativeImage> NativeImage::CreateFromPNG(
const char* buffer, v8::Isolate* isolate,
size_t length) { const base::span<const uint8_t> data) {
gfx::ImageSkia image_skia; gfx::ImageSkia image_skia;
electron::util::AddImageSkiaRepFromPNG( electron::util::AddImageSkiaRepFromPNG(&image_skia, data, 1.0);
&image_skia, reinterpret_cast<const unsigned char*>(buffer), length, 1.0);
return Create(isolate, gfx::Image(image_skia)); return Create(isolate, gfx::Image(image_skia));
} }
// static // static
gin::Handle<NativeImage> NativeImage::CreateFromJPEG(v8::Isolate* isolate, gin::Handle<NativeImage> NativeImage::CreateFromJPEG(
const char* buffer, v8::Isolate* isolate,
size_t length) { const base::span<const uint8_t> buffer) {
gfx::ImageSkia image_skia; gfx::ImageSkia image_skia;
electron::util::AddImageSkiaRepFromJPEG( electron::util::AddImageSkiaRepFromJPEG(&image_skia, buffer, 1.0);
&image_skia, reinterpret_cast<const unsigned char*>(buffer), length, 1.0);
return Create(isolate, gfx::Image(image_skia)); return Create(isolate, gfx::Image(image_skia));
} }
@ -509,7 +506,8 @@ gin::Handle<NativeImage> NativeImage::CreateFromBitmap(
auto info = SkImageInfo::MakeN32(width, height, kPremul_SkAlphaType); auto info = SkImageInfo::MakeN32(width, height, kPremul_SkAlphaType);
auto size_bytes = info.computeMinByteSize(); auto size_bytes = info.computeMinByteSize();
if (size_bytes != node::Buffer::Length(buffer)) { const auto buffer_data = electron::util::as_byte_span(buffer);
if (size_bytes != buffer_data.size()) {
thrower.ThrowError("invalid buffer size"); thrower.ThrowError("invalid buffer size");
return gin::Handle<NativeImage>(); return gin::Handle<NativeImage>();
} }
@ -522,7 +520,7 @@ gin::Handle<NativeImage> NativeImage::CreateFromBitmap(
SkBitmap bitmap; SkBitmap bitmap;
bitmap.allocN32Pixels(width, height, false); bitmap.allocN32Pixels(width, height, false);
bitmap.writePixels({info, node::Buffer::Data(buffer), bitmap.rowBytes()}); bitmap.writePixels({info, buffer_data.data(), bitmap.rowBytes()});
gfx::ImageSkia image_skia = gfx::ImageSkia image_skia =
gfx::ImageSkia::CreateFromBitmap(bitmap, scale_factor); gfx::ImageSkia::CreateFromBitmap(bitmap, scale_factor);
@ -553,8 +551,8 @@ gin::Handle<NativeImage> NativeImage::CreateFromBuffer(
gfx::ImageSkia image_skia; gfx::ImageSkia image_skia;
electron::util::AddImageSkiaRepFromBuffer( electron::util::AddImageSkiaRepFromBuffer(
&image_skia, reinterpret_cast<unsigned char*>(node::Buffer::Data(buffer)), &image_skia, electron::util::as_byte_span(buffer), width, height,
node::Buffer::Length(buffer), width, height, scale_factor); scale_factor);
return Create(args->isolate(), gfx::Image(image_skia)); return Create(args->isolate(), gfx::Image(image_skia));
} }
@ -564,9 +562,9 @@ gin::Handle<NativeImage> NativeImage::CreateFromDataURL(v8::Isolate* isolate,
std::string mime_type, charset, data; std::string mime_type, charset, data;
if (net::DataURL::Parse(url, &mime_type, &charset, &data)) { if (net::DataURL::Parse(url, &mime_type, &charset, &data)) {
if (mime_type == "image/png") if (mime_type == "image/png")
return CreateFromPNG(isolate, data.c_str(), data.size()); return CreateFromPNG(isolate, base::as_byte_span(data));
else if (mime_type == "image/jpeg") if (mime_type == "image/jpeg")
return CreateFromJPEG(isolate, data.c_str(), data.size()); return CreateFromJPEG(isolate, base::as_byte_span(data));
} }
return CreateEmpty(isolate); return CreateEmpty(isolate);

View file

@ -9,6 +9,7 @@
#include <vector> #include <vector>
#include "base/containers/flat_map.h" #include "base/containers/flat_map.h"
#include "base/containers/span.h"
#include "base/memory/raw_ptr.h" #include "base/memory/raw_ptr.h"
#include "base/values.h" #include "base/values.h"
#include "gin/wrappable.h" #include "gin/wrappable.h"
@ -61,11 +62,10 @@ class NativeImage final : public gin::Wrappable<NativeImage> {
static gin::Handle<NativeImage> Create(v8::Isolate* isolate, static gin::Handle<NativeImage> Create(v8::Isolate* isolate,
const gfx::Image& image); const gfx::Image& image);
static gin::Handle<NativeImage> CreateFromPNG(v8::Isolate* isolate, static gin::Handle<NativeImage> CreateFromPNG(v8::Isolate* isolate,
const char* buffer, base::span<const uint8_t> data);
size_t length); static gin::Handle<NativeImage> CreateFromJPEG(
static gin::Handle<NativeImage> CreateFromJPEG(v8::Isolate* isolate, v8::Isolate* isolate,
const char* buffer, base::span<const uint8_t> data);
size_t length);
static gin::Handle<NativeImage> CreateFromPath(v8::Isolate* isolate, static gin::Handle<NativeImage> CreateFromPath(v8::Isolate* isolate,
const base::FilePath& path); const base::FilePath& path);
static gin::Handle<NativeImage> CreateFromBitmap( static gin::Handle<NativeImage> CreateFromBitmap(

View file

@ -26,6 +26,19 @@
namespace electron::api { namespace electron::api {
namespace {
base::span<const uint8_t> as_byte_span(NSData* data) {
// SAFETY: There is no NSData API that passes the UNSAFE_BUFFER_USAGE
// test, so let's isolate the unsafe API use into this function. Instead of
// calling '[data bytes]' and '[data length]' directly, the rest of our
// code should prefer to use spans returned by this function.
return UNSAFE_BUFFERS(base::span{
reinterpret_cast<const uint8_t*>([data bytes]), [data length]});
}
} // namespace
NSData* bufferFromNSImage(NSImage* image) { NSData* bufferFromNSImage(NSImage* image) {
CGImageRef ref = [image CGImageForProposedRect:nil context:nil hints:nil]; CGImageRef ref = [image CGImageForProposedRect:nil context:nil hints:nil];
NSBitmapImageRep* rep = [[NSBitmapImageRep alloc] initWithCGImage:ref]; NSBitmapImageRep* rep = [[NSBitmapImageRep alloc] initWithCGImage:ref];
@ -127,9 +140,7 @@ gin::Handle<NativeImage> NativeImage::CreateFromNamedImage(gin::Arguments* args,
NSData* png_data = bufferFromNSImage(image); NSData* png_data = bufferFromNSImage(image);
if (args->GetNext(&hsl_shift) && hsl_shift.size() == 3) { if (args->GetNext(&hsl_shift) && hsl_shift.size() == 3) {
gfx::Image gfx_image = gfx::Image::CreateFrom1xPNGBytes( auto gfx_image = gfx::Image::CreateFrom1xPNGBytes(as_byte_span(png_data));
{reinterpret_cast<const uint8_t*>((char*)[png_data bytes]),
[png_data length]});
color_utils::HSL shift = {safeShift(hsl_shift[0], -1), color_utils::HSL shift = {safeShift(hsl_shift[0], -1),
safeShift(hsl_shift[1], 0.5), safeShift(hsl_shift[1], 0.5),
safeShift(hsl_shift[2], 0.5)}; safeShift(hsl_shift[2], 0.5)};
@ -139,8 +150,7 @@ gin::Handle<NativeImage> NativeImage::CreateFromNamedImage(gin::Arguments* args,
.AsNSImage()); .AsNSImage());
} }
return CreateFromPNG(args->isolate(), (char*)[png_data bytes], return CreateFromPNG(args->isolate(), as_byte_span(png_data));
[png_data length]);
} }
} }

View file

@ -4,6 +4,7 @@
#include "shell/common/node_util.h" #include "shell/common/node_util.h"
#include "base/compiler_specific.h"
#include "base/logging.h" #include "base/logging.h"
#include "gin/converter.h" #include "gin/converter.h"
#include "gin/dictionary.h" #include "gin/dictionary.h"
@ -65,4 +66,14 @@ void EmitWarning(v8::Isolate* isolate,
emit_warning.Run(warning_msg, warning_type, ""); emit_warning.Run(warning_msg, warning_type, "");
} }
// SAFETY: There is no node::Buffer API that passes the UNSAFE_BUFFER_USAGE
// test, so let's isolate the unsafe API use into this function. Instead of
// calling `Buffer::Data()` and `Buffer::Length()` directly, the rest of our
// code should prefer to use spans returned by this function.
base::span<uint8_t> as_byte_span(v8::Local<v8::Value> node_buffer) {
auto* data = reinterpret_cast<uint8_t*>(node::Buffer::Data(node_buffer));
const auto size = node::Buffer::Length(node_buffer);
return UNSAFE_BUFFERS(base::span{data, size});
}
} // namespace electron::util } // namespace electron::util

View file

@ -8,6 +8,7 @@
#include <string_view> #include <string_view>
#include <vector> #include <vector>
#include "base/containers/span.h"
#include "v8/include/v8-forward.h" #include "v8/include/v8-forward.h"
namespace node { namespace node {
@ -36,6 +37,11 @@ v8::MaybeLocal<v8::Value> CompileAndCall(
std::vector<v8::Local<v8::String>>* parameters, std::vector<v8::Local<v8::String>>* parameters,
std::vector<v8::Local<v8::Value>>* arguments); std::vector<v8::Local<v8::Value>>* arguments);
// Convenience function to view a Node buffer's data as a base::span().
// Analogous to base::as_byte_span()
[[nodiscard]] base::span<uint8_t> as_byte_span(
v8::Local<v8::Value> node_buffer);
} // namespace electron::util } // namespace electron::util
#endif // ELECTRON_SHELL_COMMON_NODE_UTIL_H_ #endif // ELECTRON_SHELL_COMMON_NODE_UTIL_H_

View file

@ -56,11 +56,10 @@ float GetScaleFactorFromPath(const base::FilePath& path) {
} }
bool AddImageSkiaRepFromPNG(gfx::ImageSkia* image, bool AddImageSkiaRepFromPNG(gfx::ImageSkia* image,
const unsigned char* data, const base::span<const uint8_t> data,
size_t size,
double scale_factor) { double scale_factor) {
SkBitmap bitmap; SkBitmap bitmap;
if (!gfx::PNGCodec::Decode(data, size, &bitmap)) if (!gfx::PNGCodec::Decode(data.data(), data.size(), &bitmap))
return false; return false;
image->AddRepresentation(gfx::ImageSkiaRep(bitmap, scale_factor)); image->AddRepresentation(gfx::ImageSkiaRep(bitmap, scale_factor));
@ -68,10 +67,9 @@ bool AddImageSkiaRepFromPNG(gfx::ImageSkia* image,
} }
bool AddImageSkiaRepFromJPEG(gfx::ImageSkia* image, bool AddImageSkiaRepFromJPEG(gfx::ImageSkia* image,
const unsigned char* data, const base::span<const uint8_t> data,
size_t size,
double scale_factor) { double scale_factor) {
auto bitmap = gfx::JPEGCodec::Decode(data, size); auto bitmap = gfx::JPEGCodec::Decode(data.data(), data.size());
if (!bitmap) if (!bitmap)
return false; return false;
@ -89,29 +87,28 @@ bool AddImageSkiaRepFromJPEG(gfx::ImageSkia* image,
} }
bool AddImageSkiaRepFromBuffer(gfx::ImageSkia* image, bool AddImageSkiaRepFromBuffer(gfx::ImageSkia* image,
const unsigned char* data, const base::span<const uint8_t> data,
size_t size,
int width, int width,
int height, int height,
double scale_factor) { double scale_factor) {
// Try PNG first. // Try PNG first.
if (AddImageSkiaRepFromPNG(image, data, size, scale_factor)) if (AddImageSkiaRepFromPNG(image, data, scale_factor))
return true; return true;
// Try JPEG second. // Try JPEG second.
if (AddImageSkiaRepFromJPEG(image, data, size, scale_factor)) if (AddImageSkiaRepFromJPEG(image, data, scale_factor))
return true; return true;
if (width == 0 || height == 0) if (width == 0 || height == 0)
return false; return false;
auto info = SkImageInfo::MakeN32(width, height, kPremul_SkAlphaType); auto info = SkImageInfo::MakeN32(width, height, kPremul_SkAlphaType);
if (size < info.computeMinByteSize()) if (data.size() < info.computeMinByteSize())
return false; return false;
SkBitmap bitmap; SkBitmap bitmap;
bitmap.allocN32Pixels(width, height, false); bitmap.allocN32Pixels(width, height, false);
bitmap.writePixels({info, data, bitmap.rowBytes()}); bitmap.writePixels({info, data.data(), bitmap.rowBytes()});
image->AddRepresentation(gfx::ImageSkiaRep(bitmap, scale_factor)); image->AddRepresentation(gfx::ImageSkiaRep(bitmap, scale_factor));
return true; return true;
@ -127,11 +124,8 @@ bool AddImageSkiaRepFromPath(gfx::ImageSkia* image,
return false; return false;
} }
const auto* data = return AddImageSkiaRepFromBuffer(image, base::as_byte_span(file_contents), 0,
reinterpret_cast<const unsigned char*>(file_contents.data()); 0, scale_factor);
size_t size = file_contents.size();
return AddImageSkiaRepFromBuffer(image, data, size, 0, 0, scale_factor);
} }
bool PopulateImageSkiaRepsFromPath(gfx::ImageSkia* image, bool PopulateImageSkiaRepsFromPath(gfx::ImageSkia* image,

View file

@ -5,9 +5,13 @@
#ifndef ELECTRON_SHELL_COMMON_SKIA_UTIL_H_ #ifndef ELECTRON_SHELL_COMMON_SKIA_UTIL_H_
#define ELECTRON_SHELL_COMMON_SKIA_UTIL_H_ #define ELECTRON_SHELL_COMMON_SKIA_UTIL_H_
#include <cstdint>
#include "base/containers/span.h"
namespace base { namespace base {
class FilePath; class FilePath;
} } // namespace base
namespace gfx { namespace gfx {
class ImageSkia; class ImageSkia;
@ -19,20 +23,17 @@ bool PopulateImageSkiaRepsFromPath(gfx::ImageSkia* image,
const base::FilePath& path); const base::FilePath& path);
bool AddImageSkiaRepFromBuffer(gfx::ImageSkia* image, bool AddImageSkiaRepFromBuffer(gfx::ImageSkia* image,
const unsigned char* data, base::span<const uint8_t> data,
size_t size,
int width, int width,
int height, int height,
double scale_factor); double scale_factor);
bool AddImageSkiaRepFromJPEG(gfx::ImageSkia* image, bool AddImageSkiaRepFromJPEG(gfx::ImageSkia* image,
const unsigned char* data, base::span<const uint8_t> data,
size_t size,
double scale_factor); double scale_factor);
bool AddImageSkiaRepFromPNG(gfx::ImageSkia* image, bool AddImageSkiaRepFromPNG(gfx::ImageSkia* image,
const unsigned char* data, base::span<const uint8_t> data,
size_t size,
double scale_factor); double scale_factor);
#if BUILDFLAG(IS_WIN) #if BUILDFLAG(IS_WIN)