ogl_beamforming

Ultrasound Beamforming Implemented with OpenGL
git clone anongit@rnpnr.xyz:ogl_beamforming.git
Log | Files | Refs | Feed | Submodules | LICENSE

Commit: bec4414befb123795eb4d38e2dead40b83860b88
Parent: 279af83c64dde4005ca28fb37720c64b1a7daccc
Author: Randy Palamar
Date:   Wed, 26 Mar 2025 08:22:36 -0600

das: make interpolation a beamform parameter

Diffstat:
Mbeamformer_parameters.h | 4++++
Mshaders/das.glsl | 22+++++++++++-----------
Mui.c | 3+++
3 files changed, 18 insertions(+), 11 deletions(-)

diff --git a/beamformer_parameters.h b/beamformer_parameters.h @@ -51,6 +51,7 @@ typedef struct { f32 off_axis_pos; /* [m] Position on screen normal to beamform in 2D HERCULES */ i32 beamform_plane; /* Plane to Beamform in 2D HERCULES */ f32 f_number; /* F# (set to 0 to disable) */ + b32 interpolate; /* Perform Cubic Interpolation of RF Samples */ } BeamformerUIParameters; typedef struct { @@ -78,9 +79,11 @@ typedef struct { f32 off_axis_pos; /* [m] Position on screen normal to beamform in 2D HERCULES */ i32 beamform_plane; /* Plane to Beamform in 2D HERCULES */ f32 f_number; /* F# (set to 0 to disable) */ + b32 interpolate; /* Perform Cubic Interpolation of RF Samples */ u32 readi_group_id; /* Which readi group this data is from */ u32 readi_group_size; /* Size of readi transmit group */ + f32 _pad[3]; } BeamformerParameters; _Static_assert((offsetof(BeamformerParameters, output_min_coordinate) & 15) == 0, @@ -113,6 +116,7 @@ layout(std140, binding = 0) uniform parameters {\n\ float off_axis_pos; /* [m] Position on screen normal to beamform in 2D HERCULES */\n\ int beamform_plane; /* Plane to Beamform in 2D HERCULES */\n\ float f_number; /* F# (set to 0 to disable) */\n\ + bool interpolate; /* Perform Cubic Interpolation of RF Samples */\n\ uint readi_group_id; /* Which readi group this data is from */\n\ uint readi_group_size; /* Size of readi transmit group */\n\ };\n\n" diff --git a/shaders/das.glsl b/shaders/das.glsl @@ -16,13 +16,6 @@ layout(location = 3) uniform uint u_cycle_t; #define TX_MODE_TX_COLS(a) (((a) & 2) != 0) #define TX_MODE_RX_COLS(a) (((a) & 1) != 0) -#if 1 -/* NOTE: interpolation is unnecessary if the data has been demodulated and not decimated */ -vec2 cubic(uint ridx, float t) -{ - return rf_data[ridx + uint(floor(t))]; -} -#else /* NOTE: See: https://cubic.org/docs/hermite.htm */ vec2 cubic(uint ridx, float x) { @@ -46,7 +39,14 @@ vec2 cubic(uint ridx, float x) vec4 C2 = vec4(P1.y, P2.y, T1.y, T2.y); return vec2(dot(S, h * C1), dot(S, h * C2)); } -#endif + +vec2 sample_rf(uint ridx, float t) +{ + vec2 result; + if (interpolate) result = cubic(ridx, t); + else result = rf_data[ridx + uint(floor(t))]; + return result; +} vec3 calc_image_point(vec3 voxel) { @@ -145,7 +145,7 @@ vec2 RCA(vec3 image_point, vec3 delta, float apodization_arg) for (uint j = 0; j < dec_data_dim.y; j++) { float sidx = sample_index(transmit_distance + length(receive_distance)); vec2 valid = vec2(sidx >= 0) * vec2(sidx < dec_data_dim.x); - sum += apodize(cubic(ridx, sidx), apodization_arg, length(receive_distance.xy)) * valid; + sum += apodize(sample_rf(ridx, sidx), apodization_arg, length(receive_distance.xy)) * valid; receive_distance -= delta; ridx += dec_data_dim.x; } @@ -197,7 +197,7 @@ vec2 HERCULES(vec3 image_point, vec3 delta, float apodization_arg) /* NOTE: tribal knowledge */ if (i == 0) valid *= inversesqrt(dec_data_dim.z); - sum += apodize(cubic(ridx, sidx), apodization_arg, + sum += apodize(sample_rf(ridx, sidx), apodization_arg, length(receive_distance.xy)) * valid; ridx += dec_data_dim.x; } @@ -230,7 +230,7 @@ vec2 uFORCES(vec3 image_point, vec3 delta, float apodization_arg) for (uint j = 0; j < dec_data_dim.y; j++) { float sidx = sample_index(transmit_dist + length(rdist)); vec2 valid = vec2(sidx >= 0) * vec2(sidx < dec_data_dim.x); - sum += apodize(cubic(ridx, sidx), apodization_arg, rdist.x) * valid; + sum += apodize(sample_rf(ridx, sidx), apodization_arg, rdist.x) * valid; rdist.x -= delta.x; ridx += dec_data_dim.x; } diff --git a/ui.c b/ui.c @@ -603,6 +603,9 @@ add_beamformer_parameters_view(Variable *parent, BeamformerCtx *ctx) add_beamformer_variable_f32(ui, result, &ui->arena, s8("F#:"), s8(""), &bp->f_number, (v2){.y = 1e3}, 1, 0.1, V_INPUT|V_TEXT|V_CAUSES_COMPUTE, ui->font); + add_beamformer_variable_b32(ui, result, &ui->arena, s8("Interpolate:"), s8("False"), s8("True"), + &bp->interpolate, V_INPUT|V_CAUSES_COMPUTE, ui->font); + return result; }