Added development shell (d8) including readline support, counters and

completion.


git-svn-id: http://v8.googlecode.com/svn/branches/bleeding_edge@533 ce2b1a6d-e550-0410-aec6-3dcde31c8c00
parent 00082564
...@@ -216,6 +216,16 @@ SAMPLE_FLAGS = { ...@@ -216,6 +216,16 @@ SAMPLE_FLAGS = {
} }
D8_FLAGS = {
'gcc': {
'console:readline': {
'LIBS': ['readline']
}
},
'msvc': { }
}
SUFFIXES = { SUFFIXES = {
'release': '', 'release': '',
'debug': '_g' 'debug': '_g'
...@@ -312,6 +322,11 @@ SIMPLE_OPTIONS = { ...@@ -312,6 +322,11 @@ SIMPLE_OPTIONS = {
'values': ['MD5', 'timestamp'], 'values': ['MD5', 'timestamp'],
'default': 'MD5', 'default': 'MD5',
'help': 'set how the build system detects file changes' 'help': 'set how the build system detects file changes'
},
'console': {
'values': ['dumb', 'readline'],
'default': 'dumb',
'help': 'the console to use for the d8 shell'
} }
} }
...@@ -364,6 +379,7 @@ class BuildContext(object): ...@@ -364,6 +379,7 @@ class BuildContext(object):
self.library_targets = [] self.library_targets = []
self.cctest_targets = [] self.cctest_targets = []
self.sample_targets = [] self.sample_targets = []
self.d8_targets = []
self.options = options self.options = options
self.env_overrides = env_overrides self.env_overrides = env_overrides
self.samples = samples self.samples = samples
...@@ -447,13 +463,15 @@ def BuildSpecific(env, mode, env_overrides): ...@@ -447,13 +463,15 @@ def BuildSpecific(env, mode, env_overrides):
dtoa_flags = context.AddRelevantFlags(library_flags, DTOA_EXTRA_FLAGS) dtoa_flags = context.AddRelevantFlags(library_flags, DTOA_EXTRA_FLAGS)
cctest_flags = context.AddRelevantFlags(v8_flags, CCTEST_EXTRA_FLAGS) cctest_flags = context.AddRelevantFlags(v8_flags, CCTEST_EXTRA_FLAGS)
sample_flags = context.AddRelevantFlags(os.environ, SAMPLE_FLAGS) sample_flags = context.AddRelevantFlags(os.environ, SAMPLE_FLAGS)
d8_flags = context.AddRelevantFlags(library_flags, D8_FLAGS)
context.flags = { context.flags = {
'v8': v8_flags, 'v8': v8_flags,
'jscre': jscre_flags, 'jscre': jscre_flags,
'dtoa': dtoa_flags, 'dtoa': dtoa_flags,
'cctest': cctest_flags, 'cctest': cctest_flags,
'sample': sample_flags 'sample': sample_flags,
'd8': d8_flags
} }
target_id = mode target_id = mode
...@@ -462,13 +480,13 @@ def BuildSpecific(env, mode, env_overrides): ...@@ -462,13 +480,13 @@ def BuildSpecific(env, mode, env_overrides):
env['LIBRARY'] = library_name env['LIBRARY'] = library_name
# Build the object files by invoking SCons recursively. # Build the object files by invoking SCons recursively.
object_files = env.SConscript( (object_files, shell_files) = env.SConscript(
join('src', 'SConscript'), join('src', 'SConscript'),
build_dir=join('obj', target_id), build_dir=join('obj', target_id),
exports='context', exports='context',
duplicate=False duplicate=False
) )
# Link the object files into a library. # Link the object files into a library.
env.Replace(**context.flags['v8']) env.Replace(**context.flags['v8'])
context.ApplyEnvOverrides(env) context.ApplyEnvOverrides(env)
...@@ -482,6 +500,11 @@ def BuildSpecific(env, mode, env_overrides): ...@@ -482,6 +500,11 @@ def BuildSpecific(env, mode, env_overrides):
library = env.SharedLibrary(library_name, object_files, PDB=pdb_name) library = env.SharedLibrary(library_name, object_files, PDB=pdb_name)
context.library_targets.append(library) context.library_targets.append(library)
d8_env = Environment()
d8_env.Replace(**context.flags['d8'])
shell = d8_env.Program('d8' + suffix, object_files + shell_files)
context.d8_targets.append(shell)
for sample in context.samples: for sample in context.samples:
sample_env = Environment(LIBRARY=library_name) sample_env = Environment(LIBRARY=library_name)
sample_env.Replace(**context.flags['sample']) sample_env.Replace(**context.flags['sample'])
...@@ -514,22 +537,25 @@ def Build(): ...@@ -514,22 +537,25 @@ def Build():
Help(opts.GenerateHelpText(env)) Help(opts.GenerateHelpText(env))
VerifyOptions(env) VerifyOptions(env)
env_overrides = ParseEnvOverrides(env['env']) env_overrides = ParseEnvOverrides(env['env'])
SourceSignatures(env['sourcesignatures']) SourceSignatures(env['sourcesignatures'])
libraries = [] libraries = []
cctests = [] cctests = []
samples = [] samples = []
d8s = []
modes = SplitList(env['mode']) modes = SplitList(env['mode'])
for mode in modes: for mode in modes:
context = BuildSpecific(env.Copy(), mode, env_overrides) context = BuildSpecific(env.Copy(), mode, env_overrides)
libraries += context.library_targets libraries += context.library_targets
cctests += context.cctest_targets cctests += context.cctest_targets
samples += context.sample_targets samples += context.sample_targets
d8s += context.d8_targets
env.Alias('library', libraries) env.Alias('library', libraries)
env.Alias('cctests', cctests) env.Alias('cctests', cctests)
env.Alias('sample', samples) env.Alias('sample', samples)
env.Alias('d8', d8s)
if env['sample']: if env['sample']:
env.Default('sample') env.Default('sample')
......
...@@ -64,6 +64,16 @@ SOURCES = { ...@@ -64,6 +64,16 @@ SOURCES = {
} }
D8_FILES = {
'all': [
'd8.cc'
],
'console:readline': [
'd8-readline.cc'
]
}
LIBRARY_FILES = ''' LIBRARY_FILES = '''
runtime.js runtime.js
v8natives.js v8natives.js
...@@ -104,11 +114,16 @@ def ConfigureObjectFiles(): ...@@ -104,11 +114,16 @@ def ConfigureObjectFiles():
# Build the standard platform-independent source files. # Build the standard platform-independent source files.
source_files = context.GetRelevantSources(SOURCES) source_files = context.GetRelevantSources(SOURCES)
d8_files = context.GetRelevantSources(D8_FILES)
d8_js = env.JS2C('d8-js.cc', 'd8.js', TYPE='D8')
d8_js_obj = context.ConfigureObject(env, d8_js, CPPPATH=['.'])
d8_objs = [context.ConfigureObject(env, [d8_files]), d8_js_obj]
# Combine the JavaScript library files into a single C++ file and # Combine the JavaScript library files into a single C++ file and
# compile it. # compile it.
library_files = [s for s in LIBRARY_FILES] library_files = [s for s in LIBRARY_FILES]
library_files.append('macros.py') library_files.append('macros.py')
libraries_src, libraries_empty_src = env.JS2C(['libraries.cc', 'libraries-empty.cc'], library_files) libraries_src, libraries_empty_src = env.JS2C(['libraries.cc', 'libraries-empty.cc'], library_files, TYPE='CORE')
libraries_obj = context.ConfigureObject(env, libraries_src, CPPPATH=['.']) libraries_obj = context.ConfigureObject(env, libraries_src, CPPPATH=['.'])
# Build JSCRE. # Build JSCRE.
...@@ -137,8 +152,9 @@ def ConfigureObjectFiles(): ...@@ -137,8 +152,9 @@ def ConfigureObjectFiles():
else: else:
snapshot_obj = empty_snapshot_obj snapshot_obj = empty_snapshot_obj
return [non_snapshot_files, libraries_obj, snapshot_obj] library_objs = [non_snapshot_files, libraries_obj, snapshot_obj]
return (library_objs, d8_objs)
library_objects = ConfigureObjectFiles() (library_objs, d8_objs) = ConfigureObjectFiles()
Return('library_objects') Return('library_objs d8_objs')
// Copyright 2008 the V8 project authors. All rights reserved.
// Redistribution and use in source and binary forms, with or without
// modification, are permitted provided that the following conditions are
// met:
//
// * Redistributions of source code must retain the above copyright
// notice, this list of conditions and the following disclaimer.
// * Redistributions in binary form must reproduce the above
// copyright notice, this list of conditions and the following
// disclaimer in the documentation and/or other materials provided
// with the distribution.
// * Neither the name of Google Inc. nor the names of its
// contributors may be used to endorse or promote products derived
// from this software without specific prior written permission.
//
// THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS
// "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT
// LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR
// A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT
// OWNER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL,
// SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT
// LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE,
// DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY
// THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
// (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE
// OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
#include <readline/readline.h>
#include <readline/history.h>
#include "d8.h"
namespace v8 {
class ReadLineEditor: public LineEditor {
public:
ReadLineEditor() : LineEditor(LineEditor::READLINE, "readline") { }
virtual i::SmartPointer<char> Prompt(const char* prompt);
virtual bool Open();
virtual bool Close();
virtual void AddHistory(const char* str);
private:
static char** AttemptedCompletion(const char* text, int start, int end);
static char* CompletionGenerator(const char* text, int state);
static char kWordBreakCharacters[];
};
static ReadLineEditor read_line_editor;
char ReadLineEditor::kWordBreakCharacters[] = {' ', '\t', '\n', '"',
'\\', '\'', '`', '@', '.', '>', '<', '=', ';', '|', '&', '{', '(',
'\0'};
bool ReadLineEditor::Open() {
rl_initialize();
rl_attempted_completion_function = AttemptedCompletion;
rl_completer_word_break_characters = kWordBreakCharacters;
rl_bind_key('\t', rl_complete);
using_history();
return read_history(Shell::kHistoryFileName) == 0;
}
bool ReadLineEditor::Close() {
return write_history(Shell::kHistoryFileName) == 0;
}
i::SmartPointer<char> ReadLineEditor::Prompt(const char* prompt) {
char* result = readline(prompt);
return i::SmartPointer<char>(result);
}
void ReadLineEditor::AddHistory(const char* str) {
add_history(str);
}
char** ReadLineEditor::AttemptedCompletion(const char* text,
int start,
int end) {
char** result = rl_completion_matches(text, CompletionGenerator);
rl_attempted_completion_over = true;
return result;
}
char* ReadLineEditor::CompletionGenerator(const char* text, int state) {
static unsigned current_index;
static Persistent<Array> current_completions;
if (state == 0) {
i::SmartPointer<char> full_text(strndup(rl_line_buffer, rl_point));
HandleScope scope;
Handle<Array> completions =
Shell::GetCompletions(String::New(text), String::New(*full_text));
current_completions = Persistent<Array>::New(completions);
current_index = 0;
}
if (current_index < current_completions->Length()) {
HandleScope scope;
Handle<Integer> index = Integer::New(current_index);
Handle<Value> str_obj = current_completions->Get(index);
current_index++;
String::Utf8Value str(str_obj);
return strdup(*str);
} else {
current_completions.Dispose();
current_completions.Clear();
return NULL;
}
}
} // namespace v8
This diff is collapsed.
// Copyright 2008 the V8 project authors. All rights reserved.
// Redistribution and use in source and binary forms, with or without
// modification, are permitted provided that the following conditions are
// met:
//
// * Redistributions of source code must retain the above copyright
// notice, this list of conditions and the following disclaimer.
// * Redistributions in binary form must reproduce the above
// copyright notice, this list of conditions and the following
// disclaimer in the documentation and/or other materials provided
// with the distribution.
// * Neither the name of Google Inc. nor the names of its
// contributors may be used to endorse or promote products derived
// from this software without specific prior written permission.
//
// THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS
// "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT
// LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR
// A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT
// OWNER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL,
// SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT
// LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE,
// DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY
// THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
// (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE
// OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
#ifndef V8_D8_H_
#define V8_D8_H_
// Disable exceptions on windows to not generate warnings from <map>.
#define _HAS_EXCEPTIONS 0
#include <map>
#include "v8.h"
namespace v8 {
namespace i = v8::internal;
class Counter {
public:
explicit Counter(const wchar_t* name)
: name_(name), value_(0) { }
int* GetValuePtr() { return &value_; }
const wchar_t* name() { return name_; }
int value() { return value_; }
private:
const wchar_t* name_;
int value_;
};
class Shell: public i::AllStatic {
public:
static bool ExecuteString(Handle<String> source,
Handle<Value> name,
bool print_result,
bool report_exceptions);
static void ReportException(TryCatch* try_catch);
static void Initialize();
static void OnExit();
static int* LookupCounter(const wchar_t* name);
static Handle<String> ReadFile(const char* name);
static void RunShell();
static int Main(int argc, char* argv[]);
static Handle<Array> GetCompletions(Handle<String> text,
Handle<String> full);
static Handle<Value> Print(const Arguments& args);
static Handle<Value> Quit(const Arguments& args);
static Handle<Value> Version(const Arguments& args);
static Handle<Value> Load(const Arguments& args);
static const char* kHistoryFileName;
static const char* kPrompt;
private:
static Persistent<Context> utility_context_;
static Persistent<Context> evaluation_context_;
typedef std::map<const wchar_t*, Counter*> CounterMap;
static CounterMap counter_map_;
};
class LineEditor {
public:
enum Type { DUMB = 0, READLINE = 1 };
LineEditor(Type type, const char* name);
virtual ~LineEditor() { }
virtual i::SmartPointer<char> Prompt(const char* prompt) = 0;
virtual bool Open() { return true; }
virtual bool Close() { return true; }
virtual void AddHistory(const char* str) { }
const char* name() { return name_; }
static LineEditor* Get();
private:
Type type_;
const char* name_;
LineEditor* next_;
static LineEditor* first_;
};
} // namespace v8
#endif // V8_D8_H_
// Copyright 2008 the V8 project authors. All rights reserved.
// Redistribution and use in source and binary forms, with or without
// modification, are permitted provided that the following conditions are
// met:
//
// * Redistributions of source code must retain the above copyright
// notice, this list of conditions and the following disclaimer.
// * Redistributions in binary form must reproduce the above
// copyright notice, this list of conditions and the following
// disclaimer in the documentation and/or other materials provided
// with the distribution.
// * Neither the name of Google Inc. nor the names of its
// contributors may be used to endorse or promote products derived
// from this software without specific prior written permission.
//
// THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS
// "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT
// LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR
// A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT
// OWNER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL,
// SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT
// LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE,
// DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY
// THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
// (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE
// OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
// How crappy is it that I have to implement completely basic stuff
// like this myself? Answer: very.
String.prototype.startsWith = function (str) {
if (str.length > this.length)
return false;
return this.substr(0, str.length) == str;
};
function ToInspectableObject(obj) {
if (!obj && typeof obj === 'object') {
return void 0;
} else {
return Object(obj);
}
}
function GetCompletions(global, last, full) {
var full_tokens = full.split();
full = full_tokens.pop();
var parts = full.split('.');
parts.pop();
var current = global;
for (var i = 0; i < parts.length; i++) {
var part = parts[i];
var next = current[part];
if (!next)
return [];
current = next;
}
var result = [];
current = ToInspectableObject(current);
while (typeof current !== 'undefined') {
var mirror = new $debug.ObjectMirror(current);
var properties = mirror.properties();
for (var i = 0; i < properties.length; i++) {
var name = properties[i].name();
if (typeof name === 'string' && name.startsWith(last))
result.push(name);
}
current = ToInspectableObject(current.__proto__);
}
return result;
}
...@@ -191,7 +191,11 @@ DEFINE_string(testing_serialization_file, "/tmp/serdes", ...@@ -191,7 +191,11 @@ DEFINE_string(testing_serialization_file, "/tmp/serdes",
"file in which to serialize heap") "file in which to serialize heap")
#endif #endif
//
// Dev shell flags
//
DEFINE_bool(dump_counters, false, "Dump counters on exit")
// //
// Debug only flags // Debug only flags
...@@ -302,7 +306,6 @@ DEFINE_string(logfile, "v8.log", "Specify the name of the log file.") ...@@ -302,7 +306,6 @@ DEFINE_string(logfile, "v8.log", "Specify the name of the log file.")
// codegen-ia32.cc / codegen-arm.cc // codegen-ia32.cc / codegen-arm.cc
DEFINE_bool(print_code, false, "print generated code") DEFINE_bool(print_code, false, "print generated code")
// Cleanup... // Cleanup...
#undef FLAG_FULL #undef FLAG_FULL
#undef FLAG_READONLY #undef FLAG_READONLY
......
...@@ -34,7 +34,12 @@ typedef bool (*NativeSourceCallback)(Vector<const char> name, ...@@ -34,7 +34,12 @@ typedef bool (*NativeSourceCallback)(Vector<const char> name,
Vector<const char> source, Vector<const char> source,
int index); int index);
class Natives { enum NativeType {
CORE, D8
};
template <NativeType type>
class NativesCollection {
public: public:
// Number of built-in scripts. // Number of built-in scripts.
static int GetBuiltinsCount(); static int GetBuiltinsCount();
...@@ -50,6 +55,8 @@ class Natives { ...@@ -50,6 +55,8 @@ class Natives {
static Vector<const char> GetScriptName(int index); static Vector<const char> GetScriptName(int index);
}; };
typedef NativesCollection<CORE> Natives;
} } // namespace v8::internal } } // namespace v8::internal
#endif // V8_NATIVES_H_ #endif // V8_NATIVES_H_
...@@ -86,7 +86,7 @@ class SmartPointer { ...@@ -86,7 +86,7 @@ class SmartPointer {
// the copy constructor it removes the pointer in the original to avoid // the copy constructor it removes the pointer in the original to avoid
// double freeing. // double freeing.
inline SmartPointer& operator=(const SmartPointer<T>& rhs) { inline SmartPointer& operator=(const SmartPointer<T>& rhs) {
ASSERT(p == NULL); ASSERT(is_empty());
T* tmp = rhs.p; // swap to handle self-assignment T* tmp = rhs.p; // swap to handle self-assignment
const_cast<SmartPointer<T>&>(rhs).p = NULL; const_cast<SmartPointer<T>&>(rhs).p = NULL;
p = tmp; p = tmp;
...@@ -94,6 +94,11 @@ class SmartPointer { ...@@ -94,6 +94,11 @@ class SmartPointer {
} }
inline bool is_empty() {
return p == NULL;
}
private: private:
T* p; T* p;
}; };
......
...@@ -200,25 +200,30 @@ namespace internal { ...@@ -200,25 +200,30 @@ namespace internal {
%(source_lines)s\ %(source_lines)s\
int Natives::GetBuiltinsCount() { template <>
int NativesCollection<%(type)s>::GetBuiltinsCount() {
return %(builtin_count)i; return %(builtin_count)i;
} }
int Natives::GetDelayCount() { template <>
int NativesCollection<%(type)s>::GetDelayCount() {
return %(delay_count)i; return %(delay_count)i;
} }
int Natives::GetIndex(const char* name) { template <>
int NativesCollection<%(type)s>::GetIndex(const char* name) {
%(get_index_cases)s\ %(get_index_cases)s\
return -1; return -1;
} }
Vector<const char> Natives::GetScriptSource(int index) { template <>
Vector<const char> NativesCollection<%(type)s>::GetScriptSource(int index) {
%(get_script_source_cases)s\ %(get_script_source_cases)s\
return Vector<const char>("", 0); return Vector<const char>("", 0);
} }
Vector<const char> Natives::GetScriptName(int index) { template <>
Vector<const char> NativesCollection<%(type)s>::GetScriptName(int index) {
%(get_script_name_cases)s\ %(get_script_name_cases)s\
return Vector<const char>("", 0); return Vector<const char>("", 0);
} }
...@@ -323,20 +328,24 @@ def JS2C(source, target, env): ...@@ -323,20 +328,24 @@ def JS2C(source, target, env):
'source_lines': "\n".join(source_lines), 'source_lines': "\n".join(source_lines),
'get_index_cases': "".join(get_index_cases), 'get_index_cases': "".join(get_index_cases),
'get_script_source_cases': "".join(get_script_source_cases), 'get_script_source_cases': "".join(get_script_source_cases),
'get_script_name_cases': "".join(get_script_name_cases) 'get_script_name_cases': "".join(get_script_name_cases),
}) 'type': env['TYPE']
output.close()
output = open(str(target[1]), "w")
output.write(HEADER_TEMPLATE % {
'builtin_count': len(ids) + len(delay_ids),
'delay_count': len(delay_ids),
'source_lines': "\n".join(source_lines_empty),
'get_index_cases': "".join(get_index_cases),
'get_script_source_cases': "".join(get_script_source_cases),
'get_script_name_cases': "".join(get_script_name_cases)
}) })
output.close() output.close()
if len(target) > 1:
output = open(str(target[1]), "w")
output.write(HEADER_TEMPLATE % {
'builtin_count': len(ids) + len(delay_ids),
'delay_count': len(delay_ids),
'source_lines': "\n".join(source_lines_empty),
'get_index_cases': "".join(get_index_cases),
'get_script_source_cases': "".join(get_script_source_cases),
'get_script_name_cases': "".join(get_script_name_cases),
'type': env['TYPE']
})
output.close()
def main(): def main():
natives = sys.argv[1] natives = sys.argv[1]
natives_empty = sys.argv[2] natives_empty = sys.argv[2]
......
Markdown is supported
0% or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment