12 #include "src/allocation.h" 13 #include "src/assembler.h" 14 #include "src/base/functional.h" 15 #include "src/base/platform/platform.h" 16 #include "src/ostreams.h" 17 #include "src/utils.h" 18 #include "src/wasm/wasm-limits.h" 24 #define FLAG_MODE_DEFINE 25 #include "src/flag-definitions.h" 28 #define FLAG_MODE_DEFINE_DEFAULTS 29 #include "src/flag-definitions.h" 55 FlagType type()
const {
return type_; }
57 const char* name()
const {
return name_; }
59 const char* comment()
const {
return cmt_; }
61 bool* bool_variable()
const {
62 DCHECK(type_ == TYPE_BOOL);
63 return reinterpret_cast<bool*
>(valptr_);
66 MaybeBoolFlag* maybe_bool_variable()
const {
67 DCHECK(type_ == TYPE_MAYBE_BOOL);
68 return reinterpret_cast<MaybeBoolFlag*
>(valptr_);
71 int* int_variable()
const {
72 DCHECK(type_ == TYPE_INT);
73 return reinterpret_cast<int*
>(valptr_);
76 unsigned int* uint_variable()
const {
77 DCHECK(type_ == TYPE_UINT);
78 return reinterpret_cast<unsigned int*
>(valptr_);
81 uint64_t* uint64_variable()
const {
82 DCHECK(type_ == TYPE_UINT64);
83 return reinterpret_cast<uint64_t*
>(valptr_);
86 double* float_variable()
const {
87 DCHECK(type_ == TYPE_FLOAT);
88 return reinterpret_cast<double*
>(valptr_);
91 size_t* size_t_variable()
const {
92 DCHECK(type_ == TYPE_SIZE_T);
93 return reinterpret_cast<size_t*
>(valptr_);
96 const char* string_value()
const {
97 DCHECK(type_ == TYPE_STRING);
98 return *
reinterpret_cast<const char**
>(valptr_);
101 void set_string_value(
const char* value,
bool owns_ptr) {
102 DCHECK(type_ == TYPE_STRING);
103 const char** ptr =
reinterpret_cast<const char**
>(valptr_);
104 if (owns_ptr_ && *ptr !=
nullptr) DeleteArray(*ptr);
106 owns_ptr_ = owns_ptr;
109 bool bool_default()
const {
110 DCHECK(type_ == TYPE_BOOL);
111 return *
reinterpret_cast<const bool*
>(defptr_);
114 int int_default()
const {
115 DCHECK(type_ == TYPE_INT);
116 return *
reinterpret_cast<const int*
>(defptr_);
119 unsigned int uint_default()
const {
120 DCHECK(type_ == TYPE_UINT);
121 return *
reinterpret_cast<const unsigned int*
>(defptr_);
124 uint64_t uint64_default()
const {
125 DCHECK(type_ == TYPE_UINT64);
126 return *
reinterpret_cast<const uint64_t*
>(defptr_);
129 double float_default()
const {
130 DCHECK(type_ == TYPE_FLOAT);
131 return *
reinterpret_cast<const double*
>(defptr_);
134 size_t size_t_default()
const {
135 DCHECK(type_ == TYPE_SIZE_T);
136 return *
reinterpret_cast<const size_t*
>(defptr_);
139 const char* string_default()
const {
140 DCHECK(type_ == TYPE_STRING);
141 return *
reinterpret_cast<const char*
const *
>(defptr_);
145 bool IsDefault()
const {
148 return *bool_variable() == bool_default();
149 case TYPE_MAYBE_BOOL:
150 return maybe_bool_variable()->has_value ==
false;
152 return *int_variable() == int_default();
154 return *uint_variable() == uint_default();
156 return *uint64_variable() == uint64_default();
158 return *float_variable() == float_default();
160 return *size_t_variable() == size_t_default();
162 const char* str1 = string_value();
163 const char* str2 = string_default();
164 if (str2 ==
nullptr)
return str1 ==
nullptr;
165 if (str1 ==
nullptr)
return str2 ==
nullptr;
166 return strcmp(str1, str2) == 0;
176 *bool_variable() = bool_default();
178 case TYPE_MAYBE_BOOL:
179 *maybe_bool_variable() = MaybeBoolFlag::Create(
false,
false);
182 *int_variable() = int_default();
185 *uint_variable() = uint_default();
188 *uint64_variable() = uint64_default();
191 *float_variable() = float_default();
194 *size_t_variable() = size_t_default();
197 set_string_value(string_default(),
false);
204 #define FLAG_MODE_META 205 #include "src/flag-definitions.h" 208 const size_t num_flags =
sizeof(flags) /
sizeof(*flags);
213 static const char* Type2String(Flag::FlagType type) {
215 case Flag::TYPE_BOOL:
return "bool";
216 case Flag::TYPE_MAYBE_BOOL:
return "maybe_bool";
217 case Flag::TYPE_INT:
return "int";
218 case Flag::TYPE_UINT:
220 case Flag::TYPE_UINT64:
222 case Flag::TYPE_FLOAT:
return "float";
223 case Flag::TYPE_SIZE_T:
225 case Flag::TYPE_STRING:
return "string";
231 std::ostream& operator<<(std::ostream& os,
const Flag& flag) {
232 switch (flag.type()) {
233 case Flag::TYPE_BOOL:
234 os << (*flag.bool_variable() ?
"true" :
"false");
236 case Flag::TYPE_MAYBE_BOOL:
237 os << (flag.maybe_bool_variable()->has_value
238 ? (flag.maybe_bool_variable()->value ?
"true" :
"false")
242 os << *flag.int_variable();
244 case Flag::TYPE_UINT:
245 os << *flag.uint_variable();
247 case Flag::TYPE_UINT64:
248 os << *flag.uint64_variable();
250 case Flag::TYPE_FLOAT:
251 os << *flag.float_variable();
253 case Flag::TYPE_SIZE_T:
254 os << *flag.size_t_variable();
256 case Flag::TYPE_STRING: {
257 const char* str = flag.string_value();
258 os << (str ? str :
"nullptr");
267 std::vector<const char*>* FlagList::argv() {
268 std::vector<const char*>* args =
new std::vector<const char*>(8);
269 for (
size_t i = 0;
i < num_flags; ++
i) {
271 if (!f->IsDefault()) {
273 bool disabled = f->type() == Flag::TYPE_BOOL && !*f->bool_variable();
274 std::ostringstream os;
275 os << (disabled ?
"--no" :
"--") << f->name();
276 args->push_back(StrDup(os.str().c_str()));
278 if (f->type() != Flag::TYPE_BOOL) {
279 std::ostringstream os;
281 args->push_back(StrDup(os.str().c_str()));
289 inline char NormalizeChar(
char ch) {
290 return ch ==
'_' ?
'-' : ch;
297 static void SplitArgument(
const char* arg,
char* buffer,
int buffer_size,
298 const char** name,
const char** value,
304 if (arg !=
nullptr && *arg ==
'-') {
309 if (arg[0] ==
'\0') {
310 const char* kJSArgumentsFlagName =
"js_arguments";
311 *name = kJSArgumentsFlagName;
315 if (arg[0] ==
'n' && arg[1] ==
'o') {
317 if (NormalizeChar(arg[0]) ==
'-') arg++;
323 while (*arg !=
'\0' && *arg !=
'=')
329 size_t n = arg - *name;
330 CHECK(n < static_cast<size_t>(buffer_size));
331 MemCopy(buffer, *name, n);
341 static bool EqualNames(
const char* a,
const char* b) {
342 for (
int i = 0; NormalizeChar(a[
i]) == NormalizeChar(b[
i]);
i++) {
351 static Flag* FindFlag(
const char* name) {
352 for (
size_t i = 0;
i < num_flags; ++
i) {
353 if (EqualNames(name, flags[
i].name()))
359 template <
typename T>
360 bool TryParseUnsigned(Flag* flag,
const char* arg,
const char* value,
361 char** endp, T* out_val) {
365 uint64_t max =
static_cast<uint64_t
>(std::numeric_limits<T>::max());
368 if (val < 0 || static_cast<uint64_t>(val) > max || errno != 0) {
370 "Error: Value for flag %s of type %s is out of bounds " 372 arg, Type2String(flag->type()), max);
375 *out_val =
static_cast<T
>(val);
380 int FlagList::SetFlagsFromCommandLine(
int* argc,
385 for (
int i = 1;
i < *argc;) {
387 const char* arg = argv[
i++];
394 SplitArgument(arg, buffer,
sizeof buffer, &name, &value, &negated);
396 if (name !=
nullptr) {
398 Flag* flag = FindFlag(name);
399 if (flag ==
nullptr) {
407 PrintF(stderr,
"Error: unrecognized flag %s\n", arg);
414 if (flag->type() != Flag::TYPE_BOOL &&
415 flag->type() != Flag::TYPE_MAYBE_BOOL && value ==
nullptr) {
420 PrintF(stderr,
"Error: missing value for flag %s of type %s\n", arg,
421 Type2String(flag->type()));
428 char* endp =
const_cast<char*
>(
"");
429 switch (flag->type()) {
430 case Flag::TYPE_BOOL:
431 *flag->bool_variable() = !negated;
433 case Flag::TYPE_MAYBE_BOOL:
434 *flag->maybe_bool_variable() = MaybeBoolFlag::Create(
true, !negated);
437 *flag->int_variable() =
static_cast<int>(strtol(value, &endp, 10));
439 case Flag::TYPE_UINT:
440 if (!TryParseUnsigned(flag, arg, value, &endp,
441 flag->uint_variable())) {
445 case Flag::TYPE_UINT64:
446 if (!TryParseUnsigned(flag, arg, value, &endp,
447 flag->uint64_variable())) {
451 case Flag::TYPE_FLOAT:
452 *flag->float_variable() = strtod(value, &endp);
454 case Flag::TYPE_SIZE_T:
455 if (!TryParseUnsigned(flag, arg, value, &endp,
456 flag->size_t_variable())) {
460 case Flag::TYPE_STRING:
461 flag->set_string_value(value ? StrDup(value) :
nullptr,
true);
466 bool is_bool_type = flag->type() == Flag::TYPE_BOOL ||
467 flag->type() == Flag::TYPE_MAYBE_BOOL;
468 if ((is_bool_type && value !=
nullptr) || (!is_bool_type && negated) ||
472 PrintF(stderr,
"Error: illegal value for flag %s of type %s\n", arg,
473 Type2String(flag->type()));
476 "To set or unset a boolean flag, use --flag or --no-flag.\n");
499 for (
int i = 1;
i < *argc;
i++) {
500 if (argv[
i] !=
nullptr) argv[j++] = argv[
i];
503 }
else if (return_code != 0) {
504 if (return_code + 1 < *argc) {
505 PrintF(stderr,
"The remaining arguments were ignored:");
506 for (
int i = return_code + 1;
i < *argc; ++
i) {
507 PrintF(stderr,
" %s", argv[
i]);
509 PrintF(stderr,
"\n");
512 if (return_code != 0) PrintF(stderr,
"Try --help for options\n");
518 static char* SkipWhiteSpace(
char* p) {
519 while (*p !=
'\0' && isspace(*p) != 0) p++;
524 static char* SkipBlackSpace(
char* p) {
525 while (*p !=
'\0' && isspace(*p) == 0) p++;
531 int FlagList::SetFlagsFromString(
const char* str,
int len) {
533 ScopedVector<char> copy0(len + 1);
534 MemCopy(copy0.start(), str, len);
538 char* copy = SkipWhiteSpace(copy0.start());
542 for (
char* p = copy; *p !=
'\0'; argc++) {
543 p = SkipBlackSpace(p);
544 p = SkipWhiteSpace(p);
548 ScopedVector<char*> argv(argc);
552 for (
char* p = copy; *p !=
'\0'; argc++) {
554 p = SkipBlackSpace(p);
555 if (*p !=
'\0') *p++ =
'\0';
556 p = SkipWhiteSpace(p);
559 return SetFlagsFromCommandLine(&argc, argv.start(),
false);
564 void FlagList::ResetAllFlags() {
565 for (
size_t i = 0;
i < num_flags; ++
i) {
572 void FlagList::PrintHelp() {
573 CpuFeatures::Probe(
false);
574 CpuFeatures::PrintTarget();
575 CpuFeatures::PrintFeatures();
579 " shell [options] [--shell] [<file>...]\n" 580 " d8 [options] [-e <string>] [--shell] [[--module] <file>...]\n\n" 581 " -e execute a string in V8\n" 582 " --shell run an interactive JavaScript shell\n" 583 " --module execute a file as a JavaScript module\n\n" 584 "Note: the --module option is implicitly enabled for *.mjs files.\n\n" 587 for (
const Flag& f : flags) {
589 for (
const char* c = f.name(); *c !=
'\0'; ++c) {
590 os << NormalizeChar(*c);
592 os <<
" (" << f.comment() <<
")\n" 593 <<
" type: " << Type2String(f.type()) <<
" default: " << f
602 void ComputeFlagListHash() {
603 std::ostringstream modified_args_as_string;
605 modified_args_as_string <<
"debug";
607 if (FLAG_embedded_builtins) {
608 modified_args_as_string <<
"embedded";
610 for (
size_t i = 0;
i < num_flags; ++
i) {
611 Flag* current = &flags[
i];
612 if (!current->IsDefault()) {
613 modified_args_as_string <<
i;
614 modified_args_as_string << *current;
617 std::string args(modified_args_as_string.str());
619 base::hash_range(args.c_str(), args.c_str() + args.length()));
624 void FlagList::EnforceFlagImplications() {
625 #define FLAG_MODE_DEFINE_IMPLICATIONS 626 #include "src/flag-definitions.h" 627 #undef FLAG_MODE_DEFINE_IMPLICATIONS 628 ComputeFlagListHash();
632 uint32_t FlagList::Hash() {
return flag_hash; }