unit time; {$mode ObjFPC}{$H+} {$CALLING SysV_ABI_CDecl} interface const CLOCK_REALTIME =0; CLOCK_VIRTUAL =1; CLOCK_PROF =2; CLOCK_MONOTONIC =4; CLOCK_UPTIME =5; // FreeBSD-specific. CLOCK_UPTIME_PRECISE =7; // FreeBSD-specific. CLOCK_UPTIME_FAST =8; // FreeBSD-specific. CLOCK_REALTIME_PRECISE =9; // FreeBSD-specific. CLOCK_REALTIME_FAST =10; // FreeBSD-specific. CLOCK_MONOTONIC_PRECISE=11; // FreeBSD-specific. CLOCK_MONOTONIC_FAST =12; // FreeBSD-specific. CLOCK_SECOND =13; // FreeBSD-specific. CLOCK_THREAD_CPUTIME_ID=14; CLOCK_PROCTIME =15; // ORBIS only CLOCK_EXT_NETWORK =16; // ORBIS only CLOCK_EXT_DEBUG_NETWORK=17; // ORBIS only CLOCK_EXT_AD_NETWORK =18; // ORBIS only CLOCK_EXT_RAW_NETWORK =19; // ORBIS only type Ptimespec=^timespec; timespec=packed record tv_sec :Int64; /// seconds tv_nsec:Int64; /// nanoseconds end; Ptimeval=^timeval; timeval=packed record tv_sec :Int64; tv_usec:Int64; //microsecond end; Ptimezone=^timezone; timezone=packed record tz_minuteswest:Integer; tz_dsttime :Integer; end; time_t=QWORD; ptime_t=^time_t; ptimesec=^timesec; timesec=packed record tz_time :time_t; tz_secwest:DWORD; tz_dstsec :DWORD; end; function timespeccmp_lt(tvp,uvp:ptimespec):Integer; inline; procedure TIMEVAL_TO_TIMESPEC(tv:ptimeval;ts:ptimespec); inline; procedure TIMESPEC_TO_TIMEVAL(tv:ptimeval;ts:ptimespec); inline; function TIMESPEC_TO_UNIT(ts:ptimespec):Int64; inline; //Unit function tvtohz(time:Int64):Int64; inline; procedure usec2timespec(ts:ptimespec;timeo:DWORD); procedure TIMESPEC_ADD(dst,src,val:ptimespec); procedure TIMESPEC_SUB(dst,src,val:ptimespec); function clock_gettime(clock_id:Integer;tp:Ptimespec):Integer; function clock_getres(clock_id:Integer;tp:Ptimespec):Integer; implementation uses trap, thr_error, kern_time; function timespeccmp_lt(tvp,uvp:ptimespec):Integer; inline; begin if (tvp^.tv_sec=uvp^.tv_sec) then begin Result:=ord(tvp^.tv_nsec < uvp^.tv_nsec); end else begin Result:=ord(tvp^.tv_sec < uvp^.tv_sec); end; end; procedure TIMEVAL_TO_TIMESPEC(tv:ptimeval;ts:ptimespec); inline; begin ts^.tv_sec :=tv^.tv_sec; ts^.tv_nsec:=tv^.tv_usec * 1000; end; procedure TIMESPEC_TO_TIMEVAL(tv:ptimeval;ts:ptimespec); inline; begin tv^.tv_sec :=ts^.tv_sec; tv^.tv_usec:=ts^.tv_nsec div 1000; end; function TIMESPEC_TO_UNIT(ts:ptimespec):Int64; inline; //Unit begin Result:=(QWORD(ts^.tv_sec)*10000000)+(QWORD(ts^.tv_nsec) div 100); end; function tvtohz(time:Int64):Int64; inline; begin Result:=time; end; procedure usec2timespec(ts:ptimespec;timeo:DWORD); begin ts^.tv_sec :=(timeo div 1000000); ts^.tv_nsec:=(timeo mod 1000000)*1000; end; procedure TIMESPEC_ADD(dst,src,val:ptimespec); begin dst^.tv_sec :=src^.tv_sec +val^.tv_sec; dst^.tv_nsec:=src^.tv_nsec+val^.tv_nsec; if (dst^.tv_nsec>=1000000000) then begin Inc(dst^.tv_sec); Dec(dst^.tv_nsec,1000000000); end; end; procedure TIMESPEC_SUB(dst,src,val:ptimespec); begin dst^.tv_sec :=src^.tv_sec -val^.tv_sec; dst^.tv_nsec:=src^.tv_nsec-val^.tv_nsec; if (dst^.tv_nsec<0) then begin Dec(dst^.tv_sec); Inc(dst^.tv_nsec,1000000000); end; end; function clock_gettime(clock_id:Integer;tp:Ptimespec):Integer; assembler; nostackframe; asm movq sys_clock_gettime,%rax call fast_syscall jmp cerror end; function clock_getres(clock_id:Integer;tp:Ptimespec):Integer; assembler; nostackframe; asm movq sys_clock_getres,%rax call fast_syscall jmp cerror end; end.