Re: C68 & detecting vsync
Posted: Sun Feb 13, 2022 8:55 am
The value is increased on every V-Sync, so the wait should be the same (except a little bit later as the OS handlers will first execute normally)
RIP Sir Clive Sinclair 1940 - 2021
https://theqlforum.com/
Code: Select all
void screen_Vsync(unsigned char wait){
volatile unsigned int c = 0;
poll_init(&c);
while (c < wait){
}
}
Code: Select all
draw_Box(screen, x1, y1, w, h, 0, PIXEL_WHITE, PIXEL_WHITE, MODE_PIXEL_SET);
draw_String(screen, col, y, max_chars, 1, 0, screen->font_8x8, PIXEL_BLACK, c, MODE_PIXEL_XOR);
screen_Vsync(10);
for (i = 0; i < 4; i++){
// Invert text and highlight
draw_Box(screen, x1, y1, w, h, 0, PIXEL_BLACK, PIXEL_BLACK, MODE_PIXEL_SET);
draw_String(screen, col, y, max_chars, 1, 0, screen->font_8x8, fill, c, MODE_PIXEL_SET);
screen_Vsync(5);
// Normal text
draw_Box(screen, x1, y1, w, h, 0, PIXEL_WHITE, PIXEL_WHITE, MODE_PIXEL_SET);
draw_String(screen, col, y, max_chars, 1, 0, screen->font_8x8, PIXEL_BLACK, c, MODE_PIXEL_XOR);
screen_Vsync(5);
}
Don't call poll_init on every cycle. It will instantiate a new polling interrupt routine every time (that's the reason for both your memory leak and the slowdown).megatron-uk wrote:Actually, I'm still seeing some odd behaviour with this poll/wait routine... but I don't know if it's an interaction with sQLux that is causing it.
With the poll/wait routine in, I observe that each time I call it, the delay gets shorter and shorter relative to the same wait value I pass it, an initial value of 10 - which is a noticeable delay at first, ultimately becomes almost imperceptible after being called more than 5 or 6 times.
Also, the poll/wait routine appears to be causing a memory leak - my free memory pool available to malloc from decreases by (approximately) 512 bytes after each call as I suspect there is a new job being created on each call.
Should there be a call to remove the poll/wait routine, or somehow reinstantiate the existing job?
The implementation is a very basic one:
Code: Select all
void screen_Vsync(unsigned char wait){ volatile unsigned int c = 0; poll_init(&c); while (c < wait){ } }
Oh that works brilliantly, thank you!tofro wrote:
Don't call poll_init on every cycle. It will instantiate a new polling interrupt routine every time (that's the reason for both your memory leak and the slowdown).
Call poll_init only once, then re-set the counter c to zero when you wait.
Thanks so much for that sample code! I don't think I'd ever have any hope on getting a simple clock ISR going!mk79 wrote: Sat Feb 12, 2022 7:40 pm Looks nice.
You can do the same thing as sleep (calling mt_susjb) to get down to 20ms granularity.megatron-uk wrote:However, the game is a bit flat looking; I haven't yet got any animation, loading icons, etc... as I'm finding it difficult to find a method to synchronise / wait on display refresh - the C68 standard library only appears to have a second-granularity sleep() and nothing else that would help.You will never read anything but 0 as these will always be handled by the appropriate interrupt handlers. For your bit-twiddling solution you would need to disable interrupts and it will never work on advanced emulators like QPC.I see that there is some documentation on https://www.chibiakumas.com/68000/sinclairql.php for reading the state of the vsync interrupt, but I am hampered by the fact that C68 (at least in the xtc68 incarnation) doesn't appear to support inline assembly.
The clean thing to do is to link in a polled interrupt service routine and for example increase an integer in it that you can busy-wait on in the main job. This is a bit tricky to do, especially as the routine must be removed when the job is killed, otherwise the system will crash. Explaining the details that must be observed is too time consuming, so I wrote some example code you can try out.