ager@chromium.org | 9258b6b | 2008-09-11 09:11:10 +0000 | [diff] [blame] | 1 | // Copyright 2006-2008 the V8 project authors. All rights reserved. |
christian.plesner.hansen | 43d26ec | 2008-07-03 15:10:15 +0000 | [diff] [blame] | 2 | // Redistribution and use in source and binary forms, with or without |
| 3 | // modification, are permitted provided that the following conditions are |
| 4 | // met: |
| 5 | // |
| 6 | // * Redistributions of source code must retain the above copyright |
| 7 | // notice, this list of conditions and the following disclaimer. |
| 8 | // * Redistributions in binary form must reproduce the above |
| 9 | // copyright notice, this list of conditions and the following |
| 10 | // disclaimer in the documentation and/or other materials provided |
| 11 | // with the distribution. |
| 12 | // * Neither the name of Google Inc. nor the names of its |
| 13 | // contributors may be used to endorse or promote products derived |
| 14 | // from this software without specific prior written permission. |
| 15 | // |
| 16 | // THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS |
| 17 | // "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT |
| 18 | // LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR |
| 19 | // A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT |
| 20 | // OWNER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, |
| 21 | // SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT |
| 22 | // LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, |
| 23 | // DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY |
| 24 | // THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT |
| 25 | // (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE |
| 26 | // OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE. |
| 27 | |
| 28 | #include <ctype.h> |
| 29 | #include <stdlib.h> |
| 30 | |
| 31 | #include "v8.h" |
| 32 | |
| 33 | #include "platform.h" |
kmillikin@chromium.org | 83e1682 | 2011-09-13 08:21:47 +0000 | [diff] [blame] | 34 | #include "smart-array-pointer.h" |
ager@chromium.org | 3bf7b91 | 2008-11-17 09:09:45 +0000 | [diff] [blame] | 35 | #include "string-stream.h" |
| 36 | |
christian.plesner.hansen | 43d26ec | 2008-07-03 15:10:15 +0000 | [diff] [blame] | 37 | |
kasperl@chromium.org | 71affb5 | 2009-05-26 05:44:31 +0000 | [diff] [blame] | 38 | namespace v8 { |
| 39 | namespace internal { |
christian.plesner.hansen | 43d26ec | 2008-07-03 15:10:15 +0000 | [diff] [blame] | 40 | |
kasperl@chromium.org | b912362 | 2008-09-17 14:05:56 +0000 | [diff] [blame] | 41 | // Define all of our flags. |
| 42 | #define FLAG_MODE_DEFINE |
| 43 | #include "flag-definitions.h" |
christian.plesner.hansen | 43d26ec | 2008-07-03 15:10:15 +0000 | [diff] [blame] | 44 | |
kasperl@chromium.org | b912362 | 2008-09-17 14:05:56 +0000 | [diff] [blame] | 45 | // Define all of our flags default values. |
| 46 | #define FLAG_MODE_DEFINE_DEFAULTS |
| 47 | #include "flag-definitions.h" |
christian.plesner.hansen | 43d26ec | 2008-07-03 15:10:15 +0000 | [diff] [blame] | 48 | |
kasperl@chromium.org | b912362 | 2008-09-17 14:05:56 +0000 | [diff] [blame] | 49 | namespace { |
christian.plesner.hansen | 43d26ec | 2008-07-03 15:10:15 +0000 | [diff] [blame] | 50 | |
kasperl@chromium.org | b912362 | 2008-09-17 14:05:56 +0000 | [diff] [blame] | 51 | // This structure represents a single entry in the flag system, with a pointer |
| 52 | // to the actual flag, default value, comment, etc. This is designed to be POD |
| 53 | // initialized as to avoid requiring static constructors. |
| 54 | struct Flag { |
ager@chromium.org | 3bf7b91 | 2008-11-17 09:09:45 +0000 | [diff] [blame] | 55 | enum FlagType { TYPE_BOOL, TYPE_INT, TYPE_FLOAT, TYPE_STRING, TYPE_ARGS }; |
kasperl@chromium.org | b912362 | 2008-09-17 14:05:56 +0000 | [diff] [blame] | 56 | |
| 57 | FlagType type_; // What type of flag, bool, int, or string. |
| 58 | const char* name_; // Name of the flag, ex "my_flag". |
| 59 | void* valptr_; // Pointer to the global flag variable. |
| 60 | const void* defptr_; // Pointer to the default value. |
| 61 | const char* cmt_; // A comment about the flags purpose. |
ager@chromium.org | 41826e7 | 2009-03-30 13:30:57 +0000 | [diff] [blame] | 62 | bool owns_ptr_; // Does the flag own its string value? |
kasperl@chromium.org | b912362 | 2008-09-17 14:05:56 +0000 | [diff] [blame] | 63 | |
| 64 | FlagType type() const { return type_; } |
| 65 | |
| 66 | const char* name() const { return name_; } |
| 67 | |
| 68 | const char* comment() const { return cmt_; } |
| 69 | |
| 70 | bool* bool_variable() const { |
| 71 | ASSERT(type_ == TYPE_BOOL); |
| 72 | return reinterpret_cast<bool*>(valptr_); |
christian.plesner.hansen | 43d26ec | 2008-07-03 15:10:15 +0000 | [diff] [blame] | 73 | } |
christian.plesner.hansen | 43d26ec | 2008-07-03 15:10:15 +0000 | [diff] [blame] | 74 | |
kasperl@chromium.org | b912362 | 2008-09-17 14:05:56 +0000 | [diff] [blame] | 75 | int* int_variable() const { |
| 76 | ASSERT(type_ == TYPE_INT); |
| 77 | return reinterpret_cast<int*>(valptr_); |
| 78 | } |
christian.plesner.hansen | 43d26ec | 2008-07-03 15:10:15 +0000 | [diff] [blame] | 79 | |
kasperl@chromium.org | b912362 | 2008-09-17 14:05:56 +0000 | [diff] [blame] | 80 | double* float_variable() const { |
| 81 | ASSERT(type_ == TYPE_FLOAT); |
| 82 | return reinterpret_cast<double*>(valptr_); |
| 83 | } |
| 84 | |
ager@chromium.org | 41826e7 | 2009-03-30 13:30:57 +0000 | [diff] [blame] | 85 | const char* string_value() const { |
kasperl@chromium.org | b912362 | 2008-09-17 14:05:56 +0000 | [diff] [blame] | 86 | ASSERT(type_ == TYPE_STRING); |
ager@chromium.org | 41826e7 | 2009-03-30 13:30:57 +0000 | [diff] [blame] | 87 | return *reinterpret_cast<const char**>(valptr_); |
| 88 | } |
| 89 | |
kasperl@chromium.org | b3284ad | 2009-05-18 06:12:45 +0000 | [diff] [blame] | 90 | void set_string_value(const char* value, bool owns_ptr) { |
ager@chromium.org | 41826e7 | 2009-03-30 13:30:57 +0000 | [diff] [blame] | 91 | ASSERT(type_ == TYPE_STRING); |
kasperl@chromium.org | b3284ad | 2009-05-18 06:12:45 +0000 | [diff] [blame] | 92 | const char** ptr = reinterpret_cast<const char**>(valptr_); |
ager@chromium.org | 41826e7 | 2009-03-30 13:30:57 +0000 | [diff] [blame] | 93 | if (owns_ptr_ && *ptr != NULL) DeleteArray(*ptr); |
| 94 | *ptr = value; |
| 95 | owns_ptr_ = owns_ptr; |
kasperl@chromium.org | b912362 | 2008-09-17 14:05:56 +0000 | [diff] [blame] | 96 | } |
| 97 | |
ager@chromium.org | 3bf7b91 | 2008-11-17 09:09:45 +0000 | [diff] [blame] | 98 | JSArguments* args_variable() const { |
| 99 | ASSERT(type_ == TYPE_ARGS); |
| 100 | return reinterpret_cast<JSArguments*>(valptr_); |
| 101 | } |
| 102 | |
kasperl@chromium.org | b912362 | 2008-09-17 14:05:56 +0000 | [diff] [blame] | 103 | bool bool_default() const { |
| 104 | ASSERT(type_ == TYPE_BOOL); |
| 105 | return *reinterpret_cast<const bool*>(defptr_); |
| 106 | } |
| 107 | |
| 108 | int int_default() const { |
| 109 | ASSERT(type_ == TYPE_INT); |
| 110 | return *reinterpret_cast<const int*>(defptr_); |
| 111 | } |
| 112 | |
| 113 | double float_default() const { |
| 114 | ASSERT(type_ == TYPE_FLOAT); |
| 115 | return *reinterpret_cast<const double*>(defptr_); |
| 116 | } |
| 117 | |
| 118 | const char* string_default() const { |
| 119 | ASSERT(type_ == TYPE_STRING); |
| 120 | return *reinterpret_cast<const char* const *>(defptr_); |
| 121 | } |
| 122 | |
ager@chromium.org | 3bf7b91 | 2008-11-17 09:09:45 +0000 | [diff] [blame] | 123 | JSArguments args_default() const { |
| 124 | ASSERT(type_ == TYPE_ARGS); |
| 125 | return *reinterpret_cast<const JSArguments*>(defptr_); |
| 126 | } |
| 127 | |
kasperl@chromium.org | b912362 | 2008-09-17 14:05:56 +0000 | [diff] [blame] | 128 | // Compare this flag's current value against the default. |
| 129 | bool IsDefault() const { |
| 130 | switch (type_) { |
| 131 | case TYPE_BOOL: |
| 132 | return *bool_variable() == bool_default(); |
| 133 | case TYPE_INT: |
| 134 | return *int_variable() == int_default(); |
| 135 | case TYPE_FLOAT: |
| 136 | return *float_variable() == float_default(); |
ager@chromium.org | 3bf7b91 | 2008-11-17 09:09:45 +0000 | [diff] [blame] | 137 | case TYPE_STRING: { |
ager@chromium.org | 41826e7 | 2009-03-30 13:30:57 +0000 | [diff] [blame] | 138 | const char* str1 = string_value(); |
kasperl@chromium.org | b912362 | 2008-09-17 14:05:56 +0000 | [diff] [blame] | 139 | const char* str2 = string_default(); |
| 140 | if (str2 == NULL) return str1 == NULL; |
| 141 | if (str1 == NULL) return str2 == NULL; |
| 142 | return strcmp(str1, str2) == 0; |
ager@chromium.org | 3bf7b91 | 2008-11-17 09:09:45 +0000 | [diff] [blame] | 143 | } |
| 144 | case TYPE_ARGS: |
| 145 | return args_variable()->argc() == 0; |
kasperl@chromium.org | b912362 | 2008-09-17 14:05:56 +0000 | [diff] [blame] | 146 | } |
| 147 | UNREACHABLE(); |
| 148 | return true; |
| 149 | } |
| 150 | |
| 151 | // Set a flag back to it's default value. |
| 152 | void Reset() { |
| 153 | switch (type_) { |
| 154 | case TYPE_BOOL: |
| 155 | *bool_variable() = bool_default(); |
| 156 | break; |
| 157 | case TYPE_INT: |
| 158 | *int_variable() = int_default(); |
| 159 | break; |
| 160 | case TYPE_FLOAT: |
| 161 | *float_variable() = float_default(); |
| 162 | break; |
| 163 | case TYPE_STRING: |
ager@chromium.org | 41826e7 | 2009-03-30 13:30:57 +0000 | [diff] [blame] | 164 | set_string_value(string_default(), false); |
kasperl@chromium.org | b912362 | 2008-09-17 14:05:56 +0000 | [diff] [blame] | 165 | break; |
ager@chromium.org | 3bf7b91 | 2008-11-17 09:09:45 +0000 | [diff] [blame] | 166 | case TYPE_ARGS: |
| 167 | *args_variable() = args_default(); |
| 168 | break; |
christian.plesner.hansen | 43d26ec | 2008-07-03 15:10:15 +0000 | [diff] [blame] | 169 | } |
| 170 | } |
kasperl@chromium.org | b912362 | 2008-09-17 14:05:56 +0000 | [diff] [blame] | 171 | }; |
| 172 | |
| 173 | Flag flags[] = { |
| 174 | #define FLAG_MODE_META |
| 175 | #include "flag-definitions.h" |
| 176 | }; |
| 177 | |
| 178 | const size_t num_flags = sizeof(flags) / sizeof(*flags); |
| 179 | |
| 180 | } // namespace |
christian.plesner.hansen | 43d26ec | 2008-07-03 15:10:15 +0000 | [diff] [blame] | 181 | |
| 182 | |
kasperl@chromium.org | b912362 | 2008-09-17 14:05:56 +0000 | [diff] [blame] | 183 | static const char* Type2String(Flag::FlagType type) { |
christian.plesner.hansen | 43d26ec | 2008-07-03 15:10:15 +0000 | [diff] [blame] | 184 | switch (type) { |
kasperl@chromium.org | b912362 | 2008-09-17 14:05:56 +0000 | [diff] [blame] | 185 | case Flag::TYPE_BOOL: return "bool"; |
| 186 | case Flag::TYPE_INT: return "int"; |
| 187 | case Flag::TYPE_FLOAT: return "float"; |
| 188 | case Flag::TYPE_STRING: return "string"; |
ager@chromium.org | 3bf7b91 | 2008-11-17 09:09:45 +0000 | [diff] [blame] | 189 | case Flag::TYPE_ARGS: return "arguments"; |
christian.plesner.hansen | 43d26ec | 2008-07-03 15:10:15 +0000 | [diff] [blame] | 190 | } |
| 191 | UNREACHABLE(); |
| 192 | return NULL; |
| 193 | } |
| 194 | |
| 195 | |
kmillikin@chromium.org | 83e1682 | 2011-09-13 08:21:47 +0000 | [diff] [blame] | 196 | static SmartArrayPointer<const char> ToString(Flag* flag) { |
ager@chromium.org | 3bf7b91 | 2008-11-17 09:09:45 +0000 | [diff] [blame] | 197 | HeapStringAllocator string_allocator; |
| 198 | StringStream buffer(&string_allocator); |
kasperl@chromium.org | b912362 | 2008-09-17 14:05:56 +0000 | [diff] [blame] | 199 | switch (flag->type()) { |
| 200 | case Flag::TYPE_BOOL: |
ager@chromium.org | 3bf7b91 | 2008-11-17 09:09:45 +0000 | [diff] [blame] | 201 | buffer.Add("%s", (*flag->bool_variable() ? "true" : "false")); |
christian.plesner.hansen | 43d26ec | 2008-07-03 15:10:15 +0000 | [diff] [blame] | 202 | break; |
kasperl@chromium.org | b912362 | 2008-09-17 14:05:56 +0000 | [diff] [blame] | 203 | case Flag::TYPE_INT: |
ager@chromium.org | 3bf7b91 | 2008-11-17 09:09:45 +0000 | [diff] [blame] | 204 | buffer.Add("%d", *flag->int_variable()); |
christian.plesner.hansen | 43d26ec | 2008-07-03 15:10:15 +0000 | [diff] [blame] | 205 | break; |
kasperl@chromium.org | b912362 | 2008-09-17 14:05:56 +0000 | [diff] [blame] | 206 | case Flag::TYPE_FLOAT: |
ager@chromium.org | 3bf7b91 | 2008-11-17 09:09:45 +0000 | [diff] [blame] | 207 | buffer.Add("%f", FmtElm(*flag->float_variable())); |
christian.plesner.hansen | 43d26ec | 2008-07-03 15:10:15 +0000 | [diff] [blame] | 208 | break; |
ager@chromium.org | 3bf7b91 | 2008-11-17 09:09:45 +0000 | [diff] [blame] | 209 | case Flag::TYPE_STRING: { |
ager@chromium.org | 41826e7 | 2009-03-30 13:30:57 +0000 | [diff] [blame] | 210 | const char* str = flag->string_value(); |
ager@chromium.org | 3bf7b91 | 2008-11-17 09:09:45 +0000 | [diff] [blame] | 211 | buffer.Add("%s", str ? str : "NULL"); |
| 212 | break; |
| 213 | } |
| 214 | case Flag::TYPE_ARGS: { |
| 215 | JSArguments args = *flag->args_variable(); |
| 216 | if (args.argc() > 0) { |
| 217 | buffer.Add("%s", args[0]); |
| 218 | for (int i = 1; i < args.argc(); i++) { |
| 219 | buffer.Add(" %s", args[i]); |
| 220 | } |
christian.plesner.hansen | 43d26ec | 2008-07-03 15:10:15 +0000 | [diff] [blame] | 221 | } |
| 222 | break; |
ager@chromium.org | 3bf7b91 | 2008-11-17 09:09:45 +0000 | [diff] [blame] | 223 | } |
christian.plesner.hansen | 43d26ec | 2008-07-03 15:10:15 +0000 | [diff] [blame] | 224 | } |
ager@chromium.org | 3bf7b91 | 2008-11-17 09:09:45 +0000 | [diff] [blame] | 225 | return buffer.ToCString(); |
christian.plesner.hansen | 43d26ec | 2008-07-03 15:10:15 +0000 | [diff] [blame] | 226 | } |
| 227 | |
| 228 | |
kasperl@chromium.org | b912362 | 2008-09-17 14:05:56 +0000 | [diff] [blame] | 229 | // static |
ager@chromium.org | 3bf7b91 | 2008-11-17 09:09:45 +0000 | [diff] [blame] | 230 | List<const char*>* FlagList::argv() { |
| 231 | List<const char*>* args = new List<const char*>(8); |
| 232 | Flag* args_flag = NULL; |
kasperl@chromium.org | b912362 | 2008-09-17 14:05:56 +0000 | [diff] [blame] | 233 | for (size_t i = 0; i < num_flags; ++i) { |
| 234 | Flag* f = &flags[i]; |
christian.plesner.hansen | 43d26ec | 2008-07-03 15:10:15 +0000 | [diff] [blame] | 235 | if (!f->IsDefault()) { |
ager@chromium.org | 3bf7b91 | 2008-11-17 09:09:45 +0000 | [diff] [blame] | 236 | if (f->type() == Flag::TYPE_ARGS) { |
| 237 | ASSERT(args_flag == NULL); |
| 238 | args_flag = f; // Must be last in arguments. |
| 239 | continue; |
christian.plesner.hansen | 43d26ec | 2008-07-03 15:10:15 +0000 | [diff] [blame] | 240 | } |
ager@chromium.org | 3bf7b91 | 2008-11-17 09:09:45 +0000 | [diff] [blame] | 241 | HeapStringAllocator string_allocator; |
| 242 | StringStream buffer(&string_allocator); |
| 243 | if (f->type() != Flag::TYPE_BOOL || *(f->bool_variable())) { |
| 244 | buffer.Add("--%s", f->name()); |
| 245 | } else { |
| 246 | buffer.Add("--no%s", f->name()); |
| 247 | } |
| 248 | args->Add(buffer.ToCString().Detach()); |
kasperl@chromium.org | b912362 | 2008-09-17 14:05:56 +0000 | [diff] [blame] | 249 | if (f->type() != Flag::TYPE_BOOL) { |
ager@chromium.org | 3bf7b91 | 2008-11-17 09:09:45 +0000 | [diff] [blame] | 250 | args->Add(ToString(f).Detach()); |
christian.plesner.hansen | 43d26ec | 2008-07-03 15:10:15 +0000 | [diff] [blame] | 251 | } |
| 252 | } |
| 253 | } |
ager@chromium.org | 3bf7b91 | 2008-11-17 09:09:45 +0000 | [diff] [blame] | 254 | if (args_flag != NULL) { |
| 255 | HeapStringAllocator string_allocator; |
| 256 | StringStream buffer(&string_allocator); |
| 257 | buffer.Add("--%s", args_flag->name()); |
| 258 | args->Add(buffer.ToCString().Detach()); |
| 259 | JSArguments jsargs = *args_flag->args_variable(); |
| 260 | for (int j = 0; j < jsargs.argc(); j++) { |
ager@chromium.org | bb29dc9 | 2009-03-24 13:25:23 +0000 | [diff] [blame] | 261 | args->Add(StrDup(jsargs[j])); |
ager@chromium.org | 3bf7b91 | 2008-11-17 09:09:45 +0000 | [diff] [blame] | 262 | } |
| 263 | } |
christian.plesner.hansen | 43d26ec | 2008-07-03 15:10:15 +0000 | [diff] [blame] | 264 | return args; |
| 265 | } |
| 266 | |
| 267 | |
kasperl@chromium.org | b912362 | 2008-09-17 14:05:56 +0000 | [diff] [blame] | 268 | // Helper function to parse flags: Takes an argument arg and splits it into |
| 269 | // a flag name and flag value (or NULL if they are missing). is_bool is set |
| 270 | // if the arg started with "-no" or "--no". The buffer may be used to NUL- |
| 271 | // terminate the name, it must be large enough to hold any possible name. |
| 272 | static void SplitArgument(const char* arg, |
| 273 | char* buffer, |
| 274 | int buffer_size, |
| 275 | const char** name, |
| 276 | const char** value, |
| 277 | bool* is_bool) { |
christian.plesner.hansen | 43d26ec | 2008-07-03 15:10:15 +0000 | [diff] [blame] | 278 | *name = NULL; |
| 279 | *value = NULL; |
| 280 | *is_bool = false; |
| 281 | |
kasperl@chromium.org | a555126 | 2010-12-07 12:49:48 +0000 | [diff] [blame] | 282 | if (arg != NULL && *arg == '-') { |
christian.plesner.hansen | 43d26ec | 2008-07-03 15:10:15 +0000 | [diff] [blame] | 283 | // find the begin of the flag name |
| 284 | arg++; // remove 1st '-' |
ager@chromium.org | 3bf7b91 | 2008-11-17 09:09:45 +0000 | [diff] [blame] | 285 | if (*arg == '-') { |
christian.plesner.hansen | 43d26ec | 2008-07-03 15:10:15 +0000 | [diff] [blame] | 286 | arg++; // remove 2nd '-' |
ager@chromium.org | 3bf7b91 | 2008-11-17 09:09:45 +0000 | [diff] [blame] | 287 | if (arg[0] == '\0') { |
| 288 | const char* kJSArgumentsFlagName = "js_arguments"; |
| 289 | *name = kJSArgumentsFlagName; |
| 290 | return; |
| 291 | } |
| 292 | } |
christian.plesner.hansen | 43d26ec | 2008-07-03 15:10:15 +0000 | [diff] [blame] | 293 | if (arg[0] == 'n' && arg[1] == 'o') { |
| 294 | arg += 2; // remove "no" |
| 295 | *is_bool = true; |
| 296 | } |
| 297 | *name = arg; |
| 298 | |
| 299 | // find the end of the flag name |
| 300 | while (*arg != '\0' && *arg != '=') |
| 301 | arg++; |
| 302 | |
| 303 | // get the value if any |
| 304 | if (*arg == '=') { |
| 305 | // make a copy so we can NUL-terminate flag name |
ager@chromium.org | c4c9272 | 2009-11-18 14:12:51 +0000 | [diff] [blame] | 306 | size_t n = arg - *name; |
| 307 | CHECK(n < static_cast<size_t>(buffer_size)); // buffer is too small |
christian.plesner.hansen | 43d26ec | 2008-07-03 15:10:15 +0000 | [diff] [blame] | 308 | memcpy(buffer, *name, n); |
| 309 | buffer[n] = '\0'; |
| 310 | *name = buffer; |
| 311 | // get the value |
| 312 | *value = arg + 1; |
| 313 | } |
| 314 | } |
| 315 | } |
| 316 | |
| 317 | |
kasperl@chromium.org | b912362 | 2008-09-17 14:05:56 +0000 | [diff] [blame] | 318 | inline char NormalizeChar(char ch) { |
| 319 | return ch == '_' ? '-' : ch; |
| 320 | } |
| 321 | |
| 322 | |
| 323 | static bool EqualNames(const char* a, const char* b) { |
| 324 | for (int i = 0; NormalizeChar(a[i]) == NormalizeChar(b[i]); i++) { |
| 325 | if (a[i] == '\0') { |
| 326 | return true; |
| 327 | } |
| 328 | } |
| 329 | return false; |
| 330 | } |
| 331 | |
| 332 | |
| 333 | static Flag* FindFlag(const char* name) { |
| 334 | for (size_t i = 0; i < num_flags; ++i) { |
| 335 | if (EqualNames(name, flags[i].name())) |
| 336 | return &flags[i]; |
| 337 | } |
| 338 | return NULL; |
| 339 | } |
| 340 | |
| 341 | |
| 342 | // static |
christian.plesner.hansen | 43d26ec | 2008-07-03 15:10:15 +0000 | [diff] [blame] | 343 | int FlagList::SetFlagsFromCommandLine(int* argc, |
| 344 | char** argv, |
| 345 | bool remove_flags) { |
yangguo@chromium.org | 5a11aaf | 2012-06-20 11:29:00 +0000 | [diff] [blame] | 346 | int return_code = 0; |
christian.plesner.hansen | 43d26ec | 2008-07-03 15:10:15 +0000 | [diff] [blame] | 347 | // parse arguments |
| 348 | for (int i = 1; i < *argc;) { |
| 349 | int j = i; // j > 0 |
| 350 | const char* arg = argv[i++]; |
| 351 | |
| 352 | // split arg into flag components |
| 353 | char buffer[1*KB]; |
| 354 | const char* name; |
| 355 | const char* value; |
| 356 | bool is_bool; |
| 357 | SplitArgument(arg, buffer, sizeof buffer, &name, &value, &is_bool); |
| 358 | |
| 359 | if (name != NULL) { |
| 360 | // lookup the flag |
kasperl@chromium.org | b912362 | 2008-09-17 14:05:56 +0000 | [diff] [blame] | 361 | Flag* flag = FindFlag(name); |
christian.plesner.hansen | 43d26ec | 2008-07-03 15:10:15 +0000 | [diff] [blame] | 362 | if (flag == NULL) { |
mads.s.ager@gmail.com | 9a4089a | 2008-09-01 08:55:01 +0000 | [diff] [blame] | 363 | if (remove_flags) { |
| 364 | // We don't recognize this flag but since we're removing |
| 365 | // the flags we recognize we assume that the remaining flags |
| 366 | // will be processed somewhere else so this flag might make |
| 367 | // sense there. |
| 368 | continue; |
| 369 | } else { |
ager@chromium.org | 3bf7b91 | 2008-11-17 09:09:45 +0000 | [diff] [blame] | 370 | fprintf(stderr, "Error: unrecognized flag %s\n" |
| 371 | "Try --help for options\n", arg); |
yangguo@chromium.org | 5a11aaf | 2012-06-20 11:29:00 +0000 | [diff] [blame] | 372 | return_code = j; |
| 373 | break; |
mads.s.ager@gmail.com | 9a4089a | 2008-09-01 08:55:01 +0000 | [diff] [blame] | 374 | } |
christian.plesner.hansen | 43d26ec | 2008-07-03 15:10:15 +0000 | [diff] [blame] | 375 | } |
| 376 | |
| 377 | // if we still need a flag value, use the next argument if available |
ager@chromium.org | 3bf7b91 | 2008-11-17 09:09:45 +0000 | [diff] [blame] | 378 | if (flag->type() != Flag::TYPE_BOOL && |
| 379 | flag->type() != Flag::TYPE_ARGS && |
| 380 | value == NULL) { |
christian.plesner.hansen | 43d26ec | 2008-07-03 15:10:15 +0000 | [diff] [blame] | 381 | if (i < *argc) { |
| 382 | value = argv[i++]; |
| 383 | } else { |
ager@chromium.org | 3bf7b91 | 2008-11-17 09:09:45 +0000 | [diff] [blame] | 384 | fprintf(stderr, "Error: missing value for flag %s of type %s\n" |
| 385 | "Try --help for options\n", |
christian.plesner.hansen | 43d26ec | 2008-07-03 15:10:15 +0000 | [diff] [blame] | 386 | arg, Type2String(flag->type())); |
yangguo@chromium.org | 5a11aaf | 2012-06-20 11:29:00 +0000 | [diff] [blame] | 387 | return_code = j; |
| 388 | break; |
christian.plesner.hansen | 43d26ec | 2008-07-03 15:10:15 +0000 | [diff] [blame] | 389 | } |
| 390 | } |
| 391 | |
| 392 | // set the flag |
| 393 | char* endp = const_cast<char*>(""); // *endp is only read |
| 394 | switch (flag->type()) { |
kasperl@chromium.org | b912362 | 2008-09-17 14:05:56 +0000 | [diff] [blame] | 395 | case Flag::TYPE_BOOL: |
christian.plesner.hansen | 43d26ec | 2008-07-03 15:10:15 +0000 | [diff] [blame] | 396 | *flag->bool_variable() = !is_bool; |
| 397 | break; |
kasperl@chromium.org | b912362 | 2008-09-17 14:05:56 +0000 | [diff] [blame] | 398 | case Flag::TYPE_INT: |
mads.s.ager@gmail.com | 9a4089a | 2008-09-01 08:55:01 +0000 | [diff] [blame] | 399 | *flag->int_variable() = strtol(value, &endp, 10); // NOLINT |
christian.plesner.hansen | 43d26ec | 2008-07-03 15:10:15 +0000 | [diff] [blame] | 400 | break; |
kasperl@chromium.org | b912362 | 2008-09-17 14:05:56 +0000 | [diff] [blame] | 401 | case Flag::TYPE_FLOAT: |
christian.plesner.hansen | 43d26ec | 2008-07-03 15:10:15 +0000 | [diff] [blame] | 402 | *flag->float_variable() = strtod(value, &endp); |
| 403 | break; |
kasperl@chromium.org | b912362 | 2008-09-17 14:05:56 +0000 | [diff] [blame] | 404 | case Flag::TYPE_STRING: |
ager@chromium.org | 41826e7 | 2009-03-30 13:30:57 +0000 | [diff] [blame] | 405 | flag->set_string_value(value ? StrDup(value) : NULL, true); |
christian.plesner.hansen | 43d26ec | 2008-07-03 15:10:15 +0000 | [diff] [blame] | 406 | break; |
ager@chromium.org | 3bf7b91 | 2008-11-17 09:09:45 +0000 | [diff] [blame] | 407 | case Flag::TYPE_ARGS: { |
| 408 | int start_pos = (value == NULL) ? i : i - 1; |
| 409 | int js_argc = *argc - start_pos; |
| 410 | const char** js_argv = NewArray<const char*>(js_argc); |
| 411 | if (value != NULL) { |
ager@chromium.org | 41826e7 | 2009-03-30 13:30:57 +0000 | [diff] [blame] | 412 | js_argv[0] = StrDup(value); |
ager@chromium.org | 3bf7b91 | 2008-11-17 09:09:45 +0000 | [diff] [blame] | 413 | } |
| 414 | for (int k = i; k < *argc; k++) { |
ager@chromium.org | 41826e7 | 2009-03-30 13:30:57 +0000 | [diff] [blame] | 415 | js_argv[k - start_pos] = StrDup(argv[k]); |
ager@chromium.org | 3bf7b91 | 2008-11-17 09:09:45 +0000 | [diff] [blame] | 416 | } |
jkummerow@chromium.org | 1456e70 | 2012-03-30 08:38:13 +0000 | [diff] [blame] | 417 | *flag->args_variable() = JSArguments::Create(js_argc, js_argv); |
ager@chromium.org | 3bf7b91 | 2008-11-17 09:09:45 +0000 | [diff] [blame] | 418 | i = *argc; // Consume all arguments |
| 419 | break; |
| 420 | } |
christian.plesner.hansen | 43d26ec | 2008-07-03 15:10:15 +0000 | [diff] [blame] | 421 | } |
| 422 | |
| 423 | // handle errors |
kasperl@chromium.org | b912362 | 2008-09-17 14:05:56 +0000 | [diff] [blame] | 424 | if ((flag->type() == Flag::TYPE_BOOL && value != NULL) || |
| 425 | (flag->type() != Flag::TYPE_BOOL && is_bool) || |
christian.plesner.hansen | 43d26ec | 2008-07-03 15:10:15 +0000 | [diff] [blame] | 426 | *endp != '\0') { |
ager@chromium.org | 3bf7b91 | 2008-11-17 09:09:45 +0000 | [diff] [blame] | 427 | fprintf(stderr, "Error: illegal value for flag %s of type %s\n" |
| 428 | "Try --help for options\n", |
christian.plesner.hansen | 43d26ec | 2008-07-03 15:10:15 +0000 | [diff] [blame] | 429 | arg, Type2String(flag->type())); |
yangguo@chromium.org | 5a11aaf | 2012-06-20 11:29:00 +0000 | [diff] [blame] | 430 | return_code = j; |
| 431 | break; |
christian.plesner.hansen | 43d26ec | 2008-07-03 15:10:15 +0000 | [diff] [blame] | 432 | } |
| 433 | |
| 434 | // remove the flag & value from the command |
ager@chromium.org | 3bf7b91 | 2008-11-17 09:09:45 +0000 | [diff] [blame] | 435 | if (remove_flags) { |
| 436 | while (j < i) { |
christian.plesner.hansen | 43d26ec | 2008-07-03 15:10:15 +0000 | [diff] [blame] | 437 | argv[j++] = NULL; |
ager@chromium.org | 3bf7b91 | 2008-11-17 09:09:45 +0000 | [diff] [blame] | 438 | } |
| 439 | } |
christian.plesner.hansen | 43d26ec | 2008-07-03 15:10:15 +0000 | [diff] [blame] | 440 | } |
| 441 | } |
| 442 | |
| 443 | // shrink the argument list |
| 444 | if (remove_flags) { |
| 445 | int j = 1; |
| 446 | for (int i = 1; i < *argc; i++) { |
| 447 | if (argv[i] != NULL) |
| 448 | argv[j++] = argv[i]; |
| 449 | } |
| 450 | *argc = j; |
| 451 | } |
| 452 | |
ager@chromium.org | 3bf7b91 | 2008-11-17 09:09:45 +0000 | [diff] [blame] | 453 | if (FLAG_help) { |
| 454 | PrintHelp(); |
iposva@chromium.org | 245aa85 | 2009-02-10 00:49:54 +0000 | [diff] [blame] | 455 | exit(0); |
ager@chromium.org | 3bf7b91 | 2008-11-17 09:09:45 +0000 | [diff] [blame] | 456 | } |
christian.plesner.hansen | 43d26ec | 2008-07-03 15:10:15 +0000 | [diff] [blame] | 457 | // parsed all flags successfully |
yangguo@chromium.org | 5a11aaf | 2012-06-20 11:29:00 +0000 | [diff] [blame] | 458 | return return_code; |
christian.plesner.hansen | 43d26ec | 2008-07-03 15:10:15 +0000 | [diff] [blame] | 459 | } |
| 460 | |
| 461 | |
| 462 | static char* SkipWhiteSpace(char* p) { |
| 463 | while (*p != '\0' && isspace(*p) != 0) p++; |
| 464 | return p; |
| 465 | } |
| 466 | |
| 467 | |
| 468 | static char* SkipBlackSpace(char* p) { |
| 469 | while (*p != '\0' && isspace(*p) == 0) p++; |
| 470 | return p; |
| 471 | } |
| 472 | |
| 473 | |
kasperl@chromium.org | b912362 | 2008-09-17 14:05:56 +0000 | [diff] [blame] | 474 | // static |
christian.plesner.hansen | 43d26ec | 2008-07-03 15:10:15 +0000 | [diff] [blame] | 475 | int FlagList::SetFlagsFromString(const char* str, int len) { |
| 476 | // make a 0-terminated copy of str |
sgjesse@chromium.org | 720dc0b | 2010-05-10 09:25:39 +0000 | [diff] [blame] | 477 | ScopedVector<char> copy0(len + 1); |
| 478 | memcpy(copy0.start(), str, len); |
christian.plesner.hansen | 43d26ec | 2008-07-03 15:10:15 +0000 | [diff] [blame] | 479 | copy0[len] = '\0'; |
| 480 | |
| 481 | // strip leading white space |
sgjesse@chromium.org | 720dc0b | 2010-05-10 09:25:39 +0000 | [diff] [blame] | 482 | char* copy = SkipWhiteSpace(copy0.start()); |
christian.plesner.hansen | 43d26ec | 2008-07-03 15:10:15 +0000 | [diff] [blame] | 483 | |
| 484 | // count the number of 'arguments' |
| 485 | int argc = 1; // be compatible with SetFlagsFromCommandLine() |
| 486 | for (char* p = copy; *p != '\0'; argc++) { |
| 487 | p = SkipBlackSpace(p); |
| 488 | p = SkipWhiteSpace(p); |
| 489 | } |
| 490 | |
| 491 | // allocate argument array |
sgjesse@chromium.org | 720dc0b | 2010-05-10 09:25:39 +0000 | [diff] [blame] | 492 | ScopedVector<char*> argv(argc); |
christian.plesner.hansen | 43d26ec | 2008-07-03 15:10:15 +0000 | [diff] [blame] | 493 | |
| 494 | // split the flags string into arguments |
| 495 | argc = 1; // be compatible with SetFlagsFromCommandLine() |
| 496 | for (char* p = copy; *p != '\0'; argc++) { |
| 497 | argv[argc] = p; |
| 498 | p = SkipBlackSpace(p); |
| 499 | if (*p != '\0') *p++ = '\0'; // 0-terminate argument |
| 500 | p = SkipWhiteSpace(p); |
| 501 | } |
| 502 | |
| 503 | // set the flags |
sgjesse@chromium.org | 720dc0b | 2010-05-10 09:25:39 +0000 | [diff] [blame] | 504 | int result = SetFlagsFromCommandLine(&argc, argv.start(), false); |
christian.plesner.hansen | 43d26ec | 2008-07-03 15:10:15 +0000 | [diff] [blame] | 505 | |
| 506 | return result; |
| 507 | } |
| 508 | |
| 509 | |
kasperl@chromium.org | b912362 | 2008-09-17 14:05:56 +0000 | [diff] [blame] | 510 | // static |
| 511 | void FlagList::ResetAllFlags() { |
| 512 | for (size_t i = 0; i < num_flags; ++i) { |
| 513 | flags[i].Reset(); |
| 514 | } |
| 515 | } |
| 516 | |
| 517 | |
| 518 | // static |
| 519 | void FlagList::PrintHelp() { |
ager@chromium.org | 3bf7b91 | 2008-11-17 09:09:45 +0000 | [diff] [blame] | 520 | printf("Usage:\n"); |
| 521 | printf(" shell [options] -e string\n"); |
| 522 | printf(" execute string in V8\n"); |
| 523 | printf(" shell [options] file1 file2 ... filek\n"); |
| 524 | printf(" run JavaScript scripts in file1, file2, ..., filek\n"); |
| 525 | printf(" shell [options]\n"); |
christian.plesner.hansen@gmail.com | 37abdec | 2009-01-06 14:43:28 +0000 | [diff] [blame] | 526 | printf(" shell [options] --shell [file1 file2 ... filek]\n"); |
| 527 | printf(" run an interactive JavaScript shell\n"); |
| 528 | printf(" d8 [options] file1 file2 ... filek\n"); |
ager@chromium.org | 3bf7b91 | 2008-11-17 09:09:45 +0000 | [diff] [blame] | 529 | printf(" d8 [options]\n"); |
christian.plesner.hansen@gmail.com | 37abdec | 2009-01-06 14:43:28 +0000 | [diff] [blame] | 530 | printf(" d8 [options] --shell [file1 file2 ... filek]\n"); |
ager@chromium.org | 3bf7b91 | 2008-11-17 09:09:45 +0000 | [diff] [blame] | 531 | printf(" run the new debugging shell\n\n"); |
| 532 | printf("Options:\n"); |
kasperl@chromium.org | b912362 | 2008-09-17 14:05:56 +0000 | [diff] [blame] | 533 | for (size_t i = 0; i < num_flags; ++i) { |
| 534 | Flag* f = &flags[i]; |
kmillikin@chromium.org | 83e1682 | 2011-09-13 08:21:47 +0000 | [diff] [blame] | 535 | SmartArrayPointer<const char> value = ToString(f); |
ager@chromium.org | 3bf7b91 | 2008-11-17 09:09:45 +0000 | [diff] [blame] | 536 | printf(" --%s (%s)\n type: %s default: %s\n", |
| 537 | f->name(), f->comment(), Type2String(f->type()), *value); |
kasperl@chromium.org | b912362 | 2008-09-17 14:05:56 +0000 | [diff] [blame] | 538 | } |
christian.plesner.hansen | 43d26ec | 2008-07-03 15:10:15 +0000 | [diff] [blame] | 539 | } |
| 540 | |
ager@chromium.org | 3bf7b91 | 2008-11-17 09:09:45 +0000 | [diff] [blame] | 541 | |
ricow@chromium.org | 7ad6522 | 2011-12-19 12:13:11 +0000 | [diff] [blame] | 542 | void FlagList::EnforceFlagImplications() { |
| 543 | #define FLAG_MODE_DEFINE_IMPLICATIONS |
| 544 | #include "flag-definitions.h" |
| 545 | } |
| 546 | |
christian.plesner.hansen | 43d26ec | 2008-07-03 15:10:15 +0000 | [diff] [blame] | 547 | } } // namespace v8::internal |