checker: fix `x := match enumexpr { .case2 { fn1 } .case2 { fn2} }`, where fn1 and fn2 have compatible signature

pull/8499/head
Delyan Angelov 2021-02-01 20:57:38 +02:00
parent e3c2604338
commit fab7b9d9d9
No known key found for this signature in database
GPG Key ID: 66886C0F12D595ED
2 changed files with 68 additions and 2 deletions

View File

@ -4071,8 +4071,11 @@ pub fn (mut c Checker) match_expr(mut node ast.MatchExpr) table.Type {
ret_type = expr_type ret_type = expr_type
stmt.typ = ret_type stmt.typ = ret_type
} else if node.is_expr && ret_type != expr_type { } else if node.is_expr && ret_type != expr_type {
sym := c.table.get_type_symbol(ret_type) if !c.check_types(ret_type, expr_type) {
c.error('return type mismatch, it should be `$sym.name`', stmt.expr.position()) ret_sym := c.table.get_type_symbol(ret_type)
c.error('return type mismatch, it should be `$ret_sym.name`',
stmt.expr.position())
}
} }
} }
else { else {

View File

@ -0,0 +1,63 @@
import math
const tau = 2 * math.pi
struct Note {
mut:
freq f32
vol f32
step int
paused bool
}
enum WaveKind {
sine
sawtooth
}
struct Config {
wave_kind WaveKind
}
fn sawtooth(note &Note, time f32, amp f32) f32 {
t := time * note.freq
f := t - int(t)
return f32(2 * (f - 0.5)) * (amp / 2)
}
fn sine(note &Note, time f32, amp f32) f32 {
return math.sinf(tau * time * note.freq) * amp
}
pub fn new_context(cfg Config) fn (&Note, f32, f32) f32 {
// Note, that here `sine` and `sawtooth`,
// are different functions, but they do have
// a compatible signature, so `next_fn` can
// become either one of them:
next_fn := match cfg.wave_kind {
.sine { sine }
.sawtooth { sawtooth }
}
return next_fn
}
fn test_match_expression_returning_fns() {
note := &Note{
freq: 432
vol: 80
step: 3
paused: false
}
x := new_context(wave_kind: .sawtooth)
assert '$x' == 'fn (Note, f32, f32) f32'
assert math.abs(x(note, 0, 0.5) + 0.25) < 0.001
assert math.abs(x(note, 0.4, 0.5) - 0.15) < 0.001
assert math.abs(x(note, 0.7, 0.5) + 0.05) < 0.001
y := new_context(wave_kind: .sine)
assert '$y' == 'fn (Note, f32, f32) f32'
assert math.abs(y(note, 0, 0.5)) - 0.475 < 0.0001
assert math.abs(y(note, 0.4, 0.5)) - 0.4755 < 0.0001
assert math.abs(y(note, 0.7, 0.5)) - 0.2939 < 0.0001
}