2020-12-11 15:01:48 +01:00
|
|
|
// SPDX-License-Identifier: Apache-2.0 OR MIT OR Unlicense
|
|
|
|
|
2020-04-21 19:30:14 -07:00
|
|
|
// This is "kernel 4" in a 4-kernel pipeline. It renders the commands
|
|
|
|
// in the per-tile command list to an image.
|
|
|
|
|
|
|
|
// Right now, this kernel stores the image in a buffer, but a better
|
|
|
|
// plan is to use a texture. This is because of limited support.
|
|
|
|
|
|
|
|
#version 450
|
|
|
|
#extension GL_GOOGLE_include_directive : enable
|
2021-04-02 18:59:07 -07:00
|
|
|
#ifdef ENABLE_IMAGE_INDICES
|
2020-11-18 15:54:11 -08:00
|
|
|
#extension GL_EXT_nonuniform_qualifier : enable
|
2021-04-02 18:59:07 -07:00
|
|
|
#endif
|
2020-04-21 19:30:14 -07:00
|
|
|
|
2020-12-11 18:30:20 +01:00
|
|
|
#include "mem.h"
|
2020-12-24 12:00:53 +01:00
|
|
|
#include "setup.h"
|
2020-06-14 23:32:59 +02:00
|
|
|
|
2021-03-23 20:54:49 +09:00
|
|
|
#define CHUNK_X 2
|
|
|
|
#define CHUNK_Y 4
|
|
|
|
#define CHUNK CHUNK_X * CHUNK_Y
|
|
|
|
#define CHUNK_DX (TILE_WIDTH_PX / CHUNK_X)
|
|
|
|
#define CHUNK_DY (TILE_HEIGHT_PX / CHUNK_Y)
|
|
|
|
layout(local_size_x = CHUNK_DX, local_size_y = CHUNK_DY) in;
|
2020-04-21 19:30:14 -07:00
|
|
|
|
2020-12-11 18:30:20 +01:00
|
|
|
layout(set = 0, binding = 1) readonly buffer ConfigBuf {
|
|
|
|
Config conf;
|
2020-04-21 19:30:14 -07:00
|
|
|
};
|
|
|
|
|
2020-12-11 18:30:20 +01:00
|
|
|
layout(rgba8, set = 0, binding = 2) uniform writeonly image2D image;
|
2020-04-21 19:30:14 -07:00
|
|
|
|
2021-04-02 18:59:07 -07:00
|
|
|
#ifdef ENABLE_IMAGE_INDICES
|
implement FillImage command and sRGB support
FillImage is like Fill, except that it takes its color from one or
more image atlases.
kernel4 uses a single image for non-Vulkan hosts, and the dynamic sized array
of image descriptors on Vulkan.
A previous version of this commit used textures. I think images are a better
choice for piet-gpu, for several reasons:
- Texture sampling, in particular textureGrad, is slow on lower spec devices
such as Google Pixel. Texture sampling is particularly slow and difficult to
implement for CPU fallbacks.
- Texture sampling need more parameters, in particular the full u,v
transformation matrix, leading to a large increase in the command size. Since
all commands use the same size, that memory penalty is paid by all scenes, not
just scenes with textures.
- It is unlikely that piet-gpu will support every kind of fill for every
client, because each kind must be added to kernel4.
With FillImage, a client will prepare the image(s) in separate shader stages,
sampling and applying transformations and special effects as needed. Textures
that align with the output pixel grid can be used directly, without
pre-processing.
Note that the pre-processing step can run concurrently with the piet-gpu pipeline;
Only the last stage, kernel4, needs the images.
Pre-processing most likely uses fixed function vertex/fragment programs,
which on some GPUs may run in parallel with piet-gpu's compute programs.
While here, fix a few validation errors:
- Explicitly enable EXT_descriptor_indexing, KHR_maintenance3,
KHR_get_physical_device_properties2.
- Specify a vkDescriptorSetVariableDescriptorCountAllocateInfo for
vkAllocateDescriptorSets. Otherwise, variable image2D arrays won't work (but
sampler2D arrays do, at least on my setup).
Updates #38
Signed-off-by: Elias Naur <mail@eliasnaur.com>
2020-12-28 22:02:39 +01:00
|
|
|
layout(rgba8, set = 0, binding = 3) uniform readonly image2D images[];
|
|
|
|
#else
|
|
|
|
layout(rgba8, set = 0, binding = 3) uniform readonly image2D images[1];
|
|
|
|
#endif
|
2020-11-18 15:54:11 -08:00
|
|
|
|
2020-04-21 19:30:14 -07:00
|
|
|
#include "ptcl.h"
|
2020-06-03 09:28:43 -07:00
|
|
|
#include "tile.h"
|
2020-04-21 19:30:14 -07:00
|
|
|
|
implement FillImage command and sRGB support
FillImage is like Fill, except that it takes its color from one or
more image atlases.
kernel4 uses a single image for non-Vulkan hosts, and the dynamic sized array
of image descriptors on Vulkan.
A previous version of this commit used textures. I think images are a better
choice for piet-gpu, for several reasons:
- Texture sampling, in particular textureGrad, is slow on lower spec devices
such as Google Pixel. Texture sampling is particularly slow and difficult to
implement for CPU fallbacks.
- Texture sampling need more parameters, in particular the full u,v
transformation matrix, leading to a large increase in the command size. Since
all commands use the same size, that memory penalty is paid by all scenes, not
just scenes with textures.
- It is unlikely that piet-gpu will support every kind of fill for every
client, because each kind must be added to kernel4.
With FillImage, a client will prepare the image(s) in separate shader stages,
sampling and applying transformations and special effects as needed. Textures
that align with the output pixel grid can be used directly, without
pre-processing.
Note that the pre-processing step can run concurrently with the piet-gpu pipeline;
Only the last stage, kernel4, needs the images.
Pre-processing most likely uses fixed function vertex/fragment programs,
which on some GPUs may run in parallel with piet-gpu's compute programs.
While here, fix a few validation errors:
- Explicitly enable EXT_descriptor_indexing, KHR_maintenance3,
KHR_get_physical_device_properties2.
- Specify a vkDescriptorSetVariableDescriptorCountAllocateInfo for
vkAllocateDescriptorSets. Otherwise, variable image2D arrays won't work (but
sampler2D arrays do, at least on my setup).
Updates #38
Signed-off-by: Elias Naur <mail@eliasnaur.com>
2020-12-28 22:02:39 +01:00
|
|
|
vec3 tosRGB(vec3 rgb) {
|
|
|
|
bvec3 cutoff = greaterThanEqual(rgb, vec3(0.0031308));
|
|
|
|
vec3 below = vec3(12.92)*rgb;
|
|
|
|
vec3 above = vec3(1.055)*pow(rgb, vec3(0.41666)) - vec3(0.055);
|
|
|
|
return mix(below, above, cutoff);
|
|
|
|
}
|
|
|
|
|
|
|
|
vec3 fromsRGB(vec3 srgb) {
|
|
|
|
// Formula from EXT_sRGB.
|
|
|
|
bvec3 cutoff = greaterThanEqual(srgb, vec3(0.04045));
|
|
|
|
vec3 below = srgb/vec3(12.92);
|
|
|
|
vec3 above = pow((srgb + vec3(0.055))/vec3(1.055), vec3(2.4));
|
|
|
|
return mix(below, above, cutoff);
|
|
|
|
}
|
|
|
|
|
|
|
|
// unpacksRGB unpacks a color in the sRGB color space to a vec4 in the linear color
|
|
|
|
// space.
|
|
|
|
vec4 unpacksRGB(uint srgba) {
|
|
|
|
vec4 color = unpackUnorm4x8(srgba).wzyx;
|
|
|
|
return vec4(fromsRGB(color.rgb), color.a);
|
|
|
|
}
|
|
|
|
|
|
|
|
// packsRGB packs a color in the linear color space into its 8-bit sRGB equivalent.
|
|
|
|
uint packsRGB(vec4 rgba) {
|
|
|
|
rgba = vec4(tosRGB(rgba.rgb), rgba.a);
|
|
|
|
return packUnorm4x8(rgba.wzyx);
|
|
|
|
}
|
|
|
|
|
2021-03-23 20:54:49 +09:00
|
|
|
uvec2 chunk_offset(uint i) {
|
|
|
|
return uvec2(i % CHUNK_X * CHUNK_DX, i / CHUNK_X * CHUNK_DY);
|
|
|
|
}
|
|
|
|
|
2021-03-18 12:47:14 +01:00
|
|
|
vec4[CHUNK] fillImage(uvec2 xy, CmdImage cmd_img) {
|
implement FillImage command and sRGB support
FillImage is like Fill, except that it takes its color from one or
more image atlases.
kernel4 uses a single image for non-Vulkan hosts, and the dynamic sized array
of image descriptors on Vulkan.
A previous version of this commit used textures. I think images are a better
choice for piet-gpu, for several reasons:
- Texture sampling, in particular textureGrad, is slow on lower spec devices
such as Google Pixel. Texture sampling is particularly slow and difficult to
implement for CPU fallbacks.
- Texture sampling need more parameters, in particular the full u,v
transformation matrix, leading to a large increase in the command size. Since
all commands use the same size, that memory penalty is paid by all scenes, not
just scenes with textures.
- It is unlikely that piet-gpu will support every kind of fill for every
client, because each kind must be added to kernel4.
With FillImage, a client will prepare the image(s) in separate shader stages,
sampling and applying transformations and special effects as needed. Textures
that align with the output pixel grid can be used directly, without
pre-processing.
Note that the pre-processing step can run concurrently with the piet-gpu pipeline;
Only the last stage, kernel4, needs the images.
Pre-processing most likely uses fixed function vertex/fragment programs,
which on some GPUs may run in parallel with piet-gpu's compute programs.
While here, fix a few validation errors:
- Explicitly enable EXT_descriptor_indexing, KHR_maintenance3,
KHR_get_physical_device_properties2.
- Specify a vkDescriptorSetVariableDescriptorCountAllocateInfo for
vkAllocateDescriptorSets. Otherwise, variable image2D arrays won't work (but
sampler2D arrays do, at least on my setup).
Updates #38
Signed-off-by: Elias Naur <mail@eliasnaur.com>
2020-12-28 22:02:39 +01:00
|
|
|
vec4 rgba[CHUNK];
|
|
|
|
for (uint i = 0; i < CHUNK; i++) {
|
2021-03-23 20:54:49 +09:00
|
|
|
ivec2 uv = ivec2(xy + chunk_offset(i)) + cmd_img.offset;
|
implement FillImage command and sRGB support
FillImage is like Fill, except that it takes its color from one or
more image atlases.
kernel4 uses a single image for non-Vulkan hosts, and the dynamic sized array
of image descriptors on Vulkan.
A previous version of this commit used textures. I think images are a better
choice for piet-gpu, for several reasons:
- Texture sampling, in particular textureGrad, is slow on lower spec devices
such as Google Pixel. Texture sampling is particularly slow and difficult to
implement for CPU fallbacks.
- Texture sampling need more parameters, in particular the full u,v
transformation matrix, leading to a large increase in the command size. Since
all commands use the same size, that memory penalty is paid by all scenes, not
just scenes with textures.
- It is unlikely that piet-gpu will support every kind of fill for every
client, because each kind must be added to kernel4.
With FillImage, a client will prepare the image(s) in separate shader stages,
sampling and applying transformations and special effects as needed. Textures
that align with the output pixel grid can be used directly, without
pre-processing.
Note that the pre-processing step can run concurrently with the piet-gpu pipeline;
Only the last stage, kernel4, needs the images.
Pre-processing most likely uses fixed function vertex/fragment programs,
which on some GPUs may run in parallel with piet-gpu's compute programs.
While here, fix a few validation errors:
- Explicitly enable EXT_descriptor_indexing, KHR_maintenance3,
KHR_get_physical_device_properties2.
- Specify a vkDescriptorSetVariableDescriptorCountAllocateInfo for
vkAllocateDescriptorSets. Otherwise, variable image2D arrays won't work (but
sampler2D arrays do, at least on my setup).
Updates #38
Signed-off-by: Elias Naur <mail@eliasnaur.com>
2020-12-28 22:02:39 +01:00
|
|
|
#ifdef ENABLE_IMAGE_INDICES
|
|
|
|
vec4 fg_rgba = imageLoad(images[cmd_img.index], uv);
|
|
|
|
#else
|
|
|
|
vec4 fg_rgba = imageLoad(images[0], uv);
|
|
|
|
#endif
|
|
|
|
fg_rgba.rgb = fromsRGB(fg_rgba.rgb);
|
|
|
|
rgba[i] = fg_rgba;
|
|
|
|
}
|
|
|
|
return rgba;
|
|
|
|
}
|
|
|
|
|
2020-04-21 19:30:14 -07:00
|
|
|
void main() {
|
2020-12-24 12:00:53 +01:00
|
|
|
if (mem_error != NO_ERROR) {
|
2020-12-11 18:30:20 +01:00
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2020-12-18 00:55:21 +01:00
|
|
|
uint tile_ix = gl_WorkGroupID.y * conf.width_in_tiles + gl_WorkGroupID.x;
|
2020-12-24 12:00:53 +01:00
|
|
|
Alloc cmd_alloc = slice_mem(conf.ptcl_alloc, tile_ix * PTCL_INITIAL_ALLOC, PTCL_INITIAL_ALLOC);
|
|
|
|
CmdRef cmd_ref = CmdRef(cmd_alloc.offset);
|
2020-04-21 19:30:14 -07:00
|
|
|
|
2021-03-30 20:16:36 +02:00
|
|
|
// Read scrach space allocation, written first in the command list.
|
|
|
|
Alloc scratch_alloc = alloc_read(cmd_alloc, cmd_ref.offset);
|
|
|
|
cmd_ref.offset += Alloc_size;
|
|
|
|
|
2021-03-23 20:54:49 +09:00
|
|
|
uvec2 xy_uint = uvec2(gl_LocalInvocationID.x + TILE_WIDTH_PX * gl_WorkGroupID.x, gl_LocalInvocationID.y + TILE_HEIGHT_PX * gl_WorkGroupID.y);
|
2020-04-21 19:30:14 -07:00
|
|
|
vec2 xy = vec2(xy_uint);
|
2021-03-22 16:06:15 +01:00
|
|
|
vec4 rgba[CHUNK];
|
2020-05-25 15:45:06 -07:00
|
|
|
for (uint i = 0; i < CHUNK; i++) {
|
2021-03-22 16:06:15 +01:00
|
|
|
rgba[i] = vec4(0.0);
|
2021-04-02 18:59:07 -07:00
|
|
|
// TODO: remove this debug image support when the actual image method is plumbed.
|
|
|
|
#ifdef DEBUG_IMAGES
|
implement FillImage command and sRGB support
FillImage is like Fill, except that it takes its color from one or
more image atlases.
kernel4 uses a single image for non-Vulkan hosts, and the dynamic sized array
of image descriptors on Vulkan.
A previous version of this commit used textures. I think images are a better
choice for piet-gpu, for several reasons:
- Texture sampling, in particular textureGrad, is slow on lower spec devices
such as Google Pixel. Texture sampling is particularly slow and difficult to
implement for CPU fallbacks.
- Texture sampling need more parameters, in particular the full u,v
transformation matrix, leading to a large increase in the command size. Since
all commands use the same size, that memory penalty is paid by all scenes, not
just scenes with textures.
- It is unlikely that piet-gpu will support every kind of fill for every
client, because each kind must be added to kernel4.
With FillImage, a client will prepare the image(s) in separate shader stages,
sampling and applying transformations and special effects as needed. Textures
that align with the output pixel grid can be used directly, without
pre-processing.
Note that the pre-processing step can run concurrently with the piet-gpu pipeline;
Only the last stage, kernel4, needs the images.
Pre-processing most likely uses fixed function vertex/fragment programs,
which on some GPUs may run in parallel with piet-gpu's compute programs.
While here, fix a few validation errors:
- Explicitly enable EXT_descriptor_indexing, KHR_maintenance3,
KHR_get_physical_device_properties2.
- Specify a vkDescriptorSetVariableDescriptorCountAllocateInfo for
vkAllocateDescriptorSets. Otherwise, variable image2D arrays won't work (but
sampler2D arrays do, at least on my setup).
Updates #38
Signed-off-by: Elias Naur <mail@eliasnaur.com>
2020-12-28 22:02:39 +01:00
|
|
|
#ifdef ENABLE_IMAGE_INDICES
|
2020-11-25 12:43:42 -08:00
|
|
|
if (xy_uint.x < 1024 && xy_uint.y < 1024) {
|
2021-03-22 16:06:15 +01:00
|
|
|
rgba[i] = imageLoad(images[gl_WorkGroupID.x / 64], ivec2(xy_uint + chunk_offset(i))/4);
|
2020-11-18 15:54:11 -08:00
|
|
|
}
|
2021-04-02 18:59:07 -07:00
|
|
|
#else
|
|
|
|
if (xy_uint.x < 1024 && xy_uint.y < 1024) {
|
|
|
|
rgb[i] = imageLoad(images[0], ivec2(xy_uint + chunk_offset(i))/4).rgb;
|
|
|
|
}
|
|
|
|
#endif
|
implement FillImage command and sRGB support
FillImage is like Fill, except that it takes its color from one or
more image atlases.
kernel4 uses a single image for non-Vulkan hosts, and the dynamic sized array
of image descriptors on Vulkan.
A previous version of this commit used textures. I think images are a better
choice for piet-gpu, for several reasons:
- Texture sampling, in particular textureGrad, is slow on lower spec devices
such as Google Pixel. Texture sampling is particularly slow and difficult to
implement for CPU fallbacks.
- Texture sampling need more parameters, in particular the full u,v
transformation matrix, leading to a large increase in the command size. Since
all commands use the same size, that memory penalty is paid by all scenes, not
just scenes with textures.
- It is unlikely that piet-gpu will support every kind of fill for every
client, because each kind must be added to kernel4.
With FillImage, a client will prepare the image(s) in separate shader stages,
sampling and applying transformations and special effects as needed. Textures
that align with the output pixel grid can be used directly, without
pre-processing.
Note that the pre-processing step can run concurrently with the piet-gpu pipeline;
Only the last stage, kernel4, needs the images.
Pre-processing most likely uses fixed function vertex/fragment programs,
which on some GPUs may run in parallel with piet-gpu's compute programs.
While here, fix a few validation errors:
- Explicitly enable EXT_descriptor_indexing, KHR_maintenance3,
KHR_get_physical_device_properties2.
- Specify a vkDescriptorSetVariableDescriptorCountAllocateInfo for
vkAllocateDescriptorSets. Otherwise, variable image2D arrays won't work (but
sampler2D arrays do, at least on my setup).
Updates #38
Signed-off-by: Elias Naur <mail@eliasnaur.com>
2020-12-28 22:02:39 +01:00
|
|
|
#endif
|
2020-05-25 15:45:06 -07:00
|
|
|
}
|
2020-04-21 19:30:14 -07:00
|
|
|
|
2021-03-18 12:47:14 +01:00
|
|
|
float area[CHUNK];
|
2021-03-30 20:16:36 +02:00
|
|
|
uint clip_depth = 0;
|
2020-04-21 19:30:14 -07:00
|
|
|
while (true) {
|
2021-03-17 10:51:38 +01:00
|
|
|
uint tag = Cmd_tag(cmd_alloc, cmd_ref).tag;
|
2020-04-21 19:30:14 -07:00
|
|
|
if (tag == Cmd_End) {
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
switch (tag) {
|
2020-04-28 11:02:19 -07:00
|
|
|
case Cmd_Stroke:
|
2020-06-28 15:37:27 +02:00
|
|
|
// Calculate distance field from all the line segments in this tile.
|
2020-12-24 12:00:53 +01:00
|
|
|
CmdStroke stroke = Cmd_Stroke_read(cmd_alloc, cmd_ref);
|
2020-05-25 15:45:06 -07:00
|
|
|
float df[CHUNK];
|
|
|
|
for (uint k = 0; k < CHUNK; k++) df[k] = 1e9;
|
2020-06-03 09:28:43 -07:00
|
|
|
TileSegRef tile_seg_ref = TileSegRef(stroke.tile_ref);
|
2020-04-28 22:25:57 -07:00
|
|
|
do {
|
2020-12-24 12:00:53 +01:00
|
|
|
TileSeg seg = TileSeg_read(new_alloc(tile_seg_ref.offset, TileSeg_size), tile_seg_ref);
|
2020-12-01 18:06:09 +01:00
|
|
|
vec2 line_vec = seg.vector;
|
2020-06-03 09:28:43 -07:00
|
|
|
for (uint k = 0; k < CHUNK; k++) {
|
2020-12-01 18:06:09 +01:00
|
|
|
vec2 dpos = xy + vec2(0.5, 0.5) - seg.origin;
|
2021-03-23 20:54:49 +09:00
|
|
|
dpos += vec2(chunk_offset(k));
|
2020-06-03 09:28:43 -07:00
|
|
|
float t = clamp(dot(line_vec, dpos) / dot(line_vec, line_vec), 0.0, 1.0);
|
|
|
|
df[k] = min(df[k], length(line_vec * t - dpos));
|
2020-05-05 18:13:07 +02:00
|
|
|
}
|
2020-06-03 09:28:43 -07:00
|
|
|
tile_seg_ref = seg.next;
|
|
|
|
} while (tile_seg_ref.offset != 0);
|
2020-05-25 15:45:06 -07:00
|
|
|
for (uint k = 0; k < CHUNK; k++) {
|
2021-03-18 12:47:14 +01:00
|
|
|
area[k] = clamp(stroke.half_width + 0.5 - df[k], 0.0, 1.0);
|
2020-05-25 15:45:06 -07:00
|
|
|
}
|
2021-03-18 20:17:04 +01:00
|
|
|
cmd_ref.offset += 4 + CmdStroke_size;
|
2020-04-28 11:02:19 -07:00
|
|
|
break;
|
2020-04-30 17:06:01 -07:00
|
|
|
case Cmd_Fill:
|
2020-12-24 12:00:53 +01:00
|
|
|
CmdFill fill = Cmd_Fill_read(cmd_alloc, cmd_ref);
|
2021-03-18 12:47:14 +01:00
|
|
|
for (uint k = 0; k < CHUNK; k++) area[k] = float(fill.backdrop);
|
|
|
|
tile_seg_ref = TileSegRef(fill.tile_ref);
|
|
|
|
// Calculate coverage based on backdrop + coverage of each line segment
|
|
|
|
do {
|
|
|
|
TileSeg seg = TileSeg_read(new_alloc(tile_seg_ref.offset, TileSeg_size), tile_seg_ref);
|
|
|
|
for (uint k = 0; k < CHUNK; k++) {
|
2021-03-23 20:54:49 +09:00
|
|
|
vec2 my_xy = xy + vec2(chunk_offset(k));
|
2021-03-18 12:47:14 +01:00
|
|
|
vec2 start = seg.origin - my_xy;
|
|
|
|
vec2 end = start + seg.vector;
|
|
|
|
vec2 window = clamp(vec2(start.y, end.y), 0.0, 1.0);
|
|
|
|
if (window.x != window.y) {
|
|
|
|
vec2 t = (window - start.y) / seg.vector.y;
|
|
|
|
vec2 xs = vec2(mix(start.x, end.x, t.x), mix(start.x, end.x, t.y));
|
|
|
|
float xmin = min(min(xs.x, xs.y), 1.0) - 1e-6;
|
|
|
|
float xmax = max(xs.x, xs.y);
|
|
|
|
float b = min(xmax, 1.0);
|
|
|
|
float c = max(b, 0.0);
|
|
|
|
float d = max(xmin, 0.0);
|
|
|
|
float a = (b + 0.5 * (d * d - c * c) - xmin) / (xmax - xmin);
|
|
|
|
area[k] += a * (window.x - window.y);
|
|
|
|
}
|
|
|
|
area[k] += sign(seg.vector.x) * clamp(my_xy.y - seg.y_edge + 1.0, 0.0, 1.0);
|
|
|
|
}
|
|
|
|
tile_seg_ref = seg.next;
|
|
|
|
} while (tile_seg_ref.offset != 0);
|
|
|
|
for (uint k = 0; k < CHUNK; k++) {
|
|
|
|
area[k] = min(abs(area[k]), 1.0);
|
|
|
|
}
|
2021-03-18 20:17:04 +01:00
|
|
|
cmd_ref.offset += 4 + CmdFill_size;
|
2021-03-18 12:47:14 +01:00
|
|
|
break;
|
|
|
|
case Cmd_Solid:
|
|
|
|
for (uint k = 0; k < CHUNK; k++) {
|
|
|
|
area[k] = 1.0;
|
|
|
|
}
|
2021-03-18 20:17:04 +01:00
|
|
|
cmd_ref.offset += 4;
|
2021-03-18 12:47:14 +01:00
|
|
|
break;
|
|
|
|
case Cmd_Alpha:
|
|
|
|
CmdAlpha alpha = Cmd_Alpha_read(cmd_alloc, cmd_ref);
|
|
|
|
for (uint k = 0; k < CHUNK; k++) {
|
|
|
|
area[k] = alpha.alpha;
|
|
|
|
}
|
2021-03-18 20:17:04 +01:00
|
|
|
cmd_ref.offset += 4 + CmdAlpha_size;
|
2021-03-18 12:47:14 +01:00
|
|
|
break;
|
|
|
|
case Cmd_Color:
|
|
|
|
CmdColor color = Cmd_Color_read(cmd_alloc, cmd_ref);
|
2021-03-22 14:12:05 +01:00
|
|
|
vec4 fg = unpacksRGB(color.rgba_color);
|
2020-05-25 15:45:06 -07:00
|
|
|
for (uint k = 0; k < CHUNK; k++) {
|
2021-03-22 14:12:05 +01:00
|
|
|
vec4 fg_k = fg * area[k];
|
2021-03-22 16:06:15 +01:00
|
|
|
rgba[k] = rgba[k] * (1.0 - fg_k.a) + fg_k;
|
2020-10-09 12:43:29 +02:00
|
|
|
}
|
2021-03-18 20:17:04 +01:00
|
|
|
cmd_ref.offset += 4 + CmdColor_size;
|
2020-10-09 12:43:29 +02:00
|
|
|
break;
|
2021-03-18 12:47:14 +01:00
|
|
|
case Cmd_Image:
|
|
|
|
CmdImage fill_img = Cmd_Image_read(cmd_alloc, cmd_ref);
|
2021-03-22 14:12:05 +01:00
|
|
|
vec4 img[CHUNK] = fillImage(xy_uint, fill_img);
|
implement FillImage command and sRGB support
FillImage is like Fill, except that it takes its color from one or
more image atlases.
kernel4 uses a single image for non-Vulkan hosts, and the dynamic sized array
of image descriptors on Vulkan.
A previous version of this commit used textures. I think images are a better
choice for piet-gpu, for several reasons:
- Texture sampling, in particular textureGrad, is slow on lower spec devices
such as Google Pixel. Texture sampling is particularly slow and difficult to
implement for CPU fallbacks.
- Texture sampling need more parameters, in particular the full u,v
transformation matrix, leading to a large increase in the command size. Since
all commands use the same size, that memory penalty is paid by all scenes, not
just scenes with textures.
- It is unlikely that piet-gpu will support every kind of fill for every
client, because each kind must be added to kernel4.
With FillImage, a client will prepare the image(s) in separate shader stages,
sampling and applying transformations and special effects as needed. Textures
that align with the output pixel grid can be used directly, without
pre-processing.
Note that the pre-processing step can run concurrently with the piet-gpu pipeline;
Only the last stage, kernel4, needs the images.
Pre-processing most likely uses fixed function vertex/fragment programs,
which on some GPUs may run in parallel with piet-gpu's compute programs.
While here, fix a few validation errors:
- Explicitly enable EXT_descriptor_indexing, KHR_maintenance3,
KHR_get_physical_device_properties2.
- Specify a vkDescriptorSetVariableDescriptorCountAllocateInfo for
vkAllocateDescriptorSets. Otherwise, variable image2D arrays won't work (but
sampler2D arrays do, at least on my setup).
Updates #38
Signed-off-by: Elias Naur <mail@eliasnaur.com>
2020-12-28 22:02:39 +01:00
|
|
|
for (uint k = 0; k < CHUNK; k++) {
|
2021-03-22 14:12:05 +01:00
|
|
|
vec4 fg_k = img[k] * area[k];
|
2021-03-22 16:06:15 +01:00
|
|
|
rgba[k] = rgba[k] * (1.0 - fg_k.a) + fg_k;
|
implement FillImage command and sRGB support
FillImage is like Fill, except that it takes its color from one or
more image atlases.
kernel4 uses a single image for non-Vulkan hosts, and the dynamic sized array
of image descriptors on Vulkan.
A previous version of this commit used textures. I think images are a better
choice for piet-gpu, for several reasons:
- Texture sampling, in particular textureGrad, is slow on lower spec devices
such as Google Pixel. Texture sampling is particularly slow and difficult to
implement for CPU fallbacks.
- Texture sampling need more parameters, in particular the full u,v
transformation matrix, leading to a large increase in the command size. Since
all commands use the same size, that memory penalty is paid by all scenes, not
just scenes with textures.
- It is unlikely that piet-gpu will support every kind of fill for every
client, because each kind must be added to kernel4.
With FillImage, a client will prepare the image(s) in separate shader stages,
sampling and applying transformations and special effects as needed. Textures
that align with the output pixel grid can be used directly, without
pre-processing.
Note that the pre-processing step can run concurrently with the piet-gpu pipeline;
Only the last stage, kernel4, needs the images.
Pre-processing most likely uses fixed function vertex/fragment programs,
which on some GPUs may run in parallel with piet-gpu's compute programs.
While here, fix a few validation errors:
- Explicitly enable EXT_descriptor_indexing, KHR_maintenance3,
KHR_get_physical_device_properties2.
- Specify a vkDescriptorSetVariableDescriptorCountAllocateInfo for
vkAllocateDescriptorSets. Otherwise, variable image2D arrays won't work (but
sampler2D arrays do, at least on my setup).
Updates #38
Signed-off-by: Elias Naur <mail@eliasnaur.com>
2020-12-28 22:02:39 +01:00
|
|
|
}
|
2021-03-18 20:17:04 +01:00
|
|
|
cmd_ref.offset += 4 + CmdImage_size;
|
implement FillImage command and sRGB support
FillImage is like Fill, except that it takes its color from one or
more image atlases.
kernel4 uses a single image for non-Vulkan hosts, and the dynamic sized array
of image descriptors on Vulkan.
A previous version of this commit used textures. I think images are a better
choice for piet-gpu, for several reasons:
- Texture sampling, in particular textureGrad, is slow on lower spec devices
such as Google Pixel. Texture sampling is particularly slow and difficult to
implement for CPU fallbacks.
- Texture sampling need more parameters, in particular the full u,v
transformation matrix, leading to a large increase in the command size. Since
all commands use the same size, that memory penalty is paid by all scenes, not
just scenes with textures.
- It is unlikely that piet-gpu will support every kind of fill for every
client, because each kind must be added to kernel4.
With FillImage, a client will prepare the image(s) in separate shader stages,
sampling and applying transformations and special effects as needed. Textures
that align with the output pixel grid can be used directly, without
pre-processing.
Note that the pre-processing step can run concurrently with the piet-gpu pipeline;
Only the last stage, kernel4, needs the images.
Pre-processing most likely uses fixed function vertex/fragment programs,
which on some GPUs may run in parallel with piet-gpu's compute programs.
While here, fix a few validation errors:
- Explicitly enable EXT_descriptor_indexing, KHR_maintenance3,
KHR_get_physical_device_properties2.
- Specify a vkDescriptorSetVariableDescriptorCountAllocateInfo for
vkAllocateDescriptorSets. Otherwise, variable image2D arrays won't work (but
sampler2D arrays do, at least on my setup).
Updates #38
Signed-off-by: Elias Naur <mail@eliasnaur.com>
2020-12-28 22:02:39 +01:00
|
|
|
break;
|
2020-11-19 11:53:59 -08:00
|
|
|
case Cmd_BeginClip:
|
2021-03-22 16:13:39 +01:00
|
|
|
uint base_ix = (scratch_alloc.offset >> 2) + CLIP_STATE_SIZE * (clip_depth * TILE_WIDTH_PX * TILE_HEIGHT_PX +
|
|
|
|
gl_LocalInvocationID.x + TILE_WIDTH_PX * gl_LocalInvocationID.y);
|
2021-03-18 12:47:14 +01:00
|
|
|
for (uint k = 0; k < CHUNK; k++) {
|
2021-03-30 20:16:36 +02:00
|
|
|
uvec2 offset = chunk_offset(k);
|
2021-03-22 16:06:15 +01:00
|
|
|
uint srgb = packsRGB(vec4(rgba[k]));
|
2021-03-22 16:13:39 +01:00
|
|
|
float alpha = clamp(abs(area[k]), 0.0, 1.0);
|
|
|
|
write_mem(scratch_alloc, base_ix + 0 + CLIP_STATE_SIZE * (offset.x + offset.y * TILE_WIDTH_PX), srgb);
|
|
|
|
write_mem(scratch_alloc, base_ix + 1 + CLIP_STATE_SIZE * (offset.x + offset.y * TILE_WIDTH_PX), floatBitsToUint(alpha));
|
2021-03-22 16:06:15 +01:00
|
|
|
rgba[k] = vec4(0.0);
|
2020-11-20 09:26:02 -08:00
|
|
|
}
|
2021-03-30 20:16:36 +02:00
|
|
|
clip_depth++;
|
2021-03-18 20:17:04 +01:00
|
|
|
cmd_ref.offset += 4;
|
2020-11-20 09:26:02 -08:00
|
|
|
break;
|
2020-11-19 11:53:59 -08:00
|
|
|
case Cmd_EndClip:
|
2021-03-30 20:16:36 +02:00
|
|
|
clip_depth--;
|
2021-03-22 16:13:39 +01:00
|
|
|
base_ix = (scratch_alloc.offset >> 2) + CLIP_STATE_SIZE * (clip_depth * TILE_WIDTH_PX * TILE_HEIGHT_PX +
|
|
|
|
gl_LocalInvocationID.x + TILE_WIDTH_PX * gl_LocalInvocationID.y);
|
2020-11-19 11:53:59 -08:00
|
|
|
for (uint k = 0; k < CHUNK; k++) {
|
2021-03-30 20:16:36 +02:00
|
|
|
uvec2 offset = chunk_offset(k);
|
2021-03-22 16:13:39 +01:00
|
|
|
uint srgb = read_mem(scratch_alloc, base_ix + 0 + CLIP_STATE_SIZE * (offset.x + offset.y * TILE_WIDTH_PX));
|
|
|
|
uint alpha = read_mem(scratch_alloc, base_ix + 1 + CLIP_STATE_SIZE * (offset.x + offset.y * TILE_WIDTH_PX));
|
2021-03-22 16:06:15 +01:00
|
|
|
vec4 bg = unpacksRGB(srgb);
|
|
|
|
vec4 fg = rgba[k] * area[k] * uintBitsToFloat(alpha);
|
|
|
|
rgba[k] = bg * (1.0 - fg.a) + fg;
|
implement FillImage command and sRGB support
FillImage is like Fill, except that it takes its color from one or
more image atlases.
kernel4 uses a single image for non-Vulkan hosts, and the dynamic sized array
of image descriptors on Vulkan.
A previous version of this commit used textures. I think images are a better
choice for piet-gpu, for several reasons:
- Texture sampling, in particular textureGrad, is slow on lower spec devices
such as Google Pixel. Texture sampling is particularly slow and difficult to
implement for CPU fallbacks.
- Texture sampling need more parameters, in particular the full u,v
transformation matrix, leading to a large increase in the command size. Since
all commands use the same size, that memory penalty is paid by all scenes, not
just scenes with textures.
- It is unlikely that piet-gpu will support every kind of fill for every
client, because each kind must be added to kernel4.
With FillImage, a client will prepare the image(s) in separate shader stages,
sampling and applying transformations and special effects as needed. Textures
that align with the output pixel grid can be used directly, without
pre-processing.
Note that the pre-processing step can run concurrently with the piet-gpu pipeline;
Only the last stage, kernel4, needs the images.
Pre-processing most likely uses fixed function vertex/fragment programs,
which on some GPUs may run in parallel with piet-gpu's compute programs.
While here, fix a few validation errors:
- Explicitly enable EXT_descriptor_indexing, KHR_maintenance3,
KHR_get_physical_device_properties2.
- Specify a vkDescriptorSetVariableDescriptorCountAllocateInfo for
vkAllocateDescriptorSets. Otherwise, variable image2D arrays won't work (but
sampler2D arrays do, at least on my setup).
Updates #38
Signed-off-by: Elias Naur <mail@eliasnaur.com>
2020-12-28 22:02:39 +01:00
|
|
|
}
|
2021-03-18 20:17:04 +01:00
|
|
|
cmd_ref.offset += 4;
|
implement FillImage command and sRGB support
FillImage is like Fill, except that it takes its color from one or
more image atlases.
kernel4 uses a single image for non-Vulkan hosts, and the dynamic sized array
of image descriptors on Vulkan.
A previous version of this commit used textures. I think images are a better
choice for piet-gpu, for several reasons:
- Texture sampling, in particular textureGrad, is slow on lower spec devices
such as Google Pixel. Texture sampling is particularly slow and difficult to
implement for CPU fallbacks.
- Texture sampling need more parameters, in particular the full u,v
transformation matrix, leading to a large increase in the command size. Since
all commands use the same size, that memory penalty is paid by all scenes, not
just scenes with textures.
- It is unlikely that piet-gpu will support every kind of fill for every
client, because each kind must be added to kernel4.
With FillImage, a client will prepare the image(s) in separate shader stages,
sampling and applying transformations and special effects as needed. Textures
that align with the output pixel grid can be used directly, without
pre-processing.
Note that the pre-processing step can run concurrently with the piet-gpu pipeline;
Only the last stage, kernel4, needs the images.
Pre-processing most likely uses fixed function vertex/fragment programs,
which on some GPUs may run in parallel with piet-gpu's compute programs.
While here, fix a few validation errors:
- Explicitly enable EXT_descriptor_indexing, KHR_maintenance3,
KHR_get_physical_device_properties2.
- Specify a vkDescriptorSetVariableDescriptorCountAllocateInfo for
vkAllocateDescriptorSets. Otherwise, variable image2D arrays won't work (but
sampler2D arrays do, at least on my setup).
Updates #38
Signed-off-by: Elias Naur <mail@eliasnaur.com>
2020-12-28 22:02:39 +01:00
|
|
|
break;
|
2020-04-25 10:15:22 -07:00
|
|
|
case Cmd_Jump:
|
2020-12-24 12:00:53 +01:00
|
|
|
cmd_ref = CmdRef(Cmd_Jump_read(cmd_alloc, cmd_ref).new_ref);
|
|
|
|
cmd_alloc.offset = cmd_ref.offset;
|
2021-04-11 19:54:40 +02:00
|
|
|
break;
|
2020-04-21 19:30:14 -07:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2020-05-25 15:45:06 -07:00
|
|
|
for (uint i = 0; i < CHUNK; i++) {
|
2021-03-22 16:06:15 +01:00
|
|
|
imageStore(image, ivec2(xy_uint + chunk_offset(i)), vec4(tosRGB(rgba[i].rgb), rgba[i].a));
|
2020-05-25 15:45:06 -07:00
|
|
|
}
|
2020-04-21 19:30:14 -07:00
|
|
|
}
|