c++: only cache constexpr calls that are constant exprs
[official-gcc.git] / gcc / analyzer / varargs.cc
blob72e1b31601cd1a05cbd56665ec35a10a17dfecac
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)
10 any later version.
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/>. */
21 #include "config.h"
22 #define INCLUDE_MEMORY
23 #include "system.h"
24 #include "coretypes.h"
25 #include "make-unique.h"
26 #include "tree.h"
27 #include "function.h"
28 #include "basic-block.h"
29 #include "gimple.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"
47 #if ENABLE_ANALYZER
49 namespace ana {
51 /* Implementation of <stdarg.h> within analyzer.
53 Objectives:
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
60 va_start has returned
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
76 a va_list.
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.
83 We have:
85 stack frame:
86 va_list: &impl_reg
87 'impl_reg': pointer to next var_arg_region
88 var_arg_region for arg 0
89 ...
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, ...);
96 here:
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
108 arguments.
110 At the call to
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)
122 va_list:
123 0x4c83700: &ALLOCA_REGION: started
125 At each call to
126 va_arg (ap, TYPE);
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
139 below). */
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. */
145 static tree
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);
150 return va_list_tree;
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,
170 const gcall *call,
171 unsigned arg_idx)
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 ())
176 arg_sval = 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;
181 return *ARG_SVAL. */
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 ())
185 src_reg_sval = cast;
186 return src_reg_sval;
188 else
190 /* va_list_arg_type_node is a va_list; return ARG_SVAL. */
191 return arg_sval;
195 namespace {
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
205 public:
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. */
221 state_t m_started;
223 /* State for a va_list that has had va_end called on it. */
224 state_t m_ended;
226 private:
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,
237 const gcall *call,
238 const svalue *arg,
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. */
254 bool
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);
265 return false;
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))
273 default:
274 break;
276 case BUILT_IN_VA_START:
277 on_va_start (sm_ctxt, node, call);
278 break;
280 case BUILT_IN_VA_COPY:
281 on_va_copy (sm_ctxt, node, call);
282 break;
284 case BUILT_IN_VA_END:
285 on_va_end (sm_ctxt, node, call);
286 break;
289 return false;
292 /* Get the svalue for which va_list_state_machine holds state on argument ARG_
293 IDX to CALL. */
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);
299 if (ap
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 ())
309 impl_sval = cast;
310 return impl_sval;
313 return NULL;
316 /* Abstract class for diagnostics relating to va_list_state_machine. */
318 class va_list_sm_diagnostic : public pending_diagnostic
320 public:
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)
330 override
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)
339 const final override
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 ();
350 protected:
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:
366 return "va_start";
367 case BUILT_IN_VA_COPY:
368 return "va_copy";
369 case BUILT_IN_VA_END:
370 return "va_end";
373 return NULL;
376 const va_list_state_machine &m_sm;
377 const svalue *m_ap_sval;
378 tree m_ap_tree;
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
386 public:
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)
419 final override
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
428 if (ev.m_expr)
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);
434 else
435 return ev.formatted_print
436 ("%qs on %qE after %qs",
437 m_usage_fnname, ev.m_expr, "va_end");
439 else
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);
445 else
446 return ev.formatted_print
447 ("%qs after %qs",
448 m_usage_fnname, "va_end");
452 private:
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
463 public:
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)
491 final override
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
503 if (ev.m_expr)
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);
509 else
510 return ev.formatted_print
511 ("missing call to %qs on %qE",
512 "va_end", ev.m_expr);
514 else
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);
520 else
521 return ev.formatted_print
522 ("missing call to %qs",
523 "va_end");
527 private:
528 diagnostic_event_id_t m_start_event;
529 const char *m_start_event_fnname;
532 /* Update state machine for a "va_start" call. */
534 void
535 va_list_state_machine::on_va_start (sm_context *sm_ctxt,
536 const supernode *,
537 const gcall *call) const
539 const svalue *arg = get_stateful_arg (sm_ctxt, call, 0);
540 if (arg)
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. */
550 void
551 va_list_state_machine::check_for_ended_va_list (sm_context *sm_ctxt,
552 const supernode *node,
553 const gcall *call,
554 const svalue *arg,
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,
569 const gcall *call,
570 unsigned arg_idx)
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);
576 return arg;
578 return NULL;
581 /* Update state machine for a "va_copy" call. */
583 void
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);
589 if (src_arg)
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);
593 if (dst_arg)
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. */
603 void
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);
609 if (arg)
610 check_for_ended_va_list (sm_ctxt, node, call, arg, "va_arg");
613 /* Update state machine for a "va_end" call. */
615 void
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);
621 if (arg)
623 state_t s = sm_ctxt->get_state (call, arg);
624 /* Transition from "started" to "ended". */
625 if (s == m_started)
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. */
645 state_machine *
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
655 public:
656 bool matches_call_types_p (const call_details &) const final override
658 return true;
660 void impl_call_pre (const call_details &cd) const final override;
663 void
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 ());
691 else
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
707 public:
708 bool matches_call_types_p (const call_details &) const final override
710 return true;
712 void impl_call_pre (const call_details &cd) const final override;
715 void
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);
723 in_va_list
724 = model->check_for_poison (in_va_list,
725 get_va_list_diag_arg (cd.get_arg_tree (1)),
726 NULL,
727 cd.get_ctxt ());
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. */
750 static int
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))
757 num_positional++;
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
770 public:
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
777 in the call. */
778 class va_arg_call_event : public call_event
780 public:
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);
799 private:
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 ()
806 == frame_reg)
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,
815 call_stmt);
816 emission_path->add_event
817 (make_unique<va_arg_call_event>
818 (eedge,
819 event_loc_info (last_stmt ? last_stmt->location : UNKNOWN_LOCATION,
820 src_point.get_fndecl (),
821 src_stack_depth),
822 num_variadic_arguments));
824 else
825 pending_diagnostic::add_call_event (eedge, emission_path);
828 protected:
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. */
854 tree m_va_list_tree;
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:
862 va_arg (AP);
863 and the type of the argument that was passed to the variadic call. */
865 class va_arg_type_mismatch : public va_arg_diagnostic
867 public:
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)
880 const final override
882 if (!va_arg_diagnostic::subclass_equal_p (base_other))
883 return false;
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". */
900 m.add_cwe (686);
901 bool warned
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);
907 return warned;
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 (),
916 m_va_list_tree);
919 private:
920 tree m_expected_type;
921 tree m_actual_type;
924 /* A subclass of pending_diagnostic for complaining about a
925 va_arg (AP);
926 after all of the args in AP have been consumed. */
928 class va_list_exhausted : public va_arg_diagnostic
930 public:
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. */
950 m.add_cwe (685);
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 ());
954 return warned;
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). */
967 static bool
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 ();
981 return NULL;
984 /* Handler for "__builtin_va_arg". */
986 class kf_va_arg : public internal_known_function
988 public:
989 void impl_call_pre (const call_details &cd) const final override;
992 void
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 ())
1005 ap_sval = 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);
1036 else
1038 if (ctxt)
1039 ctxt->warn (make_unique <va_arg_type_mismatch>
1040 (va_list_tree,
1041 arg_reg,
1042 lhs_type,
1043 arg_type));
1044 saw_problem = true;
1047 else
1049 if (ctxt)
1050 ctxt->warn (make_unique <va_list_exhausted> (va_list_tree,
1051 arg_reg));
1052 saw_problem = true;
1055 else
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
1060 it untouched. */
1061 gcc_assert (frame_reg->get_stack_depth () == 1);
1064 if (saw_problem)
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);
1071 else
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
1088 public:
1089 bool matches_call_types_p (const call_details &) const
1091 return true;
1095 /* Populate KFM with instances of known functions relating to varargs. */
1097 void
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> ());
1106 } // namespace ana
1108 #endif /* #if ENABLE_ANALYZER */