2019-06-23 05:21:30 +03:00
|
|
|
// Copyright (c) 2019 Alexander Medvednikov. All rights reserved.
|
|
|
|
// Use of this source code is governed by an MIT license
|
|
|
|
// that can be found in the LICENSE file.
|
|
|
|
|
2019-10-13 16:37:43 +03:00
|
|
|
module compiler
|
2019-06-22 21:20:28 +03:00
|
|
|
|
2019-07-12 08:37:54 +03:00
|
|
|
import os
|
|
|
|
|
2019-06-22 21:20:28 +03:00
|
|
|
struct CGen {
|
|
|
|
out os.File
|
|
|
|
out_path string
|
2019-08-30 20:19:06 +03:00
|
|
|
//types []string
|
|
|
|
thread_fns []string
|
|
|
|
//buf strings.Builder
|
|
|
|
is_user bool
|
|
|
|
mut:
|
|
|
|
lines []string
|
2019-06-22 21:20:28 +03:00
|
|
|
typedefs []string
|
|
|
|
type_aliases []string
|
|
|
|
includes []string
|
|
|
|
thread_args []string
|
|
|
|
consts []string
|
|
|
|
fns []string
|
|
|
|
so_fns []string
|
|
|
|
consts_init []string
|
2019-07-29 19:21:36 +03:00
|
|
|
pass Pass
|
2019-07-26 17:45:16 +03:00
|
|
|
nogen bool
|
2019-10-28 23:50:58 +03:00
|
|
|
prev_tmps []string
|
2019-07-26 17:45:16 +03:00
|
|
|
tmp_line string
|
|
|
|
cur_line string
|
|
|
|
prev_line string
|
|
|
|
is_tmp bool
|
|
|
|
fn_main string
|
|
|
|
stash string
|
|
|
|
file string
|
|
|
|
line int
|
|
|
|
line_directives bool
|
2019-08-28 17:35:44 +03:00
|
|
|
cut_pos int
|
2019-06-22 21:20:28 +03:00
|
|
|
}
|
|
|
|
|
2019-09-01 22:51:16 +03:00
|
|
|
fn new_cgen(out_name_c string) &CGen {
|
2019-08-23 11:42:48 +03:00
|
|
|
path := out_name_c
|
2019-07-03 22:07:42 +03:00
|
|
|
out := os.create(path) or {
|
2019-08-28 17:35:44 +03:00
|
|
|
println('failed to create $path')
|
|
|
|
return &CGen{}
|
|
|
|
}
|
2019-06-22 21:20:28 +03:00
|
|
|
gen := &CGen {
|
2019-08-28 17:35:44 +03:00
|
|
|
out_path: path
|
|
|
|
out: out
|
|
|
|
//buf: strings.new_builder(10000)
|
2019-10-09 23:38:33 +03:00
|
|
|
lines: make(0, 1000, sizeof(string))
|
2019-06-22 21:20:28 +03:00
|
|
|
}
|
|
|
|
return gen
|
|
|
|
}
|
|
|
|
|
|
|
|
fn (g mut CGen) genln(s string) {
|
2019-07-29 19:21:36 +03:00
|
|
|
if g.nogen || g.pass != .main {
|
2019-06-22 21:20:28 +03:00
|
|
|
return
|
|
|
|
}
|
|
|
|
if g.is_tmp {
|
|
|
|
g.tmp_line = '$g.tmp_line $s\n'
|
|
|
|
return
|
|
|
|
}
|
|
|
|
g.cur_line = '$g.cur_line $s'
|
|
|
|
if g.cur_line != '' {
|
2019-07-26 17:45:16 +03:00
|
|
|
if g.line_directives && g.cur_line.trim_space() != '' {
|
2019-10-12 00:04:42 +03:00
|
|
|
if g.file.len > 0 && g.line > 0 {
|
|
|
|
g.lines << '\n#line $g.line "$g.file"'
|
|
|
|
}
|
2019-07-26 17:45:16 +03:00
|
|
|
}
|
2019-06-22 21:20:28 +03:00
|
|
|
g.lines << g.cur_line
|
|
|
|
g.prev_line = g.cur_line
|
|
|
|
g.cur_line = ''
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
fn (g mut CGen) gen(s string) {
|
2019-07-29 19:21:36 +03:00
|
|
|
if g.nogen || g.pass != .main {
|
2019-06-22 21:20:28 +03:00
|
|
|
return
|
|
|
|
}
|
|
|
|
if g.is_tmp {
|
|
|
|
g.tmp_line = '$g.tmp_line $s'
|
|
|
|
}
|
|
|
|
else {
|
|
|
|
g.cur_line = '$g.cur_line $s'
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2019-07-18 21:21:23 +03:00
|
|
|
fn (g mut CGen) resetln(s string) {
|
2019-07-29 19:21:36 +03:00
|
|
|
if g.nogen || g.pass != .main {
|
2019-07-18 21:21:23 +03:00
|
|
|
return
|
|
|
|
}
|
|
|
|
if g.is_tmp {
|
|
|
|
g.tmp_line = s
|
|
|
|
}
|
|
|
|
else {
|
|
|
|
g.cur_line = s
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2019-06-22 21:20:28 +03:00
|
|
|
fn (g mut CGen) save() {
|
|
|
|
s := g.lines.join('\n')
|
2019-06-30 17:11:55 +03:00
|
|
|
g.out.writeln(s)
|
2019-06-22 21:20:28 +03:00
|
|
|
g.out.close()
|
|
|
|
}
|
|
|
|
|
|
|
|
fn (g mut CGen) start_tmp() {
|
|
|
|
if g.is_tmp {
|
2019-10-28 23:50:58 +03:00
|
|
|
g.prev_tmps << g.tmp_line
|
2019-06-22 21:20:28 +03:00
|
|
|
}
|
|
|
|
// kg.tmp_lines_pos++
|
|
|
|
g.tmp_line = ''
|
|
|
|
g.is_tmp = true
|
|
|
|
}
|
|
|
|
|
|
|
|
fn (g mut CGen) end_tmp() string {
|
|
|
|
res := g.tmp_line
|
2019-10-28 23:50:58 +03:00
|
|
|
if g.prev_tmps.len > 0 {
|
|
|
|
g.tmp_line = g.prev_tmps.last()
|
|
|
|
g.prev_tmps = g.prev_tmps[0..g.prev_tmps.len-1]
|
|
|
|
} else {
|
|
|
|
g.tmp_line = ''
|
|
|
|
g.is_tmp = false
|
|
|
|
}
|
2019-06-22 21:20:28 +03:00
|
|
|
return res
|
|
|
|
}
|
|
|
|
|
2019-09-17 13:09:58 +03:00
|
|
|
fn (g &CGen) add_placeholder() int {
|
2019-06-22 21:20:28 +03:00
|
|
|
if g.is_tmp {
|
|
|
|
return g.tmp_line.len
|
|
|
|
}
|
|
|
|
return g.cur_line.len
|
|
|
|
}
|
|
|
|
|
2019-08-13 14:50:19 +03:00
|
|
|
fn (g mut CGen) start_cut() {
|
2019-08-28 17:35:44 +03:00
|
|
|
g.cut_pos = g.add_placeholder()
|
|
|
|
}
|
2019-08-13 14:50:19 +03:00
|
|
|
|
|
|
|
fn (g mut CGen) cut() string {
|
2019-08-28 17:35:44 +03:00
|
|
|
pos := g.cut_pos
|
|
|
|
g.cut_pos = 0
|
2019-08-13 14:50:19 +03:00
|
|
|
if g.is_tmp {
|
2019-10-27 10:03:15 +03:00
|
|
|
res := g.tmp_line[pos..]
|
|
|
|
g.tmp_line = g.tmp_line[..pos]
|
2019-08-28 17:35:44 +03:00
|
|
|
return res
|
2019-08-13 14:50:19 +03:00
|
|
|
}
|
2019-10-27 10:03:15 +03:00
|
|
|
res := g.cur_line[pos..]
|
|
|
|
g.cur_line = g.cur_line[..pos]
|
2019-08-28 17:35:44 +03:00
|
|
|
return res
|
|
|
|
}
|
2019-08-13 14:50:19 +03:00
|
|
|
|
2019-06-22 21:20:28 +03:00
|
|
|
fn (g mut CGen) set_placeholder(pos int, val string) {
|
2019-07-29 19:21:36 +03:00
|
|
|
if g.nogen || g.pass != .main {
|
2019-06-22 21:20:28 +03:00
|
|
|
return
|
|
|
|
}
|
|
|
|
// g.lines.set(pos, val)
|
|
|
|
if g.is_tmp {
|
2019-10-27 10:03:15 +03:00
|
|
|
left := g.tmp_line[..pos]
|
|
|
|
right := g.tmp_line[pos..]
|
2019-06-22 21:20:28 +03:00
|
|
|
g.tmp_line = '${left}${val}${right}'
|
|
|
|
return
|
|
|
|
}
|
2019-10-27 10:03:15 +03:00
|
|
|
left := g.cur_line[..pos]
|
|
|
|
right := g.cur_line[pos..]
|
2019-06-22 21:20:28 +03:00
|
|
|
g.cur_line = '${left}${val}${right}'
|
|
|
|
// g.genln('')
|
|
|
|
}
|
|
|
|
|
|
|
|
fn (g mut CGen) insert_before(val string) {
|
2019-10-07 01:31:01 +03:00
|
|
|
if g.nogen {
|
|
|
|
return
|
|
|
|
}
|
2019-08-28 17:35:44 +03:00
|
|
|
prev := g.lines[g.lines.len - 1]
|
|
|
|
g.lines[g.lines.len - 1] = '$prev \n $val \n'
|
2019-06-22 21:20:28 +03:00
|
|
|
}
|
|
|
|
|
|
|
|
fn (g mut CGen) register_thread_fn(wrapper_name, wrapper_text, struct_text string) {
|
|
|
|
for arg in g.thread_args {
|
|
|
|
if arg.contains(wrapper_name) {
|
|
|
|
return
|
|
|
|
}
|
|
|
|
}
|
|
|
|
g.thread_args << struct_text
|
|
|
|
g.thread_args << wrapper_text
|
|
|
|
}
|
|
|
|
|
2019-09-17 13:09:58 +03:00
|
|
|
fn (v &V) prof_counters() string {
|
2019-06-22 21:20:28 +03:00
|
|
|
mut res := []string
|
|
|
|
// Global fns
|
2019-06-30 23:44:15 +03:00
|
|
|
//for f in c.table.fns {
|
|
|
|
//res << 'double ${c.table.cgen_name(f)}_time;'
|
|
|
|
//}
|
2019-06-22 21:20:28 +03:00
|
|
|
// Methods
|
2019-09-01 14:31:43 +03:00
|
|
|
/*
|
2019-06-22 21:20:28 +03:00
|
|
|
for typ in c.table.types {
|
|
|
|
// println('')
|
|
|
|
for f in typ.methods {
|
|
|
|
// res << f.cgen_name()
|
|
|
|
res << 'double ${c.table.cgen_name(f)}_time;'
|
|
|
|
// println(f.cgen_name())
|
|
|
|
}
|
|
|
|
}
|
2019-09-01 14:31:43 +03:00
|
|
|
*/
|
2019-06-22 21:20:28 +03:00
|
|
|
return res.join(';\n')
|
|
|
|
}
|
|
|
|
|
2019-09-17 13:09:58 +03:00
|
|
|
fn (p &Parser) print_prof_counters() string {
|
2019-06-22 21:20:28 +03:00
|
|
|
mut res := []string
|
|
|
|
// Global fns
|
2019-06-30 23:44:15 +03:00
|
|
|
//for f in p.table.fns {
|
|
|
|
//counter := '${p.table.cgen_name(f)}_time'
|
|
|
|
//res << 'if ($counter) printf("%%f : $f.name \\n", $counter);'
|
|
|
|
//}
|
2019-06-22 21:20:28 +03:00
|
|
|
// Methods
|
2019-09-01 14:31:43 +03:00
|
|
|
/*
|
2019-06-22 21:20:28 +03:00
|
|
|
for typ in p.table.types {
|
|
|
|
// println('')
|
|
|
|
for f in typ.methods {
|
|
|
|
counter := '${p.table.cgen_name(f)}_time'
|
|
|
|
res << 'if ($counter) printf("%%f : ${p.table.cgen_name(f)} \\n", $counter);'
|
|
|
|
// res << 'if ($counter) printf("$f.name : %%f\\n", $counter);'
|
|
|
|
// res << f.cgen_name()
|
|
|
|
// res << 'double ${f.cgen_name()}_time;'
|
|
|
|
// println(f.cgen_name())
|
|
|
|
}
|
|
|
|
}
|
2019-09-01 14:31:43 +03:00
|
|
|
*/
|
2019-06-22 21:20:28 +03:00
|
|
|
return res.join(';\n')
|
|
|
|
}
|
|
|
|
|
|
|
|
fn (p mut Parser) gen_typedef(s string) {
|
2019-07-29 19:21:36 +03:00
|
|
|
if !p.first_pass() {
|
2019-06-22 21:20:28 +03:00
|
|
|
return
|
|
|
|
}
|
|
|
|
p.cgen.typedefs << s
|
|
|
|
}
|
|
|
|
|
|
|
|
fn (p mut Parser) gen_type_alias(s string) {
|
2019-07-29 19:21:36 +03:00
|
|
|
if !p.first_pass() {
|
2019-06-22 21:20:28 +03:00
|
|
|
return
|
|
|
|
}
|
|
|
|
p.cgen.type_aliases << s
|
|
|
|
}
|
|
|
|
|
|
|
|
fn (g mut CGen) add_to_main(s string) {
|
|
|
|
g.fn_main = g.fn_main + s
|
|
|
|
}
|
|
|
|
|
2019-07-10 14:27:35 +03:00
|
|
|
|
2019-09-23 00:51:59 +03:00
|
|
|
fn build_thirdparty_obj_file(path string, moduleflags []CFlag) {
|
2019-09-06 15:12:04 +03:00
|
|
|
obj_path := os.realpath(path)
|
2019-07-10 14:27:35 +03:00
|
|
|
if os.file_exists(obj_path) {
|
2019-08-28 17:35:44 +03:00
|
|
|
return
|
|
|
|
}
|
|
|
|
println('$obj_path not found, building it...')
|
2019-09-06 15:12:04 +03:00
|
|
|
parent := os.dir(obj_path)
|
2019-10-17 14:30:05 +03:00
|
|
|
files := os.ls(parent) or { panic(err) }
|
2019-08-28 17:35:44 +03:00
|
|
|
mut cfiles := ''
|
2019-07-10 14:27:35 +03:00
|
|
|
for file in files {
|
2019-08-28 17:35:44 +03:00
|
|
|
if file.ends_with('.c') {
|
2019-10-12 22:31:05 +03:00
|
|
|
cfiles += '"' + os.realpath( parent + os.path_separator + file ) + '" '
|
2019-08-28 17:35:44 +03:00
|
|
|
}
|
|
|
|
}
|
2019-08-23 11:42:48 +03:00
|
|
|
cc := find_c_compiler()
|
|
|
|
cc_thirdparty_options := find_c_compiler_thirdparty_options()
|
2019-09-23 00:51:59 +03:00
|
|
|
btarget := moduleflags.c_options_before_target()
|
|
|
|
atarget := moduleflags.c_options_after_target()
|
|
|
|
cmd := '$cc $cc_thirdparty_options $btarget -c -o "$obj_path" $cfiles $atarget '
|
2019-09-03 16:09:43 +03:00
|
|
|
res := os.exec(cmd) or {
|
|
|
|
println('failed thirdparty object build cmd: $cmd')
|
2019-09-24 00:40:34 +03:00
|
|
|
verror(err)
|
2019-08-29 03:30:17 +03:00
|
|
|
return
|
2019-08-08 08:30:05 +03:00
|
|
|
}
|
2019-10-28 12:43:57 +03:00
|
|
|
if res.exit_code != 0 {
|
|
|
|
println('failed thirdparty object build cmd: $cmd')
|
|
|
|
verror(res.output)
|
|
|
|
return
|
|
|
|
}
|
2019-08-28 17:35:44 +03:00
|
|
|
println(res.output)
|
|
|
|
}
|
2019-07-10 14:27:35 +03:00
|
|
|
|
2019-08-28 17:35:44 +03:00
|
|
|
fn os_name_to_ifdef(name string) string {
|
2019-10-24 19:19:03 +03:00
|
|
|
match name {
|
|
|
|
'windows' { return '_WIN32'}
|
|
|
|
'mac' { return '__APPLE__'}
|
|
|
|
'linux' { return '__linux__'}
|
|
|
|
'freebsd' { return '__FreeBSD__'}
|
|
|
|
'openbsd'{ return '__OpenBSD__'}
|
|
|
|
'netbsd'{ return '__NetBSD__'}
|
|
|
|
'dragonfly'{ return '__DragonFly__'}
|
|
|
|
'msvc'{ return '_MSC_VER'}
|
|
|
|
'android'{ return '__BIONIC__'}
|
|
|
|
'js' {return '_VJS'}
|
|
|
|
'solaris'{ return '__sun'}
|
2019-08-28 17:35:44 +03:00
|
|
|
}
|
2019-09-24 00:40:34 +03:00
|
|
|
verror('bad os ifdef name "$name"')
|
2019-08-29 03:30:17 +03:00
|
|
|
return ''
|
2019-08-28 17:35:44 +03:00
|
|
|
}
|
2019-08-13 14:50:19 +03:00
|
|
|
|
|
|
|
fn platform_postfix_to_ifdefguard(name string) string {
|
2019-10-23 08:18:44 +03:00
|
|
|
s := match name {
|
|
|
|
'.v' { '' }// no guard needed
|
|
|
|
'_win.v', '_windows.v' { '#ifdef _WIN32' }
|
|
|
|
'_nix.v' { '#ifndef _WIN32' }
|
|
|
|
'_lin.v', '_linux.v' { '#ifdef __linux__' }
|
|
|
|
'_mac.v', '_darwin.v' { '#ifdef __APPLE__' }
|
|
|
|
'_solaris.v' { '#ifdef __sun' }
|
|
|
|
else {
|
|
|
|
|
|
|
|
//verror('bad platform_postfix "$name"')
|
|
|
|
// TODO
|
|
|
|
''
|
|
|
|
}
|
2019-09-06 15:12:04 +03:00
|
|
|
}
|
2019-10-23 08:18:44 +03:00
|
|
|
if s == '' {
|
|
|
|
verror('bad platform_postfix "$name"')
|
|
|
|
}
|
|
|
|
return s
|
2019-08-13 14:50:19 +03:00
|
|
|
}
|
|
|
|
|
2019-08-28 17:35:44 +03:00
|
|
|
// C struct definitions, ordered
|
2019-08-29 01:52:32 +03:00
|
|
|
// Sort the types, make sure types that are referenced by other types
|
|
|
|
// are added before them.
|
2019-09-17 13:09:58 +03:00
|
|
|
fn (v &V) type_definitions() string {
|
2019-09-04 19:49:53 +03:00
|
|
|
mut types := []Type // structs that need to be sorted
|
|
|
|
mut builtin_types := []Type // builtin types
|
2019-09-01 14:31:43 +03:00
|
|
|
// builtin types need to be on top
|
|
|
|
builtins := ['string', 'array', 'map', 'Option']
|
|
|
|
for builtin in builtins {
|
|
|
|
typ := v.table.typesmap[builtin]
|
|
|
|
builtin_types << typ
|
|
|
|
}
|
2019-09-04 19:49:53 +03:00
|
|
|
// everything except builtin will get sorted
|
|
|
|
for t_name, t in v.table.typesmap {
|
|
|
|
if t_name in builtins {
|
2019-08-29 01:52:32 +03:00
|
|
|
continue
|
|
|
|
}
|
|
|
|
types << t
|
2019-08-28 17:35:44 +03:00
|
|
|
}
|
2019-09-04 19:49:53 +03:00
|
|
|
// sort structs
|
|
|
|
types_sorted := sort_structs(types)
|
2019-08-28 17:35:44 +03:00
|
|
|
// Generate C code
|
2019-09-17 22:41:58 +03:00
|
|
|
res := types_to_c(builtin_types,v.table) + '\n//----\n' +
|
2019-09-04 19:49:53 +03:00
|
|
|
types_to_c(types_sorted, v.table)
|
2019-09-17 22:41:58 +03:00
|
|
|
return res
|
2019-08-29 01:52:32 +03:00
|
|
|
}
|
|
|
|
|
2019-09-03 19:11:21 +03:00
|
|
|
// sort structs by dependant fields
|
2019-09-04 10:35:08 +03:00
|
|
|
fn sort_structs(types []Type) []Type {
|
|
|
|
mut dep_graph := new_dep_graph()
|
|
|
|
// types name list
|
2019-09-03 19:11:21 +03:00
|
|
|
mut type_names := []string
|
2019-09-04 10:35:08 +03:00
|
|
|
for t in types {
|
|
|
|
type_names << t.name
|
2019-09-03 19:11:21 +03:00
|
|
|
}
|
|
|
|
// loop over types
|
2019-09-04 10:35:08 +03:00
|
|
|
for t in types {
|
2019-09-03 19:11:21 +03:00
|
|
|
// create list of deps
|
2019-09-04 10:35:08 +03:00
|
|
|
mut field_deps := []string
|
2019-09-03 19:11:21 +03:00
|
|
|
for field in t.fields {
|
2019-09-04 10:35:08 +03:00
|
|
|
// skip if not in types list or already in deps
|
|
|
|
if !(field.typ in type_names) || field.typ in field_deps {
|
2019-09-03 19:11:21 +03:00
|
|
|
continue
|
|
|
|
}
|
2019-09-04 10:35:08 +03:00
|
|
|
field_deps << field.typ
|
2019-09-03 19:11:21 +03:00
|
|
|
}
|
|
|
|
// add type and dependant types to graph
|
2019-09-04 10:35:08 +03:00
|
|
|
dep_graph.add(t.name, field_deps)
|
|
|
|
}
|
|
|
|
// sort graph
|
|
|
|
dep_graph_sorted := dep_graph.resolve()
|
|
|
|
if !dep_graph_sorted.acyclic {
|
2019-10-24 13:23:11 +03:00
|
|
|
verror('cgen.sort_structs(): the following structs form a dependency cycle:\n' +
|
2019-10-24 07:53:39 +03:00
|
|
|
dep_graph_sorted.display_cycles() +
|
2019-10-24 11:18:50 +03:00
|
|
|
'\nyou can solve this by making one or both of the dependant struct fields references, eg: field &MyStruct' +
|
2019-10-24 07:53:39 +03:00
|
|
|
'\nif you feel this is an error, please create a new issue here: https://github.com/vlang/v/issues and tag @joe-conigliaro')
|
2019-09-04 10:35:08 +03:00
|
|
|
}
|
|
|
|
// sort types
|
|
|
|
mut types_sorted := []Type
|
|
|
|
for node in dep_graph_sorted.nodes {
|
|
|
|
for t in types {
|
2019-09-03 19:11:21 +03:00
|
|
|
if t.name == node.name {
|
2019-09-04 10:35:08 +03:00
|
|
|
types_sorted << t
|
2019-08-29 01:52:32 +03:00
|
|
|
continue
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2019-09-04 10:35:08 +03:00
|
|
|
return types_sorted
|
2019-08-29 01:52:32 +03:00
|
|
|
}
|