stopwatch: auto_start option, make pause cumulative

pull/5132/head
JalonSolov 2020-05-30 03:20:54 -04:00 committed by GitHub
parent 077e06b44e
commit b7dc5b2f7b
No known key found for this signature in database
GPG Key ID: 4AEE18F83AFDEB23
7 changed files with 52 additions and 36 deletions

View File

@ -80,7 +80,7 @@ fn measure(cmd string) int {
exec(cmd) exec(cmd)
} }
println('Building...') println('Building...')
sw := time.new_stopwatch() sw := time.new_stopwatch({})
exec(cmd) exec(cmd)
return int(sw.elapsed().milliseconds()) return int(sw.elapsed().milliseconds())
} }

View File

@ -1494,7 +1494,7 @@ You can also use stopwatches to measure just portions of your code explicitly:
```v ```v
import time import time
fn main(){ fn main(){
sw := time.new_stopwatch() sw := time.new_stopwatch({})
println('Hello world') println('Hello world')
println('Greeting the world took: ${sw.elapsed().nanoseconds()}ns') println('Greeting the world took: ${sw.elapsed().nanoseconds()}ns')
} }

View File

@ -70,14 +70,14 @@ pub mut:
pub fn new_benchmark() Benchmark { pub fn new_benchmark() Benchmark {
return Benchmark{ return Benchmark{
bench_timer: time.new_stopwatch() bench_timer: time.new_stopwatch({})
verbose: true verbose: true
} }
} }
pub fn new_benchmark_no_cstep() Benchmark { pub fn new_benchmark_no_cstep() Benchmark {
return Benchmark{ return Benchmark{
bench_timer: time.new_stopwatch() bench_timer: time.new_stopwatch({})
verbose: true verbose: true
no_cstep: true no_cstep: true
} }
@ -85,7 +85,7 @@ pub fn new_benchmark_no_cstep() Benchmark {
pub fn new_benchmark_pointer() &Benchmark { pub fn new_benchmark_pointer() &Benchmark {
return &Benchmark{ return &Benchmark{
bench_timer: time.new_stopwatch() bench_timer: time.new_stopwatch({})
verbose: true verbose: true
} }
} }

View File

@ -49,7 +49,7 @@ fn elog(r &live.LiveReloadInfo, s string){
} }
fn compile_and_reload_shared_lib(r mut live.LiveReloadInfo) ?bool { fn compile_and_reload_shared_lib(r mut live.LiveReloadInfo) ?bool {
sw := time.new_stopwatch() sw := time.new_stopwatch({})
new_lib_path := compile_lib(mut r) or { new_lib_path := compile_lib(mut r) or {
return error('errors while compiling $r.original') return error('errors while compiling $r.original')
} }
@ -63,7 +63,7 @@ fn compile_lib(r mut live.LiveReloadInfo) ?string {
new_lib_path, new_lib_path_with_extension := current_shared_library_path(mut r) new_lib_path, new_lib_path_with_extension := current_shared_library_path(mut r)
cmd := '$r.vexe $r.vopts -o $new_lib_path $r.original' cmd := '$r.vexe $r.vopts -o $new_lib_path $r.original'
elog(r,'> compilation cmd: $cmd') elog(r,'> compilation cmd: $cmd')
cwatch := time.new_stopwatch() cwatch := time.new_stopwatch({})
recompilation_result := os.exec( cmd ) or { recompilation_result := os.exec( cmd ) or {
eprintln('recompilation failed') eprintln('recompilation failed')
return none return none

View File

@ -3,50 +3,65 @@
// that can be found in the LICENSE file. // that can be found in the LICENSE file.
module time module time
pub struct StopWatchOptions {
auto_start bool = true
}
pub struct StopWatch { pub struct StopWatch {
mut: mut:
pause_time u64 elapsed u64
pub mut: pub mut:
start u64 start u64
end u64 end u64
} }
pub fn new_stopwatch() StopWatch { pub fn new_stopwatch(opts StopWatchOptions) StopWatch {
return StopWatch{pause_time: 0, start: time.sys_mono_now(), end: 0} mut initial := u64(0)
if opts.auto_start {
initial = time.sys_mono_now()
}
return StopWatch{elapsed: 0, start: initial, end: 0}
} }
// start Starts the timer. If the timer was paused, restarts counting. // start Starts the timer. If the timer was paused, restarts counting.
pub fn (mut t StopWatch) start() { pub fn (mut t StopWatch) start() {
if t.pause_time == 0 {
t.start = time.sys_mono_now() t.start = time.sys_mono_now()
} else {
t.start += time.sys_mono_now() - t.pause_time
}
t.end = 0 t.end = 0
t.pause_time = 0
} }
pub fn (mut t StopWatch) restart() { pub fn (mut t StopWatch) restart() {
t.end = 0
t.pause_time = 0
t.start = time.sys_mono_now() t.start = time.sys_mono_now()
t.end = 0
t.elapsed = 0
} }
pub fn (mut t StopWatch) stop() { pub fn (mut t StopWatch) stop() {
t.end = time.sys_mono_now() t.end = time.sys_mono_now()
t.pause_time = 0
} }
pub fn (mut t StopWatch) pause() { pub fn (mut t StopWatch) pause() {
t.pause_time = time.sys_mono_now() if t.start > 0 {
t.end = t.pause_time // so elapsed keeps track of actual running time if t.end == 0 {
t.elapsed += time.sys_mono_now() - t.start
} else {
t.elapsed += t.end - t.start
}
}
t.start = 0
} }
// elapsed returns the Duration since the last start call // elapsed returns the Duration since the last start call
pub fn (t StopWatch) elapsed() Duration { pub fn (t StopWatch) elapsed() Duration {
if t.start > 0 {
if t.end == 0 { if t.end == 0 {
return Duration(time.sys_mono_now() - t.start) return Duration(time.sys_mono_now() - t.start + t.elapsed)
} else { } else {
return Duration(t.end - t.start) return Duration(t.end - t.start + t.elapsed)
} }
}
return Duration(t.elapsed)
} }

View File

@ -4,7 +4,7 @@ import time
// time than you have specified. To avoid false positives from CI test // time than you have specified. To avoid false positives from CI test
// failures, some of the asserts will be run only if you pass `-d stopwatch` // failures, some of the asserts will be run only if you pass `-d stopwatch`
fn test_stopwatch_works_as_intended() { fn test_stopwatch_works_as_intended() {
sw := time.new_stopwatch() sw := time.new_stopwatch({})
// sample code that you want to measure: // sample code that you want to measure:
println('Hello world') println('Hello world')
time.sleep_ms(1) time.sleep_ms(1)
@ -14,20 +14,21 @@ fn test_stopwatch_works_as_intended() {
} }
fn test_stopwatch_time_between_pause_and_start_should_be_skipped_in_elapsed() { fn test_stopwatch_time_between_pause_and_start_should_be_skipped_in_elapsed() {
sw := time.new_stopwatch() println('Testing pause function')
sw := time.new_stopwatch({})
time.sleep_ms(10) // A time.sleep_ms(10) // A
// eprintln('${sw.elapsed().milliseconds()}ms') eprintln('Elapsed after 10ms nap: ${sw.elapsed().milliseconds()}ms')
assert sw.elapsed().milliseconds() >= 10 assert sw.elapsed().milliseconds() >= 10
sw.pause() sw.pause()
time.sleep_ms(10) time.sleep_ms(10)
// eprintln('${sw.elapsed().milliseconds()}ms') eprintln('Elapsed after pause and another 10ms nap: ${sw.elapsed().milliseconds()}ms')
assert sw.elapsed().milliseconds() >= 10 assert sw.elapsed().milliseconds() >= 10
$if stopwatch ? { $if stopwatch ? {
assert sw.elapsed().milliseconds() < 20 assert sw.elapsed().milliseconds() < 20
} }
sw.start() sw.start()
time.sleep_ms(10) // B time.sleep_ms(10) // B
// Here, sw.elapsed() should be ~10ms (from A) + ~10ms (from B) >= 20ms eprintln('Elapsed after resume and another 10ms nap: ${sw.elapsed().milliseconds()}ms')
assert sw.elapsed().milliseconds() >= 20 assert sw.elapsed().milliseconds() >= 20
$if stopwatch ? { $if stopwatch ? {
assert sw.elapsed().milliseconds() < 30 assert sw.elapsed().milliseconds() < 30

View File

@ -29,7 +29,7 @@ pub fn compile(command string, pref &pref.Preferences) {
println('builder.compile() pref:') println('builder.compile() pref:')
// println(pref) // println(pref)
} }
mut sw := time.new_stopwatch() mut sw := time.new_stopwatch({})
match pref.backend { match pref.backend {
.c { b.compile_c() } .c { b.compile_c() }
.js { b.compile_js() } .js { b.compile_js() }