30
30
#include " source/val/instruction.h"
31
31
#include " source/val/validate.h"
32
32
#include " source/val/validation_state.h"
33
+ #include " spirv-tools/libspirv.h"
33
34
#include " spirv/unified1/NonSemanticClspvReflection.h"
34
35
35
36
namespace spvtools {
@@ -283,6 +284,50 @@ spv_result_t ValidateOperandDebugType(
283
284
<< " is not a valid debug type" ;
284
285
}
285
286
287
+ spv_result_t ValidateOperandDebugSource (
288
+ ValidationState_t& _, const Instruction* inst, uint32_t source_index,
289
+ uint32_t line_index, uint32_t column_index,
290
+ spv_ext_inst_type_t ext_inst_type,
291
+ const std::function<std::string()>& ext_inst_name) {
292
+ auto * debug_source_inst = _.FindDef (inst->word (source_index));
293
+ const std::vector<uint32_t >& line_lengths =
294
+ _.GetDebugSourceLineLength (debug_source_inst->id (), false );
295
+ if (line_lengths.empty ())
296
+ return SPV_SUCCESS; // Text not provide in DebugSource
297
+
298
+ const bool vulkanDebugInfo =
299
+ ext_inst_type == SPV_EXT_INST_TYPE_NONSEMANTIC_SHADER_DEBUGINFO_100;
300
+ uint32_t line = 0 ;
301
+ uint32_t column = 0 ;
302
+ // NonSemantic uses OpConstant for all operands
303
+ if (vulkanDebugInfo) {
304
+ bool is_int32 = false , is_const_int32 = false ;
305
+ std::tie (is_int32, is_const_int32, line) =
306
+ _.EvalInt32IfConst (inst->word (line_index));
307
+ std::tie (is_int32, is_const_int32, column) =
308
+ _.EvalInt32IfConst (inst->word (column_index));
309
+ } else {
310
+ line = inst->word (line_index);
311
+ column = inst->word (column_index);
312
+ }
313
+
314
+ if (line > line_lengths.size ()) {
315
+ return _.diag (SPV_ERROR_INVALID_DATA, inst)
316
+ << ext_inst_name () << " : operand Line (" << line
317
+ << " ) is larger then the " << line_lengths.size ()
318
+ << " lines found in the DebugSource text" ;
319
+ } else if (line != 0 ) {
320
+ const uint32_t line_length = line_lengths[line - 1 ];
321
+ if (column > line_length) {
322
+ return _.diag (SPV_ERROR_INVALID_DATA, inst)
323
+ << ext_inst_name () << " : operand Column End (" << column
324
+ << " ) is larger then Line " << line << " column length of "
325
+ << line_length << " found in the DebugSource text" ;
326
+ }
327
+ }
328
+ return SPV_SUCCESS;
329
+ }
330
+
286
331
spv_result_t ValidateClspvReflectionKernel (ValidationState_t& _,
287
332
const Instruction* inst,
288
333
uint32_t version) {
@@ -1008,6 +1053,11 @@ spv_result_t ValidateClspvReflectionInstruction(ValidationState_t& _,
1008
1053
// inside
1009
1054
void BuildDebugSourceLineLength (ValidationState_t& _, const Instruction* inst,
1010
1055
uint32_t ext_inst_index) {
1056
+ if (ext_inst_index == NonSemanticShaderDebugInfo100DebugSource &&
1057
+ inst->words ().size () < 7 ) {
1058
+ return ; // The optional text was not provided
1059
+ }
1060
+
1011
1061
const bool use_last_id =
1012
1062
ext_inst_index == NonSemanticShaderDebugInfo100DebugSourceContinued;
1013
1063
@@ -3304,38 +3354,39 @@ spv_result_t ValidateExtInst(ValidationState_t& _, const Instruction* inst) {
3304
3354
auto * debug_source_inst = _.FindDef (inst->word (5 ));
3305
3355
const std::vector<uint32_t >& line_lengths =
3306
3356
_.GetDebugSourceLineLength (debug_source_inst->id (), false );
3307
-
3308
- if (line_end > line_lengths.size ()) {
3309
- return _.diag (SPV_ERROR_INVALID_DATA, inst)
3310
- << ext_inst_name () << " : operand Line End (" << line_end
3311
- << " ) is larger then the " << line_lengths.size ()
3312
- << " lines found in the DebugSource text" ;
3313
- }
3314
- if (line_start == line_end) {
3315
- const uint32_t columns = line_lengths[line_end - 1 ];
3316
- if (column_end > columns) {
3357
+ if (!line_lengths.empty ()) {
3358
+ if (line_end > line_lengths.size ()) {
3317
3359
return _.diag (SPV_ERROR_INVALID_DATA, inst)
3318
- << ext_inst_name () << " : operand Column End ("
3319
- << column_end << " ) is larger then Line " << line_end
3320
- << " column length of " << columns
3321
- << " found in the DebugSource text" ;
3360
+ << ext_inst_name () << " : operand Line End (" << line_end
3361
+ << " ) is larger then the " << line_lengths.size ()
3362
+ << " lines found in the DebugSource text" ;
3322
3363
}
3323
- } else {
3324
- uint32_t columns = line_lengths[line_start - 1 ];
3325
- if (column_start > columns) {
3326
- return _.diag (SPV_ERROR_INVALID_DATA, inst)
3327
- << ext_inst_name () << " : operand Column Start ("
3328
- << column_start << " ) is larger then Line " << line_start
3329
- << " column length of " << columns
3330
- << " found in the DebugSource text" ;
3331
- }
3332
- columns = line_lengths[line_end - 1 ];
3333
- if (column_end > columns) {
3334
- return _.diag (SPV_ERROR_INVALID_DATA, inst)
3335
- << ext_inst_name () << " : operand Column End ("
3336
- << column_end << " ) is larger then Line " << line_end
3337
- << " column length of " << columns
3338
- << " found in the DebugSource text" ;
3364
+ if (line_start == line_end) {
3365
+ const uint32_t line_length = line_lengths[line_end - 1 ];
3366
+ if (column_end > line_length) {
3367
+ return _.diag (SPV_ERROR_INVALID_DATA, inst)
3368
+ << ext_inst_name () << " : operand Column End ("
3369
+ << column_end << " ) is larger then Line " << line_end
3370
+ << " column length of " << line_length
3371
+ << " found in the DebugSource text" ;
3372
+ }
3373
+ } else {
3374
+ uint32_t line_length = line_lengths[line_start - 1 ];
3375
+ if (column_start > line_length) {
3376
+ return _.diag (SPV_ERROR_INVALID_DATA, inst)
3377
+ << ext_inst_name () << " : operand Column Start ("
3378
+ << column_start << " ) is larger then Line " << line_start
3379
+ << " column length of " << line_length
3380
+ << " found in the DebugSource text" ;
3381
+ }
3382
+ line_length = line_lengths[line_end - 1 ];
3383
+ if (column_end > line_length) {
3384
+ return _.diag (SPV_ERROR_INVALID_DATA, inst)
3385
+ << ext_inst_name () << " : operand Column End ("
3386
+ << column_end << " ) is larger then Line " << line_end
3387
+ << " column length of " << line_length
3388
+ << " found in the DebugSource text" ;
3389
+ }
3339
3390
}
3340
3391
}
3341
3392
@@ -3519,6 +3570,10 @@ spv_result_t ValidateExtInst(ValidationState_t& _, const Instruction* inst) {
3519
3570
CHECK_DEBUG_OPERAND (" Source" , CommonDebugInfoDebugSource, 7 );
3520
3571
CHECK_CONST_UINT_OPERAND (" Line" , 8 );
3521
3572
CHECK_CONST_UINT_OPERAND (" Column" , 9 );
3573
+ if (auto error = ValidateOperandDebugSource (
3574
+ _, inst, 7 , 8 , 9 , ext_inst_type, ext_inst_name))
3575
+ return error;
3576
+
3522
3577
auto validate_parent =
3523
3578
ValidateOperandLexicalScope (_, " Parent" , inst, 10 , ext_inst_name);
3524
3579
if (validate_parent != SPV_SUCCESS) return validate_parent;
@@ -3557,6 +3612,10 @@ spv_result_t ValidateExtInst(ValidationState_t& _, const Instruction* inst) {
3557
3612
CHECK_DEBUG_OPERAND (" Source" , CommonDebugInfoDebugSource, 7 );
3558
3613
CHECK_CONST_UINT_OPERAND (" Line" , 8 );
3559
3614
CHECK_CONST_UINT_OPERAND (" Column" , 9 );
3615
+ if (auto error = ValidateOperandDebugSource (
3616
+ _, inst, 7 , 8 , 9 , ext_inst_type, ext_inst_name))
3617
+ return error;
3618
+
3560
3619
auto validate_parent =
3561
3620
ValidateOperandLexicalScope (_, " Parent" , inst, 10 , ext_inst_name);
3562
3621
if (validate_parent != SPV_SUCCESS) return validate_parent;
@@ -3580,6 +3639,10 @@ spv_result_t ValidateExtInst(ValidationState_t& _, const Instruction* inst) {
3580
3639
CHECK_DEBUG_OPERAND (" Source" , CommonDebugInfoDebugSource, 7 );
3581
3640
CHECK_CONST_UINT_OPERAND (" Line" , 8 );
3582
3641
CHECK_CONST_UINT_OPERAND (" Column" , 9 );
3642
+ if (auto error = ValidateOperandDebugSource (
3643
+ _, inst, 7 , 8 , 9 , ext_inst_type, ext_inst_name))
3644
+ return error;
3645
+
3583
3646
auto validate_parent =
3584
3647
ValidateOperandLexicalScope (_, " Parent" , inst, 10 , ext_inst_name);
3585
3648
if (validate_parent != SPV_SUCCESS) return validate_parent;
@@ -3621,6 +3684,10 @@ spv_result_t ValidateExtInst(ValidationState_t& _, const Instruction* inst) {
3621
3684
CHECK_DEBUG_OPERAND (" Source" , CommonDebugInfoDebugSource, 7 );
3622
3685
CHECK_CONST_UINT_OPERAND (" Line" , 8 );
3623
3686
CHECK_CONST_UINT_OPERAND (" Column" , 9 );
3687
+ if (auto error = ValidateOperandDebugSource (
3688
+ _, inst, 7 , 8 , 9 , ext_inst_type, ext_inst_name))
3689
+ return error;
3690
+
3624
3691
// NonSemantic.Shader.DebugInfo doesn't have the Parent operand
3625
3692
if (vulkanDebugInfo) {
3626
3693
CHECK_OPERAND (" Offset" , spv::Op::OpConstant, 10 );
@@ -3673,6 +3740,10 @@ spv_result_t ValidateExtInst(ValidationState_t& _, const Instruction* inst) {
3673
3740
CHECK_DEBUG_OPERAND (" Source" , CommonDebugInfoDebugSource, 7 );
3674
3741
CHECK_CONST_UINT_OPERAND (" Line" , 8 );
3675
3742
CHECK_CONST_UINT_OPERAND (" Column" , 9 );
3743
+ if (auto error = ValidateOperandDebugSource (
3744
+ _, inst, 7 , 8 , 9 , ext_inst_type, ext_inst_name))
3745
+ return error;
3746
+
3676
3747
auto validate_parent =
3677
3748
ValidateOperandLexicalScope (_, " Parent" , inst, 10 , ext_inst_name);
3678
3749
if (validate_parent != SPV_SUCCESS) return validate_parent;
@@ -3708,6 +3779,10 @@ spv_result_t ValidateExtInst(ValidationState_t& _, const Instruction* inst) {
3708
3779
CHECK_DEBUG_OPERAND (" Source" , CommonDebugInfoDebugSource, 7 );
3709
3780
CHECK_CONST_UINT_OPERAND (" Line" , 8 );
3710
3781
CHECK_CONST_UINT_OPERAND (" Column" , 9 );
3782
+ if (auto error = ValidateOperandDebugSource (
3783
+ _, inst, 7 , 8 , 9 , ext_inst_type, ext_inst_name))
3784
+ return error;
3785
+
3711
3786
auto validate_parent =
3712
3787
ValidateOperandLexicalScope (_, " Parent" , inst, 10 , ext_inst_name);
3713
3788
if (validate_parent != SPV_SUCCESS) return validate_parent;
@@ -3719,6 +3794,10 @@ spv_result_t ValidateExtInst(ValidationState_t& _, const Instruction* inst) {
3719
3794
CHECK_DEBUG_OPERAND (" Source" , CommonDebugInfoDebugSource, 5 );
3720
3795
CHECK_CONST_UINT_OPERAND (" Line" , 6 );
3721
3796
CHECK_CONST_UINT_OPERAND (" Column" , 7 );
3797
+ if (auto error = ValidateOperandDebugSource (
3798
+ _, inst, 5 , 6 , 7 , ext_inst_type, ext_inst_name))
3799
+ return error;
3800
+
3722
3801
auto validate_parent =
3723
3802
ValidateOperandLexicalScope (_, " Parent" , inst, 8 , ext_inst_name);
3724
3803
if (validate_parent != SPV_SUCCESS) return validate_parent;
@@ -3744,6 +3823,10 @@ spv_result_t ValidateExtInst(ValidationState_t& _, const Instruction* inst) {
3744
3823
CHECK_DEBUG_OPERAND (" Source" , CommonDebugInfoDebugSource, 7 );
3745
3824
CHECK_CONST_UINT_OPERAND (" Line" , 8 );
3746
3825
CHECK_CONST_UINT_OPERAND (" Column" , 9 );
3826
+ if (auto error = ValidateOperandDebugSource (
3827
+ _, inst, 7 , 8 , 9 , ext_inst_type, ext_inst_name))
3828
+ return error;
3829
+
3747
3830
auto validate_parent =
3748
3831
ValidateOperandLexicalScope (_, " Parent" , inst, 10 , ext_inst_name);
3749
3832
if (validate_parent != SPV_SUCCESS) return validate_parent;
@@ -3823,6 +3906,9 @@ spv_result_t ValidateExtInst(ValidationState_t& _, const Instruction* inst) {
3823
3906
CHECK_DEBUG_OPERAND (" Source" , CommonDebugInfoDebugSource, 8 );
3824
3907
CHECK_CONST_UINT_OPERAND (" Line" , 9 );
3825
3908
CHECK_CONST_UINT_OPERAND (" Column" , 10 );
3909
+ if (auto error = ValidateOperandDebugSource (
3910
+ _, inst, 8 , 9 , 10 , ext_inst_type, ext_inst_name))
3911
+ return error;
3826
3912
break ;
3827
3913
}
3828
3914
case CommonDebugInfoDebugGlobalVariable: {
@@ -3833,6 +3919,10 @@ spv_result_t ValidateExtInst(ValidationState_t& _, const Instruction* inst) {
3833
3919
CHECK_DEBUG_OPERAND (" Source" , CommonDebugInfoDebugSource, 7 );
3834
3920
CHECK_CONST_UINT_OPERAND (" Line" , 8 );
3835
3921
CHECK_CONST_UINT_OPERAND (" Column" , 9 );
3922
+ if (auto error = ValidateOperandDebugSource (
3923
+ _, inst, 7 , 8 , 9 , ext_inst_type, ext_inst_name))
3924
+ return error;
3925
+
3836
3926
auto validate_scope =
3837
3927
ValidateOperandLexicalScope (_, " Scope" , inst, 10 , ext_inst_name);
3838
3928
if (validate_scope != SPV_SUCCESS) return validate_scope;
0 commit comments