[Q]: Отдача таймслайсов. Паскаль с ассемблером. [A]: Vadim Rumyantsev (2:5030/301) Более новая версия с пофиксенным зависанием при редком стечении обстоятельств в полночь в ДОСе :) И ещё чуть-чуть список операционных систем расширен. ──────────────────────────────────[Cut Here]────────────────────────────────── { Written by Vadim Rumyantsev, 2:5030/301. } { Generic DELAY unit -- release timeslices } { if under OS/2 2.0, Windows 3.0, DesqView, } { DoubleDOS and probably DOS 5.0 (?!), else } { do nothing. } { It is assumed that program receives time } { quantums every day... so, don't run this } { unit on slow systems! ;-) } { Virtual Pascal compatible now! } { Delphi 2.0 compatible now. } { You may use this **without restrictions** } UNIT USLDelay; {$I-} INTERFACE type OS_Type = (OS_MSDOS, OS_DOUBLEDOS, OS_TOPVIEW, OS_DESQVIEW, OS_OS2_1, OS_OS2_2, OS_WINDOWS, OS_WIN32, OS_MACOS); const AccessDenied : set of byte = [5 {$IFNDEF DOS} , 32 {$ENDIF} ]; var Running_OS_Name : string; {$IFDEF OS2} const Running_OS = OS_OS2_2; {$ENDIF} {$IFDEF WIN32} const Running_OS = OS_WIN32; {$ENDIF} {$IFDEF MSDOS} var Running_OS : OS_Type; {$ENDIF} {$IFDEF DPMI} var Running_OS : OS_Type; {$ENDIF} procedure Delay (n : longint); IMPLEMENTATION {$IFDEF OS2} uses {$IFDEF VIRTUALPASCAL} Os2base {$ELSE} Doscalls {$ENDIF}; var Buf : packed array [5..12] of longint; Sgn : string; f : file; fp : longint; sp : longint; p1, p2 : integer; {$ENDIF} {$IFDEF WIN32} uses SysUtils, Windows; const UnknownPlatform = 'Win32'; UnknownWin95 = 'Win9x'; var VersionInfo : TOsVersionInfoA; vb : string [10]; {$ENDIF} {$IFDEF MSDOS} uses Dos; { Define Seg0040 for backward compatibility with TP 4.0 .. TP 6.0 } const Seg0040 = $0040; var r : Registers; dosvh, dosvl : byte; osvh, osvl : byte; vendor : string [3]; {$DEFINE DOSMODE} {$ENDIF} {$IFDEF DPMI} uses Dos; { Define Seg0040 for backward compatibility with TP 4.0 .. TP 6.0 } var r : Registers; dosvh, dosvl : byte; osvh, osvl : byte; vendor : string [3]; {$DEFINE DOSMODE} {$ENDIF} function Version (vh, vl : longint) : string; var vhs, vls : string [2]; begin str (vh, vhs); str (vl, vls); if length (vls) = 1 then vls := '0' + vls; if vls [length (vls)] = '0' then dec (vls [0]); Version := vhs + '.' + vls end; {$IFDEF OS2} procedure Delay; begin if DosSleep (n) <> 0 then; end; BEGIN Running_OS_Name := 'OS/2'; if DosQuerySysInfo (5, 12, Buf, sizeof (Buf)) = 0 then begin FileMode := open_access_ReadOnly + open_share_DenyNone; assign (f, chr (64 + Buf [5]) + ':\OS2KRNL'); reset (f, 1); seek (f, $3C); blockread (f, fp, 4); seek (f, fp+$88); blockread (f, fp, 4); seek (f, fp); blockread (f, Sgn [0], 1); blockread (f, Sgn [1], length (Sgn)); p1 := pos ('@#', Sgn); p2 := pos ('#@', Sgn); if (IoResult = 0) and (p1 <> 0) and (p2 <> 0) and (p2 > (p1+2)) then begin Sgn := copy (Sgn, p1+2, p2-p1-2); p1 := pos (':', Sgn); if p1 <> 0 then Sgn := copy (Sgn, p1+1, 255); Running_OS_Name := Running_OS_Name + ' Revision ' + Sgn end else begin Buf [11] := Buf [11] div 10; if (Buf [11] = 2) and (Buf [12] >= 30) and (Buf [12] < 90) then begin Buf [11] := Buf [12] div 10; Buf [12] := Buf [12] mod 10 end; Running_OS_Name := Running_OS_Name + ' ' + Version (Buf [11], Buf [12]) end; close (f); if IoResult <> 0 then; end; {$ENDIF} {$IFDEF WIN32} procedure Delay; begin Sleep (n); end; BEGIN with VersionInfo do begin dwOsVersionInfoSize := sizeof (VersionInfo); if not GetVersionExA (VersionInfo) then Running_OS_Name := UnknownPlatform else begin str (dwBuildNumber and $FFFF, vb); case dwPlatformId of VER_PLATFORM_WIN32_WINDOWS: if (dwMajorVersion = 4) and (dwMinorVersion = 0) then Running_OS_Name := 'Windows 95' else if (dwMajorVersion = 4) and (dwMinorVersion = 10) then Running_OS_Name := 'Windows 98' else if (dwMajorVersion = 4) and (dwMinorVersion = 90) then Running_OS_Name := 'Windows Me' else Running_OS_Name := UnknownWin95; VER_PLATFORM_WIN32_NT: if (dwMajorVersion = 5) then Running_OS_Name := 'Windows 2000' else Running_OS_Name := 'Windows NT' else Running_OS_Name := UnknownPlatform end; Running_OS_Name := Running_OS_Name + ' ' + Version (dwMajorVersion, dwMinorVersion) + '/' + vb; if szCsdVersion [0] <> #0 then Running_OS_Name := Running_OS_Name + ' ' + StrPas (@szCsdVersion [0]) end end; {$ENDIF} {$IFDEF DOSMODE} procedure Delay (n : longint); const TicksPerDay = 1572480; var DelayQnt : longint; DoneTime : longint; LastTime : longint; ThisTime : longint; DateFlag : boolean; nh, nl : word; begin if Running_OS = OS_OS2_2 then begin {$IFDEF VER70} nh := n shr 8 shr 8; {$ELSE} nh := n shr 16; {$ENDIF} nl := n and $FFFF; asm mov dx, nh; mov ax, nl; hlt; db $35,$CA end; exit end; DoneTime := MemW [Seg0040:$006C]; { What time is it? } DelayQnt := round (n / 1000 * 18.2); { How many ticks wait? } DateFlag := (DoneTime + DelayQnt) >= TicksPerDay; { Skip midnight? } DoneTime := (DoneTime + DelayQnt) mod TicksPerDay; { When we'll finish? } LastTime := MemW [Seg0040:$006C]; while (DateFlag or (LastTime < DoneTime)) do begin { probably fixed damned midnight freeze } ThisTime := MemW [Seg0040:$006C]; if ThisTime < LastTime then { A new day! } DateFlag := false; LastTime := ThisTime; { Release timeslice } case Running_OS of OS_TOPVIEW, OS_DESQVIEW: begin r.AX := $1000; Intr ($15, r) end; OS_DOUBLEDOS: begin r.AH := $EE; if DelayQnt > 767 then r.AL := $FF else r.AL := DelayQnt div 3; dec (DelayQnt, r.AL * 3); Intr ($21, r) end else begin r.AX := $1680; Intr ($2F, r) end; end end end; BEGIN r.AX := $3000; MsDos (r); dosvh := r.AL; dosvl := r.AH; if r.BH = $00 then vendor := 'PC' else if r.BH = $66 then vendor := 'PTS' else if r.BH = $FF then vendor := 'MS' else vendor := 'OEM'; { Check for Novell NetWare to eliminate conflict with DoubleDOS detection } r.AX := $DC00; Intr ($21, r); if r.AL = 0 then begin { NetWare is not installed, so we can check for DoubleDOS } r.AX := $E400; Intr ($21, r); if r.AL <> 0 then begin { Yes, DoubleDos } Running_OS := OS_DOUBLEDOS; Running_OS_Name := 'DoubleDos'; exit end; end; { Check for DesqView } r.AX := $1022; r.BX := $0000; Intr ($15, r); if r.BX <> 0 then begin { Yes, DesqView or TopView } if r.BX <> $0A01 then begin Running_OS := OS_TOPVIEW; Running_OS_Name := 'TopView ' + Version (r.BL, r.BH) end else begin Running_OS := OS_DESQVIEW; r.CX := $4445; { 'DE', Serg Projzogin uses it } r.DX := $5351; { 'SQ', Serg Projzogin uses it } r.AX := $2B01; Intr ($21, r); Running_OS_Name := 'DesqView ' + Version (r.BH, r.BL) end; exit end; { Check for OS/2 } r.AX := $4010; r.BX := $0000; Intr ($2F, r); if r.BX <> 0 then begin { Yes, OS/2 } if r.BH >= 20 then Running_OS := OS_OS2_2 else Running_OS := OS_OS2_1; Include (AccessDenied, 162); if (r.BH <> dosvh) or (r.BL <> dosvl) then begin { DOS VMB under OS/2 } osvh := r.BH div 10; osvl := r.BL; if (osvh = 2) and (osvl >= 30) and (osvl < 90) then begin osvh := osvl div 10; osvl := osvl mod 10 end; Running_OS_Name := vendor + ' DOS ' + Version (dosvh, dosvl) + ' under OS/2 ' + Version (osvh, osvl); exit end; dosvh := dosvh div 10; if (dosvh = 2) and (dosvl >= 30) and (dosvl < 90) then begin dosvh := dosvl div 10; dosvl := dosvl mod 10 end; Running_OS_Name := 'OS/2 ' + Version (dosvh, dosvl); exit end; r.AX := $1600; Intr ($2F, r); if r.AL <> 0 then begin { Yes, Windows } Running_OS := OS_WINDOWS; if r.AX = $0004 then Running_OS_Name := 'Windows 95' else if r.AX = $0A04 then Running_OS_Name := 'Windows 98' else if r.AX = $5A04 then Running_OS_Name := 'Windows Me' else Running_OS_Name := 'Windows ' + Version (r.AL, r.AH); exit end; Running_OS := OS_MSDOS; Running_OS_Name := vendor + ' DOS ' + Version (dosvh, dosvl); {$ENDIF} END. ──────────────────────────────────[Cut Here]──────────────────────────────────