-
Notifications
You must be signed in to change notification settings - Fork 806
New issue
Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.
By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.
Already on GitHub? Sign in to your account
[CGD2D] Render buffer-shared G8 images as A8. #1804
Changes from all commits
ed27f7a
011ba77
079cecc
8d3df05
9769497
1565b7a
a396419
File filter
Filter by extension
Conversations
Jump to
Diff view
Diff view
There are no files selected for viewing
Original file line number | Diff line number | Diff line change |
---|---|---|
|
@@ -2782,10 +2782,43 @@ bool CGContextIsPointInPath(CGContextRef context, bool eoFill, CGFloat x, CGFloa | |
struct __CGBitmapContext : CoreFoundation::CppBase<__CGBitmapContext, __CGContext> { | ||
woc::unique_cf<CGImageRef> _image; | ||
|
||
void* _data; | ||
|
||
size_t _width; | ||
size_t _height; | ||
size_t _stride; | ||
|
||
CGBitmapContextReleaseDataCallback _releaseDataCallback; | ||
There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. nit: make these private? nobody should be allowed to muck around with 'em |
||
void* _releaseInfo; | ||
|
||
__CGBitmapContext(ID2D1RenderTarget* renderTarget, REFWICPixelFormatGUID outputPixelFormat) | ||
: Parent(renderTarget), _outputPixelFormat(outputPixelFormat) { | ||
} | ||
|
||
__CGBitmapContext(ID2D1RenderTarget* renderTarget, | ||
REFWICPixelFormatGUID outputPixelFormat, | ||
void* data, | ||
size_t width, | ||
size_t height, | ||
size_t stride, | ||
CGBitmapContextReleaseDataCallback releaseDataCallback, | ||
void* releaseInfo) | ||
: Parent(renderTarget), | ||
_outputPixelFormat(outputPixelFormat), | ||
_data(data), | ||
_width(width), | ||
_height(height), | ||
_stride(stride), | ||
_releaseDataCallback(releaseDataCallback), | ||
_releaseInfo(releaseInfo) { | ||
} | ||
|
||
~__CGBitmapContext() { | ||
if (_data && _releaseDataCallback) { | ||
_releaseDataCallback(_releaseInfo, _data); | ||
There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. will the reference platform call the release callback if data == nullptr? There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. No; there's no data to be released. |
||
} | ||
} | ||
|
||
inline void SetImage(CGImageRef image) { | ||
_image.reset(CGImageRetain(image)); | ||
} | ||
|
@@ -2814,39 +2847,55 @@ CGContextRef CGBitmapContextCreate(void* data, | |
|
||
/** | ||
@Status Caveat | ||
@Notes releaseCallback and releaseInfo is ignored. | ||
We only support formats that are 32 bits per pixel, colorspace and bitmapinfo that are ARGB. | ||
@Notes If data is provided, it can only be in one of the few pixel formats Direct2D can render to in system memory: (P)RGBA, (P)BGRA, or Alpha8. If a buffer is | ||
provided for a grayscale image, render operations will be carried out into an Alpha8 buffer instead. | ||
There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. Grayscale + 8bpp. Our support for grayscale colorspace for > 8bpp doesn't exist. |
||
Luminance values will be discarded in favour of alpha values. | ||
*/ | ||
CGContextRef CGBitmapContextCreateWithData(void* data, | ||
size_t width, | ||
size_t height, | ||
size_t bitsPerComponent, | ||
size_t bytesPerRow, | ||
CGColorSpaceRef space, | ||
CGColorSpaceRef colorSpace, | ||
uint32_t bitmapInfo, | ||
CGBitmapContextReleaseDataCallback releaseCallback, | ||
void* releaseInfo) { | ||
RETURN_NULL_IF(!width); | ||
RETURN_NULL_IF(!height); | ||
RETURN_NULL_IF(!space); | ||
RETURN_NULL_IF(!colorSpace); | ||
|
||
// bitsperpixel = ((bytesPerRow/width) * 8bits/byte) | ||
size_t bitsPerPixel = ((bytesPerRow / width) << 3); | ||
WICPixelFormatGUID outputPixelFormat; | ||
RETURN_NULL_IF_FAILED(_CGImageGetWICPixelFormatFromImageProperties(bitsPerComponent, bitsPerPixel, space, bitmapInfo, &outputPixelFormat)); | ||
WICPixelFormatGUID pixelFormat = outputPixelFormat; | ||
WICPixelFormatGUID requestedPixelFormat; | ||
RETURN_NULL_IF_FAILED( | ||
_CGImageGetWICPixelFormatFromImageProperties(bitsPerComponent, bitsPerPixel, colorSpace, bitmapInfo, &requestedPixelFormat)); | ||
WICPixelFormatGUID internalPixelFormat = requestedPixelFormat; | ||
|
||
// If we clear this, CGIWICBitmap will allocate its own buffer. | ||
void* wicImageBackingBuffer = data; | ||
|
||
if (!_CGIsValidRenderTargetPixelFormat(internalPixelFormat)) { | ||
internalPixelFormat = GUID_WICPixelFormat32bppPRGBA; | ||
|
||
if (!_CGIsValidRenderTargetPixelFormat(pixelFormat)) { | ||
if (data) { | ||
UNIMPLEMENTED_WITH_MSG( | ||
"CGBitmapContext does not currently support input conversion and can only render into 32bpp PRGBA buffers."); | ||
return nullptr; | ||
if (requestedPixelFormat == GUID_WICPixelFormat8bppGray) { | ||
TraceWarning(TAG, | ||
L"Shared-buffer 8bpp grayscale context requested at %dx%d. Since Direct2D does not support this, the context " | ||
L"will be 8bpp alpha-only. There may be minor color aberrations.", | ||
width, | ||
height); | ||
|
||
// Set requestedPixelFormat here as so that we don't do a copy on CGBitmapContextCGBitmapContextGetImage(). | ||
requestedPixelFormat = internalPixelFormat = GUID_WICPixelFormat8bppAlpha; | ||
} else { | ||
UNIMPLEMENTED_WITH_MSG( | ||
"CGBitmapContext does not currently support format conversion for shared buffers and can only render into 32bpp " | ||
"(P)RGBA, 32bpp (P)BGRA or 8bpp alpha-only buffers."); | ||
return nullptr; | ||
} | ||
} | ||
pixelFormat = GUID_WICPixelFormat32bppPRGBA; | ||
} | ||
|
||
// if data is null, enough memory is allocated via CGIWICBitmap | ||
ComPtr<IWICBitmap> customBitmap = Make<CGIWICBitmap>(data, pixelFormat, height, width); | ||
ComPtr<IWICBitmap> customBitmap = Make<CGIWICBitmap>(wicImageBackingBuffer, internalPixelFormat, height, width); | ||
RETURN_NULL_IF(!customBitmap); | ||
|
||
woc::unique_cf<CGImageRef> image(_CGImageCreateWithWICBitmap(customBitmap.Get())); | ||
|
@@ -2857,7 +2906,11 @@ CGContextRef CGBitmapContextCreateWithData(void* data, | |
|
||
ComPtr<ID2D1RenderTarget> renderTarget; | ||
RETURN_NULL_IF_FAILED(factory->CreateWicBitmapRenderTarget(customBitmap.Get(), D2D1::RenderTargetProperties(), &renderTarget)); | ||
CGContextRef context = _CGBitmapContextCreateWithRenderTarget(renderTarget.Get(), image.get(), outputPixelFormat); | ||
|
||
__CGBitmapContext* context = __CGBitmapContext::CreateInstance( | ||
kCFAllocatorDefault, renderTarget.Get(), requestedPixelFormat, data, width, height, bytesPerRow, releaseCallback, releaseInfo); | ||
__CGContextPrepareDefaults(context); | ||
context->SetImage(image); | ||
There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more.
image.get()? |
||
return context; | ||
} | ||
|
||
|
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
nit: I know this one has been around a bit longer and CG has been using it, but I like the m_* naming convention for C++ structs/classes since it makes it more obvious what's C++ and what's objc.
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
@aballway what is the value in distinguishing Objective-C and C++ code? Members and instance variables are, for all intents and purposes, identical. A fully-qualified out-of-body member function refers to members just like an Objective-C method refers to ivars.
They have similar lifetime requirements, as well.
our CF stuff is closer to Objective-C due to the reference counting; it's only C++ as a measure of convenience.
Also, our coding convention states