ogl_beamforming

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

Commit: 1c66618f924819eb3dd2a35dbe4e2dd52365d4e6
Parent: 6dc7bc9c4ebd3b24556d56f7d41776d425be3613
Author: Randy Palamar
Date:   Thu, 27 Mar 2025 19:25:54 -0600

core: convert beamformer parameters to X macro, add ACE_HERCULES id

I keep having to add parameters in 3 places and its getting annoying

Diffstat:
Mbeamformer.c | 11++++++++++-
Mbeamformer_parameters.h | 123++++++++++++++++++++++++++++++-------------------------------------------------
Mshaders/das.glsl | 4++--
3 files changed, 58 insertions(+), 80 deletions(-)

diff --git a/beamformer.c b/beamformer.c @@ -503,7 +503,16 @@ static s8 push_compute_shader_header(Arena *a, ComputeShaderID shader) { s8 result = {.data = a->beg}; - push_s8(a, s8(COMPUTE_SHADER_HEADER)); + + #define X(name, type, size, gltype, glsize, comment) "\t" #gltype " " #name #glsize "; " comment "\n" + push_s8(a, s8("#version 460 core\n\n" + "layout(std140, binding = 0) uniform parameters {\n" + BEAMFORMER_PARAMS_HEAD + BEAMFORMER_UI_PARAMS + BEAMFORMER_PARAMS_TAIL + "};\n\n")); + #undef X + switch (shader) { case CS_DAS: { push_s8(a, s8("layout(" diff --git a/beamformer_parameters.h b/beamformer_parameters.h @@ -26,97 +26,66 @@ typedef enum { /* X(type, id, pretty name, fixed transmits) */ #define DAS_TYPES \ - X(FORCES, 0, "FORCES", 1) \ - X(UFORCES, 1, "UFORCES", 0) \ - X(HERCULES, 2, "HERCULES", 1) \ - X(RCA_VLS, 3, "VLS", 0) \ - X(RCA_TPW, 4, "TPW", 0) \ - X(UHERCULES, 5, "UHERCULES", 0) + X(FORCES, 0, "FORCES", 1) \ + X(UFORCES, 1, "UFORCES", 0) \ + X(HERCULES, 2, "HERCULES", 1) \ + X(RCA_VLS, 3, "VLS", 0) \ + X(RCA_TPW, 4, "TPW", 0) \ + X(UHERCULES, 5, "UHERCULES", 0) \ + X(ACE_HERCULES, 6, "ACE-HERCULES", 1) #define DAS_LOCAL_SIZE_X 32 #define DAS_LOCAL_SIZE_Y 1 #define DAS_LOCAL_SIZE_Z 32 #define MAX_BEAMFORMED_SAVED_FRAMES 16 -/* NOTE: This struct follows the OpenGL std140 layout. DO NOT modify unless you have - * read and understood the rules, particulary with regards to _member alignment_ */ -typedef struct { - v4 output_min_coordinate; /* [m] Back-Top-Left corner of output region (w ignored) */ - v4 output_max_coordinate; /* [m] Front-Bottom-Right corner of output region (w ignored)*/ - uv4 output_points; /* Width * Height * Depth * (Frame Average Count) */ - f32 sampling_frequency; /* [Hz] */ - f32 center_frequency; /* [Hz] */ - f32 speed_of_sound; /* [m/s] */ - 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; +/* TODO(rnp): actually use a substruct but generate a header compatible with MATLAB */ +/* X(name, type, size, gltype, glsize, comment) */ +#define BEAMFORMER_UI_PARAMS \ + X(output_min_coordinate, v4, , vec4, , "/* [m] Back-Top-Left corner of output region */") \ + X(output_max_coordinate, v4, , vec4, , "/* [m] Front-Bottom-Right corner of output region */") \ + X(output_points, uv4, , uvec4, , "/* Width * Height * Depth * (Frame Average Count) */") \ + X(sampling_frequency, f32, , float, , "/* [Hz] */") \ + X(center_frequency, f32, , float, , "/* [Hz] */") \ + X(speed_of_sound, f32, , float, , "/* [m/s] */") \ + X(off_axis_pos, f32, , float, , "/* [m] Position on screen normal to beamform in TPW/VLSHERCULES */") \ + X(beamform_plane, i32, , int, , "/* Plane to Beamform in TPW/VLS/HERCULES */") \ + X(f_number, f32, , float, , "/* F# (set to 0 to disable) */") \ + X(interpolate, b32, , bool, , "/* Perform Cubic Interpolation of RF Samples */") -typedef struct { - u16 channel_mapping[256]; /* Transducer Channel to Verasonics Channel */ - u16 uforces_channels[256]; /* Channels used for virtual UFORCES elements */ - f32 focal_depths[256]; /* [m] Focal Depths for each transmit of a RCA imaging scheme*/ - f32 transmit_angles[256]; /* [radians] Transmit Angles for each transmit of a RCA imaging scheme*/ - f32 xdc_transform[16]; /* IMPORTANT: column major order */ - uv4 dec_data_dim; /* Samples * Channels * Acquisitions; last element ignored */ - f32 xdc_element_pitch[2]; /* [m] Transducer Element Pitch {row, col} */ - uv2 rf_raw_dim; /* Raw Data Dimensions */ - i32 transmit_mode; /* Method/Orientation of Transmit */ - u32 decode; /* Decode or just reshape data */ - u32 das_shader_id; - f32 time_offset; /* pulse length correction time [s] */ +#define BEAMFORMER_PARAMS_HEAD \ + X(channel_mapping, u16, [256], uvec4, [32], "/* Transducer Channel to Verasonics Channel */") \ + X(uforces_channels, u16, [256], uvec4, [32], "/* Channels used for virtual UFORCES elements */") \ + X(focal_depths, f32, [256], vec4, [64], "/* [m] Focal Depths for each transmit of a RCA imaging scheme*/") \ + X(transmit_angles, f32, [256], vec4, [64], "/* [radians] Transmit Angles for each transmit of a RCA imaging scheme*/") \ + X(xdc_transform, f32, [16] , mat4, , "/* IMPORTANT: column major order */") \ + X(dec_data_dim, uv4, , uvec4, , "/* Samples * Channels * Acquisitions; last element ignored */") \ + X(xdc_element_pitch, f32, [2] , vec2, , "/* [m] Transducer Element Pitch {row, col} */") \ + X(rf_raw_dim, uv2, , uvec2, , "/* Raw Data Dimensions */") \ + X(transmit_mode, i32, , int, , "/* Method/Orientation of Transmit */") \ + X(decode, u32, , uint, , "/* Decode or just reshape data */") \ + X(das_shader_id, u32, , uint, , "") \ + X(time_offset, f32, , float, , "/* pulse length correction time [s] */") + +#define BEAMFORMER_PARAMS_TAIL \ + X(readi_group_id, u32, , uint, , "/* Which readi group this data is from */") \ + X(readi_group_size, u32, , uint, , "/* Size of readi transmit group */") - /* TODO(rnp): actually use a substruct but generate a header compatible with MATLAB */ - /* UI Parameters */ - v4 output_min_coordinate; /* [m] Back-Top-Left corner of output region (w ignored) */ - v4 output_max_coordinate; /* [m] Front-Bottom-Right corner of output region (w ignored)*/ - uv4 output_points; /* Width * Height * Depth * (Frame Average Count) */ - f32 sampling_frequency; /* [Hz] */ - f32 center_frequency; /* [Hz] */ - f32 speed_of_sound; /* [m/s] */ - 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 */ +#define X(name, type, size, gltype, glsize, comment) type name size; +typedef struct { BEAMFORMER_UI_PARAMS } BeamformerUIParameters; - u32 readi_group_id; /* Which readi group this data is from */ - u32 readi_group_size; /* Size of readi transmit group */ +/* NOTE: This struct follows the OpenGL std140 layout. DO NOT modify unless you have + * read and understood the rules, particulary with regards to _member alignment_ */ +typedef struct { + BEAMFORMER_PARAMS_HEAD + BEAMFORMER_UI_PARAMS + BEAMFORMER_PARAMS_TAIL f32 _pad[3]; } BeamformerParameters; +#undef X _Static_assert((offsetof(BeamformerParameters, output_min_coordinate) & 15) == 0, "BeamformerParameters.output_min_coordinate must lie on a 16 byte boundary"); _Static_assert((sizeof(BeamformerParameters) & 15) == 0, "sizeof(BeamformerParameters) must be a multiple of 16"); - -#define COMPUTE_SHADER_HEADER "\ -#version 460 core\n\ -\n\ -layout(std140, binding = 0) uniform parameters {\n\ - uvec4 channel_mapping[32]; /* Transducer Channel to Verasonics Channel */\n\ - uvec4 uforces_channels[32]; /* Channels used for virtual UFORCES elements */\n\ - vec4 focal_depths[64]; /* [m] Focal Depths for each transmit of a RCA imaging scheme*/\n\ - vec4 transmit_angles[64]; /* [radians] Transmit Angles for each transmit of a RCA imaging scheme*/\n\ - mat4 xdc_transform; /* IMPORTANT: column major order */\n\ - uvec4 dec_data_dim; /* Samples * Channels * Acquisitions; last element ignored */\n\ - vec2 xdc_element_pitch; /* [m] Transducer Element Pitch {row, col} */\n\ - uvec2 rf_raw_dim; /* Raw Data Dimensions */\n\ - int transmit_mode; /* Method/Orientation of Transmit */\n\ - uint decode; /* Decode or just reshape data */\n\ - uint das_shader_id;\n\ - float time_offset; /* pulse length correction time [s] */\n\ - vec4 output_min_coord; /* [m] Top left corner of output region */\n\ - vec4 output_max_coord; /* [m] Bottom right corner of output region */\n\ - uvec4 output_points; /* Width * Height * Depth * (Frame Average Count) */\n\ - float sampling_frequency; /* [Hz] */\n\ - float center_frequency; /* [Hz] */\n\ - float speed_of_sound; /* [m/s] */\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 @@ -51,8 +51,8 @@ vec2 sample_rf(uint ridx, float t) vec3 calc_image_point(vec3 voxel) { ivec3 out_data_dim = imageSize(u_out_data_tex); - vec4 output_size = abs(output_max_coord - output_min_coord); - vec3 image_point = output_min_coord.xyz + voxel * output_size.xyz / out_data_dim; + vec4 output_size = abs(output_max_coordinate - output_min_coordinate); + vec3 image_point = output_min_coordinate.xyz + voxel * output_size.xyz / out_data_dim; switch (das_shader_id) { case DAS_ID_FORCES: