1 /* Implementation of <stdarg.h> within analyzer.
2 Copyright (C) 2022-2023 Free Software Foundation, Inc.
3 Contributed by David Malcolm <dmalcolm@redhat.com>.
5 This file is part of GCC.
7 GCC is free software; you can redistribute it and/or modify it
8 under the terms of the GNU General Public License as published by
9 the Free Software Foundation; either version 3, or (at your option)
12 GCC is distributed in the hope that it will be useful, but
13 WITHOUT ANY WARRANTY; without even the implied warranty of
14 MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
15 General Public License for more details.
17 You should have received a copy of the GNU General Public License
18 along with GCC; see the file COPYING3. If not see
19 <http://www.gnu.org/licenses/>. */
22 #define INCLUDE_MEMORY
24 #include "coretypes.h"
25 #include "make-unique.h"
28 #include "basic-block.h"
30 #include "diagnostic-path.h"
31 #include "analyzer/analyzer.h"
32 #include "analyzer/analyzer-logging.h"
33 #include "analyzer/sm.h"
34 #include "analyzer/pending-diagnostic.h"
35 #include "analyzer/call-string.h"
36 #include "analyzer/program-point.h"
37 #include "analyzer/store.h"
38 #include "analyzer/region-model.h"
39 #include "analyzer/program-state.h"
40 #include "analyzer/checker-path.h"
41 #include "analyzer/supergraph.h"
42 #include "analyzer/diagnostic-manager.h"
43 #include "analyzer/exploded-graph.h"
44 #include "diagnostic-metadata.h"
45 #include "analyzer/call-details.h"
51 /* Implementation of <stdarg.h> within analyzer.
54 - detection of interprocedural type errors involving va_arg
55 - tracking of symbolic values interprocedurally from variadic call
56 through to va_arg unpacking
57 - detection of missing va_end
58 - detection of va_arg outside of a va_start/va_end pair
59 - detection of uses of a va_list after the frame in containing the
62 The analyzer runs *before* the "stdarg" and "lower_vaarg" gimple
63 passes, which have target-dependent effects.
65 This file implements a state machine on svalues for tracking when
66 va_start has been called, so that we can detect missing va_end,
67 and misplaced va_arg, etc.
68 To do this requires an svalue that can have state, so we implement va_start
69 by creating a stack-allocated region, and use a pointer to that region
70 as the svalue that has state.
72 We call this stack-allocated region the "impl_reg". Allocating it on
73 the stack ensures that it is invalidated when the frame containing
74 the va_start returns, leading to
75 -Wanalyzer-use-of-pointer-in-stale-stack-frame on attempts to use such
78 To track svalues from variadic calls interprocedurally, we implement
79 variadic arguments via new child regions of the callee's frame_region,
80 var_arg_region, each one representing a storage slot for one of the
81 variadic arguments, accessed by index.
87 'impl_reg': pointer to next var_arg_region
88 var_arg_region for arg 0
90 var_arg_region for arg N-1
92 Hence given test_1 in stdarg-1.c, at the call to:
94 __analyzer_called_by_test_1 (int placeholder, ...);
98 __analyzer_called_by_test_1 (42, "foo", 1066, '@');
100 we push this frame for the called function:
101 clusters within frame: ‘__analyzer_called_by_test_1’@2
102 cluster for: placeholder: (int)42
103 cluster for: VAR_ARG_REG(frame: ‘__analyzer_called_by_test_1’@2, arg_idx: 0): &"foo" (TOUCHED)
104 cluster for: VAR_ARG_REG(frame: ‘__analyzer_called_by_test_1’@2, arg_idx: 1): (int)1066 (TOUCHED)
105 cluster for: VAR_ARG_REG(frame: ‘__analyzer_called_by_test_1’@2, arg_idx: 2): (int)64 (TOUCHED)
106 where the called function's frame has been populated with both the value
107 of the regular argument "placeholder", and with values for 3 variadic
111 va_start (ap, placeholder);
112 we allocate a region ALLOCA_REGION for ap to point to, populate that
113 region with the address of variadic argument 0, and set sm-state of
114 &ALLOCA_REGION to "started":
115 clusters within frame: ‘__analyzer_called_by_test_1’@2
116 cluster for: placeholder: (int)42
117 cluster for: VAR_ARG_REG(frame: ‘__analyzer_called_by_test_1’@2, arg_idx: 0): &"foo" (TOUCHED)
118 cluster for: VAR_ARG_REG(frame: ‘__analyzer_called_by_test_1’@2, arg_idx: 1): (int)1066 (TOUCHED)
119 cluster for: VAR_ARG_REG(frame: ‘__analyzer_called_by_test_1’@2, arg_idx: 2): (int)64 (TOUCHED)
120 cluster for: ap: &ALLOCA_REGION
121 cluster for: ALLOCA_REGION: &VAR_ARG_REG(frame: ‘__analyzer_called_by_test_1’@2, arg_idx: 0) (TOUCHED)
123 0x4c83700: &ALLOCA_REGION: started
127 we can look within *ap, locate the region holding the next variadic
128 argument to be extracted, extract the svalue, and advance the index
129 by effectively updating *ap.
131 At the va_end, we can set &ALLOCA_REGION's state to "ended".
133 The various __builtin_va_* accept ap by pointer, so we have e.g.:
135 __builtin_va_start (&ap, [...]);
137 except for the 2nd param of __builtin_va_copy, where the type
138 is already target-dependent (see the discussion of get_va_copy_arg
141 /* Get a tree for diagnostics.
142 Typically we have "&ap", but it will make more sense to
143 the user as just "ap", so strip off the ADDR_EXPR. */
146 get_va_list_diag_arg (tree va_list_tree
)
148 if (TREE_CODE (va_list_tree
) == ADDR_EXPR
)
149 va_list_tree
= TREE_OPERAND (va_list_tree
, 0);
153 /* Get argument ARG_IDX of va_copy.
155 builtin-types.def has:
156 DEF_PRIMITIVE_TYPE (BT_VALIST_ARG, va_list_arg_type_node)
158 and c_common_nodes_and_builtins initializes va_list_arg_type_node
159 based on whether TREE_CODE (va_list_type_node) is of ARRAY_TYPE or
160 not, giving either one or zero levels of indirection.
162 Alternatively we could be dealing with __builtin_ms_va_copy or
163 __builtin_sysv_va_copy.
165 Handle this by looking at the types of the argument in question. */
167 static const svalue
*
168 get_va_copy_arg (const region_model
*model
,
169 region_model_context
*ctxt
,
173 tree arg
= gimple_call_arg (call
, arg_idx
);
174 const svalue
*arg_sval
= model
->get_rvalue (arg
, ctxt
);
175 if (const svalue
*cast
= arg_sval
->maybe_undo_cast ())
177 if (TREE_CODE (TREE_TYPE (arg
)) == POINTER_TYPE
178 && TREE_CODE (TREE_TYPE (TREE_TYPE (arg
))) == ARRAY_TYPE
)
180 /* va_list_arg_type_node is a pointer to a va_list;
182 const region
*src_reg
= model
->deref_rvalue (arg_sval
, arg
, ctxt
);
183 const svalue
*src_reg_sval
= model
->get_store_value (src_reg
, ctxt
);
184 if (const svalue
*cast
= src_reg_sval
->maybe_undo_cast ())
190 /* va_list_arg_type_node is a va_list; return ARG_SVAL. */
197 /* A state machine for tracking the state of a va_list, so that
198 we can enforce that each va_start is paired with a va_end,
199 and va_arg only happens within a va_start/va_end pair.
200 Specifically, this tracks the state of the &ALLOCA_BUFFER
201 that va_start/va_copy allocate. */
203 class va_list_state_machine
: public state_machine
206 va_list_state_machine (logger
*logger
);
208 bool inherited_state_p () const final override
{ return false; }
210 bool on_stmt (sm_context
*sm_ctxt
,
211 const supernode
*node
,
212 const gimple
*stmt
) const final override
;
214 bool can_purge_p (state_t s
) const final override
216 return s
!= m_started
;
218 std::unique_ptr
<pending_diagnostic
> on_leak (tree var
) const final override
;
220 /* State for a va_list that is the result of a va_start or va_copy. */
223 /* State for a va_list that has had va_end called on it. */
227 void on_va_start (sm_context
*sm_ctxt
, const supernode
*node
,
228 const gcall
*call
) const;
229 void on_va_copy (sm_context
*sm_ctxt
, const supernode
*node
,
230 const gcall
*call
) const;
231 void on_va_arg (sm_context
*sm_ctxt
, const supernode
*node
,
232 const gcall
*call
) const;
233 void on_va_end (sm_context
*sm_ctxt
, const supernode
*node
,
234 const gcall
*call
) const;
235 void check_for_ended_va_list (sm_context
*sm_ctxt
,
236 const supernode
*node
,
239 const char *usage_fnname
) const;
242 /* va_list_state_machine's ctor. */
244 va_list_state_machine::va_list_state_machine (logger
*logger
)
245 : state_machine ("va_list", logger
)
247 m_started
= add_state ("started");
248 m_ended
= add_state ("ended");
251 /* Implementation of the various "va_*" functions for
252 va_list_state_machine. */
255 va_list_state_machine::on_stmt (sm_context
*sm_ctxt
,
256 const supernode
*node
,
257 const gimple
*stmt
) const
259 if (const gcall
*call
= dyn_cast
<const gcall
*> (stmt
))
261 if (gimple_call_internal_p (call
)
262 && gimple_call_internal_fn (call
) == IFN_VA_ARG
)
264 on_va_arg (sm_ctxt
, node
, call
);
268 if (tree callee_fndecl
= sm_ctxt
->get_fndecl_for_call (call
))
269 if (fndecl_built_in_p (callee_fndecl
, BUILT_IN_NORMAL
)
270 && gimple_builtin_call_types_compatible_p (call
, callee_fndecl
))
271 switch (DECL_UNCHECKED_FUNCTION_CODE (callee_fndecl
))
276 case BUILT_IN_VA_START
:
277 on_va_start (sm_ctxt
, node
, call
);
280 case BUILT_IN_VA_COPY
:
281 on_va_copy (sm_ctxt
, node
, call
);
284 case BUILT_IN_VA_END
:
285 on_va_end (sm_ctxt
, node
, call
);
292 /* Get the svalue for which va_list_state_machine holds state on argument ARG_
295 static const svalue
*
296 get_stateful_arg (sm_context
*sm_ctxt
, const gcall
*call
, unsigned arg_idx
)
298 tree ap
= gimple_call_arg (call
, arg_idx
);
300 && POINTER_TYPE_P (TREE_TYPE (ap
)))
302 if (const program_state
*new_state
= sm_ctxt
->get_new_program_state ())
304 const region_model
*new_model
= new_state
->m_region_model
;
305 const svalue
*ptr_sval
= new_model
->get_rvalue (ap
, NULL
);
306 const region
*reg
= new_model
->deref_rvalue (ptr_sval
, ap
, NULL
);
307 const svalue
*impl_sval
= new_model
->get_store_value (reg
, NULL
);
308 if (const svalue
*cast
= impl_sval
->maybe_undo_cast ())
316 /* Abstract class for diagnostics relating to va_list_state_machine. */
318 class va_list_sm_diagnostic
: public pending_diagnostic
321 bool subclass_equal_p (const pending_diagnostic
&base_other
) const override
323 const va_list_sm_diagnostic
&other
324 = (const va_list_sm_diagnostic
&)base_other
;
325 return (m_ap_sval
== other
.m_ap_sval
326 && same_tree_p (m_ap_tree
, other
.m_ap_tree
));
329 label_text
describe_state_change (const evdesc::state_change
&change
)
332 if (const char *fnname
= maybe_get_fnname (change
))
333 return change
.formatted_print ("%qs called here", fnname
);
334 return label_text ();
337 diagnostic_event::meaning
338 get_meaning_for_state_change (const evdesc::state_change
&change
)
341 if (change
.m_new_state
== m_sm
.m_started
)
342 return diagnostic_event::meaning (diagnostic_event::VERB_acquire
,
343 diagnostic_event::NOUN_resource
);
344 if (change
.m_new_state
== m_sm
.m_ended
)
345 return diagnostic_event::meaning (diagnostic_event::VERB_release
,
346 diagnostic_event::NOUN_resource
);
347 return diagnostic_event::meaning ();
351 va_list_sm_diagnostic (const va_list_state_machine
&sm
,
352 const svalue
*ap_sval
, tree ap_tree
)
353 : m_sm (sm
), m_ap_sval (ap_sval
), m_ap_tree (ap_tree
)
356 static const char *maybe_get_fnname (const evdesc::state_change
&change
)
358 if (change
.m_event
.m_stmt
)
359 if (const gcall
*call
= as_a
<const gcall
*> (change
.m_event
.m_stmt
))
360 if (tree callee_fndecl
= gimple_call_fndecl (call
))
362 if (fndecl_built_in_p (callee_fndecl
, BUILT_IN_NORMAL
))
363 switch (DECL_UNCHECKED_FUNCTION_CODE (callee_fndecl
))
365 case BUILT_IN_VA_START
:
367 case BUILT_IN_VA_COPY
:
369 case BUILT_IN_VA_END
:
376 const va_list_state_machine
&m_sm
;
377 const svalue
*m_ap_sval
;
381 /* Concrete class for -Wanalyzer-va-list-use-after-va-end:
382 complain about use of a va_list after va_end has been called on it. */
384 class va_list_use_after_va_end
: public va_list_sm_diagnostic
387 va_list_use_after_va_end (const va_list_state_machine
&sm
,
388 const svalue
*ap_sval
, tree ap_tree
,
389 const char *usage_fnname
)
390 : va_list_sm_diagnostic (sm
, ap_sval
, ap_tree
),
391 m_usage_fnname (usage_fnname
)
395 int get_controlling_option () const final override
397 return OPT_Wanalyzer_va_list_use_after_va_end
;
400 bool operator== (const va_list_use_after_va_end
&other
) const
402 return (va_list_sm_diagnostic::subclass_equal_p (other
)
403 && 0 == strcmp (m_usage_fnname
, other
.m_usage_fnname
));
406 bool emit (rich_location
*rich_loc
, logger
*) final override
408 auto_diagnostic_group d
;
409 return warning_at (rich_loc
, get_controlling_option (),
410 "%qs after %qs", m_usage_fnname
, "va_end");
413 const char *get_kind () const final override
415 return "va_list_use_after_va_end";
418 label_text
describe_state_change (const evdesc::state_change
&change
)
421 if (change
.m_new_state
== m_sm
.m_ended
)
422 m_va_end_event
= change
.m_event_id
;
423 return va_list_sm_diagnostic::describe_state_change (change
);
426 label_text
describe_final_event (const evdesc::final_event
&ev
) final override
430 if (m_va_end_event
.known_p ())
431 return ev
.formatted_print
432 ("%qs on %qE after %qs at %@",
433 m_usage_fnname
, ev
.m_expr
, "va_end", &m_va_end_event
);
435 return ev
.formatted_print
436 ("%qs on %qE after %qs",
437 m_usage_fnname
, ev
.m_expr
, "va_end");
441 if (m_va_end_event
.known_p ())
442 return ev
.formatted_print
443 ("%qs after %qs at %@",
444 m_usage_fnname
, "va_end", &m_va_end_event
);
446 return ev
.formatted_print
448 m_usage_fnname
, "va_end");
453 diagnostic_event_id_t m_va_end_event
;
454 const char *m_usage_fnname
;
457 /* Concrete class for -Wanalyzer-va-list-leak:
458 complain about a va_list in the "started" state that doesn't get after
459 va_end called on it. */
461 class va_list_leak
: public va_list_sm_diagnostic
464 va_list_leak (const va_list_state_machine
&sm
,
465 const svalue
*ap_sval
, tree ap_tree
)
466 : va_list_sm_diagnostic (sm
, ap_sval
, ap_tree
),
467 m_start_event_fnname (NULL
)
471 int get_controlling_option () const final override
473 return OPT_Wanalyzer_va_list_leak
;
476 bool operator== (const va_list_leak
&other
) const
478 return va_list_sm_diagnostic::subclass_equal_p (other
);
481 bool emit (rich_location
*rich_loc
, logger
*) final override
483 auto_diagnostic_group d
;
484 return warning_at (rich_loc
, get_controlling_option (),
485 "missing call to %qs", "va_end");
488 const char *get_kind () const final override
{ return "va_list_leak"; }
490 label_text
describe_state_change (const evdesc::state_change
&change
)
493 if (change
.m_new_state
== m_sm
.m_started
)
495 m_start_event
= change
.m_event_id
;
496 m_start_event_fnname
= maybe_get_fnname (change
);
498 return va_list_sm_diagnostic::describe_state_change (change
);
501 label_text
describe_final_event (const evdesc::final_event
&ev
) final override
505 if (m_start_event
.known_p () && m_start_event_fnname
)
506 return ev
.formatted_print
507 ("missing call to %qs on %qE to match %qs at %@",
508 "va_end", ev
.m_expr
, m_start_event_fnname
, &m_start_event
);
510 return ev
.formatted_print
511 ("missing call to %qs on %qE",
512 "va_end", ev
.m_expr
);
516 if (m_start_event
.known_p () && m_start_event_fnname
)
517 return ev
.formatted_print
518 ("missing call to %qs to match %qs at %@",
519 "va_end", m_start_event_fnname
, &m_start_event
);
521 return ev
.formatted_print
522 ("missing call to %qs",
528 diagnostic_event_id_t m_start_event
;
529 const char *m_start_event_fnname
;
532 /* Update state machine for a "va_start" call. */
535 va_list_state_machine::on_va_start (sm_context
*sm_ctxt
,
537 const gcall
*call
) const
539 const svalue
*arg
= get_stateful_arg (sm_ctxt
, call
, 0);
542 /* Transition from start state to "started". */
543 if (sm_ctxt
->get_state (call
, arg
) == m_start
)
544 sm_ctxt
->set_next_state (call
, arg
, m_started
);
548 /* Complain if ARG is in the "ended" state. */
551 va_list_state_machine::check_for_ended_va_list (sm_context
*sm_ctxt
,
552 const supernode
*node
,
555 const char *usage_fnname
) const
557 if (sm_ctxt
->get_state (call
, arg
) == m_ended
)
558 sm_ctxt
->warn (node
, call
, arg
,
559 make_unique
<va_list_use_after_va_end
>
560 (*this, arg
, NULL_TREE
, usage_fnname
));
563 /* Get the svalue with associated va_list_state_machine state for
564 ARG_IDX of CALL to va_copy, if SM_CTXT supports this,
565 or NULL otherwise. */
567 static const svalue
*
568 get_stateful_va_copy_arg (sm_context
*sm_ctxt
,
572 if (const program_state
*new_state
= sm_ctxt
->get_new_program_state ())
574 const region_model
*new_model
= new_state
->m_region_model
;
575 const svalue
*arg
= get_va_copy_arg (new_model
, NULL
, call
, arg_idx
);
581 /* Update state machine for a "va_copy" call. */
584 va_list_state_machine::on_va_copy (sm_context
*sm_ctxt
,
585 const supernode
*node
,
586 const gcall
*call
) const
588 const svalue
*src_arg
= get_stateful_va_copy_arg (sm_ctxt
, call
, 1);
590 check_for_ended_va_list (sm_ctxt
, node
, call
, src_arg
, "va_copy");
592 const svalue
*dst_arg
= get_stateful_arg (sm_ctxt
, call
, 0);
595 /* Transition from start state to "started". */
596 if (sm_ctxt
->get_state (call
, dst_arg
) == m_start
)
597 sm_ctxt
->set_next_state (call
, dst_arg
, m_started
);
601 /* Update state machine for a "va_arg" call. */
604 va_list_state_machine::on_va_arg (sm_context
*sm_ctxt
,
605 const supernode
*node
,
606 const gcall
*call
) const
608 const svalue
*arg
= get_stateful_arg (sm_ctxt
, call
, 0);
610 check_for_ended_va_list (sm_ctxt
, node
, call
, arg
, "va_arg");
613 /* Update state machine for a "va_end" call. */
616 va_list_state_machine::on_va_end (sm_context
*sm_ctxt
,
617 const supernode
*node
,
618 const gcall
*call
) const
620 const svalue
*arg
= get_stateful_arg (sm_ctxt
, call
, 0);
623 state_t s
= sm_ctxt
->get_state (call
, arg
);
624 /* Transition from "started" to "ended". */
626 sm_ctxt
->set_next_state (call
, arg
, m_ended
);
627 else if (s
== m_ended
)
628 check_for_ended_va_list (sm_ctxt
, node
, call
, arg
, "va_end");
632 /* Implementation of state_machine::on_leak vfunc for va_list_state_machine
633 (for complaining about leaks of values in state 'started'). */
635 std::unique_ptr
<pending_diagnostic
>
636 va_list_state_machine::on_leak (tree var
) const
638 return make_unique
<va_list_leak
> (*this, NULL
, var
);
641 } // anonymous namespace
643 /* Internal interface to this file. */
646 make_va_list_state_machine (logger
*logger
)
648 return new va_list_state_machine (logger
);
651 /* Handler for "__builtin_va_start". */
653 class kf_va_start
: public known_function
656 bool matches_call_types_p (const call_details
&) const final override
660 void impl_call_pre (const call_details
&cd
) const final override
;
664 kf_va_start::impl_call_pre (const call_details
&cd
) const
666 region_model
*model
= cd
.get_model ();
667 region_model_manager
*mgr
= cd
.get_manager ();
668 const svalue
*out_ptr
= cd
.get_arg_svalue (0);
669 const region
*out_reg
670 = model
->deref_rvalue (out_ptr
, cd
.get_arg_tree (0), cd
.get_ctxt ());
671 const frame_region
*frame
= model
->get_current_frame ();
673 /* "*out_ptr = &IMPL_REGION;". */
674 const region
*impl_reg
= mgr
->create_region_for_alloca (frame
);
676 /* We abuse the types here, since va_list_type isn't
677 necessarily anything to do with a pointer. */
678 const svalue
*ptr_to_impl_reg
= mgr
->get_ptr_svalue (NULL_TREE
, impl_reg
);
679 model
->set_value (out_reg
, ptr_to_impl_reg
, cd
.get_ctxt ());
681 if (model
->get_stack_depth () > 1)
683 /* The interprocedural case: the frame containing the va_start call
684 will have been populated with any variadic aruguments.
685 Initialize IMPL_REGION with a ptr to var_arg_region 0. */
686 const region
*init_var_arg_reg
= mgr
->get_var_arg_region (frame
, 0);
687 const svalue
*ap_sval
688 = mgr
->get_ptr_svalue (NULL_TREE
, init_var_arg_reg
);
689 model
->set_value (impl_reg
, ap_sval
, cd
.get_ctxt ());
693 /* The frame containing va_start is an entry-point to the analysis,
694 so there won't be any specific var_arg_regions populated within it.
695 Initialize IMPL_REGION as the UNKNOWN_SVALUE to avoid state
696 explosions on repeated calls to va_arg. */
697 const svalue
*unknown_sval
698 = mgr
->get_or_create_unknown_svalue (NULL_TREE
);
699 model
->set_value (impl_reg
, unknown_sval
, cd
.get_ctxt ());
703 /* Handler for "__builtin_va_copy". */
705 class kf_va_copy
: public known_function
708 bool matches_call_types_p (const call_details
&) const final override
712 void impl_call_pre (const call_details
&cd
) const final override
;
716 kf_va_copy::impl_call_pre (const call_details
&cd
) const
718 region_model
*model
= cd
.get_model ();
719 region_model_manager
*mgr
= cd
.get_manager ();
720 const svalue
*out_dst_ptr
= cd
.get_arg_svalue (0);
721 const svalue
*in_va_list
722 = get_va_copy_arg (model
, cd
.get_ctxt (), cd
.get_call_stmt (), 1);
724 = model
->check_for_poison (in_va_list
,
725 get_va_list_diag_arg (cd
.get_arg_tree (1)),
729 const region
*out_dst_reg
730 = model
->deref_rvalue (out_dst_ptr
, cd
.get_arg_tree (0), cd
.get_ctxt ());
732 /* "*out_dst_ptr = &NEW_IMPL_REGION;". */
733 const region
*new_impl_reg
734 = mgr
->create_region_for_alloca (model
->get_current_frame ());
735 const svalue
*ptr_to_new_impl_reg
736 = mgr
->get_ptr_svalue (NULL_TREE
, new_impl_reg
);
737 model
->set_value (out_dst_reg
, ptr_to_new_impl_reg
, cd
.get_ctxt ());
739 if (const region
*old_impl_reg
= in_va_list
->maybe_get_region ())
741 /* "(NEW_IMPL_REGION) = (OLD_IMPL_REGION);". */
742 const svalue
*existing_sval
743 = model
->get_store_value (old_impl_reg
, cd
.get_ctxt ());
744 model
->set_value (new_impl_reg
, existing_sval
, cd
.get_ctxt ());
748 /* Get the number of variadic arguments to CALLEE_FNDECL at CALL_STMT. */
751 get_num_variadic_arguments (tree callee_fndecl
,
752 const gcall
*call_stmt
)
754 int num_positional
= 0;
755 for (tree iter_parm
= DECL_ARGUMENTS (callee_fndecl
); iter_parm
;
756 iter_parm
= DECL_CHAIN (iter_parm
))
758 return gimple_call_num_args (call_stmt
) - num_positional
;
761 /* An abstract subclass of pending_diagnostic for diagnostics relating
762 to bad va_arg invocations.
764 This shows the number of variadic arguments at the call of interest.
765 Ideally we'd also be able to highlight individual arguments, but
766 that location information isn't generally available from the middle end. */
768 class va_arg_diagnostic
: public pending_diagnostic
771 /* Override of pending_diagnostic::add_call_event,
772 adding a custom call_event subclass. */
773 void add_call_event (const exploded_edge
&eedge
,
774 checker_path
*emission_path
) override
776 /* As per call_event, but show the number of variadic arguments
778 class va_arg_call_event
: public call_event
781 va_arg_call_event (const exploded_edge
&eedge
,
782 const event_loc_info
&loc_info
,
783 int num_variadic_arguments
)
784 : call_event (eedge
, loc_info
),
785 m_num_variadic_arguments (num_variadic_arguments
)
789 label_text
get_desc (bool can_colorize
) const override
791 return make_label_text_n
792 (can_colorize
, m_num_variadic_arguments
,
793 "calling %qE from %qE with %i variadic argument",
794 "calling %qE from %qE with %i variadic arguments",
795 get_callee_fndecl (),
796 get_caller_fndecl (),
797 m_num_variadic_arguments
);
800 int m_num_variadic_arguments
;
803 const frame_region
*frame_reg
= m_var_arg_reg
->get_frame_region ();
804 const exploded_node
*dst_node
= eedge
.m_dest
;
805 if (dst_node
->get_state ().m_region_model
->get_current_frame ()
808 const exploded_node
*src_node
= eedge
.m_src
;
809 const program_point
&src_point
= src_node
->get_point ();
810 const int src_stack_depth
= src_point
.get_stack_depth ();
811 const gimple
*last_stmt
= src_point
.get_supernode ()->get_last_stmt ();
812 const gcall
*call_stmt
= as_a
<const gcall
*> (last_stmt
);
813 int num_variadic_arguments
814 = get_num_variadic_arguments (dst_node
->get_function ()->decl
,
816 emission_path
->add_event
817 (make_unique
<va_arg_call_event
>
819 event_loc_info (last_stmt
? last_stmt
->location
: UNKNOWN_LOCATION
,
820 src_point
.get_fndecl (),
822 num_variadic_arguments
));
825 pending_diagnostic::add_call_event (eedge
, emission_path
);
829 va_arg_diagnostic (tree va_list_tree
, const var_arg_region
*var_arg_reg
)
830 : m_va_list_tree (va_list_tree
), m_var_arg_reg (var_arg_reg
)
833 bool subclass_equal_p (const pending_diagnostic
&base_other
) const override
835 const va_arg_diagnostic
&other
= (const va_arg_diagnostic
&)base_other
;
836 return (same_tree_p (m_va_list_tree
, other
.m_va_list_tree
)
837 && m_var_arg_reg
== other
.m_var_arg_reg
);
840 /* Get the number of arguments consumed so far from the va_list
841 (*before* this va_arg call). */
842 unsigned get_num_consumed () const
844 return m_var_arg_reg
->get_index ();
847 /* Get a 1-based index of which variadic argument is being consumed. */
848 unsigned get_variadic_index_for_diagnostic () const
850 return get_num_consumed () + 1;
853 /* User-readable expr for the va_list argument to va_arg. */
856 /* The region that the va_arg attempted to access. */
857 const var_arg_region
*m_var_arg_reg
;
860 /* A subclass of pending_diagnostic for complaining about a type mismatch
861 between the result of:
863 and the type of the argument that was passed to the variadic call. */
865 class va_arg_type_mismatch
: public va_arg_diagnostic
868 va_arg_type_mismatch (tree va_list_tree
, const var_arg_region
*var_arg_reg
,
869 tree expected_type
, tree actual_type
)
870 : va_arg_diagnostic (va_list_tree
, var_arg_reg
),
871 m_expected_type (expected_type
), m_actual_type (actual_type
)
874 const char *get_kind () const final override
876 return "va_arg_type_mismatch";
879 bool subclass_equal_p (const pending_diagnostic
&base_other
)
882 if (!va_arg_diagnostic::subclass_equal_p (base_other
))
884 const va_arg_type_mismatch
&other
885 = (const va_arg_type_mismatch
&)base_other
;
886 return (same_tree_p (m_expected_type
, other
.m_expected_type
)
887 && same_tree_p (m_actual_type
, other
.m_actual_type
));
890 int get_controlling_option () const final override
892 return OPT_Wanalyzer_va_arg_type_mismatch
;
895 bool emit (rich_location
*rich_loc
, logger
*) final override
897 auto_diagnostic_group d
;
898 diagnostic_metadata m
;
899 /* "CWE-686: Function Call With Incorrect Argument Type". */
902 = warning_meta (rich_loc
, m
, get_controlling_option (),
903 "%<va_arg%> expected %qT but received %qT"
904 " for variadic argument %i of %qE",
905 m_expected_type
, m_actual_type
,
906 get_variadic_index_for_diagnostic (), m_va_list_tree
);
910 label_text
describe_final_event (const evdesc::final_event
&ev
) final override
912 return ev
.formatted_print ("%<va_arg%> expected %qT but received %qT"
913 " for variadic argument %i of %qE",
914 m_expected_type
, m_actual_type
,
915 get_variadic_index_for_diagnostic (),
920 tree m_expected_type
;
924 /* A subclass of pending_diagnostic for complaining about a
926 after all of the args in AP have been consumed. */
928 class va_list_exhausted
: public va_arg_diagnostic
931 va_list_exhausted (tree va_list_tree
, const var_arg_region
*var_arg_reg
)
932 : va_arg_diagnostic (va_list_tree
, var_arg_reg
)
935 const char *get_kind () const final override
937 return "va_list_exhausted";
940 int get_controlling_option () const final override
942 return OPT_Wanalyzer_va_list_exhausted
;
945 bool emit (rich_location
*rich_loc
, logger
*) final override
947 auto_diagnostic_group d
;
948 diagnostic_metadata m
;
949 /* CWE-685: Function Call With Incorrect Number of Arguments. */
951 bool warned
= warning_meta (rich_loc
, m
, get_controlling_option (),
952 "%qE has no more arguments (%i consumed)",
953 m_va_list_tree
, get_num_consumed ());
957 label_text
describe_final_event (const evdesc::final_event
&ev
) final override
959 return ev
.formatted_print ("%qE has no more arguments (%i consumed)",
960 m_va_list_tree
, get_num_consumed ());
964 /* Return true if it's OK to copy a value from ARG_TYPE to LHS_TYPE via
965 va_arg (where argument promotion has already happened). */
968 va_arg_compatible_types_p (tree lhs_type
, tree arg_type
)
970 return compat_types_p (arg_type
, lhs_type
);
973 /* If AP_SVAL is a pointer to a var_arg_region, return that var_arg_region.
974 Otherwise return NULL. */
976 static const var_arg_region
*
977 maybe_get_var_arg_region (const svalue
*ap_sval
)
979 if (const region
*reg
= ap_sval
->maybe_get_region ())
980 return reg
->dyn_cast_var_arg_region ();
984 /* Handler for "__builtin_va_arg". */
986 class kf_va_arg
: public internal_known_function
989 void impl_call_pre (const call_details
&cd
) const final override
;
993 kf_va_arg::impl_call_pre (const call_details
&cd
) const
995 region_model_context
*ctxt
= cd
.get_ctxt ();
996 region_model
*model
= cd
.get_model ();
997 region_model_manager
*mgr
= cd
.get_manager ();
999 const svalue
*in_ptr
= cd
.get_arg_svalue (0);
1000 const region
*ap_reg
1001 = model
->deref_rvalue (in_ptr
, cd
.get_arg_tree (0), ctxt
);
1003 const svalue
*ap_sval
= model
->get_store_value (ap_reg
, ctxt
);
1004 if (const svalue
*cast
= ap_sval
->maybe_undo_cast ())
1007 tree va_list_tree
= get_va_list_diag_arg (cd
.get_arg_tree (0));
1008 ap_sval
= model
->check_for_poison (ap_sval
, va_list_tree
, ap_reg
, ctxt
);
1010 if (const region
*impl_reg
= ap_sval
->maybe_get_region ())
1012 const svalue
*old_impl_sval
= model
->get_store_value (impl_reg
, ctxt
);
1013 if (const var_arg_region
*arg_reg
1014 = maybe_get_var_arg_region (old_impl_sval
))
1016 bool saw_problem
= false;
1018 const frame_region
*frame_reg
= arg_reg
->get_frame_region ();
1019 unsigned next_arg_idx
= arg_reg
->get_index ();
1021 if (frame_reg
->get_stack_depth () > 1)
1023 /* The interprocedural case: the called frame will have been
1024 populated with any variadic aruguments.
1025 Attempt to extract arg_reg to cd's return region (which already
1026 has a conjured_svalue), or warn if there's a problem
1027 (incompatible types, or if we've run out of args). */
1028 if (const svalue
*arg_sval
1029 = model
->get_store ()->get_any_binding
1030 (mgr
->get_store_manager (), arg_reg
))
1032 tree lhs_type
= cd
.get_lhs_type ();
1033 tree arg_type
= arg_sval
->get_type ();
1034 if (va_arg_compatible_types_p (lhs_type
, arg_type
))
1035 cd
.maybe_set_lhs (arg_sval
);
1039 ctxt
->warn (make_unique
<va_arg_type_mismatch
>
1050 ctxt
->warn (make_unique
<va_list_exhausted
> (va_list_tree
,
1057 /* This frame is an entry-point to the analysis, so there won't be
1058 any specific var_arg_regions populated within it.
1059 We already have a conjured_svalue for the result, so leave
1061 gcc_assert (frame_reg
->get_stack_depth () == 1);
1066 /* Set impl_reg to UNKNOWN to suppress further warnings. */
1067 const svalue
*new_ap_sval
1068 = mgr
->get_or_create_unknown_svalue (impl_reg
->get_type ());
1069 model
->set_value (impl_reg
, new_ap_sval
, ctxt
);
1073 /* Update impl_reg to advance to the next arg. */
1074 const region
*next_var_arg_region
1075 = mgr
->get_var_arg_region (frame_reg
, next_arg_idx
+ 1);
1076 const svalue
*new_ap_sval
1077 = mgr
->get_ptr_svalue (NULL_TREE
, next_var_arg_region
);
1078 model
->set_value (impl_reg
, new_ap_sval
, ctxt
);
1084 /* Handler for "__builtin_va_end". */
1086 class kf_va_end
: public known_function
1089 bool matches_call_types_p (const call_details
&) const
1095 /* Populate KFM with instances of known functions relating to varargs. */
1098 register_varargs_builtins (known_function_manager
&kfm
)
1100 kfm
.add (BUILT_IN_VA_START
, make_unique
<kf_va_start
> ());
1101 kfm
.add (BUILT_IN_VA_COPY
, make_unique
<kf_va_copy
> ());
1102 kfm
.add (IFN_VA_ARG
, make_unique
<kf_va_arg
> ());
1103 kfm
.add (BUILT_IN_VA_END
, make_unique
<kf_va_end
> ());
1108 #endif /* #if ENABLE_ANALYZER */