187 lines
4.8 KiB
NASM
187 lines
4.8 KiB
NASM
org 100h
|
|
cpu 8086
|
|
|
|
jmp start
|
|
|
|
; Variablen
|
|
sollwert db tmin
|
|
tcounts db 0 ; Impulszaehler fuer Servo, Vorwaertszaehler
|
|
|
|
anzeige times 3 db 0 ; Puffer fuer Ausgabe
|
|
strlen equ $ - anzeige ; Laenge Zeichenpuffer
|
|
|
|
; Konstanten
|
|
intab0 equ 20h ; Adresse Interrupttabelle PIT, Kanal 1
|
|
eoi equ 20h ; End Of Interrupt (EOI)
|
|
clrscr equ 0 ; Clear Screen
|
|
getkey equ 1 ; Funktion auf Tastatureingabe warten
|
|
ascii equ 1 ; Funktion ASCII-Zeichenausgabe
|
|
hexbyte equ 4 ; HEX-Byte Ausgabe
|
|
conin equ 5 ; Console IN
|
|
conout equ 6 ; Console OUT
|
|
pitc equ 0a6h ; Steuerkanal PIT
|
|
pit1 equ 0a2h ; Counter 1 PIT
|
|
pit2 equ 0a4h ; Counter 2 PIT
|
|
ppi_ctl equ 0b6h ; Steuerkanal PPI (Parallelinterface)
|
|
ppi_a equ 0b0h ; Kanal A PPI
|
|
ppi_pa0 equ 1 ; LED 0
|
|
ppi_pa1 equ 2 ; LED 1
|
|
ppi_pa2 equ 4 ; LED 2
|
|
ppi_pa3 equ 8 ; Lautsprecher
|
|
ppi_pa6 equ 1 << 6 ; Servomotor
|
|
ppi_b equ 0b2h ; Kanal B PPI
|
|
ppi_c equ 0b4h ; Kanal C PPI
|
|
ocw_2_3 equ 0c0h ; PIC (Interruptcontroller), OCW2,3
|
|
ocw_1 equ 0c2h ; PIC (Interruptcontroller), OCW1
|
|
icw_1 equ 0c0h ; PIC (Interruptcontroller), ICW1
|
|
icw_2_4 equ 0c2h ; PIC (Interruptcontroller), ICW2,4
|
|
leds equ 0 ; LED Port
|
|
schalter equ 0 ; Schalterport
|
|
keybd equ 80h ; SBC-86 Tastatur
|
|
gokey equ 11h ; Taste "GO"
|
|
outkey equ 15h ; Taste "OUT"
|
|
sseg7 equ 9eh ; Segmentanzeige 7
|
|
tcpwm equ 184 ; 1843200 Hz / 184 = ca. 10000 Hz = 0.1 ms
|
|
; Taktzyklus
|
|
; Zeitkonstante fuer PWM-Interrupt
|
|
tpwm equ 200 ; Periodendauer des PWM-Signals (* 0.1 ms)
|
|
tmax equ 25 ; Impulsdauer fuer 0 Grad (* 0.1 ms)
|
|
tmin equ 6 ; Impulsdauer fuer 180 Grad (* 0.1 ms)
|
|
divider equ 13 ; Vorteiler Wandlerwert
|
|
; 0..255 -> 0..19 -> + tmin -> tmin..tmax
|
|
ad_start1 equ 00110000b ; AD-Wandler starten und auf "read"
|
|
ad_start2 equ 00100111b ; AD-Wandler nur "read" und LED's an
|
|
|
|
start:
|
|
|
|
; Initialisierung
|
|
|
|
mov ah, clrscr ; Anzeige aus
|
|
int conout
|
|
|
|
call init ; Controller und Interruptsystem scharfmachen
|
|
|
|
mov al, 0
|
|
out leds, al
|
|
mov al, ad_start1 ; ADU init.
|
|
out ppi_a, al
|
|
mov al, ad_start2
|
|
out ppi_a, al
|
|
|
|
; Hintergrundprogramm: Lesen des Wandlerwertes, Verarbeitung und Displayausgabe
|
|
|
|
again:
|
|
in al, ppi_b ; Wandlerwert
|
|
; ...
|
|
; ...
|
|
; ...
|
|
jmp again
|
|
|
|
; Ausgabe einer ASCII-Zeichenkette
|
|
|
|
strout:
|
|
; ...
|
|
; ...
|
|
; ...
|
|
ret
|
|
|
|
; 8-Bit Division AL / BL
|
|
; Return: Quotient in AL, Rest in AH
|
|
; Zerstoert AX, BH
|
|
; Algorithmus siehe:
|
|
; https://www-user.tu-chemnitz.de/~heha/Mikrocontroller/Division.htm
|
|
; Wuerde auch mit dem DIV-Befehl der CPU funktionieren, wieso dann das hier?
|
|
|
|
divide: xor ah, ah ; High-Teil Dividend
|
|
mov bh, 8 ; Anzahl Bits Quotient
|
|
_div: shl ax, 1 ; Dividend * 2
|
|
cmp ah, bl ; teilbar?
|
|
jb _smaller ; nein
|
|
sub ah, bl ; ja, abziehen
|
|
inc al ; Dividend, Bit 0 = 1
|
|
_smaller:
|
|
dec bh ; fertig?
|
|
jnz _div ; nein, naechste Stelle
|
|
ret
|
|
|
|
; Hornerschema rueckwaerts (Teilen des Wertes durch die Zielbasis bis nix mehr
|
|
; da ist; die Reste der Division sind die Ziffern, von "hinten" (LOW) beginnend)
|
|
; Hexwert (in AL) nach dezimal (ASCII-String) wandeln und in "anzeige" speichern
|
|
; Zerstoert AX, BX, CX, DI
|
|
|
|
val2astr:
|
|
mov di, anzeige + strlen - 1 ; Zielpuffer rueckwaerts beschreiben
|
|
mov bl, 10 ; Dezimalwandlung (Zielbasis)
|
|
xor cx, cx
|
|
v1: call divide ; AL / BL, Rest in AH
|
|
add ah, '0' ; --> ASCII
|
|
mov [di], ah ; Zeichen in Puffer
|
|
dec di ; vorherige Pufferstelle
|
|
inc cl ; Zaehler++
|
|
or al, al ; fertig? (Dividend zerdividiert?)
|
|
jnz v1 ; noe, nochmal
|
|
mov al, cl ; Anzahl
|
|
mov cl, strlen ; max. Stringlaenge
|
|
sub cl, al ; Anzahl Reststellen
|
|
; out leds, al ; Kontrollausgabe
|
|
jcxz v3 ; fertig?
|
|
v2: mov byte [di], ' ' ; Rest loeschen
|
|
dec di
|
|
loop v2 ; fertig?
|
|
v3: ret
|
|
|
|
; Initialisierung Controller und Interruptsystem
|
|
|
|
init:
|
|
cli ; Interrupts aus
|
|
|
|
; PIT-Init.
|
|
mov al, 01110110b ; Kanal 1, Mode 3, 16-Bit ZK
|
|
out pitc, al ; Steuerkanal
|
|
mov al, tcpwm & 0ffh ; Low-Teil Zeitkonstante
|
|
out pit1, al
|
|
mov al, tcpwm >> 8 ; High-Teil Zeitkonstante
|
|
out pit1, al
|
|
|
|
; PPI-Init.
|
|
mov al, 10001011b ; PPI A/B/C Mode 0, A Output, sonst Input
|
|
out ppi_ctl, al
|
|
jmp short $+2 ; I/O-Delay
|
|
mov al, 0 ; LED's aus (high aktiv)
|
|
out ppi_a, al
|
|
|
|
; PIC-Init.
|
|
mov al, 00010011b ; ICW1, ICW4 benoetigt, Bit 2 egal,
|
|
; Flankentriggerung
|
|
out icw_1, al
|
|
jmp short $+2 ; I/O-Delay
|
|
mov al, 00001000b ; ICW2, auf INT 8 gemapped
|
|
out icw_2_4, al
|
|
jmp short $+2 ; I/O-Delay
|
|
mov al, 00010001b ; ICW4, MCS-86, EOI, non-buffered,
|
|
; fully nested
|
|
out icw_2_4, al
|
|
jmp short $+2 ; I/O-Delay
|
|
mov al, 11111110b ; Kanal 0 am PIC demaskieren
|
|
; PIT K1
|
|
out ocw_1, al
|
|
|
|
; Interrupttabelle init.
|
|
mov word [intab0], isr_servotimer ; Interrupttabelle (Timer K1)
|
|
; initialisieren (Offset)
|
|
mov [intab0 + 2], cs ; (Segmentadresse)
|
|
sti ; ab jetzt Interrupts
|
|
ret
|
|
|
|
isr_servotimer: ; Timer fuer Servo, besser exklusiv (CLI)
|
|
push ax
|
|
; ...
|
|
; ...
|
|
; ...
|
|
|
|
isr_servotimer_out:
|
|
mov al, eoi ; EOI an PIC
|
|
out ocw_2_3, al
|
|
pop ax
|
|
iret
|