Added real-time clock functions to the cx16 library.
This commit is contained in:
parent
f067c4530f
commit
b56ba8f073
6 changed files with 229 additions and 48 deletions
36
libsrc/cx16/getres.s
Normal file
36
libsrc/cx16/getres.s
Normal file
|
@ -0,0 +1,36 @@
|
||||||
|
;
|
||||||
|
; 2019-12-26, Greg King
|
||||||
|
;
|
||||||
|
; int __fastcall__ clock_getres (clockid_t clk_id, struct timespec *res);
|
||||||
|
;
|
||||||
|
|
||||||
|
.include "time.inc"
|
||||||
|
|
||||||
|
.importzp ptr1
|
||||||
|
.import incsp1, return0
|
||||||
|
|
||||||
|
|
||||||
|
;----------------------------------------------------------------------------
|
||||||
|
|
||||||
|
.proc _clock_getres
|
||||||
|
|
||||||
|
sta ptr1
|
||||||
|
stx ptr1+1
|
||||||
|
|
||||||
|
ldy #.sizeof(timespec) - 1
|
||||||
|
@L1: lda time,y
|
||||||
|
sta (ptr1),y
|
||||||
|
dey
|
||||||
|
bpl @L1
|
||||||
|
|
||||||
|
jsr incsp1
|
||||||
|
jmp return0
|
||||||
|
|
||||||
|
.endproc
|
||||||
|
|
||||||
|
;----------------------------------------------------------------------------
|
||||||
|
; timespec struct with tv_nsec set to approximately 1/60 of a second
|
||||||
|
.rodata
|
||||||
|
|
||||||
|
time: .dword 0
|
||||||
|
.dword 17 * 1000 * 1000
|
56
libsrc/cx16/gettime.s
Normal file
56
libsrc/cx16/gettime.s
Normal file
|
@ -0,0 +1,56 @@
|
||||||
|
;
|
||||||
|
; 2019-12-27, Greg King
|
||||||
|
;
|
||||||
|
; int __fastcall__ clock_gettime (clockid_t clk_id, struct timespec *tp);
|
||||||
|
;
|
||||||
|
|
||||||
|
.include "time.inc"
|
||||||
|
.include "cx16.inc"
|
||||||
|
|
||||||
|
.import pushax, pusheax, tosmul0ax, steaxspidx, incsp1, return0
|
||||||
|
.import TM, load_jiffy
|
||||||
|
.import CLOCK_GET_DATE_TIME
|
||||||
|
|
||||||
|
|
||||||
|
;----------------------------------------------------------------------------
|
||||||
|
|
||||||
|
.proc _clock_gettime
|
||||||
|
|
||||||
|
jsr pushax
|
||||||
|
jsr pushax
|
||||||
|
|
||||||
|
jsr CLOCK_GET_DATE_TIME
|
||||||
|
|
||||||
|
lda gREG::r0L
|
||||||
|
sta TM + tm::tm_year
|
||||||
|
lda gREG::r0H
|
||||||
|
dec a
|
||||||
|
sta TM + tm::tm_mon
|
||||||
|
lda gREG::r1L
|
||||||
|
sta TM + tm::tm_mday
|
||||||
|
|
||||||
|
lda gREG::r1H
|
||||||
|
sta TM + tm::tm_hour
|
||||||
|
lda gREG::r2L
|
||||||
|
sta TM + tm::tm_min
|
||||||
|
lda gREG::r2H
|
||||||
|
sta TM + tm::tm_sec
|
||||||
|
|
||||||
|
lda #<TM
|
||||||
|
ldx #>TM
|
||||||
|
jsr _mktime
|
||||||
|
ldy #timespec::tv_sec
|
||||||
|
jsr steaxspidx ; Pops address pushed by 2. pushax
|
||||||
|
|
||||||
|
jsr load_jiffy
|
||||||
|
jsr pusheax
|
||||||
|
lda gREG::r3L
|
||||||
|
ldx #>$0000
|
||||||
|
jsr tosmul0ax
|
||||||
|
ldy #timespec::tv_nsec
|
||||||
|
jsr steaxspidx ; Pops address pushed by 1. pushax
|
||||||
|
|
||||||
|
jsr incsp1
|
||||||
|
jmp return0
|
||||||
|
|
||||||
|
.endproc
|
55
libsrc/cx16/settime.s
Normal file
55
libsrc/cx16/settime.s
Normal file
|
@ -0,0 +1,55 @@
|
||||||
|
;
|
||||||
|
; 2019-12-27, Greg King
|
||||||
|
;
|
||||||
|
; int __fastcall__ clock_settime (clockid_t clk_id, const struct timespec *tp);
|
||||||
|
;
|
||||||
|
|
||||||
|
.include "time.inc"
|
||||||
|
.include "cx16.inc"
|
||||||
|
|
||||||
|
.importzp ptr1
|
||||||
|
.import pushax, pusheax, ldax0sp, ldeaxidx
|
||||||
|
.import tosdiveax, incsp3, return0
|
||||||
|
.import load_jiffy
|
||||||
|
.import CLOCK_SET_DATE_TIME
|
||||||
|
|
||||||
|
|
||||||
|
.macro COPY reg, offset
|
||||||
|
ldy #offset
|
||||||
|
lda (ptr1),y
|
||||||
|
sta gREG::reg
|
||||||
|
.endmac
|
||||||
|
|
||||||
|
;----------------------------------------------------------------------------
|
||||||
|
|
||||||
|
.proc _clock_settime
|
||||||
|
|
||||||
|
jsr pushax
|
||||||
|
|
||||||
|
.assert timespec::tv_sec = 0, error
|
||||||
|
jsr _localtime
|
||||||
|
sta ptr1
|
||||||
|
stx ptr1+1
|
||||||
|
|
||||||
|
COPY r0L, tm::tm_year
|
||||||
|
COPY r0H, tm::tm_mon
|
||||||
|
inc gREG::r0H
|
||||||
|
COPY r1L, tm::tm_mday
|
||||||
|
COPY r1H, tm::tm_hour
|
||||||
|
COPY r2L, tm::tm_min
|
||||||
|
COPY r2H, tm::tm_sec
|
||||||
|
|
||||||
|
jsr ldax0sp ; Get tp
|
||||||
|
ldy #timespec::tv_nsec+3
|
||||||
|
jsr ldeaxidx ; Get nanoseconds
|
||||||
|
jsr pusheax
|
||||||
|
jsr load_jiffy
|
||||||
|
jsr tosdiveax
|
||||||
|
sta gREG::r3L ; Put number of jiffies
|
||||||
|
|
||||||
|
jsr CLOCK_SET_DATE_TIME
|
||||||
|
|
||||||
|
jsr incsp3
|
||||||
|
jmp return0
|
||||||
|
|
||||||
|
.endproc
|
|
@ -1,16 +0,0 @@
|
||||||
;
|
|
||||||
; 2019-11-05, Greg King
|
|
||||||
;
|
|
||||||
|
|
||||||
.export ST: zp
|
|
||||||
|
|
||||||
.segment "EXTZP": zp
|
|
||||||
|
|
||||||
; This is a temporary hack.
|
|
||||||
|
|
||||||
; A zero-page copy of the IEC status byte.
|
|
||||||
; This is needed because the Commander X16's Kernal's status
|
|
||||||
; variable was moved out of the zero page. But, the common
|
|
||||||
; CBM file function modules import this as a zero-page variable.
|
|
||||||
|
|
||||||
ST: .res 1
|
|
40
libsrc/cx16/tmcommon.s
Normal file
40
libsrc/cx16/tmcommon.s
Normal file
|
@ -0,0 +1,40 @@
|
||||||
|
;
|
||||||
|
; 2019-12-27, Greg King
|
||||||
|
;
|
||||||
|
; Common stuff for the clock routines
|
||||||
|
;
|
||||||
|
|
||||||
|
.export TM, load_jiffy
|
||||||
|
|
||||||
|
.importzp sreg
|
||||||
|
|
||||||
|
|
||||||
|
;----------------------------------------------------------------------------
|
||||||
|
; Load .EAX with the approximate number of nanoseconds
|
||||||
|
; in one jiffy (1/60th of a second).
|
||||||
|
|
||||||
|
.proc load_jiffy
|
||||||
|
|
||||||
|
lda #<(17 * 1000 * 1000 / $10000)
|
||||||
|
ldx #>(17 * 1000 * 1000 / $10000)
|
||||||
|
sta sreg
|
||||||
|
stx sreg+1
|
||||||
|
lda #<(17 * 1000 * 1000)
|
||||||
|
ldx #>(17 * 1000 * 1000)
|
||||||
|
rts
|
||||||
|
|
||||||
|
.endproc
|
||||||
|
|
||||||
|
;----------------------------------------------------------------------------
|
||||||
|
; TM struct with "is daylight-saving time" set to "unknown"
|
||||||
|
.data
|
||||||
|
|
||||||
|
TM: .word 0 ; tm_sec
|
||||||
|
.word 0 ; tm_min
|
||||||
|
.word 0 ; tm_hour
|
||||||
|
.word 0 ; tm_mday
|
||||||
|
.word 0 ; tm_mon
|
||||||
|
.word 0 ; tm_year
|
||||||
|
.word 0 ; tm_wday
|
||||||
|
.word 0 ; tm_yday
|
||||||
|
.word .loword(-1) ; tm_isdst
|
|
@ -1,6 +1,7 @@
|
||||||
/* Clock test program
|
/* Calendar-clock test program
|
||||||
*
|
**
|
||||||
* 25-Sep-2018, chris@groessler.org
|
** 2018-Sep-25, chris@groessler.org
|
||||||
|
** 2019-Dec-27, Greg King
|
||||||
*/
|
*/
|
||||||
|
|
||||||
#include <stdio.h>
|
#include <stdio.h>
|
||||||
|
@ -12,72 +13,81 @@
|
||||||
#ifdef __CC65__
|
#ifdef __CC65__
|
||||||
#include <conio.h>
|
#include <conio.h>
|
||||||
#include <cc65.h>
|
#include <cc65.h>
|
||||||
#endif /* #ifdef __CC65__ */
|
#endif
|
||||||
|
|
||||||
static void print_time(void)
|
static int print_time(void)
|
||||||
{
|
{
|
||||||
struct tm *cur_tm;
|
struct tm *cur_tm;
|
||||||
time_t cur_time = time(NULL);
|
time_t cur_time = time(NULL);
|
||||||
|
|
||||||
if (cur_time == -1) {
|
if (cur_time == -1) {
|
||||||
printf("time() failed: %s\n", strerror(errno));
|
printf("time() failed: %s\n", strerror(errno));
|
||||||
return;
|
return 1;
|
||||||
}
|
}
|
||||||
cur_tm = localtime(&cur_time);
|
cur_tm = localtime(&cur_time);
|
||||||
|
|
||||||
printf("time: %s\n", asctime(cur_tm));
|
printf("time: %s\n", asctime(cur_tm));
|
||||||
// DEBUG:
|
// DEBUG:
|
||||||
printf("mday=%d mon=%d year=%d\nhour=%d min=%d sec=%d\n", cur_tm->tm_mday, cur_tm->tm_mon, cur_tm->tm_year, cur_tm->tm_hour, cur_tm->tm_min, cur_tm->tm_sec);
|
printf("year=%d, mon=%d, mday=%d\nhour=%d, min=%d, sec=%d\n",
|
||||||
|
cur_tm->tm_year, cur_tm->tm_mon, cur_tm->tm_mday,
|
||||||
|
cur_tm->tm_hour, cur_tm->tm_min, cur_tm->tm_sec);
|
||||||
|
return 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
int main(int argc, char **argv)
|
int main(int argc, char **argv)
|
||||||
{
|
{
|
||||||
char c = 0;
|
char c;
|
||||||
int s;
|
int s;
|
||||||
struct tm cur_time;
|
struct tm cur_time;
|
||||||
struct timespec new_time;
|
struct timespec new_time;
|
||||||
|
|
||||||
#ifdef __CC65__
|
#ifdef __CC65__
|
||||||
/* if DOS will automatically clear the screen after the program exits, wait for a keypress... */
|
/* If DOS automatically will clear the screen after the program exits,
|
||||||
|
** then wait for a key-press.
|
||||||
|
*/
|
||||||
if (doesclrscrafterexit())
|
if (doesclrscrafterexit())
|
||||||
atexit((void (*)(void))cgetc);
|
atexit((void (*)(void))cgetc);
|
||||||
#endif
|
#endif
|
||||||
|
|
||||||
if (argc <= 1) {
|
if (argc == 1) {
|
||||||
print_time();
|
return print_time();
|
||||||
return 0;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
if (argc != 3 || strcasecmp(*(argv + 1), "set")) {
|
if (argc != 2) {
|
||||||
printf("usage: CLOCKTST [set DD-MM-YY-HH-MM-SS]\n");
|
#ifdef __CC65__
|
||||||
|
printf("Usage: run:rem [YY-MM-DD-HH-MM-SS]\n");
|
||||||
|
#else
|
||||||
|
printf("Usage: %s [YY-MM-DD-HH-MM-SS]\n", argv[0]);
|
||||||
|
#endif
|
||||||
return 1;
|
return 1;
|
||||||
}
|
}
|
||||||
|
|
||||||
memset(&cur_time, 0, sizeof(cur_time));
|
memset(&cur_time, 0, sizeof cur_time);
|
||||||
s = sscanf(*(argv + 2), "%d-%d-%d-%d-%d-%d", &cur_time.tm_mday, &cur_time.tm_mon, &cur_time.tm_year, &cur_time.tm_hour, &cur_time.tm_min, &cur_time.tm_sec);
|
s = sscanf(argv[1], "%d-%d-%d-%d-%d-%d",
|
||||||
if (s != 6 || cur_time.tm_year > 99 /* other input values aren't being verified... */) {
|
&cur_time.tm_year, &cur_time.tm_mon, &cur_time.tm_mday,
|
||||||
printf("invalid time/date format\n");
|
&cur_time.tm_hour, &cur_time.tm_min, &cur_time.tm_sec);
|
||||||
|
if (s != 6 || cur_time.tm_year > 99 /* other input values aren't being verified */) {
|
||||||
|
printf("Invalid date-time format\n");
|
||||||
return 1;
|
return 1;
|
||||||
}
|
}
|
||||||
|
cur_time.tm_year += 100; /* assume 21st century */
|
||||||
--cur_time.tm_mon;
|
--cur_time.tm_mon;
|
||||||
if (cur_time.tm_year < 79)
|
|
||||||
cur_time.tm_year += 100; /* adjust century */
|
|
||||||
|
|
||||||
memset(&new_time, 0, sizeof(new_time));
|
memset(&new_time, 0, sizeof new_time);
|
||||||
new_time.tv_sec = mktime(&cur_time);
|
new_time.tv_sec = mktime(&cur_time);
|
||||||
|
|
||||||
printf("\nyou are about to set the time to\n--> %s\n\nContinue (y/n)?", ctime(&new_time.tv_sec));
|
printf("\nYou are about to set the time to\n--> %s\nContinue (y/n)? ", ctime(&new_time.tv_sec));
|
||||||
|
do {
|
||||||
while (c != 'y' && c != 'Y' && c != 'n' && c != 'N') {
|
|
||||||
#ifdef __CC65__
|
#ifdef __CC65__
|
||||||
c = cgetc();
|
c = cgetc();
|
||||||
#else
|
#else
|
||||||
c = getchar();
|
c = getchar();
|
||||||
#endif
|
#endif
|
||||||
}
|
} while (c != 'y' && c != 'Y' && c != 'n' && c != 'N');
|
||||||
printf("%c\n", c);
|
printf("%c\n", c);
|
||||||
|
|
||||||
if (c == 'n' || c == 'N') {
|
if (c == 'n' || c == 'N') {
|
||||||
printf("user abort\n");
|
printf("User abort\n");
|
||||||
return 0;
|
return 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -86,11 +96,11 @@ int main(int argc, char **argv)
|
||||||
printf("clock_settime() failed: %s\n", strerror(errno));
|
printf("clock_settime() failed: %s\n", strerror(errno));
|
||||||
return 1;
|
return 1;
|
||||||
}
|
}
|
||||||
printf("time set!\n");
|
printf("Time set!\n\n");
|
||||||
|
|
||||||
//DEBUG test begin
|
//DEBUG test begin
|
||||||
print_time();
|
return print_time();
|
||||||
//DEBUG test end
|
//DEBUG test end
|
||||||
return 0;
|
|
||||||
}
|
}
|
||||||
/* Local Variables: */
|
/* Local Variables: */
|
||||||
/* c-file-style: "cpg" */
|
/* c-file-style: "cpg" */
|
||||||
|
|
Loading…
Add table
Reference in a new issue