2020-04-21 19:30:14 -07:00
|
|
|
// This is "kernel 3" in a 4-kernel pipeline. It walks the active items
|
|
|
|
// for the tilegroup and produces a per-tile command list for each tile.
|
|
|
|
|
2020-04-21 17:55:17 -07:00
|
|
|
#version 450
|
|
|
|
#extension GL_GOOGLE_include_directive : enable
|
|
|
|
|
|
|
|
layout(local_size_x = 32, local_size_y = 1) in;
|
|
|
|
|
|
|
|
layout(set = 0, binding = 0) readonly buffer SceneBuf {
|
|
|
|
uint[] scene;
|
|
|
|
};
|
|
|
|
|
|
|
|
// TODO: this should have a `readonly` qualifier, but then inclusion
|
|
|
|
// of ptcl.h would fail because of the writers.
|
|
|
|
layout(set = 0, binding = 1) buffer TilegroupBuf {
|
|
|
|
uint[] tilegroup;
|
|
|
|
};
|
|
|
|
|
|
|
|
layout(set = 0, binding = 2) buffer PtclBuf {
|
|
|
|
uint[] ptcl;
|
|
|
|
};
|
|
|
|
|
2020-04-25 10:15:22 -07:00
|
|
|
layout(set = 0, binding = 3) buffer AllocBuf {
|
|
|
|
uint alloc;
|
|
|
|
};
|
|
|
|
|
2020-04-21 17:55:17 -07:00
|
|
|
#include "scene.h"
|
|
|
|
#include "tilegroup.h"
|
|
|
|
#include "ptcl.h"
|
|
|
|
|
2020-04-24 13:06:47 -07:00
|
|
|
#include "setup.h"
|
2020-04-21 17:55:17 -07:00
|
|
|
|
2020-04-25 10:15:22 -07:00
|
|
|
void alloc_cmd(inout CmdRef cmd_ref, inout uint cmd_limit) {
|
|
|
|
if (cmd_ref.offset > cmd_limit) {
|
|
|
|
uint new_cmd = atomicAdd(alloc, PTCL_INITIAL_ALLOC);
|
|
|
|
CmdJump jump = CmdJump(new_cmd);
|
|
|
|
Cmd_Jump_write(cmd_ref, jump);
|
|
|
|
cmd_ref = CmdRef(new_cmd);
|
|
|
|
cmd_limit = new_cmd + PTCL_INITIAL_ALLOC - 2 * Cmd_size;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2020-04-21 17:55:17 -07:00
|
|
|
void main() {
|
|
|
|
uint tile_ix = gl_GlobalInvocationID.y * WIDTH_IN_TILES + gl_GlobalInvocationID.x;
|
|
|
|
uint tilegroup_ix = gl_GlobalInvocationID.y * WIDTH_IN_TILEGROUPS
|
|
|
|
+ (gl_GlobalInvocationID.x / TILEGROUP_WIDTH_TILES);
|
|
|
|
vec2 xy0 = vec2(gl_GlobalInvocationID.xy) * vec2(TILE_WIDTH_PX, TILE_HEIGHT_PX);
|
|
|
|
TileGroupRef tg_ref = TileGroupRef(tilegroup_ix * TILEGROUP_INITIAL_ALLOC);
|
|
|
|
CmdRef cmd_ref = CmdRef(tile_ix * PTCL_INITIAL_ALLOC);
|
2020-04-25 10:15:22 -07:00
|
|
|
uint cmd_limit = cmd_ref.offset + PTCL_INITIAL_ALLOC - 2 * Cmd_size;
|
2020-04-21 17:55:17 -07:00
|
|
|
|
|
|
|
while (true) {
|
|
|
|
uint tg_tag = TileGroup_tag(tg_ref);
|
|
|
|
if (tg_tag == TileGroup_End) {
|
|
|
|
break;
|
|
|
|
}
|
2020-04-25 10:15:22 -07:00
|
|
|
if (tg_tag == TileGroup_Jump) {
|
|
|
|
tg_ref = TileGroupRef(TileGroup_Jump_read(tg_ref).new_ref);
|
|
|
|
continue;
|
|
|
|
}
|
2020-04-21 17:55:17 -07:00
|
|
|
// Assume tg_tag is `Instance`, though there will be more cases.
|
|
|
|
Instance ins = TileGroup_Instance_read(tg_ref);
|
|
|
|
PietItemRef item_ref = PietItemRef(ins.item_ref);
|
|
|
|
uint item_tag = PietItem_tag(item_ref);
|
|
|
|
switch (item_tag) {
|
|
|
|
case PietItem_Circle:
|
|
|
|
PietCircle circle = PietItem_Circle_read(item_ref);
|
|
|
|
vec2 center = ins.offset + circle.center.xy;
|
|
|
|
float r = circle.radius;
|
|
|
|
if (max(center.x - r, xy0.x) < min(center.x + r, xy0.x + float(TILE_WIDTH_PX))
|
|
|
|
&& max(center.y - r, xy0.y) < min(center.y + r, xy0.y + float(TILE_HEIGHT_PX)))
|
|
|
|
{
|
|
|
|
CmdCircle cmd = CmdCircle(center, r, circle.rgba_color);
|
2020-04-25 10:15:22 -07:00
|
|
|
alloc_cmd(cmd_ref, cmd_limit);
|
2020-04-21 17:55:17 -07:00
|
|
|
Cmd_Circle_write(cmd_ref, cmd);
|
|
|
|
cmd_ref.offset += Cmd_size;
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
tg_ref.offset += TileGroup_size;
|
|
|
|
}
|
|
|
|
Cmd_End_write(cmd_ref);
|
|
|
|
}
|