223 lines
5.5 KiB
V
223 lines
5.5 KiB
V
// Copyright (c) 2019-2020 Alexander Medvednikov. All rights reserved.
|
|
// Use of this source code is governed by an MIT license
|
|
// that can be found in the LICENSE file.
|
|
module main
|
|
|
|
import (
|
|
//internal.compile
|
|
internal.help
|
|
os
|
|
os.cmdline
|
|
v.table
|
|
v.doc
|
|
v.pref
|
|
v.util
|
|
v.builder
|
|
)
|
|
|
|
const (
|
|
simple_cmd = ['fmt',
|
|
'up', 'self',
|
|
'test', 'test-fmt', 'test-compiler', 'test-fixed',
|
|
'bin2v',
|
|
'repl',
|
|
'build-tools', 'build-examples', 'build-vbinaries',
|
|
'setup-freetype']
|
|
|
|
list_of_flags_that_allow_duplicates = ['cc','d','define','cf','cflags']
|
|
//list_of_flags contains a list of flags where an argument is expected past it.
|
|
list_of_flags_with_param = [
|
|
'o', 'output', 'd', 'define', 'b', 'backend', 'cc', 'os', 'target-os', 'arch',
|
|
'csource', 'cf', 'cflags', 'path'
|
|
]
|
|
)
|
|
|
|
fn main() {
|
|
args := os.args[1..]
|
|
//args = 123
|
|
if args.len == 0 || args[0] in ['-', 'repl'] {
|
|
// Running `./v` without args launches repl
|
|
println('For usage information, quit V REPL using `exit` and use `v help`')
|
|
util.launch_tool(false, 'vrepl')
|
|
return
|
|
}
|
|
if args.len > 0 && (args[0] in ['version', '-V', '-version', '--version'] || (args[0] == '-v' && args.len == 1) ) {
|
|
// `-v` flag is for setting verbosity, but without any args it prints the version, like Clang
|
|
println(util.full_v_version())
|
|
return
|
|
}
|
|
prefs, command := parse_args(args)
|
|
if prefs.is_verbose {
|
|
println('command = "$command"')
|
|
println(util.full_v_version())
|
|
}
|
|
if prefs.is_verbose {
|
|
//println('args= ')
|
|
//println(args) // QTODO
|
|
//println('prefs= ')
|
|
//println(prefs) // QTODO
|
|
}
|
|
// Start calling the correct functions/external tools
|
|
// Note for future contributors: Please add new subcommands in the `match` block below.
|
|
if command in simple_cmd {
|
|
// External tools
|
|
util.launch_tool(prefs.is_verbose, 'v' + command)
|
|
return
|
|
}
|
|
match command {
|
|
'help' {
|
|
invoke_help_and_exit(args)
|
|
}
|
|
'new', 'init' {
|
|
util.launch_tool(prefs.is_verbose, 'vcreate')
|
|
return
|
|
}
|
|
'translate' {
|
|
println('Translating C to V will be available in V 0.3')
|
|
return
|
|
}
|
|
'search', 'install', 'update', 'remove' {
|
|
util.launch_tool(prefs.is_verbose, 'vpm')
|
|
return
|
|
}
|
|
'get' {
|
|
println('V Error: Use `v install` to install modules from vpm.vlang.io')
|
|
exit(1)
|
|
}
|
|
'symlink' {
|
|
create_symlink()
|
|
return
|
|
}
|
|
'doc' {
|
|
if args.len == 1 {
|
|
println('v doc [module]')
|
|
exit(1)
|
|
}
|
|
table := table.new_table()
|
|
println(doc.doc(args[1], table))
|
|
return
|
|
}
|
|
'help' {
|
|
invoke_help_and_exit(args)
|
|
return
|
|
}
|
|
else {}
|
|
}
|
|
if command in ['run', 'build'] || command.ends_with('.v') || os.exists(command) {
|
|
builder.compile(command, prefs)
|
|
return
|
|
}
|
|
eprintln('v $command: unknown command\nRun "v help" for usage.')
|
|
exit(1)
|
|
}
|
|
|
|
fn parse_args(args []string) (&pref.Preferences, string) {
|
|
mut res := &pref.Preferences{}
|
|
mut command := ''
|
|
mut command_pos := 0
|
|
//for i, arg in args {
|
|
for i := 0 ; i < args.len; i ++ {
|
|
arg := args[i]
|
|
match arg {
|
|
'-v' { res.is_verbose = true }
|
|
'-cg' { res.is_debug = true }
|
|
'-live' { res.is_solive = true }
|
|
'-autofree' { res.autofree = true }
|
|
'-compress' { res.compress = true }
|
|
'-freestanding' { res.is_bare = true }
|
|
'-prod' { res.is_prod = true }
|
|
'-stats' { res.is_stats = true }
|
|
'-obfuscate' { res.obfuscate = true }
|
|
'-translated' { res.translated = true }
|
|
//'-x64' { res.translated = true }
|
|
'-os' {
|
|
//TODO Remove `tmp` variable when it doesn't error out in C.
|
|
target_os := cmdline.option(args, '-os', '')
|
|
tmp := pref.os_from_string(target_os) or {
|
|
println('unknown operating system target `$target_os`')
|
|
exit(1)
|
|
}
|
|
res.os = tmp
|
|
i++
|
|
}
|
|
'-cc' {
|
|
res.ccompiler = cmdline.option(args, '-cc', 'cc')
|
|
i++
|
|
}
|
|
'-o' {
|
|
res.out_name = cmdline.option(args, '-o', '')
|
|
i++
|
|
}
|
|
else {
|
|
mut should_continue := false
|
|
for flag_with_param in list_of_flags_with_param {
|
|
if '-$flag_with_param' == arg {
|
|
should_continue = true
|
|
i++
|
|
break
|
|
}
|
|
}
|
|
if should_continue {
|
|
continue
|
|
}
|
|
if !arg.starts_with('-') && command == '' {
|
|
command = arg
|
|
command_pos = i
|
|
}
|
|
}
|
|
}
|
|
}
|
|
if command.ends_with('.v') || os.exists(command) {
|
|
res.path = command
|
|
}
|
|
else if command == 'run' {
|
|
res.is_run = true
|
|
res.path = args[command_pos+1]
|
|
res.run_args = args[command_pos+1..]
|
|
}
|
|
if res.is_verbose {
|
|
println('setting pref.path to "$res.path"')
|
|
}
|
|
res.fill_with_defaults()
|
|
return res, command
|
|
}
|
|
|
|
fn invoke_help_and_exit(remaining []string) {
|
|
match remaining.len {
|
|
0, 1 {
|
|
help.print_and_exit('default')
|
|
}
|
|
2 {
|
|
help.print_and_exit(remaining[1])
|
|
}
|
|
else {}
|
|
}
|
|
println('V Error: Expected only one help topic to be provided.')
|
|
println('For usage information, use `v help`.')
|
|
exit(1)
|
|
}
|
|
|
|
fn create_symlink() {
|
|
$if windows {
|
|
return
|
|
}
|
|
vexe := pref.vexe_path()
|
|
mut link_path := '/usr/local/bin/v'
|
|
mut ret := os.exec('ln -sf $vexe $link_path') or { panic(err) }
|
|
if ret.exit_code == 0 {
|
|
println('Symlink "$link_path" has been created')
|
|
}
|
|
else if os.system('uname -o | grep -q \'[A/a]ndroid\'') == 0 {
|
|
println('Failed to create symlink "$link_path". Trying again with Termux path for Android.')
|
|
link_path = '/data/data/com.termux/files/usr/bin/v'
|
|
ret = os.exec('ln -sf $vexe $link_path') or { panic(err) }
|
|
if ret.exit_code == 0 {
|
|
println('Symlink "$link_path" has been created')
|
|
} else {
|
|
println('Failed to create symlink "$link_path". Try again with sudo.')
|
|
}
|
|
} else {
|
|
println('Failed to create symlink "$link_path". Try again with sudo.')
|
|
}
|
|
}
|