mirror of
https://github.com/LostRuins/koboldcpp.git
synced 2025-09-11 17:44:38 +00:00
vulkan: optimize rms_norm, and allow the work to spread across multiple SMs (#15281)
* vulkan: optimize rms_norm, and allow the work to spread across multiple SMs There are really two parts to this change: (1) Some optimizations similar to what we have in soft_max, to unroll with different numbers of iterations. (2) A fusion optimization where we detect add followed by rms_norm, and make the add shader atomically accumulate the values^2 into memory. Then the rms_norm shader can just load that sum. This allows the rms_norm to be parallelized across multiple workgroups, it just becomes a simple per-element multiply. The fusion optimization is currently only applied when the rms_norm is on a single vector. This previously always ran on a single SM. It could apply more broadly, but when there are other dimensions the work can already spread across SMs, and there would be some complexity to tracking multiple atomic sums. * Change add+rms_norm optimization to write out an array of partial sums rather than using atomic add, to make it deterministic. The rms_norm shader fetches a subgroup's worth in parallel and uses subgroupAdd to add them up. * complete rebase against fused adds - multi_add shader can also compute partial sums * fix validation errors * disable add_rms_fusion for Intel due to possible driver bug * resolve against #15489, sync after clearing partial sums
This commit is contained in:
parent
b1afcab804
commit
611f419cff
7 changed files with 379 additions and 50 deletions
|
@ -102,9 +102,9 @@ static bool is_pow2(uint32_t x) { return x > 1 && (x & (x-1)) == 0; }
|
||||||
|
|
||||||
struct ggml_backend_vk_context;
|
struct ggml_backend_vk_context;
|
||||||
|
|
||||||
#define MAX_PARAMETER_COUNT 8
|
#define MAX_PARAMETER_COUNT 12
|
||||||
// Max number of adds that can be fused without exceeding MAX_PARAMETER_COUNT.
|
// Max number of adds that can be fused without exceeding MAX_PARAMETER_COUNT.
|
||||||
#define MAX_FUSED_ADDS (MAX_PARAMETER_COUNT - 2)
|
#define MAX_FUSED_ADDS (MAX_PARAMETER_COUNT - 3)
|
||||||
|
|
||||||
struct vk_pipeline_struct {
|
struct vk_pipeline_struct {
|
||||||
std::string name;
|
std::string name;
|
||||||
|
@ -381,6 +381,9 @@ struct vk_device_struct {
|
||||||
bool subgroup_shuffle;
|
bool subgroup_shuffle;
|
||||||
bool multi_add;
|
bool multi_add;
|
||||||
|
|
||||||
|
bool add_rms_fusion;
|
||||||
|
uint32_t partials_binding_alignment;
|
||||||
|
|
||||||
bool integer_dot_product;
|
bool integer_dot_product;
|
||||||
|
|
||||||
bool subgroup_size_control;
|
bool subgroup_size_control;
|
||||||
|
@ -460,9 +463,12 @@ struct vk_device_struct {
|
||||||
vk_pipeline pipeline_mul_norepeat[2][2][2];
|
vk_pipeline pipeline_mul_norepeat[2][2][2];
|
||||||
vk_pipeline pipeline_div[2][2][2];
|
vk_pipeline pipeline_div[2][2][2];
|
||||||
vk_pipeline pipeline_div_norepeat[2][2][2];
|
vk_pipeline pipeline_div_norepeat[2][2][2];
|
||||||
|
vk_pipeline pipeline_add_rms[2][2][2];
|
||||||
|
vk_pipeline pipeline_add_rms_norepeat[2][2][2];
|
||||||
|
|
||||||
// indexed by num_additional_fused_ops == num_adds - 1
|
// indexed by num_additional_fused_ops == num_adds - 1
|
||||||
vk_pipeline pipeline_multi_add[MAX_FUSED_ADDS];
|
vk_pipeline pipeline_multi_add[MAX_FUSED_ADDS];
|
||||||
|
vk_pipeline pipeline_multi_add_rms[MAX_FUSED_ADDS];
|
||||||
|
|
||||||
vk_pipeline pipeline_add_id_f32;
|
vk_pipeline pipeline_add_id_f32;
|
||||||
|
|
||||||
|
@ -486,6 +492,8 @@ struct vk_device_struct {
|
||||||
vk_pipeline pipeline_group_norm_f32;
|
vk_pipeline pipeline_group_norm_f32;
|
||||||
vk_pipeline pipeline_rms_norm_f32;
|
vk_pipeline pipeline_rms_norm_f32;
|
||||||
vk_pipeline pipeline_rms_norm_mul_f32;
|
vk_pipeline pipeline_rms_norm_mul_f32;
|
||||||
|
vk_pipeline pipeline_rms_norm_partials_f32;
|
||||||
|
vk_pipeline pipeline_rms_norm_mul_partials_f32;
|
||||||
vk_pipeline pipeline_rms_norm_back_f32;
|
vk_pipeline pipeline_rms_norm_back_f32;
|
||||||
vk_pipeline pipeline_l2_norm_f32;
|
vk_pipeline pipeline_l2_norm_f32;
|
||||||
|
|
||||||
|
@ -823,8 +831,13 @@ struct vk_op_multi_add_push_constants {
|
||||||
uint32_t ne20; uint32_t ne21; uint32_t ne22; uint32_t ne23;
|
uint32_t ne20; uint32_t ne21; uint32_t ne22; uint32_t ne23;
|
||||||
|
|
||||||
// strides for srcs+dst
|
// strides for srcs+dst
|
||||||
uint32_t nb[8][4];
|
uint32_t nb[MAX_PARAMETER_COUNT][4];
|
||||||
|
|
||||||
|
uint32_t rms_partials;
|
||||||
};
|
};
|
||||||
|
// update multi_add.comp if this changes
|
||||||
|
static_assert(MAX_PARAMETER_COUNT == 12);
|
||||||
|
static_assert(sizeof(vk_op_multi_add_push_constants) <= 256);
|
||||||
|
|
||||||
struct vk_op_add_id_push_constants {
|
struct vk_op_add_id_push_constants {
|
||||||
uint32_t ne0;
|
uint32_t ne0;
|
||||||
|
@ -1208,6 +1221,12 @@ class vk_perf_logger {
|
||||||
timings[name].push_back(time);
|
timings[name].push_back(time);
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
if (node->op == GGML_OP_RMS_NORM) {
|
||||||
|
std::string name = ggml_op_name(node->op);
|
||||||
|
name += "(" + std::to_string(node->ne[0]) + "," + std::to_string(node->ne[1]) + "," + std::to_string(node->ne[2]) + "," + std::to_string(node->ne[3]) + ")";
|
||||||
|
timings[name].push_back(time);
|
||||||
|
return;
|
||||||
|
}
|
||||||
timings[ggml_op_name(node->op)].push_back(time);
|
timings[ggml_op_name(node->op)].push_back(time);
|
||||||
}
|
}
|
||||||
private:
|
private:
|
||||||
|
@ -1222,10 +1241,13 @@ struct ggml_backend_vk_context {
|
||||||
|
|
||||||
size_t semaphore_idx, event_idx;
|
size_t semaphore_idx, event_idx;
|
||||||
ggml_vk_garbage_collector gc;
|
ggml_vk_garbage_collector gc;
|
||||||
size_t prealloc_size_x, prealloc_size_y, prealloc_size_split_k;
|
size_t prealloc_size_x, prealloc_size_y, prealloc_size_split_k, prealloc_size_add_rms_partials, prealloc_size_add_rms_partials_offset;
|
||||||
vk_buffer prealloc_x, prealloc_y, prealloc_split_k;
|
vk_buffer prealloc_x, prealloc_y, prealloc_split_k, prealloc_add_rms_partials;
|
||||||
vk::Fence fence, almost_ready_fence;
|
vk::Fence fence, almost_ready_fence;
|
||||||
bool almost_ready_fence_pending {};
|
bool almost_ready_fence_pending {};
|
||||||
|
// Set before op_add and unset after op_rms_norm to indicate that the add should
|
||||||
|
// write partial sums to accumulate the square of the vector components
|
||||||
|
bool do_add_rms_partials;
|
||||||
|
|
||||||
// Cache most recent tensor that was converted into prealloc_y, and what pipeline it used to convert.
|
// Cache most recent tensor that was converted into prealloc_y, and what pipeline it used to convert.
|
||||||
vk_pipeline_struct * prealloc_y_last_pipeline_used {};
|
vk_pipeline_struct * prealloc_y_last_pipeline_used {};
|
||||||
|
@ -2987,8 +3009,12 @@ static void ggml_vk_load_shaders(vk_device& device) {
|
||||||
|
|
||||||
ggml_vk_create_pipeline(device, device->pipeline_norm_f32, "norm_f32", norm_f32_len, norm_f32_data, "main", 2, sizeof(vk_op_push_constants), {1, 1, 1}, {}, 1);
|
ggml_vk_create_pipeline(device, device->pipeline_norm_f32, "norm_f32", norm_f32_len, norm_f32_data, "main", 2, sizeof(vk_op_push_constants), {1, 1, 1}, {}, 1);
|
||||||
ggml_vk_create_pipeline(device, device->pipeline_group_norm_f32, "group_norm_f32", group_norm_f32_len, group_norm_f32_data, "main", 2, sizeof(vk_op_push_constants), {1, 1, 1}, {}, 1);
|
ggml_vk_create_pipeline(device, device->pipeline_group_norm_f32, "group_norm_f32", group_norm_f32_len, group_norm_f32_data, "main", 2, sizeof(vk_op_push_constants), {1, 1, 1}, {}, 1);
|
||||||
ggml_vk_create_pipeline(device, device->pipeline_rms_norm_f32, "rms_norm_f32", rms_norm_f32_len, rms_norm_f32_data, "main", 3, sizeof(vk_op_binary_push_constants), {1, 1, 1}, {0, 0}, 1);
|
|
||||||
ggml_vk_create_pipeline(device, device->pipeline_rms_norm_mul_f32, "rms_norm_mul_f32", rms_norm_f32_len, rms_norm_f32_data, "main", 3, sizeof(vk_op_binary_push_constants), {1, 1, 1}, {0, 1}, 1);
|
ggml_vk_create_pipeline(device, device->pipeline_rms_norm_f32, "rms_norm_f32", rms_norm_f32_len, rms_norm_f32_data, "main", 4, sizeof(vk_op_binary_push_constants), {1, 1, 1}, {0, 0}, 1, true);
|
||||||
|
ggml_vk_create_pipeline(device, device->pipeline_rms_norm_mul_f32, "rms_norm_mul_f32", rms_norm_f32_len, rms_norm_f32_data, "main", 4, sizeof(vk_op_binary_push_constants), {1, 1, 1}, {0, 1}, 1, true);
|
||||||
|
ggml_vk_create_pipeline(device, device->pipeline_rms_norm_partials_f32, "rms_norm_partials_f32", rms_norm_partials_f32_len, rms_norm_partials_f32_data, "main", 4, sizeof(vk_op_binary_push_constants), {1, 1, 1}, {0, 0}, 1, true);
|
||||||
|
ggml_vk_create_pipeline(device, device->pipeline_rms_norm_mul_partials_f32, "rms_norm_mul_partials_f32", rms_norm_partials_f32_len, rms_norm_partials_f32_data, "main", 4, sizeof(vk_op_binary_push_constants), {1, 1, 1}, {0, 1}, 1, true);
|
||||||
|
|
||||||
ggml_vk_create_pipeline(device, device->pipeline_rms_norm_back_f32, "rms_norm_back_f32", rms_norm_back_f32_len, rms_norm_back_f32_data, "main", 3, sizeof(vk_op_push_constants), {1, 1, 1}, {}, 1);
|
ggml_vk_create_pipeline(device, device->pipeline_rms_norm_back_f32, "rms_norm_back_f32", rms_norm_back_f32_len, rms_norm_back_f32_data, "main", 3, sizeof(vk_op_push_constants), {1, 1, 1}, {}, 1);
|
||||||
ggml_vk_create_pipeline(device, device->pipeline_l2_norm_f32, "l2_norm_f32", l2_norm_f32_len, l2_norm_f32_data, "main", 2, sizeof(vk_op_push_constants), {1, 1, 1}, {}, 1);
|
ggml_vk_create_pipeline(device, device->pipeline_l2_norm_f32, "l2_norm_f32", l2_norm_f32_len, l2_norm_f32_data, "main", 2, sizeof(vk_op_push_constants), {1, 1, 1}, {}, 1);
|
||||||
|
|
||||||
|
@ -3058,25 +3084,28 @@ static void ggml_vk_load_shaders(vk_device& device) {
|
||||||
};
|
};
|
||||||
|
|
||||||
bool rte = device->float_controls_rte_fp16;
|
bool rte = device->float_controls_rte_fp16;
|
||||||
#define CREATE_BINARY(name, namemod, spec) \
|
#define CREATE_BINARY(name, namemod, spec, bindings) \
|
||||||
for (int s0 : {0,1}) for (int s1 : {0,1}) for (int d : {0,1}) \
|
for (int s0 : {0,1}) for (int s1 : {0,1}) for (int d : {0,1}) \
|
||||||
ggml_vk_create_pipeline(device, device->pipeline_ ## name ## namemod[s0][s1][d], \
|
ggml_vk_create_pipeline(device, device->pipeline_ ## name ## namemod[s0][s1][d], \
|
||||||
#name + get_suffix(s0, s1, d) + #namemod, name ## _len[s0][s1][d][rte], name ## _data[s0][s1][d][rte], \
|
#name + get_suffix(s0, s1, d) + #namemod, name ## _len[s0][s1][d][rte], name ## _data[s0][s1][d][rte], \
|
||||||
"main", 3, sizeof(vk_op_binary_push_constants), {512, 1, 1}, spec, 1);
|
"main", (bindings), sizeof(vk_op_binary_push_constants), {512, 1, 1}, spec, 1);
|
||||||
|
|
||||||
CREATE_BINARY(add, , {0})
|
CREATE_BINARY(add, , {0}, 4)
|
||||||
CREATE_BINARY(add, _norepeat, {1})
|
CREATE_BINARY(add, _norepeat, {1}, 4)
|
||||||
CREATE_BINARY(sub, , {0})
|
CREATE_BINARY(sub, , {0}, 3)
|
||||||
CREATE_BINARY(sub, _norepeat, {1})
|
CREATE_BINARY(sub, _norepeat, {1}, 3)
|
||||||
CREATE_BINARY(mul, , {0})
|
CREATE_BINARY(mul, , {0}, 3)
|
||||||
CREATE_BINARY(mul, _norepeat, {1})
|
CREATE_BINARY(mul, _norepeat, {1}, 3)
|
||||||
CREATE_BINARY(div, , {0})
|
CREATE_BINARY(div, , {0}, 3)
|
||||||
CREATE_BINARY(div, _norepeat, {1})
|
CREATE_BINARY(div, _norepeat, {1}, 3)
|
||||||
|
CREATE_BINARY(add_rms, , {0}, 4)
|
||||||
|
CREATE_BINARY(add_rms, _norepeat, {1}, 4)
|
||||||
#undef CREATE_BINARY
|
#undef CREATE_BINARY
|
||||||
|
|
||||||
if (device->multi_add) {
|
if (device->multi_add) {
|
||||||
for (uint32_t i = 0; i < MAX_FUSED_ADDS; ++i) {
|
for (uint32_t i = 0; i < MAX_FUSED_ADDS; ++i) {
|
||||||
ggml_vk_create_pipeline(device, device->pipeline_multi_add[i], "multi_add_f32_" + std::to_string(i+1), multi_add_f32_len, multi_add_f32_data, "main", MAX_PARAMETER_COUNT, sizeof(vk_op_multi_add_push_constants), {512, 1, 1}, {i+2}, 1);
|
ggml_vk_create_pipeline(device, device->pipeline_multi_add[i], "multi_add_f32_" + std::to_string(i+1), multi_add_f32_len, multi_add_f32_data, "main", MAX_PARAMETER_COUNT, sizeof(vk_op_multi_add_push_constants), {512, 1, 1}, {i+2}, 1);
|
||||||
|
ggml_vk_create_pipeline(device, device->pipeline_multi_add_rms[i], "multi_add_rms_f32_" + std::to_string(i+1), multi_add_rms_f32_len, multi_add_rms_f32_data, "main", MAX_PARAMETER_COUNT, sizeof(vk_op_multi_add_push_constants), {512, 1, 1}, {i+2}, 1);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -3944,6 +3973,12 @@ static vk_device ggml_vk_get_device(size_t idx) {
|
||||||
|
|
||||||
device->disable_fusion = getenv("GGML_VK_DISABLE_FUSION") != nullptr;
|
device->disable_fusion = getenv("GGML_VK_DISABLE_FUSION") != nullptr;
|
||||||
|
|
||||||
|
device->add_rms_fusion = !device->disable_fusion &&
|
||||||
|
device->subgroup_add &&
|
||||||
|
device->vendor_id != VK_VENDOR_ID_INTEL;
|
||||||
|
device->partials_binding_alignment =
|
||||||
|
std::max(4u, (uint32_t)device->properties.limits.minStorageBufferOffsetAlignment);
|
||||||
|
|
||||||
return device;
|
return device;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -7080,7 +7115,7 @@ static std::array<uint32_t, 3> ggml_vk_get_conv_elements(const ggml_tensor *dst)
|
||||||
return elements;
|
return elements;
|
||||||
}
|
}
|
||||||
|
|
||||||
static vk_pipeline ggml_vk_op_get_pipeline(ggml_backend_vk_context * ctx, const ggml_tensor * src0, const ggml_tensor * src1, const ggml_tensor * src2, ggml_tensor * dst, ggml_op op) {
|
static vk_pipeline ggml_vk_op_get_pipeline(ggml_backend_vk_context * ctx, const ggml_tensor * src0, const ggml_tensor * src1, const ggml_tensor * src2, const ggml_tensor * dst, ggml_op op) {
|
||||||
switch (op) {
|
switch (op) {
|
||||||
case GGML_OP_GET_ROWS:
|
case GGML_OP_GET_ROWS:
|
||||||
GGML_ASSERT(src1->type == GGML_TYPE_I32);
|
GGML_ASSERT(src1->type == GGML_TYPE_I32);
|
||||||
|
@ -7109,10 +7144,19 @@ static vk_pipeline ggml_vk_op_get_pipeline(ggml_backend_vk_context * ctx, const
|
||||||
case GGML_OP_ADD:
|
case GGML_OP_ADD:
|
||||||
{
|
{
|
||||||
if (ctx->num_additional_fused_ops > 0) {
|
if (ctx->num_additional_fused_ops > 0) {
|
||||||
return ctx->device->pipeline_multi_add[ctx->num_additional_fused_ops];
|
if (ctx->do_add_rms_partials) {
|
||||||
|
return ctx->device->pipeline_multi_add_rms[ctx->num_additional_fused_ops];
|
||||||
|
} else {
|
||||||
|
return ctx->device->pipeline_multi_add[ctx->num_additional_fused_ops];
|
||||||
|
}
|
||||||
|
}
|
||||||
|
if (ctx->do_add_rms_partials) {
|
||||||
|
auto pipelines = ggml_are_same_shape(src0, src1) ? ctx->device->pipeline_add_rms_norepeat : ctx->device->pipeline_add_rms;
|
||||||
|
return pipelines[src0->type == GGML_TYPE_F16][src1->type == GGML_TYPE_F16][dst->type == GGML_TYPE_F16];
|
||||||
|
} else {
|
||||||
|
auto pipelines = ggml_are_same_shape(src0, src1) ? ctx->device->pipeline_add_norepeat : ctx->device->pipeline_add;
|
||||||
|
return pipelines[src0->type == GGML_TYPE_F16][src1->type == GGML_TYPE_F16][dst->type == GGML_TYPE_F16];
|
||||||
}
|
}
|
||||||
auto pipelines = ggml_are_same_shape(src0, src1) ? ctx->device->pipeline_add_norepeat : ctx->device->pipeline_add;
|
|
||||||
return pipelines[src0->type == GGML_TYPE_F16][src1->type == GGML_TYPE_F16][dst->type == GGML_TYPE_F16];
|
|
||||||
}
|
}
|
||||||
case GGML_OP_SUB:
|
case GGML_OP_SUB:
|
||||||
{
|
{
|
||||||
|
@ -7235,7 +7279,11 @@ static vk_pipeline ggml_vk_op_get_pipeline(ggml_backend_vk_context * ctx, const
|
||||||
return nullptr;
|
return nullptr;
|
||||||
case GGML_OP_RMS_NORM:
|
case GGML_OP_RMS_NORM:
|
||||||
if (src0->type == GGML_TYPE_F32 && dst->type == GGML_TYPE_F32) {
|
if (src0->type == GGML_TYPE_F32 && dst->type == GGML_TYPE_F32) {
|
||||||
return ctx->num_additional_fused_ops > 0 ? ctx->device->pipeline_rms_norm_mul_f32 : ctx->device->pipeline_rms_norm_f32;
|
if (ctx->do_add_rms_partials) {
|
||||||
|
return ctx->num_additional_fused_ops > 0 ? ctx->device->pipeline_rms_norm_mul_partials_f32 : ctx->device->pipeline_rms_norm_partials_f32;
|
||||||
|
} else {
|
||||||
|
return ctx->num_additional_fused_ops > 0 ? ctx->device->pipeline_rms_norm_mul_f32 : ctx->device->pipeline_rms_norm_f32;
|
||||||
|
}
|
||||||
}
|
}
|
||||||
return nullptr;
|
return nullptr;
|
||||||
case GGML_OP_RMS_NORM_BACK:
|
case GGML_OP_RMS_NORM_BACK:
|
||||||
|
@ -7748,7 +7796,12 @@ static void ggml_vk_op_f32(ggml_backend_vk_context * ctx, vk_context& subctx, co
|
||||||
}
|
}
|
||||||
} break;
|
} break;
|
||||||
case GGML_OP_RMS_NORM:
|
case GGML_OP_RMS_NORM:
|
||||||
elements = { (uint32_t)ne01, (uint32_t)ne02, (uint32_t)ne03 };
|
if (ctx->do_add_rms_partials) {
|
||||||
|
// Run one element per thread, 128 threads per workgroup
|
||||||
|
elements = { (uint32_t)CEIL_DIV(ne00, 128), 1, 1 };
|
||||||
|
} else {
|
||||||
|
elements = { (uint32_t)ne01, (uint32_t)ne02, (uint32_t)ne03 };
|
||||||
|
}
|
||||||
break;
|
break;
|
||||||
|
|
||||||
case GGML_OP_SUM:
|
case GGML_OP_SUM:
|
||||||
|
@ -7897,7 +7950,16 @@ static void ggml_vk_op_f32(ggml_backend_vk_context * ctx, vk_context& subctx, co
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
if (op == GGML_OP_GLU) {
|
if (op == GGML_OP_ADD || op == GGML_OP_RMS_NORM) {
|
||||||
|
vk_buffer d_A = ctx->do_add_rms_partials ? ctx->prealloc_add_rms_partials : d_X;
|
||||||
|
size_t a_buf_offset = ctx->do_add_rms_partials ? ctx->prealloc_size_add_rms_partials_offset : 0;
|
||||||
|
ggml_vk_dispatch_pipeline(ctx, subctx, pipeline,
|
||||||
|
{ vk_subbuffer{ d_X, x_buf_offset, x_sz },
|
||||||
|
vk_subbuffer{ d_Y, y_buf_offset, y_sz },
|
||||||
|
vk_subbuffer{ d_D, d_buf_offset, d_sz },
|
||||||
|
vk_subbuffer{ d_A, a_buf_offset, VK_WHOLE_SIZE },
|
||||||
|
}, pc, elements);
|
||||||
|
} else if (op == GGML_OP_GLU) {
|
||||||
// Empty src1 is possible in glu, but the shader needs a buffer
|
// Empty src1 is possible in glu, but the shader needs a buffer
|
||||||
vk_subbuffer subbuf_y;
|
vk_subbuffer subbuf_y;
|
||||||
if (use_src1) {
|
if (use_src1) {
|
||||||
|
@ -7998,7 +8060,7 @@ static void ggml_vk_multi_add(ggml_backend_vk_context * ctx, vk_context& subctx,
|
||||||
const ggml_tensor *tensors[MAX_PARAMETER_COUNT];
|
const ggml_tensor *tensors[MAX_PARAMETER_COUNT];
|
||||||
uint32_t num_srcs = ctx->num_additional_fused_ops + 2;
|
uint32_t num_srcs = ctx->num_additional_fused_ops + 2;
|
||||||
uint32_t num_tensors = num_srcs + 1;
|
uint32_t num_tensors = num_srcs + 1;
|
||||||
GGML_ASSERT(num_tensors <= MAX_PARAMETER_COUNT);
|
GGML_ASSERT(num_tensors + ctx->do_add_rms_partials <= MAX_PARAMETER_COUNT);
|
||||||
|
|
||||||
tensors[0] = first_node->src[0];
|
tensors[0] = first_node->src[0];
|
||||||
tensors[1] = first_node->src[1];
|
tensors[1] = first_node->src[1];
|
||||||
|
@ -8025,8 +8087,9 @@ static void ggml_vk_multi_add(ggml_backend_vk_context * ctx, vk_context& subctx,
|
||||||
pc.nb[i][2] = (uint32_t)t->nb[2] / sizeof(float);
|
pc.nb[i][2] = (uint32_t)t->nb[2] / sizeof(float);
|
||||||
pc.nb[i][3] = (uint32_t)t->nb[3] / sizeof(float);
|
pc.nb[i][3] = (uint32_t)t->nb[3] / sizeof(float);
|
||||||
}
|
}
|
||||||
|
pc.rms_partials = ctx->do_add_rms_partials;
|
||||||
|
|
||||||
vk_pipeline pipeline = ctx->device->pipeline_multi_add[ctx->num_additional_fused_ops];
|
vk_pipeline pipeline = ggml_vk_op_get_pipeline(ctx, tensors[0], tensors[1], nullptr, dst, dst->op);
|
||||||
|
|
||||||
if (pipeline == nullptr) {
|
if (pipeline == nullptr) {
|
||||||
std::cerr << "ggml_vulkan: Error: Missing multi_add";
|
std::cerr << "ggml_vulkan: Error: Missing multi_add";
|
||||||
|
@ -8064,6 +8127,10 @@ static void ggml_vk_multi_add(ggml_backend_vk_context * ctx, vk_context& subctx,
|
||||||
buf[i] = buf[0];
|
buf[i] = buf[0];
|
||||||
offset[i] = 0;
|
offset[i] = 0;
|
||||||
}
|
}
|
||||||
|
if (ctx->do_add_rms_partials) {
|
||||||
|
buf[num_tensors] = ctx->prealloc_add_rms_partials;
|
||||||
|
offset[num_tensors] = ctx->prealloc_size_add_rms_partials_offset;
|
||||||
|
}
|
||||||
|
|
||||||
std::array<uint32_t, 3> elements;
|
std::array<uint32_t, 3> elements;
|
||||||
|
|
||||||
|
@ -8076,6 +8143,7 @@ static void ggml_vk_multi_add(ggml_backend_vk_context * ctx, vk_context& subctx,
|
||||||
elements = { ne, 1, 1 };
|
elements = { ne, 1, 1 };
|
||||||
}
|
}
|
||||||
|
|
||||||
|
static_assert(MAX_PARAMETER_COUNT == 12);
|
||||||
ggml_vk_dispatch_pipeline(ctx, subctx, pipeline,
|
ggml_vk_dispatch_pipeline(ctx, subctx, pipeline,
|
||||||
{
|
{
|
||||||
vk_subbuffer{ buf[0], offset[0], VK_WHOLE_SIZE },
|
vk_subbuffer{ buf[0], offset[0], VK_WHOLE_SIZE },
|
||||||
|
@ -8086,6 +8154,10 @@ static void ggml_vk_multi_add(ggml_backend_vk_context * ctx, vk_context& subctx,
|
||||||
vk_subbuffer{ buf[5], offset[5], VK_WHOLE_SIZE },
|
vk_subbuffer{ buf[5], offset[5], VK_WHOLE_SIZE },
|
||||||
vk_subbuffer{ buf[6], offset[6], VK_WHOLE_SIZE },
|
vk_subbuffer{ buf[6], offset[6], VK_WHOLE_SIZE },
|
||||||
vk_subbuffer{ buf[7], offset[7], VK_WHOLE_SIZE },
|
vk_subbuffer{ buf[7], offset[7], VK_WHOLE_SIZE },
|
||||||
|
vk_subbuffer{ buf[8], offset[8], VK_WHOLE_SIZE },
|
||||||
|
vk_subbuffer{ buf[9], offset[9], VK_WHOLE_SIZE },
|
||||||
|
vk_subbuffer{ buf[10], offset[10], VK_WHOLE_SIZE },
|
||||||
|
vk_subbuffer{ buf[11], offset[11], VK_WHOLE_SIZE },
|
||||||
}, pc, elements);
|
}, pc, elements);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -8100,7 +8172,7 @@ static void ggml_vk_add(ggml_backend_vk_context * ctx, vk_context& subctx, const
|
||||||
(uint32_t)src1->ne[0], (uint32_t)src1->ne[1], (uint32_t)src1->ne[2],(uint32_t)src1->ne[3], (uint32_t)src1->nb[0] / src1_type_size, (uint32_t)src1->nb[1] / src1_type_size, (uint32_t)src1->nb[2] / src1_type_size, (uint32_t)src1->nb[3] / src1_type_size,
|
(uint32_t)src1->ne[0], (uint32_t)src1->ne[1], (uint32_t)src1->ne[2],(uint32_t)src1->ne[3], (uint32_t)src1->nb[0] / src1_type_size, (uint32_t)src1->nb[1] / src1_type_size, (uint32_t)src1->nb[2] / src1_type_size, (uint32_t)src1->nb[3] / src1_type_size,
|
||||||
(uint32_t) dst->ne[0], (uint32_t) dst->ne[1], (uint32_t) dst->ne[2],(uint32_t) dst->ne[3], (uint32_t) dst->nb[0] / dst_type_size, (uint32_t) dst->nb[1] / dst_type_size, (uint32_t) dst->nb[2] / dst_type_size, (uint32_t) dst->nb[3] / dst_type_size,
|
(uint32_t) dst->ne[0], (uint32_t) dst->ne[1], (uint32_t) dst->ne[2],(uint32_t) dst->ne[3], (uint32_t) dst->nb[0] / dst_type_size, (uint32_t) dst->nb[1] / dst_type_size, (uint32_t) dst->nb[2] / dst_type_size, (uint32_t) dst->nb[3] / dst_type_size,
|
||||||
0,
|
0,
|
||||||
0.0f, 0.0f, 0,
|
0.0f, 0.0f, ctx->do_add_rms_partials,
|
||||||
}, dryrun);
|
}, dryrun);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -8558,19 +8630,39 @@ static void ggml_vk_group_norm(ggml_backend_vk_context * ctx, vk_context& subctx
|
||||||
ggml_vk_op_f32<vk_op_push_constants>(ctx, subctx, src0, nullptr, nullptr, dst, GGML_OP_GROUP_NORM, { group_size, 0, eps, 0.0f }, dryrun);
|
ggml_vk_op_f32<vk_op_push_constants>(ctx, subctx, src0, nullptr, nullptr, dst, GGML_OP_GROUP_NORM, { group_size, 0, eps, 0.0f }, dryrun);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
static uint32_t ggml_vk_rms_num_partials(ggml_backend_vk_context * ctx, const ggml_tensor *node) {
|
||||||
|
const uint32_t ne = (uint32_t)node->ne[0];
|
||||||
|
const uint32_t denom = ctx->device->pipeline_add_rms[0][0][0]->wg_denoms[0];
|
||||||
|
const uint32_t num_partials = CEIL_DIV(ne, denom);
|
||||||
|
return num_partials;
|
||||||
|
}
|
||||||
|
|
||||||
|
static uint32_t ggml_vk_rms_partials_size(ggml_backend_vk_context * ctx, const ggml_tensor *node) {
|
||||||
|
const uint32_t num_partials = ggml_vk_rms_num_partials(ctx, node);
|
||||||
|
const uint32_t num_bytes = ROUNDUP_POW2(num_partials * sizeof(uint32_t), ctx->device->partials_binding_alignment);
|
||||||
|
return num_bytes;
|
||||||
|
}
|
||||||
|
|
||||||
static void ggml_vk_rms_norm(ggml_backend_vk_context * ctx, vk_context& subctx, const ggml_tensor * src0, const ggml_tensor * src1, ggml_tensor * dst, float * op_params, bool dryrun = false) {
|
static void ggml_vk_rms_norm(ggml_backend_vk_context * ctx, vk_context& subctx, const ggml_tensor * src0, const ggml_tensor * src1, ggml_tensor * dst, float * op_params, bool dryrun = false) {
|
||||||
const uint32_t src0_type_size = ggml_type_size(src0->type);
|
const uint32_t src0_type_size = ggml_type_size(src0->type);
|
||||||
const uint32_t src1_type_size = ggml_type_size(src1->type);
|
const uint32_t src1_type_size = ggml_type_size(src1->type);
|
||||||
const uint32_t dst_type_size = ggml_type_size(dst->type);
|
const uint32_t dst_type_size = ggml_type_size(dst->type);
|
||||||
|
|
||||||
|
uint32_t param3 = ctx->do_add_rms_partials ? ggml_vk_rms_num_partials(ctx, dst) : 0;
|
||||||
|
|
||||||
ggml_vk_op_f32<vk_op_binary_push_constants>(ctx, subctx, src0, src1, nullptr, dst, GGML_OP_RMS_NORM, {
|
ggml_vk_op_f32<vk_op_binary_push_constants>(ctx, subctx, src0, src1, nullptr, dst, GGML_OP_RMS_NORM, {
|
||||||
(uint32_t)ggml_nelements(src0),
|
(uint32_t)ggml_nelements(src0),
|
||||||
(uint32_t)src0->ne[0], (uint32_t)src0->ne[1], (uint32_t)src0->ne[2],(uint32_t)src0->ne[3], (uint32_t)src0->nb[0] / src0_type_size, (uint32_t)src0->nb[1] / src0_type_size, (uint32_t)src0->nb[2] / src0_type_size, (uint32_t)src0->nb[3] / src0_type_size,
|
(uint32_t)src0->ne[0], (uint32_t)src0->ne[1], (uint32_t)src0->ne[2],(uint32_t)src0->ne[3], (uint32_t)src0->nb[0] / src0_type_size, (uint32_t)src0->nb[1] / src0_type_size, (uint32_t)src0->nb[2] / src0_type_size, (uint32_t)src0->nb[3] / src0_type_size,
|
||||||
(uint32_t)src1->ne[0], (uint32_t)src1->ne[1], (uint32_t)src1->ne[2],(uint32_t)src1->ne[3], (uint32_t)src1->nb[0] / src1_type_size, (uint32_t)src1->nb[1] / src1_type_size, (uint32_t)src1->nb[2] / src1_type_size, (uint32_t)src1->nb[3] / src1_type_size,
|
(uint32_t)src1->ne[0], (uint32_t)src1->ne[1], (uint32_t)src1->ne[2],(uint32_t)src1->ne[3], (uint32_t)src1->nb[0] / src1_type_size, (uint32_t)src1->nb[1] / src1_type_size, (uint32_t)src1->nb[2] / src1_type_size, (uint32_t)src1->nb[3] / src1_type_size,
|
||||||
(uint32_t) dst->ne[0], (uint32_t) dst->ne[1], (uint32_t) dst->ne[2],(uint32_t) dst->ne[3], (uint32_t) dst->nb[0] / dst_type_size, (uint32_t) dst->nb[1] / dst_type_size, (uint32_t) dst->nb[2] / dst_type_size, (uint32_t) dst->nb[3] / dst_type_size,
|
(uint32_t) dst->ne[0], (uint32_t) dst->ne[1], (uint32_t) dst->ne[2],(uint32_t) dst->ne[3], (uint32_t) dst->nb[0] / dst_type_size, (uint32_t) dst->nb[1] / dst_type_size, (uint32_t) dst->nb[2] / dst_type_size, (uint32_t) dst->nb[3] / dst_type_size,
|
||||||
0,
|
0,
|
||||||
op_params[0], 0.0f, 0,
|
op_params[0], 0.0f, (int32_t)param3,
|
||||||
}, dryrun);
|
}, dryrun);
|
||||||
|
|
||||||
|
if (ctx->do_add_rms_partials) {
|
||||||
|
ctx->prealloc_size_add_rms_partials_offset += ggml_vk_rms_partials_size(ctx, src0);
|
||||||
|
ctx->do_add_rms_partials = false;
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
static void ggml_vk_rms_norm_back(ggml_backend_vk_context * ctx, vk_context& subctx, const ggml_tensor * src0, const ggml_tensor * src1, ggml_tensor * dst, bool dryrun = false) {
|
static void ggml_vk_rms_norm_back(ggml_backend_vk_context * ctx, vk_context& subctx, const ggml_tensor * src0, const ggml_tensor * src1, ggml_tensor * dst, bool dryrun = false) {
|
||||||
|
@ -9848,6 +9940,14 @@ static void ggml_vk_preallocate_buffers(ggml_backend_vk_context * ctx) {
|
||||||
}
|
}
|
||||||
ctx->prealloc_split_k = ggml_vk_create_buffer_device(ctx->device, ctx->prealloc_size_split_k);
|
ctx->prealloc_split_k = ggml_vk_create_buffer_device(ctx->device, ctx->prealloc_size_split_k);
|
||||||
}
|
}
|
||||||
|
if (ctx->prealloc_add_rms_partials == nullptr || (ctx->prealloc_size_add_rms_partials > 0 && ctx->prealloc_add_rms_partials->size < ctx->prealloc_size_add_rms_partials)) {
|
||||||
|
VK_LOG_MEMORY("ggml_vk_preallocate_buffers(add_partials_size: " << ctx->prealloc_add_rms_partials << ")");
|
||||||
|
// Resize buffer
|
||||||
|
if (ctx->prealloc_add_rms_partials != nullptr) {
|
||||||
|
ggml_vk_destroy_buffer(ctx->prealloc_add_rms_partials);
|
||||||
|
}
|
||||||
|
ctx->prealloc_add_rms_partials = ggml_vk_create_buffer_device(ctx->device, ctx->prealloc_size_add_rms_partials);
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
static bool ggml_vk_compute_forward(ggml_backend_vk_context* ctx, ggml_cgraph * cgraph, ggml_tensor* tensor, int tensor_idx, bool use_fence, bool almost_ready);
|
static bool ggml_vk_compute_forward(ggml_backend_vk_context* ctx, ggml_cgraph * cgraph, ggml_tensor* tensor, int tensor_idx, bool use_fence, bool almost_ready);
|
||||||
|
@ -9904,10 +10004,23 @@ static bool ggml_vk_build_graph(ggml_backend_vk_context * ctx, ggml_cgraph * cgr
|
||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
break;
|
break;
|
||||||
|
case GGML_OP_ADD:
|
||||||
|
{
|
||||||
|
int next_node_idx = node_idx + 1 + ctx->num_additional_fused_ops;
|
||||||
|
if (next_node_idx < cgraph->n_nodes &&
|
||||||
|
cgraph->nodes[next_node_idx]->op == GGML_OP_RMS_NORM &&
|
||||||
|
cgraph->nodes[next_node_idx]->src[0] == cgraph->nodes[next_node_idx - 1] &&
|
||||||
|
ggml_nrows(cgraph->nodes[next_node_idx]) == 1 &&
|
||||||
|
ctx->device->add_rms_fusion) {
|
||||||
|
if (dryrun) {
|
||||||
|
ctx->prealloc_size_add_rms_partials += ggml_vk_rms_partials_size(ctx, cgraph->nodes[node_idx]);
|
||||||
|
}
|
||||||
|
ctx->do_add_rms_partials = true;
|
||||||
|
}
|
||||||
|
} break;
|
||||||
case GGML_OP_REPEAT:
|
case GGML_OP_REPEAT:
|
||||||
case GGML_OP_REPEAT_BACK:
|
case GGML_OP_REPEAT_BACK:
|
||||||
case GGML_OP_GET_ROWS:
|
case GGML_OP_GET_ROWS:
|
||||||
case GGML_OP_ADD:
|
|
||||||
case GGML_OP_ADD_ID:
|
case GGML_OP_ADD_ID:
|
||||||
case GGML_OP_ACC:
|
case GGML_OP_ACC:
|
||||||
case GGML_OP_SUB:
|
case GGML_OP_SUB:
|
||||||
|
@ -10029,6 +10142,9 @@ static bool ggml_vk_build_graph(ggml_backend_vk_context * ctx, ggml_cgraph * cgr
|
||||||
// do the only thing needed for the dryrun.
|
// do the only thing needed for the dryrun.
|
||||||
vk_pipeline pipeline = ggml_vk_op_get_pipeline(ctx, src0, src1, src2, node, node->op);
|
vk_pipeline pipeline = ggml_vk_op_get_pipeline(ctx, src0, src1, src2, node, node->op);
|
||||||
ggml_pipeline_request_descriptor_sets(ctx, pipeline, 1);
|
ggml_pipeline_request_descriptor_sets(ctx, pipeline, 1);
|
||||||
|
if (node->op == GGML_OP_RMS_NORM) {
|
||||||
|
ctx->do_add_rms_partials = false;
|
||||||
|
}
|
||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
default:
|
default:
|
||||||
|
@ -11098,6 +11214,10 @@ static ggml_status ggml_backend_vk_graph_compute(ggml_backend_t backend, ggml_cg
|
||||||
vk_instance.pfn_vkQueueBeginDebugUtilsLabelEXT(ctx->device->compute_queue.queue, reinterpret_cast<VkDebugUtilsLabelEXT*>(&dul));
|
vk_instance.pfn_vkQueueBeginDebugUtilsLabelEXT(ctx->device->compute_queue.queue, reinterpret_cast<VkDebugUtilsLabelEXT*>(&dul));
|
||||||
}
|
}
|
||||||
|
|
||||||
|
ctx->prealloc_size_add_rms_partials = 0;
|
||||||
|
ctx->prealloc_size_add_rms_partials_offset = 0;
|
||||||
|
ctx->do_add_rms_partials = false;
|
||||||
|
|
||||||
uint64_t total_mat_mul_bytes = 0;
|
uint64_t total_mat_mul_bytes = 0;
|
||||||
for (int i = 0; i < cgraph->n_nodes; i++) {
|
for (int i = 0; i < cgraph->n_nodes; i++) {
|
||||||
if (!ctx->device->disable_fusion) {
|
if (!ctx->device->disable_fusion) {
|
||||||
|
@ -11166,6 +11286,19 @@ static ggml_status ggml_backend_vk_graph_compute(ggml_backend_t backend, ggml_cg
|
||||||
ctx->prealloc_y_last_pipeline_used = nullptr;
|
ctx->prealloc_y_last_pipeline_used = nullptr;
|
||||||
ctx->prealloc_y_last_tensor_used = nullptr;
|
ctx->prealloc_y_last_tensor_used = nullptr;
|
||||||
|
|
||||||
|
if (ctx->prealloc_size_add_rms_partials) {
|
||||||
|
if (ctx->compute_ctx.expired()) {
|
||||||
|
compute_ctx = ggml_vk_create_context(ctx, ctx->compute_cmd_pool);
|
||||||
|
ctx->compute_ctx = compute_ctx;
|
||||||
|
ggml_vk_ctx_begin(ctx->device, compute_ctx);
|
||||||
|
} else {
|
||||||
|
compute_ctx = ctx->compute_ctx.lock();
|
||||||
|
}
|
||||||
|
// initialize partial sums to zero.
|
||||||
|
ggml_vk_buffer_memset_async(compute_ctx, ctx->prealloc_add_rms_partials, 0, 0, ctx->prealloc_size_add_rms_partials);
|
||||||
|
ggml_vk_sync_buffers(ctx, compute_ctx);
|
||||||
|
}
|
||||||
|
|
||||||
// Submit after enough work has accumulated, to overlap CPU cmdbuffer generation with GPU execution.
|
// Submit after enough work has accumulated, to overlap CPU cmdbuffer generation with GPU execution.
|
||||||
// Estimate the amount of matmul work by looking at the weight matrix size, and submit every 100MB
|
// Estimate the amount of matmul work by looking at the weight matrix size, and submit every 100MB
|
||||||
// (and scaled down based on model size, so smaller models submit earlier).
|
// (and scaled down based on model size, so smaller models submit earlier).
|
||||||
|
|
|
@ -1,20 +1,34 @@
|
||||||
#version 450
|
#version 450
|
||||||
|
|
||||||
#extension GL_EXT_shader_16bit_storage : require
|
#extension GL_EXT_shader_16bit_storage : require
|
||||||
|
#if ADD_RMS
|
||||||
|
#extension GL_KHR_shader_subgroup_arithmetic : enable
|
||||||
|
#extension GL_KHR_shader_subgroup_basic : enable
|
||||||
|
#endif
|
||||||
|
|
||||||
#include "types.comp"
|
#include "types.comp"
|
||||||
#include "generic_binary_head.comp"
|
#include "generic_binary_head.comp"
|
||||||
|
|
||||||
const uint num_threads = 256;
|
const uint num_threads = 256;
|
||||||
|
|
||||||
|
layout (binding = 3, std430) buffer PartialBuf {float partial_sums[];};
|
||||||
|
|
||||||
layout(local_size_x = num_threads, local_size_y = 1, local_size_z = 1) in;
|
layout(local_size_x = num_threads, local_size_y = 1, local_size_z = 1) in;
|
||||||
|
|
||||||
|
#if ADD_RMS
|
||||||
|
// XXX TODO this could be sized based on number of subgroups, but that't not considered a constant
|
||||||
|
shared FLOAT_TYPE sumsh[num_threads];
|
||||||
|
#endif
|
||||||
|
|
||||||
void main() {
|
void main() {
|
||||||
uint idx = get_idx();
|
uint idx = get_idx();
|
||||||
|
uint orig_idx = idx;
|
||||||
|
|
||||||
// num_threads * num_iter must equal 512, to match the wg_denoms and get_idx calculation
|
// num_threads * num_iter must equal 512, to match the wg_denoms and get_idx calculation
|
||||||
const uint num_iter = 2;
|
const uint num_iter = 2;
|
||||||
|
|
||||||
|
FLOAT_TYPE sum_sq = 0;
|
||||||
|
|
||||||
[[unroll]] for (uint i = 0; i < num_iter; ++i) {
|
[[unroll]] for (uint i = 0; i < num_iter; ++i) {
|
||||||
if (idx >= p.ne) {
|
if (idx >= p.ne) {
|
||||||
continue;
|
continue;
|
||||||
|
@ -22,8 +36,34 @@ void main() {
|
||||||
uint i00, i01, i02, i03;
|
uint i00, i01, i02, i03;
|
||||||
get_indices(idx, i00, i01, i02, i03);
|
get_indices(idx, i00, i01, i02, i03);
|
||||||
|
|
||||||
data_d[get_doffset() + dst_idx(i00, i01, i02, i03)] = D_TYPE(FLOAT_TYPE(data_a[get_aoffset() + src0_idx(i00, i01, i02, i03)]) + FLOAT_TYPE(data_b[get_boffset() + src1_idx(i00, i01, i02, i03)]));
|
FLOAT_TYPE sum = FLOAT_TYPE(data_a[get_aoffset() + src0_idx(i00, i01, i02, i03)]) + FLOAT_TYPE(data_b[get_boffset() + src1_idx(i00, i01, i02, i03)]);
|
||||||
|
sum_sq += sum*sum;
|
||||||
|
|
||||||
|
data_d[get_doffset() + dst_idx(i00, i01, i02, i03)] = D_TYPE(sum);
|
||||||
|
|
||||||
idx += num_threads;
|
idx += num_threads;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
#if ADD_RMS
|
||||||
|
if (p.param3 != 0) {
|
||||||
|
// reduce the sum within each subgroup, then across subgroups
|
||||||
|
const uint NumSubgroups = num_threads / gl_SubgroupSize;
|
||||||
|
sum_sq = subgroupAdd(sum_sq);
|
||||||
|
if (gl_SubgroupInvocationID == 0) {
|
||||||
|
sumsh[gl_SubgroupID] = sum_sq;
|
||||||
|
}
|
||||||
|
barrier();
|
||||||
|
[[unroll]] for (uint s = NumSubgroups / 2; s > 0; s >>= 1) {
|
||||||
|
if (gl_SubgroupID < s && gl_SubgroupInvocationID == 0) {
|
||||||
|
sum_sq += sumsh[gl_SubgroupID + s];
|
||||||
|
sumsh[gl_SubgroupID] = sum_sq;
|
||||||
|
}
|
||||||
|
barrier();
|
||||||
|
}
|
||||||
|
|
||||||
|
if (gl_SubgroupID == 0 && gl_SubgroupInvocationID == 0) {
|
||||||
|
partial_sums[orig_idx / (num_iter * num_threads)] = sum_sq;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
#endif
|
||||||
}
|
}
|
||||||
|
|
|
@ -3,6 +3,10 @@
|
||||||
#extension GL_EXT_shader_16bit_storage : require
|
#extension GL_EXT_shader_16bit_storage : require
|
||||||
#extension GL_EXT_nonuniform_qualifier : enable
|
#extension GL_EXT_nonuniform_qualifier : enable
|
||||||
#extension GL_EXT_control_flow_attributes : require
|
#extension GL_EXT_control_flow_attributes : require
|
||||||
|
#if ADD_RMS
|
||||||
|
#extension GL_KHR_shader_subgroup_arithmetic : enable
|
||||||
|
#extension GL_KHR_shader_subgroup_basic : enable
|
||||||
|
#endif
|
||||||
|
|
||||||
#include "rte.comp"
|
#include "rte.comp"
|
||||||
#include "types.comp"
|
#include "types.comp"
|
||||||
|
@ -14,12 +18,16 @@ layout (push_constant) uniform parameter2
|
||||||
uint ne20; uint ne21; uint ne22; uint ne23;
|
uint ne20; uint ne21; uint ne22; uint ne23;
|
||||||
|
|
||||||
// strides for srcs+dst
|
// strides for srcs+dst
|
||||||
uint nb[8][4];
|
uint nb[12][4];
|
||||||
|
|
||||||
|
uint rms_partials;
|
||||||
} p;
|
} p;
|
||||||
|
|
||||||
layout (binding = 0) readonly buffer A {A_TYPE data_a[];} a[];
|
layout (binding = 0) readonly buffer A {A_TYPE data_a[];} a[];
|
||||||
layout (binding = 0) writeonly buffer D {D_TYPE data_d[];} d[];
|
layout (binding = 0) writeonly buffer D {D_TYPE data_d[];} d[];
|
||||||
|
|
||||||
|
layout (binding = 0, std430) buffer PartialBuf {float partial_sums[];} partials[];
|
||||||
|
|
||||||
layout(constant_id = 0) const uint num_srcs = 2;
|
layout(constant_id = 0) const uint num_srcs = 2;
|
||||||
|
|
||||||
uint src_idx(uint s, uint i00, uint i01, uint i02, uint i03) {
|
uint src_idx(uint s, uint i00, uint i01, uint i02, uint i03) {
|
||||||
|
@ -42,14 +50,22 @@ const uint num_threads = 256;
|
||||||
|
|
||||||
layout(local_size_x = num_threads, local_size_y = 1, local_size_z = 1) in;
|
layout(local_size_x = num_threads, local_size_y = 1, local_size_z = 1) in;
|
||||||
|
|
||||||
|
#if ADD_RMS
|
||||||
|
// XXX TODO this could be sized based on number of subgroups, but that't not considered a constant
|
||||||
|
shared FLOAT_TYPE sumsh[num_threads];
|
||||||
|
#endif
|
||||||
|
|
||||||
void main() {
|
void main() {
|
||||||
uint idx = get_idx();
|
uint idx = get_idx();
|
||||||
|
uint orig_idx = idx;
|
||||||
|
|
||||||
uint ne = p.ne20 * p.ne21 * p.ne22 * p.ne23;
|
uint ne = p.ne20 * p.ne21 * p.ne22 * p.ne23;
|
||||||
|
|
||||||
// num_threads * num_iter must equal 512, to match the wg_denoms and get_idx calculation
|
// num_threads * num_iter must equal 512, to match the wg_denoms and get_idx calculation
|
||||||
const uint num_iter = 2;
|
const uint num_iter = 2;
|
||||||
|
|
||||||
|
FLOAT_TYPE sum_sq = 0;
|
||||||
|
|
||||||
[[unroll]] for (uint i = 0; i < num_iter; ++i) {
|
[[unroll]] for (uint i = 0; i < num_iter; ++i) {
|
||||||
if (idx >= ne) {
|
if (idx >= ne) {
|
||||||
continue;
|
continue;
|
||||||
|
@ -61,8 +77,32 @@ void main() {
|
||||||
[[unroll]] for (uint s = 0; s < num_srcs; ++s) {
|
[[unroll]] for (uint s = 0; s < num_srcs; ++s) {
|
||||||
sum += FLOAT_TYPE(a[s].data_a[src_idx(s, i00, i01, i02, i03)]);
|
sum += FLOAT_TYPE(a[s].data_a[src_idx(s, i00, i01, i02, i03)]);
|
||||||
}
|
}
|
||||||
|
sum_sq += sum*sum;
|
||||||
d[num_srcs].data_d[dst_idx(i00, i01, i02, i03)] = D_TYPE(sum);
|
d[num_srcs].data_d[dst_idx(i00, i01, i02, i03)] = D_TYPE(sum);
|
||||||
|
|
||||||
idx += num_threads;
|
idx += num_threads;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
#if ADD_RMS
|
||||||
|
if (p.rms_partials != 0) {
|
||||||
|
// reduce the sum within each subgroup, then across subgroups
|
||||||
|
const uint NumSubgroups = num_threads / gl_SubgroupSize;
|
||||||
|
sum_sq = subgroupAdd(sum_sq);
|
||||||
|
if (gl_SubgroupInvocationID == 0) {
|
||||||
|
sumsh[gl_SubgroupID] = sum_sq;
|
||||||
|
}
|
||||||
|
barrier();
|
||||||
|
[[unroll]] for (uint s = NumSubgroups / 2; s > 0; s >>= 1) {
|
||||||
|
if (gl_SubgroupID < s && gl_SubgroupInvocationID == 0) {
|
||||||
|
sum_sq += sumsh[gl_SubgroupID + s];
|
||||||
|
sumsh[gl_SubgroupID] = sum_sq;
|
||||||
|
}
|
||||||
|
barrier();
|
||||||
|
}
|
||||||
|
|
||||||
|
if (gl_SubgroupID == 0 && gl_SubgroupInvocationID == 0) {
|
||||||
|
partials[num_srcs + 1].partial_sums[orig_idx / (num_iter * num_threads)] = sum_sq;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
#endif
|
||||||
}
|
}
|
||||||
|
|
|
@ -10,9 +10,9 @@ layout (constant_id = 1) const bool do_multiply = false;
|
||||||
|
|
||||||
layout(local_size_x = BLOCK_SIZE, local_size_y = 1, local_size_z = 1) in;
|
layout(local_size_x = BLOCK_SIZE, local_size_y = 1, local_size_z = 1) in;
|
||||||
|
|
||||||
shared FLOAT_TYPE sum[BLOCK_SIZE];
|
shared FLOAT_TYPE sumsh[BLOCK_SIZE];
|
||||||
|
|
||||||
void main() {
|
void rms_norm(uint num_iters) {
|
||||||
const uint ncols = p.ne00;
|
const uint ncols = p.ne00;
|
||||||
const uint nrows = gl_NumWorkGroups.x;
|
const uint nrows = gl_NumWorkGroups.x;
|
||||||
const uint nchannels = gl_NumWorkGroups.y;
|
const uint nchannels = gl_NumWorkGroups.y;
|
||||||
|
@ -30,38 +30,76 @@ void main() {
|
||||||
uint32_t b_offset = src1_idx(0, row, channel, samp) + get_boffset();
|
uint32_t b_offset = src1_idx(0, row, channel, samp) + get_boffset();
|
||||||
uint32_t d_offset = ((samp*nchannels + channel)*nrows + row)*ncols + get_doffset();
|
uint32_t d_offset = ((samp*nchannels + channel)*nrows + row)*ncols + get_doffset();
|
||||||
|
|
||||||
sum[tid] = FLOAT_TYPE(0.0f); // partial sum for thread in warp
|
FLOAT_TYPE sum = FLOAT_TYPE(0.0f); // partial sum for thread in warp
|
||||||
|
|
||||||
[[unroll]] for (uint col = tid; col < ncols; col += BLOCK_SIZE) {
|
[[unroll]] for (uint col = tid, idx = 0; idx < num_iters; col += BLOCK_SIZE, ++idx) {
|
||||||
const FLOAT_TYPE xi = FLOAT_TYPE(data_a[a_offset + col]);
|
FLOAT_TYPE xi = FLOAT_TYPE(0);
|
||||||
sum[tid] += xi * xi;
|
if (col < ncols) {
|
||||||
|
xi = FLOAT_TYPE(data_a[a_offset + col]);
|
||||||
|
}
|
||||||
|
sum += xi * xi;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
sumsh[tid] = sum;
|
||||||
// sum up partial sums and write back result
|
// sum up partial sums and write back result
|
||||||
barrier();
|
barrier();
|
||||||
[[unroll]] for (int s = BLOCK_SIZE / 2; s > 0; s >>= 1) {
|
[[unroll]] for (int s = BLOCK_SIZE / 2; s > 0; s >>= 1) {
|
||||||
if (tid < s) {
|
if (tid < s) {
|
||||||
sum[tid] += sum[tid + s];
|
sum += sumsh[tid + s];
|
||||||
|
sumsh[tid] = sum;
|
||||||
}
|
}
|
||||||
barrier();
|
barrier();
|
||||||
}
|
}
|
||||||
|
sum = sumsh[0];
|
||||||
|
|
||||||
const FLOAT_TYPE mean = sum[0] / FLOAT_TYPE(ncols);
|
const FLOAT_TYPE mean = sum / FLOAT_TYPE(ncols);
|
||||||
const FLOAT_TYPE scale = inversesqrt(mean + FLOAT_TYPE(p.param1));
|
const FLOAT_TYPE scale = inversesqrt(mean + FLOAT_TYPE(p.param1));
|
||||||
|
|
||||||
if (do_multiply) {
|
if (do_multiply) {
|
||||||
if (ncols > p.ne10) {
|
if (ncols > p.ne10) {
|
||||||
[[unroll]] for (uint col = tid; col < ncols; col += BLOCK_SIZE) {
|
[[unroll]] for (uint col = tid, idx = 0; idx < num_iters; col += BLOCK_SIZE, ++idx) {
|
||||||
|
if (col >= ncols) {
|
||||||
|
continue;
|
||||||
|
}
|
||||||
data_d[d_offset + col] = D_TYPE(scale * FLOAT_TYPE(data_a[a_offset + col]) * FLOAT_TYPE(data_b[b_offset + fastmod(col, p.ne10)]));
|
data_d[d_offset + col] = D_TYPE(scale * FLOAT_TYPE(data_a[a_offset + col]) * FLOAT_TYPE(data_b[b_offset + fastmod(col, p.ne10)]));
|
||||||
}
|
}
|
||||||
} else {
|
} else {
|
||||||
[[unroll]] for (uint col = tid; col < ncols; col += BLOCK_SIZE) {
|
[[unroll]] for (uint col = tid, idx = 0; idx < num_iters; col += BLOCK_SIZE, ++idx) {
|
||||||
|
if (col >= ncols) {
|
||||||
|
continue;
|
||||||
|
}
|
||||||
data_d[d_offset + col] = D_TYPE(scale * FLOAT_TYPE(data_a[a_offset + col]) * FLOAT_TYPE(data_b[b_offset + col]));
|
data_d[d_offset + col] = D_TYPE(scale * FLOAT_TYPE(data_a[a_offset + col]) * FLOAT_TYPE(data_b[b_offset + col]));
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
} else {
|
} else {
|
||||||
[[unroll]] for (uint col = tid; col < ncols; col += BLOCK_SIZE) {
|
[[unroll]] for (uint col = tid, idx = 0; idx < num_iters; col += BLOCK_SIZE, ++idx) {
|
||||||
|
if (col >= ncols) {
|
||||||
|
continue;
|
||||||
|
}
|
||||||
data_d[d_offset + col] = D_TYPE(scale * FLOAT_TYPE(data_a[a_offset + col]));
|
data_d[d_offset + col] = D_TYPE(scale * FLOAT_TYPE(data_a[a_offset + col]));
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
void main() {
|
||||||
|
// instantiate the rms_norm function for several different
|
||||||
|
// dimensions, to allow loop unrolling
|
||||||
|
uint num_blocks = (p.ne00 + BLOCK_SIZE - 1) / BLOCK_SIZE;
|
||||||
|
if (num_blocks > 32) {
|
||||||
|
rms_norm(num_blocks);
|
||||||
|
} else if (num_blocks > 16) {
|
||||||
|
rms_norm(32);
|
||||||
|
} else if (num_blocks > 8) {
|
||||||
|
rms_norm(16);
|
||||||
|
} else if (num_blocks > 4) {
|
||||||
|
rms_norm(8);
|
||||||
|
} else if (num_blocks == 4) {
|
||||||
|
rms_norm(4);
|
||||||
|
} else if (num_blocks == 3) {
|
||||||
|
rms_norm(3);
|
||||||
|
} else if (num_blocks == 2) {
|
||||||
|
rms_norm(2);
|
||||||
|
} else if (num_blocks == 1) {
|
||||||
|
rms_norm(1);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
65
ggml/src/ggml-vulkan/vulkan-shaders/rms_norm_partials.comp
Normal file
65
ggml/src/ggml-vulkan/vulkan-shaders/rms_norm_partials.comp
Normal file
|
@ -0,0 +1,65 @@
|
||||||
|
#version 450
|
||||||
|
|
||||||
|
#include "generic_binary_head.comp"
|
||||||
|
#include "types.comp"
|
||||||
|
|
||||||
|
#extension GL_EXT_control_flow_attributes : enable
|
||||||
|
#extension GL_KHR_shader_subgroup_arithmetic : enable
|
||||||
|
#extension GL_KHR_shader_subgroup_basic : enable
|
||||||
|
|
||||||
|
#define BLOCK_SIZE 128
|
||||||
|
|
||||||
|
layout (constant_id = 1) const bool do_multiply = false;
|
||||||
|
|
||||||
|
layout(local_size_x = BLOCK_SIZE, local_size_y = 1, local_size_z = 1) in;
|
||||||
|
|
||||||
|
layout (binding = 3, std430) readonly buffer PartialsBuf {float partial_sums[];};
|
||||||
|
|
||||||
|
shared FLOAT_TYPE sumsh[BLOCK_SIZE];
|
||||||
|
|
||||||
|
void main() {
|
||||||
|
const uint ncols = p.ne00;
|
||||||
|
const uint nrows = gl_NumWorkGroups.x;
|
||||||
|
const uint nchannels = gl_NumWorkGroups.y;
|
||||||
|
|
||||||
|
const uint row = 0;
|
||||||
|
const uint channel = gl_WorkGroupID.y;
|
||||||
|
const uint samp = gl_WorkGroupID.z;
|
||||||
|
// The work is split across multiple workgroups in the x dimension. Each invocation
|
||||||
|
// processes one element
|
||||||
|
const uint tid = gl_GlobalInvocationID.x;
|
||||||
|
|
||||||
|
const uint stride_row = p.nb01;
|
||||||
|
const uint stride_channel = p.nb02;
|
||||||
|
const uint stride_sample = p.nb03;
|
||||||
|
|
||||||
|
uint32_t a_offset = samp*stride_sample + channel*stride_channel + row*stride_row + get_aoffset();
|
||||||
|
uint32_t b_offset = src1_idx(0, row, channel, samp) + get_boffset();
|
||||||
|
uint32_t d_offset = ((samp*nchannels + channel)*nrows + row)*ncols + get_doffset();
|
||||||
|
|
||||||
|
FLOAT_TYPE sum = FLOAT_TYPE(0.0f); // partial sum for thread in warp
|
||||||
|
|
||||||
|
uint32_t num_partials = p.param3;
|
||||||
|
for (uint32_t i = gl_SubgroupInvocationID; i < num_partials; i += gl_SubgroupSize) {
|
||||||
|
sum += partial_sums[i];
|
||||||
|
}
|
||||||
|
sum = subgroupAdd(sum);
|
||||||
|
|
||||||
|
uint col = tid;
|
||||||
|
if (col >= ncols) {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
const FLOAT_TYPE mean = sum / FLOAT_TYPE(ncols);
|
||||||
|
const FLOAT_TYPE scale = inversesqrt(mean + FLOAT_TYPE(p.param1));
|
||||||
|
|
||||||
|
if (do_multiply) {
|
||||||
|
if (ncols > p.ne10) {
|
||||||
|
data_d[d_offset + col] = D_TYPE(scale * FLOAT_TYPE(data_a[a_offset + col]) * FLOAT_TYPE(data_b[b_offset + fastmod(col, p.ne10)]));
|
||||||
|
} else {
|
||||||
|
data_d[d_offset + col] = D_TYPE(scale * FLOAT_TYPE(data_a[a_offset + col]) * FLOAT_TYPE(data_b[b_offset + col]));
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
data_d[d_offset + col] = D_TYPE(scale * FLOAT_TYPE(data_a[a_offset + col]));
|
||||||
|
}
|
||||||
|
}
|
|
@ -503,6 +503,7 @@ void process_shaders() {
|
||||||
string_to_spv("norm_f32", "norm.comp", merge_maps(base_dict, {{"A_TYPE", "float"}, {"D_TYPE", "float"}}));
|
string_to_spv("norm_f32", "norm.comp", merge_maps(base_dict, {{"A_TYPE", "float"}, {"D_TYPE", "float"}}));
|
||||||
string_to_spv("group_norm_f32", "group_norm.comp", merge_maps(base_dict, {{"A_TYPE", "float"}, {"D_TYPE", "float"}}));
|
string_to_spv("group_norm_f32", "group_norm.comp", merge_maps(base_dict, {{"A_TYPE", "float"}, {"D_TYPE", "float"}}));
|
||||||
string_to_spv("rms_norm_f32", "rms_norm.comp", merge_maps(base_dict, {{"A_TYPE", "float"}, {"B_TYPE", "float"}, {"D_TYPE", "float"}}));
|
string_to_spv("rms_norm_f32", "rms_norm.comp", merge_maps(base_dict, {{"A_TYPE", "float"}, {"B_TYPE", "float"}, {"D_TYPE", "float"}}));
|
||||||
|
string_to_spv("rms_norm_partials_f32", "rms_norm_partials.comp", merge_maps(base_dict, {{"A_TYPE", "float"}, {"B_TYPE", "float"}, {"D_TYPE", "float"}}));
|
||||||
string_to_spv("rms_norm_back_f32", "rms_norm_back.comp", merge_maps(base_dict, {{"A_TYPE", "float"}, {"B_TYPE", "float"}, {"D_TYPE", "float"}}));
|
string_to_spv("rms_norm_back_f32", "rms_norm_back.comp", merge_maps(base_dict, {{"A_TYPE", "float"}, {"B_TYPE", "float"}, {"D_TYPE", "float"}}));
|
||||||
string_to_spv("l2_norm_f32", "l2_norm.comp", merge_maps(base_dict, {{"A_TYPE", "float"}, {"D_TYPE", "float"}}));
|
string_to_spv("l2_norm_f32", "l2_norm.comp", merge_maps(base_dict, {{"A_TYPE", "float"}, {"D_TYPE", "float"}}));
|
||||||
|
|
||||||
|
@ -538,13 +539,15 @@ void process_shaders() {
|
||||||
s += std::string(dst_f16 ? "_f16" : "_f32");
|
s += std::string(dst_f16 ? "_f16" : "_f32");
|
||||||
return s;
|
return s;
|
||||||
};
|
};
|
||||||
for (std::string op : {"add", "sub", "mul", "div"}) {
|
for (std::string op : {"add", "sub", "mul", "div", "add_rms", }) {
|
||||||
for (auto src0_f16 : {false, true}) {
|
for (auto src0_f16 : {false, true}) {
|
||||||
for (auto src1_f16 : {false, true}) {
|
for (auto src1_f16 : {false, true}) {
|
||||||
for (auto dst_f16 : {false, true}) {
|
for (auto dst_f16 : {false, true}) {
|
||||||
for (auto rte : {false, true}) {
|
for (auto rte : {false, true}) {
|
||||||
|
auto source = op == "add_rms" ? std::string("add") : op;
|
||||||
auto name = op + get_suffix(src0_f16, src1_f16, dst_f16) + (rte ? "_rte" : "");
|
auto name = op + get_suffix(src0_f16, src1_f16, dst_f16) + (rte ? "_rte" : "");
|
||||||
string_to_spv(name.c_str(), op + ".comp", {{"A_TYPE", get_type_str(src0_f16)}, {"B_TYPE", get_type_str(src1_f16)}, {"D_TYPE", get_type_str(dst_f16)}, {"FLOAT_TYPE", "float"}, {"RTE16", rte ? "1" : "0"}});
|
auto add_rms = op == "add_rms" ? "1" : "0";
|
||||||
|
string_to_spv(name.c_str(), source + ".comp", {{"A_TYPE", get_type_str(src0_f16)}, {"B_TYPE", get_type_str(src1_f16)}, {"D_TYPE", get_type_str(dst_f16)}, {"FLOAT_TYPE", "float"}, {"RTE16", rte ? "1" : "0"}, {"ADD_RMS" , add_rms}});
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -687,7 +690,8 @@ void process_shaders() {
|
||||||
|
|
||||||
string_to_spv("add_id_f32", "add_id.comp", merge_maps(base_dict, {{"A_TYPE", "float"}, {"B_TYPE", "float"}, {"D_TYPE", "float"}}));
|
string_to_spv("add_id_f32", "add_id.comp", merge_maps(base_dict, {{"A_TYPE", "float"}, {"B_TYPE", "float"}, {"D_TYPE", "float"}}));
|
||||||
|
|
||||||
string_to_spv("multi_add_f32", "multi_add.comp", {{"A_TYPE", "float"}, {"B_TYPE", "float"}, {"D_TYPE", "float"}, {"FLOAT_TYPE", "float"}, {"RTE16", "1"}});
|
string_to_spv("multi_add_f32", "multi_add.comp", {{"A_TYPE", "float"}, {"B_TYPE", "float"}, {"D_TYPE", "float"}, {"FLOAT_TYPE", "float"}, {"RTE16", "1"}, {"ADD_RMS" , "0"}});
|
||||||
|
string_to_spv("multi_add_rms_f32", "multi_add.comp", {{"A_TYPE", "float"}, {"B_TYPE", "float"}, {"D_TYPE", "float"}, {"FLOAT_TYPE", "float"}, {"RTE16", "1"}, {"ADD_RMS" , "1"}});
|
||||||
|
|
||||||
for (auto &c : compiles) {
|
for (auto &c : compiles) {
|
||||||
c.wait();
|
c.wait();
|
||||||
|
@ -745,7 +749,7 @@ void write_output_files() {
|
||||||
}
|
}
|
||||||
|
|
||||||
std::string suffixes[2] = {"_f32", "_f16"};
|
std::string suffixes[2] = {"_f32", "_f16"};
|
||||||
for (const char *op : {"add", "sub", "mul", "div"}) {
|
for (const char *op : {"add", "sub", "mul", "div", "add_rms"}) {
|
||||||
fprintf(hdr, "extern unsigned char *%s_data[2][2][2][2];\n", op);
|
fprintf(hdr, "extern unsigned char *%s_data[2][2][2][2];\n", op);
|
||||||
fprintf(hdr, "extern uint64_t %s_len[2][2][2][2];\n", op);
|
fprintf(hdr, "extern uint64_t %s_len[2][2][2][2];\n", op);
|
||||||
std::string data = "unsigned char *" + std::string(op) + "_data[2][2][2][2] = ";
|
std::string data = "unsigned char *" + std::string(op) + "_data[2][2][2][2] = ";
|
||||||
|
|
|
@ -2858,6 +2858,7 @@ struct test_rms_norm_mul_add : public test_case {
|
||||||
const std::array<int64_t, 4> ne;
|
const std::array<int64_t, 4> ne;
|
||||||
const float eps;
|
const float eps;
|
||||||
const bool broadcast;
|
const bool broadcast;
|
||||||
|
const bool multi_add; // test a sequence of adds feeding into rms_norm
|
||||||
|
|
||||||
std::string op_desc(ggml_tensor * t) override {
|
std::string op_desc(ggml_tensor * t) override {
|
||||||
GGML_UNUSED(t);
|
GGML_UNUSED(t);
|
||||||
|
@ -2867,13 +2868,13 @@ struct test_rms_norm_mul_add : public test_case {
|
||||||
bool run_whole_graph() override { return true; }
|
bool run_whole_graph() override { return true; }
|
||||||
|
|
||||||
std::string vars() override {
|
std::string vars() override {
|
||||||
return VARS_TO_STR4(type, ne, eps, broadcast);
|
return VARS_TO_STR5(type, ne, eps, broadcast, multi_add);
|
||||||
}
|
}
|
||||||
|
|
||||||
test_rms_norm_mul_add(ggml_type type = GGML_TYPE_F32,
|
test_rms_norm_mul_add(ggml_type type = GGML_TYPE_F32,
|
||||||
std::array<int64_t, 4> ne = {64, 5, 4, 3},
|
std::array<int64_t, 4> ne = {64, 5, 4, 3},
|
||||||
float eps = 1e-6f, bool broadcast = false)
|
float eps = 1e-6f, bool broadcast = false, bool multi_add = false)
|
||||||
: type(type), ne(ne), eps(eps), broadcast(broadcast) {}
|
: type(type), ne(ne), eps(eps), broadcast(broadcast), multi_add(multi_add) {}
|
||||||
|
|
||||||
ggml_tensor * build_graph(ggml_context * ctx) override {
|
ggml_tensor * build_graph(ggml_context * ctx) override {
|
||||||
std::array<int64_t, 4> broadcast_dims = {ne[0]*2, ne[1]*3, ne[2]*3, ne[3]*4};
|
std::array<int64_t, 4> broadcast_dims = {ne[0]*2, ne[1]*3, ne[2]*3, ne[3]*4};
|
||||||
|
@ -2891,6 +2892,9 @@ struct test_rms_norm_mul_add : public test_case {
|
||||||
|
|
||||||
// Use a, b and c early, so we don't end up with an OP_NONE between rms_norm and mul
|
// Use a, b and c early, so we don't end up with an OP_NONE between rms_norm and mul
|
||||||
a = ggml_add(ctx, ggml_add(ctx, a, b), c);
|
a = ggml_add(ctx, ggml_add(ctx, a, b), c);
|
||||||
|
if (multi_add) {
|
||||||
|
a = ggml_add(ctx, ggml_add(ctx, a, b), c);
|
||||||
|
}
|
||||||
ggml_tensor * out = ggml_add(ctx, ggml_mul(ctx, ggml_rms_norm(ctx, a, eps), b), c);
|
ggml_tensor * out = ggml_add(ctx, ggml_mul(ctx, ggml_rms_norm(ctx, a, eps), b), c);
|
||||||
ggml_set_name(out, "out");
|
ggml_set_name(out, "out");
|
||||||
|
|
||||||
|
@ -5842,6 +5846,11 @@ static std::vector<std::unique_ptr<test_case>> make_test_cases_eval() {
|
||||||
test_cases.emplace_back(new test_rms_norm_mul_add(GGML_TYPE_F32, {64, 5, 4, 3}, eps));
|
test_cases.emplace_back(new test_rms_norm_mul_add(GGML_TYPE_F32, {64, 5, 4, 3}, eps));
|
||||||
test_cases.emplace_back(new test_rms_norm_mul_add(GGML_TYPE_F32, {64, 5, 4, 3}, eps, true));
|
test_cases.emplace_back(new test_rms_norm_mul_add(GGML_TYPE_F32, {64, 5, 4, 3}, eps, true));
|
||||||
}
|
}
|
||||||
|
for (uint32_t n : {1, 511, 1025, 8192, 33*512}) {
|
||||||
|
for (bool multi_add : {false, true}) {
|
||||||
|
test_cases.emplace_back(new test_rms_norm_mul_add(GGML_TYPE_F32, {n, 1, 1, 1}, 1e-6f, false, multi_add));
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
test_cases.emplace_back(new test_l2_norm(GGML_TYPE_F32, {64, 5, 4, 3}, 1e-12f));
|
test_cases.emplace_back(new test_l2_norm(GGML_TYPE_F32, {64, 5, 4, 3}, 1e-12f));
|
||||||
|
|
||||||
|
|
Loading…
Add table
Add a link
Reference in a new issue