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
|
|
|
|
|
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
|
|
|
|
2021-04-19 18:56:57 +02:00
|
|
|
layout(set = 0, binding = 1) restrict readonly buffer ConfigBuf {
|
2020-12-11 18:30:20 +01:00
|
|
|
Config conf;
|
2020-04-21 19:30:14 -07:00
|
|
|
};
|
|
|
|
|
2021-04-19 18:56:57 +02:00
|
|
|
layout(rgba8, set = 0, binding = 2) uniform restrict writeonly image2D image;
|
2020-04-21 19:30:14 -07:00
|
|
|
|
2021-06-23 11:50:51 -07:00
|
|
|
layout(rgba8, set = 0, binding = 3) uniform restrict readonly image2D image_atlas;
|
|
|
|
|
|
|
|
layout(rgba8, set = 0, binding = 4) uniform restrict readonly image2D gradients;
|
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
|
|
|
|
2021-03-30 12:59:49 +09:00
|
|
|
#define MAX_BLEND_STACK 128
|
2021-04-19 18:56:57 +02:00
|
|
|
mediump vec3 tosRGB(mediump vec3 rgb) {
|
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
|
|
|
bvec3 cutoff = greaterThanEqual(rgb, vec3(0.0031308));
|
2021-04-19 18:56:57 +02:00
|
|
|
mediump vec3 below = vec3(12.92)*rgb;
|
|
|
|
mediump vec3 above = vec3(1.055)*pow(rgb, vec3(0.41666)) - vec3(0.055);
|
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
|
|
|
return mix(below, above, cutoff);
|
|
|
|
}
|
|
|
|
|
2021-04-19 18:56:57 +02:00
|
|
|
mediump vec3 fromsRGB(mediump vec3 srgb) {
|
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
|
|
|
// Formula from EXT_sRGB.
|
|
|
|
bvec3 cutoff = greaterThanEqual(srgb, vec3(0.04045));
|
2021-04-19 18:56:57 +02:00
|
|
|
mediump vec3 below = srgb/vec3(12.92);
|
|
|
|
mediump vec3 above = pow((srgb + vec3(0.055))/vec3(1.055), vec3(2.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
|
|
|
return mix(below, above, cutoff);
|
|
|
|
}
|
|
|
|
|
|
|
|
// unpacksRGB unpacks a color in the sRGB color space to a vec4 in the linear color
|
|
|
|
// space.
|
2021-04-19 18:56:57 +02:00
|
|
|
mediump vec4 unpacksRGB(uint srgba) {
|
|
|
|
mediump vec4 color = unpackUnorm4x8(srgba).wzyx;
|
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
|
|
|
return vec4(fromsRGB(color.rgb), color.a);
|
|
|
|
}
|
|
|
|
|
|
|
|
// packsRGB packs a color in the linear color space into its 8-bit sRGB equivalent.
|
2021-04-19 18:56:57 +02:00
|
|
|
uint packsRGB(mediump vec4 rgba) {
|
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
|
|
|
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-04-19 18:56:57 +02:00
|
|
|
mediump vec4[CHUNK] fillImage(uvec2 xy, CmdImage cmd_img) {
|
|
|
|
mediump vec4 rgba[CHUNK];
|
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 i = 0; i < CHUNK; i++) {
|
2021-03-23 20:54:49 +09:00
|
|
|
ivec2 uv = ivec2(xy + chunk_offset(i)) + cmd_img.offset;
|
2021-04-19 18:56:57 +02:00
|
|
|
mediump vec4 fg_rgba;
|
2021-06-23 11:50:51 -07:00
|
|
|
fg_rgba = imageLoad(image_atlas, uv);
|
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
|
|
|
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-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-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-04-19 18:56:57 +02:00
|
|
|
mediump vec4 rgba[CHUNK];
|
2021-03-30 12:59:49 +09:00
|
|
|
uint blend_stack[MAX_BLEND_STACK][CHUNK];
|
|
|
|
mediump float blend_alpha_stack[MAX_BLEND_STACK][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);
|
2020-05-25 15:45:06 -07:00
|
|
|
}
|
2020-04-21 19:30:14 -07:00
|
|
|
|
2021-04-19 18:56:57 +02:00
|
|
|
mediump float area[CHUNK];
|
2021-03-30 20:16:36 +02:00
|
|
|
uint clip_depth = 0;
|
2021-04-12 14:41:03 +02:00
|
|
|
bool mem_ok = mem_error == NO_ERROR;
|
|
|
|
while (mem_ok) {
|
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);
|
2021-04-19 18:56:57 +02:00
|
|
|
mediump float df[CHUNK];
|
2020-05-25 15:45:06 -07:00
|
|
|
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 {
|
2021-04-12 14:41:03 +02:00
|
|
|
TileSeg seg = TileSeg_read(new_alloc(tile_seg_ref.offset, TileSeg_size, mem_ok), 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 {
|
2021-04-12 14:41:03 +02:00
|
|
|
TileSeg seg = TileSeg_read(new_alloc(tile_seg_ref.offset, TileSeg_size, mem_ok), tile_seg_ref);
|
2021-03-18 12:47:14 +01:00
|
|
|
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-04-19 18:56:57 +02:00
|
|
|
mediump vec4 fg = unpacksRGB(color.rgba_color);
|
2020-05-25 15:45:06 -07:00
|
|
|
for (uint k = 0; k < CHUNK; k++) {
|
2021-04-19 18:56:57 +02:00
|
|
|
mediump 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-06-23 11:50:51 -07:00
|
|
|
case Cmd_LinGrad:
|
|
|
|
CmdLinGrad lin = Cmd_LinGrad_read(cmd_alloc, cmd_ref);
|
|
|
|
float d = lin.line_x * float(xy.x) + lin.line_y * float(xy.y) + lin.line_c;
|
|
|
|
for (uint k = 0; k < CHUNK; k++) {
|
|
|
|
vec2 chunk_xy = vec2(chunk_offset(k));
|
|
|
|
float my_d = d + lin.line_x * chunk_xy.x + lin.line_y * chunk_xy.y;
|
|
|
|
int x = int(round(clamp(my_d, 0.0, 1.0) * float(GRADIENT_WIDTH - 1)));
|
|
|
|
mediump vec4 fg_rgba = imageLoad(gradients, ivec2(x, int(lin.index)));
|
|
|
|
fg_rgba.rgb = fromsRGB(fg_rgba.rgb);
|
|
|
|
rgba[k] = fg_rgba;
|
|
|
|
}
|
|
|
|
cmd_ref.offset += 4 + CmdLinGrad_size;
|
|
|
|
break;
|
2021-03-18 12:47:14 +01:00
|
|
|
case Cmd_Image:
|
|
|
|
CmdImage fill_img = Cmd_Image_read(cmd_alloc, cmd_ref);
|
2021-04-19 18:56:57 +02:00
|
|
|
mediump 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-04-19 18:56:57 +02:00
|
|
|
mediump 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-18 12:47:14 +01:00
|
|
|
for (uint k = 0; k < CHUNK; k++) {
|
2021-03-30 12:59:49 +09:00
|
|
|
// We reject any inputs that might overflow in render_ctx.rs.
|
|
|
|
// The following is a sanity check so we don't corrupt memory should there be malformed inputs.
|
|
|
|
uint d = min(clip_depth, MAX_BLEND_STACK - 1);
|
|
|
|
blend_stack[d][k] = packsRGB(vec4(rgba[k]));
|
|
|
|
blend_alpha_stack[d][k] = clamp(abs(area[k]), 0.0, 1.0);
|
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--;
|
2020-11-19 11:53:59 -08:00
|
|
|
for (uint k = 0; k < CHUNK; k++) {
|
2021-03-30 12:59:49 +09:00
|
|
|
uint d = min(clip_depth, MAX_BLEND_STACK - 1);
|
|
|
|
mediump vec4 bg = unpacksRGB(blend_stack[d][k]);
|
|
|
|
mediump vec4 fg = rgba[k] * area[k] * blend_alpha_stack[d][k];
|
2021-03-22 16:06:15 +01:00
|
|
|
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
|
|
|
}
|