blob: 67b9277a62631211d36aa50c21ab25407b41f4ca [file] [log] [blame]
sewardj45f4e7c2005-09-27 19:20:21 +00001
2/*--------------------------------------------------------------------*/
3/*--- Command line handling. m_commandline.c ---*/
4/*--------------------------------------------------------------------*/
5
6/*
7 This file is part of Valgrind, a dynamic binary instrumentation
8 framework.
9
sewardjb3a1e4b2015-08-21 11:32:26 +000010 Copyright (C) 2000-2015 Julian Seward
sewardj45f4e7c2005-09-27 19:20:21 +000011 jseward@acm.org
12
13 This program is free software; you can redistribute it and/or
14 modify it under the terms of the GNU General Public License as
15 published by the Free Software Foundation; either version 2 of the
16 License, or (at your option) any later version.
17
18 This program is distributed in the hope that it will be useful, but
19 WITHOUT ANY WARRANTY; without even the implied warranty of
20 MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
21 General Public License for more details.
22
23 You should have received a copy of the GNU General Public License
24 along with this program; if not, write to the Free Software
25 Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA
26 02111-1307, USA.
27
28 The GNU General Public License is contained in the file COPYING.
29*/
30
31#include "pub_core_basics.h"
sewardj4cfea4f2006-10-14 19:26:10 +000032#include "pub_core_vki.h"
sewardj45f4e7c2005-09-27 19:20:21 +000033#include "pub_core_libcassert.h"
34#include "pub_core_libcbase.h"
35#include "pub_core_libcfile.h"
36#include "pub_core_libcprint.h"
37#include "pub_core_libcproc.h"
38#include "pub_core_mallocfree.h"
sewardj14c7cc52007-02-25 15:08:24 +000039#include "pub_core_xarray.h"
sewardj45f4e7c2005-09-27 19:20:21 +000040#include "pub_core_clientstate.h"
sewardj14c7cc52007-02-25 15:08:24 +000041#include "pub_core_commandline.h" /* self */
sewardj45f4e7c2005-09-27 19:20:21 +000042
43
44/* Add a string to an expandable array of strings. */
45
sewardj14c7cc52007-02-25 15:08:24 +000046static void add_string ( XArray* /* of HChar* */xa, HChar* str )
sewardj45f4e7c2005-09-27 19:20:21 +000047{
sewardj0f631482007-02-27 16:40:53 +000048 (void) VG_(addToXA)( xa, (void*)(&str) );
sewardj45f4e7c2005-09-27 19:20:21 +000049}
50
51
52/* Read the contents of .valgrindrc in 'dir' into malloc'd memory. */
53// Note that we deliberately don't free the malloc'd memory. See
54// comment at call site.
55
florian6bd9dc12012-11-23 16:17:43 +000056static HChar* read_dot_valgrindrc ( const HChar* dir )
sewardj45f4e7c2005-09-27 19:20:21 +000057{
58 Int n;
59 SysRes fd;
dirka656f3d2008-11-22 12:03:19 +000060 struct vg_stat stat_buf;
sewardj45f4e7c2005-09-27 19:20:21 +000061 HChar* f_clo = NULL;
florian7d3ae572014-09-27 18:01:19 +000062 const HChar dot_valgrindrc[] = ".valgrindrc";
sewardj45f4e7c2005-09-27 19:20:21 +000063
florian7d3ae572014-09-27 18:01:19 +000064 vg_assert(dir != NULL);
65
66 HChar filename[VG_(strlen)(dir) + 1 + VG_(strlen)(dot_valgrindrc) + 1];
67 VG_(sprintf)(filename, "%s/%s", dir, dot_valgrindrc);
68
sewardj45f4e7c2005-09-27 19:20:21 +000069 fd = VG_(open)(filename, 0, VKI_S_IRUSR);
njncda2f0f2009-05-18 02:12:08 +000070 if ( !sr_isError(fd) ) {
71 Int res = VG_(fstat)( sr_Res(fd), &stat_buf );
dirka656f3d2008-11-22 12:03:19 +000072 // Ignore if not owned by current user or world writeable (CVE-2008-4865)
njn9c20ece2009-05-20 02:02:30 +000073 if (!res && stat_buf.uid == VG_(geteuid)()
74 && (!(stat_buf.mode & VKI_S_IWOTH))) {
75 if ( stat_buf.size > 0 ) {
76 f_clo = VG_(malloc)("commandline.rdv.1", stat_buf.size+1);
njn9c20ece2009-05-20 02:02:30 +000077 n = VG_(read)(sr_Res(fd), f_clo, stat_buf.size);
dirka656f3d2008-11-22 12:03:19 +000078 if (n == -1) n = 0;
njn9c20ece2009-05-20 02:02:30 +000079 vg_assert(n >= 0 && n <= stat_buf.size+1);
dirka656f3d2008-11-22 12:03:19 +000080 f_clo[n] = '\0';
81 }
sewardj45f4e7c2005-09-27 19:20:21 +000082 }
dirka656f3d2008-11-22 12:03:19 +000083 else
84 VG_(message)(Vg_UserMsg,
florian885d4792014-06-24 13:08:44 +000085 "%s was not read as it is either world writeable or not "
86 "owned by the current user\n", filename);
dirka656f3d2008-11-22 12:03:19 +000087
njncda2f0f2009-05-18 02:12:08 +000088 VG_(close)(sr_Res(fd));
sewardj45f4e7c2005-09-27 19:20:21 +000089 }
90 return f_clo;
91}
92
93
94// Add args from a string into VG_(args_for_valgrind), splitting the
95// string at whitespace and adding each component as a separate arg.
96
97static void add_args_from_string ( HChar* s )
98{
99 HChar* tmp;
100 HChar* cp = s;
101 vg_assert(cp);
102 while (True) {
103 // We have alternating sequences: blanks, non-blanks, blanks...
104 // copy the non-blanks sequences, and add terminating '\0'
105 while (VG_(isspace)(*cp)) cp++;
106 if (*cp == 0) break;
107 tmp = cp;
108 while ( !VG_(isspace)(*cp) && *cp != 0 ) cp++;
109 if ( *cp != 0 ) *cp++ = '\0'; // terminate if not the last
sewardj14c7cc52007-02-25 15:08:24 +0000110 add_string( VG_(args_for_valgrind), tmp );
sewardj45f4e7c2005-09-27 19:20:21 +0000111 }
112}
113
114
115/* Split up the args presented by the launcher to m_main.main(), and
116 park them in VG_(args_for_client) and VG_(args_for_valgrind).
117
118 The resulting arg list is the concatenation of the following:
119 - contents of ~/.valgrindrc
120 - contents of $VALGRIND_OPTS
121 - contents of ./.valgrindrc
122 - args from the command line
123 in the stated order.
124
125 VG_(args_for_valgrind_noexecpass) is set to be the number of items
126 in the first three categories. They are not passed to child invokations
127 at exec, whereas the last group is.
128
129 If the last group contains --command-line-only=yes, then the
130 first three groups are left empty.
131
132 Scheme: first examine the last group (the supplied argc/argv).
133 It should look like this.
134
135 args-for-v exe_name args-for-c
136
137 args-for-v are taken until either they don't start with '-' or
138 a "--" is seen.
139
140 The exe name and args-for-c are recorded without further ado.
141 Note that args-for-c[0] is the first real arg for the client, not
142 its executable name.
143
144 args-for-v are then copied into tmp_xarray.
145
146 if args-for-v does not include --command-line-only=yes:
147 contents of ~/.valgrindrc, $VALGRIND_OPTS and ./.valgrindrc
148 are copied into VG_(args_for_valgrind).
149 else
150 VG_(args_for_valgrind) is made empty.
151
152 Finally, tmp_xarray is copied onto the end of VG_(args_for_valgrind).
153*/
154
155void VG_(split_up_argv)( Int argc, HChar** argv )
156{
157 Int i;
158 Bool augment = True;
159 static Bool already_called = False;
160
sewardj14c7cc52007-02-25 15:08:24 +0000161 XArray* /* of HChar* */ tmp_xarray;
sewardj45f4e7c2005-09-27 19:20:21 +0000162
163 /* This function should be called once, at startup, and then never
164 again. */
165 vg_assert(!already_called);
166 already_called = True;
167
sewardj9c606bd2008-09-18 18:12:50 +0000168 tmp_xarray = VG_(newXA)( VG_(malloc), "commandline.sua.1",
169 VG_(free), sizeof(HChar*) );
sewardj14c7cc52007-02-25 15:08:24 +0000170
171 vg_assert( ! VG_(args_for_valgrind) );
172 VG_(args_for_valgrind)
sewardj9c606bd2008-09-18 18:12:50 +0000173 = VG_(newXA)( VG_(malloc), "commandline.sua.2",
174 VG_(free), sizeof(HChar*) );
sewardj14c7cc52007-02-25 15:08:24 +0000175
176 vg_assert( ! VG_(args_for_client) );
177 VG_(args_for_client)
sewardj9c606bd2008-09-18 18:12:50 +0000178 = VG_(newXA)( VG_(malloc), "commandline.sua.3",
179 VG_(free), sizeof(HChar*) );
sewardj14c7cc52007-02-25 15:08:24 +0000180
sewardj45f4e7c2005-09-27 19:20:21 +0000181 /* Collect up the args-for-V. */
182 i = 1; /* skip the exe (stage2) name. */
183 for (; i < argc; i++) {
184 vg_assert(argv[i]);
185 if (0 == VG_(strcmp)(argv[i], "--")) {
186 i++;
187 break;
188 }
189 if (0 == VG_(strcmp)(argv[i], "--command-line-only=yes"))
190 augment = False;
191 if (argv[i][0] != '-')
192 break;
sewardj14c7cc52007-02-25 15:08:24 +0000193 add_string( tmp_xarray, argv[i] );
sewardj45f4e7c2005-09-27 19:20:21 +0000194 }
195
196 /* Should now be looking at the exe name. */
197 if (i < argc) {
198 vg_assert(argv[i]);
199 VG_(args_the_exename) = argv[i];
200 i++;
201 }
202
203 /* The rest are args for the client. */
204 for (; i < argc; i++) {
205 vg_assert(argv[i]);
sewardj14c7cc52007-02-25 15:08:24 +0000206 add_string( VG_(args_for_client), argv[i] );
sewardj45f4e7c2005-09-27 19:20:21 +0000207 }
208
sewardj45f4e7c2005-09-27 19:20:21 +0000209 /* Get extra args from ~/.valgrindrc, $VALGRIND_OPTS and
210 ./.valgrindrc into VG_(args_for_valgrind). */
211 if (augment) {
212 // read_dot_valgrindrc() allocates the return value with
213 // VG_(malloc)(). We do not free f1_clo and f2_clo as they get
214 // put into VG_(args_for_valgrind) and so must persist.
sewardj02ab2772006-12-06 17:38:32 +0000215 HChar* home = VG_(getenv)("HOME");
216 HChar* f1_clo = home ? read_dot_valgrindrc( home ) : NULL;
sewardj9c606bd2008-09-18 18:12:50 +0000217 HChar* env_clo = VG_(strdup)( "commandline.sua.4",
218 VG_(getenv)(VALGRIND_OPTS) );
njn64bd24a2007-03-26 22:05:30 +0000219 HChar* f2_clo = NULL;
220
221 // Don't read ./.valgrindrc if "." is the same as "$HOME", else its
222 // contents will be applied twice. (bug #142488)
223 if (home) {
florian29d82f62014-09-27 17:42:07 +0000224 const HChar *cwd = VG_(get_startup_wd)();
225 f2_clo = ( VG_STREQ(home, cwd)
njn64bd24a2007-03-26 22:05:30 +0000226 ? NULL : read_dot_valgrindrc(".") );
227 }
sewardj45f4e7c2005-09-27 19:20:21 +0000228
229 if (f1_clo) add_args_from_string( f1_clo );
230 if (env_clo) add_args_from_string( env_clo );
231 if (f2_clo) add_args_from_string( f2_clo );
232 }
233
234 /* .. and record how many extras we got. */
sewardj14c7cc52007-02-25 15:08:24 +0000235 VG_(args_for_valgrind_noexecpass)
236 = VG_(sizeXA)( VG_(args_for_valgrind) );
sewardj45f4e7c2005-09-27 19:20:21 +0000237
238 /* Finally, copy tmp_xarray onto the end. */
sewardj14c7cc52007-02-25 15:08:24 +0000239 for (i = 0; i < VG_(sizeXA)( tmp_xarray ); i++)
240 add_string( VG_(args_for_valgrind),
241 * (HChar**)VG_(indexXA)( tmp_xarray, i ) );
sewardj45f4e7c2005-09-27 19:20:21 +0000242
sewardj14c7cc52007-02-25 15:08:24 +0000243 VG_(deleteXA)( tmp_xarray );
sewardj45f4e7c2005-09-27 19:20:21 +0000244}
245
246/*--------------------------------------------------------------------*/
247/*--- end ---*/
248/*--------------------------------------------------------------------*/