string cloning; fix `foo.str += 's'`

pull/3064/head
Alexander Medvednikov 2019-12-12 04:09:31 +03:00
parent e182274fe7
commit 576618d8cc
6 changed files with 34 additions and 23 deletions

View File

@ -76,18 +76,6 @@ pub fn panic(s string) {
C.exit(1) C.exit(1)
} }
pub fn println(s string) {
// Should never happen
if isnil(s.str) {
panic('println(NIL)')
}
$if windows {
C._putws(s.to_wide())
} $else {
C.printf('%.*s\n', s.len, s.str)
}
}
pub fn eprintln(s string) { pub fn eprintln(s string) {
if isnil(s.str) { if isnil(s.str) {
panic('eprintln(NIL)') panic('eprintln(NIL)')

View File

@ -4,6 +4,10 @@
module builtin module builtin
pub fn println(s string) {
C.printf('%.*s\n', s.len, s.str)
}
fn print_backtrace_skipping_top_frames_msvc(skipframes int) bool { fn print_backtrace_skipping_top_frames_msvc(skipframes int) bool {
println('not implemented, see builtin_windows.v') println('not implemented, see builtin_windows.v')
return false return false

View File

@ -133,3 +133,8 @@ fn print_backtrace_skipping_top_frames_nix(skipframes int) bool {
println('not implemented, see builtin_nix.v') println('not implemented, see builtin_nix.v')
return false return false
} }
pub fn println(s string) {
C._putws(s.to_wide())
}

View File

@ -87,7 +87,7 @@ fn (v mut V) cc() {
//'-Wno-unused-but-set-variable', //'-Wno-unused-but-set-variable',
'-Wno-unused-parameter', '-Wno-unused-parameter',
'-Wno-unused-result', '-Wno-unused-result',
'-Wunused-function', '-Wno-unused-function',
'-Wno-missing-braces', '-Wno-missing-braces',
'-Wno-unused-label'] '-Wno-unused-label']
// TCC on Linux by default, unless -cc was provided // TCC on Linux by default, unless -cc was provided

View File

@ -1163,6 +1163,10 @@ fn (p mut Parser) close_scope() {
fn (p mut Parser) free_var(v Var) { fn (p mut Parser) free_var(v Var) {
// Clean up memory, only do this if -autofree was passed for now // Clean up memory, only do this if -autofree was passed for now
//if p.fileis('mem.v') {println('free_var() $v.name')} //if p.fileis('mem.v') {println('free_var() $v.name')}
//println(p.cur_fn.name)
if p.cur_fn.name in ['add', 'clone', 'free'] {
return
}
mut free_fn := 'free' mut free_fn := 'free'
if v.typ.starts_with('array_') { if v.typ.starts_with('array_') {
free_fn = 'v_array_free' free_fn = 'v_array_free'
@ -1181,11 +1185,19 @@ fn (p mut Parser) free_var(v Var) {
if p.returns { if p.returns {
// Don't free a variable that's being returned // Don't free a variable that's being returned
if !v.is_returned && v.typ != 'FILE*' { //!v.is_c { if !v.is_returned && v.typ != 'FILE*' { //!v.is_c {
prev_line := p.cgen.lines[p.cgen.lines.len-2] //p.cgen.cur_line = '/* free */' + p.cgen.cur_line
p.cgen.lines[p.cgen.lines.len-2] = //p.cgen.set_placeholder(0, '/*free2*/')
'$free_fn ($v.name); /* :) close_scope free $v.typ */' + prev_line prev_line := p.cgen.lines[p.cgen.lines.len-1]
free := '$free_fn ($v.name); /* :) close_scope free $v.typ */'
p.cgen.lines[p.cgen.lines.len-1] = free + '\n' + prev_line
//'$free_fn ($v.name); /* :) close_scope free $v.typ */\n' + prev_line
} }
} else { } else if p.mod != 'strings' { //&& p.mod != 'builtin' {
/*
prev_line := p.cgen.lines[p.cgen.lines.len-1]
free := '$free_fn ($v.name); /* :) close_scope free $v.typ */'
p.cgen.lines[p.cgen.lines.len-1] = free + '\n' + prev_line
*/
//if p.fileis('mem.v') {println(v.name)} //if p.fileis('mem.v') {println(v.name)}
p.genln('$free_fn ($v.name); // close_scope free') p.genln('$free_fn ($v.name); // close_scope free')
} }
@ -1359,7 +1371,7 @@ fn (p mut Parser) assign_statement(v Var, ph int, is_map bool) {
if v.is_arg { if v.is_arg {
if p.cur_fn.args.len > 0 && p.cur_fn.args[0].name == v.name { if p.cur_fn.args.len > 0 && p.cur_fn.args[0].name == v.name {
println('make the receiver `$v.name` mutable: println('make the receiver `$v.name` mutable:
fn ($v.name mut $v.typ) $p.cur_fn.name (...) { fn ($v.name mut $v.typ) ${p.cur_fn.name}(...) {
') ')
} }
} }
@ -1368,8 +1380,8 @@ fn ($v.name mut $v.typ) $p.cur_fn.name (...) {
if !v.is_changed { if !v.is_changed {
p.mark_var_changed(v) p.mark_var_changed(v)
} }
is_str := v.typ == 'string' is_str := p.expected_type == 'string'
is_ustr := v.typ == 'ustring' is_ustr := p.expected_type == 'ustring'
match tok { match tok {
.assign { .assign {
if !is_map && !p.is_empty_c_struct_init { if !is_map && !p.is_empty_c_struct_init {
@ -1378,7 +1390,8 @@ fn ($v.name mut $v.typ) $p.cur_fn.name (...) {
} }
.plus_assign { .plus_assign {
if is_str && !p.is_js { if is_str && !p.is_js {
p.gen('= string_add($v.name, ')// TODO can't do `foo.bar += '!'` expr := p.cgen.cur_line
p.gen('= string_add($expr, ')
} }
else if is_ustr { else if is_ustr {
p.gen('= ustring_add($v.name, ') p.gen('= ustring_add($v.name, ')

View File

@ -45,7 +45,8 @@ pub fn dice_coefficient(s1, s2 string) f32 {
mut first_bigrams := map[string]int mut first_bigrams := map[string]int
for i := 0; i < a.len-1; i++ { for i := 0; i < a.len-1; i++ {
bigram := a[i..i+2] bigram := a[i..i+2]
q := if bigram in first_bigrams { first_bigrams[bigram]+1 } else { 1 } q := if bigram in first_bigrams {
first_bigrams[bigram]+1 } else { 1 }
first_bigrams[bigram] = q first_bigrams[bigram] = q
} }
mut intersection_size := 0 mut intersection_size := 0