Tämä on lyhyt ohjeistus TTK-91 -kielen alkeisiin ja tarkoitettu lähinnä alkuun pääsemiseen. Lisätietoa kannattaa lukea kurssin materiaaleista.
TTK-91:ssä voit käyttää vain kokonaislukuja. Siinä ei ole tukea merkkijonojen ilmaisemiselle eikä liukuluvuille. Tulostetaan siis Hello Worldin sijaan luku 2.
Aina, kun lasket jotain tai käytät I/O:ta (tulostat näytölle tai luet luvun näppäimistöltä), on käytettävä jotakin rekisteriä!
Rekistereitä joita voit käyttää vapaasti ovat R1 - R5. Myös rekisteriä R0 voi käyttää, mutta se voi käyttäytyä välillä oudosti. Vältä sitä toistaiseksi.
Ladataan nyt luku 2 rekisteriin LOAD -käskyllä:
LOAD R1, =2
=2 tarkoittaa lukua 2. Jos käytät konekäskyä
LOAD R1, 2
, se lataa rekisteriin R1 muistiosoitteessa 2 olevan arvon!
Pienet ja isot kirjaimet eivät ole merkityksellisiä. Voit siis kirjoittaa koodia vaikka näin load r1, =2
tai LoAd R1, =2
.
Käsky tallettaa rekisteriin R1 kokonaisluvun 2. Huomaat että ennen lukua 2 on = -merkki. Se ilmaisee että sitä seuraava arvo on välitön operandi. Operandi tarkoittaa operaation kohdetta. Operaationa tässä on LOAD, ensimmäisenä operandina on R1:n arvo ja toisena operandina luku 2. KAIKKI TTK-91 -kielen operaatiot ovat binäärioperaatioita eli operaatioita joissa on kaksi (siksi binääri) operaation kohdetta eli operandia. Tällainen termistö voi tuntua vieraalta, mutta olet käyttänyt operaatioita ja operandeja alakoulusta lähtien. Esimerkiksi aivan tavallinen laskutoimitus 4 + 7, missä 4 ja 7 ovat operandeja ja + on operaatio.
Terminologiaa: Käsky on sama kuin operaatio, esimerkiksi LOAD. Konekäsky on koko rimpsu operandeineen ja rekistereineen, tässä tapauksessa LOAD R1, =2.
Tarkalleen ottaen LOAD R1, =2
on konekäsky LOAD R1, =2(R0)
. Konekäskyssä on aina operaatio, ensimmäinen rekisteri, osoitusmoodi (yllä siis välitön; muut ovat suora ja epäsuora, joihin palataan kohta), osoiteosa (välitön operandi tai muistiosoite) ja indeksirekisteri. Indeksirekisteri on R0 jos sitä ei ole merkitty. Jos indeksirekisterinä käytetään R0:aa, sen arvona on silloin 0, vaikka rekisterissä R0 olisi jokin muu luku. Siispä LOAD R1, =2(R0) lataa rekisteriin R1 arvon, joka saadaan laskemalla yhteen 2 + 0 eli 2. Sen sijaan kun käytät rekistereitä R1 - R5 indeksirekisterinä, esimerkiksi näin: LOAD R1, =2(R2)
, ja R2:n sisältö on esimerkiksi 5, ladataan R1:een arvo 2 + 5.
Seuraavaksi tulostetaan rekisteriin R1 talletettu arvo näytölle OUT -käskyllä:
OUT R1, =CRT
Tämä konekäsky lähettää CRT -laitteelle eli näytölle rekisterin R1 sisällön tulostettavaksi. Siis tuon siihen aiemmin ladatun luvun 2.
Muistutus: ensimmäinen operandi on aina jossakin rekisterissä!
Ohjelma pitää aina lopettaa käyttöjärjestelmän kutsumiskäskyllä SVC antamalla ensimmäiseksi operandiksi pino-osoitin (Stack Pointer, SP) ja toiseksi operandiksi ohjelman lopetusrutiinin osoite HALT näin:
SVC SP, =HALT
Tämä konekäsky lopettaa ohjelman.
Titokoneella voidaan käsitellä vain kokonaislukuja. Luku luetaan näppäimistöltä näin:
IN R1, =KBD
Tämän konekäskyn suorituksen jälkeen rekisterin R1 arvo on näpppäimistöltä luettu luku.
Kun ohjelma lukee käyttäjän syötteen operaatiolla IN
tai kun määritellään muuttujan tai muuttujana tallennettavan vakion arvo, on käytettävissä 32 bittiä luvun ilmaisuun. Koska käytössä on myös negatiiviset luvut, lukualue tällöin seuraava:
- Pienin kokonaisluku on - 2^31 = - 2147483648
- Suurin kokonaisluku on 2^31 - 1 = 2147483647
Kun kokonaislukua käytetään konekäskyn yhteydessä, on käytössä 16 bittiä luvun ilmaisuun. Koska käytössä on myös negatiiviset luvut, lukualue on tällöin seuraava:
- Pienin kokonaisluku on - 2^15 = -32768
- Suurin kokonaisluku on 2^15 - 1 = 32767
Tässä on taulukko konekäskyn binääriesityksestä, josta voitte havaita, että konekäskyn yhteydessä on vain 16 bittiä käytössä kokonaisluvun ilmaisua varten.
Konekäskyn rakenteesta on lisätietoa täällä. Alla oleva taulukko on suoraan kyseisestä lähteestä.
8 bittiä 3b 2b 3b 16 bittiä
+----------------+------+----+------+--------------------------------+
| Operaatiokoodi | Rj | M | Ri | osoite / välitön operandi |
+----------------+------+----+------+--------------------------------+
31 24 23 16 15 0
Lähde: Tietokoneen toiminta -kurssin kotisivu, Helsingin yliopisto
https://www.cs.helsinki.fi/group/titokone/ttk91_ref_fi.html
Konekäskyn binääriesityksessä vasemmalta lukien ensimmäiset 8 bittiä ovat operaatiokoodi, seuraavat 3 bittiä ensimmäinen rekisteri, sitten 2 bittiä osoitustapa, sitten 3 bittiä toinen rekisteri ja lopuksi 16 bittiä muistiosoitetta tai välitöntä operandia varten. Luvun ilmaisu konekäskyn yhteydessä on siis mahdollista käyttämällä 16 bittiä. Koska TTK-91:ssä on sekä negatiiviset että positiiviset kokonaisluvut, pienin luku mitä konekäskyn yhteydessä voidaan käyttää, on - 2^15 = - 32768 ja suurin luku on 2^15 - 1 = 32767.
Konekäskyn voi purkaa osiin ohjelmallisesti käyttämällä bittitason AND ja SHR (Javassa & ja >>>) -operaatioita. Niistä voit lukea lisää täältä.
Seuraavassa kappaleessa näytetään miten voidaan käyttää pienempiä lukuja kuin -32768 ja suurempia lukuja kuin 32767.
Muuttujat alustetaan pseudokäskyllä DC (Data Constant) näin:
A DC 0 ; Kuten Java-kielellä: int A = 0;
B DC 42 ; Kuten Java-kielellä: int B = 42;
Konekäskyn yhteydessä osoiteosaan mahtuu vain välillä -32786 ... 32767 olevia lukuja. Pseudokäskyn DC avulla voi määritellä muuttujan arvoksi tämän välin ulkopuolisia lukuja, joita voi sitten käyttää ohjelmassa.
MILTSI DC 1000000 ; int MILTSI = 1000000;
PIENIN DC -2147483648 ; Pienin TTK-91:n hyväksymä kokonaisluku eli - 2^31.
SUURIN DC 2147483647 ; Suurin TTK-91:n hyväksymä kokonaisluku eli 2^31 - 1.
Muuttujaan alustettuja isoja lukuja voi käyttää näin:
LOAD R1, MILTSI ; R1:n arvoksi tulee muuttujan MILTSI arvo 1000000.
; Tämä ei toimisi: LOAD R1, =1000000
Konekäskyn yhteydessä pienin mahdollinen luku on -32768 ja suurin 32767.
Lukua operaatiolla
IN
luettaessa ja muuttujan tai muuttujana talletettavan vakion määrittelyssä pienin mahdollinen luku on -2147483648 ja suurin 2147483647. Muuttujana talletettavalla vakiolla tarkoitetaan lukua, joka on aivan kuten muuttuja, mutta ohjelmoijan tulee muistaa, että sen arvoa ei saa muuttaa.
Pseudokäsky ei ole "oikea" konekäsky. Se huomioidaan vain käännösvaiheessa. Esimerkiksi A DC 0
pseudokäsky varaa käännösvaiheessa muuttujalle A muistiosoitteen ja asettaa muuttujan arvoksi 0. Tällöin symbolin A arvo on sille varattu muistiosoite ja muuttujan A arvo on 0. Kääntäjä asettaa symbolitauluun tiedon muuttujan nimestä ja osoitteesta.
Muuttujan määrittely ja alustaminen
A DC 0
. Samantapaisesti siis kuten Javassaint a = 0;
Muuttujan arvoa voidaan muuttaa, vaikka tuo DC käsky on lyhenne sanoista data constant. Yleensä constant on vakio jota ei voi muuttaa. Muuttujaa voidaan kuitenkin käyttää halutessaan kuin vakiota: silloin ohjelmoijan on muistettava, että kyse on vakiosta, jota ei saa muuttaa STORE -käskyllä.
Vakiot määritellään pseudokäskyllä EQU näin:
VASTAUS EQU 42
Vakiota ei voi muuttaa. Sen arvo on aina se mihin se ohjelmassa määritellään. Tätä vastaa Javascriptin uudemmissa versioissa käytettävä
const vastaus = 42
. Kuten DC:n, myös EQU:n avulla määritellyn vakion nimi ja arvo talletetaan symbolitauluun käännösvaiheessa.
Esimerkki EQU-vakioiden käyttämisestä ja vertailua muuttujan tai muuttujana alustetun vakion käyttämiseen.
MILJARDI DC 1000000000 ; Tämä on aivan tavallinen muuttuja, jota käytetään tässä kuin vakiota.
suurinVakio EQU 32767
pieninVakio EQU -32768
LOAD R1, =suurinVakio ; Käännöksen jälkeen: LOAD R1, =32767
; Käännösvaiheessa suurinVakio on korvattu vakion ARVOLLA!
OUT R1, =CRT
LOAD R1, =pieninVakio ; Käännöksen jälkeen: LOAD R1, =-32768
OUT R1, =CRT
LOAD R1, MILJARDI ; Käännöksen jälkeen sanan MILJARDI tilalla on muuttujan osoite, eli käsky on muodossa LOAD R1, 10
; (jos muuttujan MILJARDI osoite on 10)
OUT R1, =CRT ; Rekisterin R1 arvoksi asettiin osoitteessa 10 (eli osoitteessa MILJARDI) oleva arvo 1000000000.
SVC SP, =HALT
Tästä kannattaa huomata erityisesti, että pseudokäskyllä EQU voi asettaa vakiolle arvoja vain välillä -32768..32767. Sallitut arvot ovat siis samat kuin mitä lukuja osoite/vakio-osaan voi muutenkin kirjoittaa. Tämän välin ulkopuolisia lukuja voi tallettaa muuttujien arvoiksi ja käyttää niitä vakioina, kuten tässä esimerkissä tehtiin muuttujalle MILJARDI. Tällöin ohjelmoijan on muistettava, ettei vahingossa muuta muuttujana talletetun vakion arvoa.
Tietueelle ja taulukolle varataan tilaa pseudokäskyllä DS (Data Segment) näin:
PERSON DS 4
Tämä pseudokäsky alustaa taulukon PERSON nelialkioiseksi. Taulukolle varataan siis tilaa samantyyppisesti kuin Javassa int person = new int[4]
.
Taulukon läpikäyminen on kätevintä tehdä indeksirekisterin avulla. Rekisteriä, jonka sisältämää arvoa käytetään taulukkoa läpikäydessä indeksinä, kutsutaan indeksirekisteriksi.
Esimerkki:
PERSON DS 4 ; Varataan taulukolle muistialue, kooltaan 4. Kääntäjä asettaa taulukon alkuosoitteeksi
; jonkin muistiosoitteen.
LOAD R2, =0 ; Käytetään rekisteriä R2 indeksirekisterinä. Indeksiksi asetetaan aluksi 0.
LOAD R1, =3 ; Ladataan rekisteriin R1 jokin arvo, tässä tapauksessa luku 3.
STORE R1, PERSON(R2) ; Tämä käsky tallettaa (STORE) rekisterin R1 sisällön
; muistiosoitteeseen, joka saadaan laskemalla yhteen
; indeksirekisterin R2 arvo ja PERSON -taulukon osoite.
; Alussa on asetettu, että R2=0, joten tallennus tapahtui osoitteeseen
; PERSON+R2 eli PERSON-taulukon indeksiin 0.
; Jos esim. R2=2, tallennus tapahtuu PERSON-taulukon indeksiin 2.
TTK-91:ssä on käytössä seuraavat laskutoimitukset.
ADD |
Yhteenlasku | ADD R1, =5 lisää rekisterin R1 arvoon luvun 5. |
SUB |
Vähennyslasku | SUB R1, 2 vähentää rekisterin R1 arvosta MUISTIOSOITTEEN 2 sisältämän arvon. |
MUL |
Kertolasku | MUL R1, X kertoo rekisterin R1 arvon muistiosoitteessa X olevalla arvolla (muuttujan X arvolla). |
DIV |
Jakolasku | DIV R1, X jakaa rekisterin R1 arvon muistiosoitteessa X olevalla arvolla. Huom! Esim. kun R1:n arvo on 9 ja X:n arvo 2, eli suoritetaan laskutoimitus 9/2, tuloksena on 4. Luku siis pyöristyy kokonaisluvuksi alaspäin. |
MOD |
Modulo (jakojäännös) | MOD R1, =3 muuttaa rekisterin R1 arvoksi jakojäännöksen. Esim. R1:n arvo on 5, niin jakojäännös 3:lla jaettaessa on 2. |
LOAD R1, =40 ; Ladataan rekisteriin R1 luku 40.
ADD R1, =2 ; Lisätään rekisterissä R1 olevaan arvoon luku 2.
A DC 2 ; int a = 2
B DC 42 ; int b = 42
LOAD R1, A ; Ladataan rekisteriin R1 muuttujan A arvo. Toisin sanoen ladataan R1:een muistiosoitteen A sisältö.
MUL R1, B ; Kerrotaan rekisterin R1 sisältämä luku muuttujan B arvolla.
STORE R1, A ; Talletetaan rekisterin R1 arvo muuttujan A arvoksi
Mikä ihmeen STORE
? No, sillä talletetaan rekisterin arvo muuttujan arvoksi. Täsmällisesti ilmaistuna konekäskyllä STORE R1, A
talletetaan rekisterin R1 arvo muistipaikkaan A.
Ttk-91:ssä on käytössä bittimanipulointiin bittitason loogiset operaatiot AND, OR, XOR ja NOT sekä bittien siirto-operaatiot SHL, SHR ja SHRA. Bittimanipulointia käytetään esimerkiksi osan biteistä eristämiseksi kokonaisluvusta, salausmenetelmissä ja tiedon eheyden varmentamisessa.
Tarkastellaan näistä käskyistä tässä toistaiseksi vain AND-käskyä.
Operaatio AND suorittaa bittitason loogisen JA -toiminnon. Logiikassa a JA b on tosi ainoastaan silloin, kun a ja b ovat tosia, eli sekä a että b ovat tosia. Biteillä ilmaisten:
1 JA 1 = 1
1 JA 0 = 0
0 JA 1 = 0
0 JA 0 = 0
Bittitason AND-operaatio toimii siten, että kahden kokonaisluvun binääriesityksen samassa kohdassa olevien bittien kesken suoritetaan yllä kuvatulla tavalla JA eli AND -operaatio. Tässä esimerkki:
11101110
AND 00111000
============
00101000
Esimerkissä AND-operaation avulla saatiin eristettyä ylemmästä luvusta 11101110 ne bitit, joiden kohdalla luvussa 00111000 bitit ovat ykkösiä. Siis ykkösten kohdalla olevat bitit säilyivät alkuperäisinä ja muut nollautuivat.
Ennen kuin näytän, miten ttk-91:ssä käytetään AND-operaatiota, tarkastellaan miten sitä käytetään korkean tason kielessä.
Java-kielessä bittitason AND tehdään käyttämällä & -merkkiä. Esimerkki:
int anna = 0b11101110;
int mask = 0b00111000;
int b = anna & mask;
// Nyt b = 0b00101000;
Esimerkissä AND-operaation avulla saatiin eristettyä muuttujan anna
arvosta ne bitit, joiden kohdissa muuttujan mask
arvossa on bitit ykkösiä.
AND-operaation ymmärtämistä ja maskin käyttöä tarvitset muun muassa tietoliikenteen sovelluksissa (aliverkon peite eli subnet mask). Sen avulla saa myös vaikkapa 32-bittisestä konekäskystä eristettyä osan tietoa.
luku DC 327175712 ; Alla luku binäärilukuna:
; 0001 0011 1000 0000 0100 1110 0010 0000
; Välilyönnit on lisätty helpottamaan luvun lukemista.
maski DC 65535 ; 0000 0000 0000 0000 1111 1111 1111 1111
LOAD R1, luku
AND R1, maski
; Lopputuloksena R1:ssä on nyt binääriluku 0000 0000 0000 0000 0100 1110 0010 0000.
Tuo esimerkin luku 327175712 on erään ttk-91-kielen 32-bittisen konekäskyn numeerinen esitysmuoto. Tuolla AND-operaatiolla saatiin säilytettyä luvun oikeanpuoleiset 16 bittiä ja nollattua muut bitit. Siis saatiin eristettyä ttk-91-konekäskyn vakio-/osoiteosa!
Bittien siirtämistä varten ttk-91:sä on käskyt SHL, SHR ja SHRA. Niitä vastaa esimerkiksi Javassa operaatiot <<, >>> ja >>.
Käskyllä SHL saadaan siirrettyä bittejä vasemmalle haluttu määrä. Samalla oikealle tulee nollia siirron määrän verran. Esimerkki valaiskoon asiaa:
Ella DC 25 ; 25 = 0b 0000 0000 0000 0000 0000 0000 0001 1001
LOAD R1, Ella
SHL R1, =2
; Seurauksena R1 = 0b 0000 0000 0000 0000 0000 0000 0110 0100
Javassa sama tehdään näin:
int ella = 25;
int bella = ella << 2;
Käskyllä SHR saadaan siirrettyä bittejä oikealle haluttu määrä. Vastaavasti vasemmalle puolelle tulee nollia siirron määrän verran. Esimerkki:
Heini DC 2000000000
; Luku 2000000000 = 0b 0111 0111 0011 0101 1001 0100 0000 0000
LOAD R1, Heini
SHR R1, =24
; Seurauksena R1 = 0b 0000 0000 0000 0000 0000 0000 0111 0111
Kuten esimerkistä näkyy, kun 32-bittisen binääriluvun 0111 0111 0011 0101 1001 0100 0000 0000 bittejä siirretään oikealle 24 bitin verran, saadaan luku 0000 0000 0000 0000 0000 0000 0111 0111. Näin saadaan eristettyä luvun 8 vasemmanpuoleista bittiä ja selvitettyä niiden arvo.
Javalla sama tehdään näin:
int heini = 0b0111_0111_0011_0101_1001_0100_0000_0000;
int siirretty = heini >>> 24;
Negatiivisten lukujen bittien siirto oikealle Pythonia käytettäessä
Javaa käytettäessä int-tyypin luvut on talletettu 32-bittisessä Big Endian kahden komplementtimuodossa kuten ttk-91:ssäkin. Pythonissa kokonaisluvut ovat myös Big Endian kahden komplementtimuodossa, mutta lukujen bittimäärää ei ole rajoitettu. Niinpä esimerkiksi luvun -1, jonka 32-bittinen esitysmuoto on 0b1111_1111_1111_1111_1111_1111_1111_1111, bittien siirto oikealle >>> -operaatiolla Javassa aikaansaa sen, että vasemmalta luku täyttyy nollilla. Siis esimerkiksi -1 >>> 1 = 2147483647 eli 32-bittinen binääriluku, jossa vasemmanpuoleisin bitti on 0 ja muut 1. Pythonissa ei ole >>> -operaatiota, vaan vain >> -operaatio, joka täyttää vasemmalta bittejä sen mukaan, mikä on vasemmanpuoleisin bitti alkuperäisessä luvussa. Koska Pythonissa negatiivissa luvuissa tulkitaan olevan mielivaltainen määrä ykkösiä vasemmalla, niin -1 >> 1 on -1. Pythonin >> -operaatio toimii siis saman tyyppisesti kuin alla esitelty SHRA eli artimeettinen oikealle siirto. Pythonissa voi kuitenkin saada tehtyä bittisiirron käyttämällä maskia. Voitte kokeilla pythonissa seuraavaa:
luku = -1
print(luku >> 1)
# tämä tulostaa -1
print(luku & 0xffffffff) >> 1
# tämä tulostaa luvun 2147483647
Käsky SHRA (Javassa >>) toimii kuten SHR, mutta sen sijaan, että siirrossa vasemmalle jäävät bitit korvataan nollilla, ne korvataan sillä bitillä, joka on vasemmanpuoleisin bitti. Siis nollalla jos vasemmanpuoleisin on 0 ja ykkösellä jos vasemmanpuoleisin on 1. Tällä on merkitystä luvun merkin (+ vai -) säilymisen kannalta.
NOT-operaation avulla saadaan vaihdettua luvun bitit siten, että jokaisesta nollasta tulee yksi ja ykkösistä nolla. Esimerkki:
A DC 1 ; 0b 0000 0000 0000 0000 0000 0000 0000 0001
LOAD R1, A
NOT R1
; Saadaan R1 = 0b 1111 1111 1111 1111 1111 1111 1111 1110
OR-operaation (looginen TAI) avulla saadaan esimerkiksi lisättyä binääriluvun nollia sisältäviin kohtiin halutut bitit. Esimerkki:
IRA DC 230 ; 0b 0000 0000 0000 0000 0000 0000 1110 0110
LUKU DC 2147483392 ; 0b 0111 1111 1111 1111 1111 1111 0000 0000
LOAD R1, LUKU
OR R1, IRA
; Saadaan R1 = 0b 0111 1111 1111 1111 1111 1111 1110 0110
Konekielessä ehtolauseet ja niiden mukaan toimiminen tehdään hyppäämällä ehtojen toteutumisen mukaisiin muistiosoitteisiin.
Esitellään uusia käskyjä vertailukäsky
COMP
ja ehdollinen hyppykäskyJLES
(Jump if Less). Lisäksi: voit laittaa konekäskyn eteen sanan ja koodissa hypätä siihen.
Tämä ohjelma tulostaa luvut 0...9:
LOAD R1, =0 ; Ladataan rekisteriin R1 luku 0.
UUSIKSI OUT R1, =CRT ; Tulostetaan R1:ssä oleva luku.
ADD R1, =1 ; Kasvatetaan R1:n arvoa yhdellä.
COMP R1, =10 ; Verrataan R1:n arvoa lukuun 10.
JLES UUSIKSI ; (Jump if Less) Jos vertailukäskyn COMP suorituksen jälkeen
; tilarekisterin bitti L (less) on 1 ja E (equals) on 0,
; eli R1:n arvo on alle 10, hypätään osoitteeseen UUSIKSI.
SVC SP, =HALT ; Tämä lopettaa ohjelman.
Rivin alussa oleva sanaa, joka ei ole mikään kielen käskykannan operaatio, voi käyttää ohjelmoidessa rivin muistiosoitteena. Käännösvaiheessa kääntäjä huomaa nämä sanat ja asettaa koodissa jokaiseen kohtaan, missä sana on konekäskyn osana (kuten yllä olevassa koodissa kohdassa JUMP UUSIKSI
), kyseisen sanan tilalle sen rivin muistiosoitteen minkä alussa sana on.
Näitä sanoja voi käyttää kuten muuttujia. Voit esimerkiksi saada ladattua rekisteriin R1 muistipaikassa UUSIKSI olevan konekäskyn (LOAD R1, UUSIKSI
) ja muistipaikan UUSIKSI osoitteen (LOAD R1, =UUSIKSI
).
Osoitteen ilmaisimena ei saa käyttää operaatioiden nimiä (ei saa olla esim. LOAD, ADD, OUT, ...).
Huom! Jos rivin alussa määritetään sana jonka perusteella siihen voi hypätä, on sillä rivillä oltava konekäsky. Siinä voi käyttää halutessaan konekäskyä
NOP
eli No Operation, joka tosin sitten kuluttaa yhden kierroksen käskyjen nouto- ja suoritussyklissä.
Näin ei siis voi tehdä:
LOAD R1, =0
UUSIKSI ; **Ei näin! Tällä rivillä on oltava myös konekäsky.**
OUT R1, =CRT
Tässä esimerkkiohjelmassa pyydetään näppäimistöltä lukua. Jos luku on 0, sitä ei hyväksytä vaan pyydetään uudelleen lukua. Sen jälkeen se talletetaan muuttujan A arvoksi ja verrataan muuttujien A ja B arvoa keskenään. Huomatkaa että vertailussa vähintään toisen luvuista pitää olla rekisterissä! Jos vertailun COMP R1, B
tulos on että R1 > B, JGRE suurempi
-konekäskyn avulla siirrytään osoitteeseen suurempi
.
A DC 0 ; Alustetaan muuttuja A, A=0.
B DC 100 ; Alustetaan muuttuja B, B=100.
kysyLuku IN R1, =KBD ; Pyydetään luku näppäimistöltä.
JZER R1, kysyLuku ; Jump if ZERo. Jos R1==0, hyppää kysyLuku.
STORE R1, A ; Talletetaan kysytty luku muuttujan A arvoksi.
COMP R1, B ; Verrataan kysyttyä lukua muuttujan B arvoon.
JGRE suurempi ; Jump if GREater. Jos R1:n arvo on suurempi
; B:n arvo, hypätään kohtaan suurempi.
LOAD R1, B ; Ei ollut suurempi, tehdään tässä jotain muuta.
ADD R1, =A
STORE R1, A
OUT R1, =CRT
JUMP lopetus ; Hypätään ehdottomalla JUMP -käskyllä kohtaan lopetus.
; Ilman tätä ehdotonta hyppyä siirryttäisiin seuraavalle riville.
suurempi LOAD R1, =1000 ; Tehdään tässä asioita jos vertailun tulos oli suurempi.
DIV R1, A
OUT R1, =CRT
lopetus SVC SP, =HALT ; Lopetus pitää merkitä jos sinne halutaan hypätä.
JUMP | Ehdoton hyppy | JUMP 20 |
Hypätään osoitteeseen 20. |
JUMP lopetus |
Hypätään suorittamaan konekäskyä jonka eteen on kirjoitettu lopetus . |
Muut hypyt ovat ehdollisia hyppyjä. Jos ehto toteutuu, hypätään käskyn yhteydessä annettavaan osoitteeseen. Muutoin jatketaan seuraavan käskyn suoritukseen, eli lähdekoodissa siis seuraavalla rivillä olevan käskyn suorittamiseen.
Näitä käskyjä käytetään rekisterin kanssa ilman mitään muuta vertailukäskyä.
Operaatio | Merkitys | Esimerkki | Selite |
---|---|---|---|
JNEG | Jump if NEGative. | JNEG R1, tuliNegaa |
Jos R1<0, hyppää osoitteeseen tuliNegaa . |
JZER | Jump if ZERo | JZER R3, onNolla |
Jos R3==0, hyppää osoitteeseen onNolla . |
JPOS | Jump if POSitive | JPOS R2, onPosii |
Jos R2>0, hyppää osoitteeseen onPosii . |
JNNEG | Jump if Not NEGative | JNNEG R5, eiNega |
Jos R5 >= 0, hyppää osoitteseen eiNega |
JNZER | Jump if Not ZERo) | JNZER R4, eiNolla |
Jos R4 erisuuri kuin 0, hyppää osoitteeseen eiNolla |
JNPOS | Jump if Not POSitive | JNPOS R1, eiPosii |
Jos R1 <= 0, hyppää eiPosii |
Tulostetaan luvut 9...0 luupin avulla.
LOAD R1, =9
Tulosta OUT R1, =CRT
SUB R1, =1
JNNEG R1, Tulosta ; Jos R1>=0, hyppää kohtaan TULOSTA
SVC SP, =HALT
Vertailukäskyn (esim. COMP R1, =10
) jälkeen tilarekisterin (SR) bittien G (greater), L (less) ja E (equals) arvot muuttuvat sen mukaisiksi mikä vertailun tulos on ollut.
Esimerkki:
COMP R1, =10 ; Verrataan COMP -käskyllä R1:n arvoa lukuun 10. Vertailun tulos
; merkitään tilarekisteriin.
JLES lopetus ; Jump if LESs, hypätään osoitteeseen lopetus, jos vertailun tulos
; oli se, että R1<10. Muutoin jatketaan seuraavan rivin suorittamiseen.
; Tämä hyppykäsky tarkastaa vertailun tuloksen tilarekisteristä (SR).
COMP R1, A ; Verrataan R1:n arvoa muuttujan A arvoon. Vertailun tulos
; merkitään tilarekisteriin.
JGRE suurempi ; Jump if Greater. Jos vertailun tulos oli se, että R1:n arvo on
; suurempi kuin A:n arvo, hypätään osoitteeseen suurempi.
Operaatio | Merkitys | Esimerkki | Selite | SR:n bitit G,L,E |
---|---|---|---|---|
JLES | Jump if LESs | JLES pienempi |
Jos vertailun tulos oli pienempi. | L asetettu, ei E |
JEQU | Jump if EQUal | JEQU tasan |
Jos vertailun tulos oli tasan | E, ei muut |
JGRE | Jump if GREater | JGRE suurempi |
Jos vertailun tulos oli suurempi | G, ei muut |
JNLES | Jump if Not LESs | JNLES eiAlle |
Jos vertailun tulos oli suurempi tai yhtäsuuri | E tai G |
JNEQU | Jump if Not EQUal | JNEQU eiSama |
Jos vertailun tulos oli erisuuri | L tai G |
JNGRE | Jump if Not GREater | JNGRE eiSuurempi |
Jos vertailun tulos oli pienempi tai yhtäsuuri | L tai E |
Muista, että näitä suuruusvertailuhyppykäskyjä käytettäessä on aina ensin tehtävä vertailu
COMP
-käskyllä. Esimerkkejä:COMP R1, X
taiCOMP R1, =10
.
Joissakin tapauksissa on kätevä käyttää hyppytauluja. Hyppytaulu on taulukko, joka sisältää hyppykäskyjä johonkin kohtaan koodia. Sen sijaan, että käytettäisiin vertailuoperaattoria kunkin vaikkapa syötteenä saadun arvon mukaan hyppäämiseen, voidaan hypätä suoraan syötteen perusteella hyppytaulun kohtaan, jossa on hyppykäsky haluttuun kohtaan koodia. Usein on tarpeen toki selvittää, onko annettu syöte oikeellinen. Alla olevassa esimerkissä hypätään parillisuuden perustella oikeaan kohtaan koodia, kunhan syöte on kelvollinen. Tässä syötteeksi hyväksytään vain ei-negatiivinen luku.
Tässä ensin esimerkin koodi ilman kommentteja. Mieti mitä eri kohdissa tapahtuu. Tämän koodilohkon alla on vielä sama koodi kommentein varustettuna.
jump ALKU
HYPPYTAULU jump CASE0
jump CASE1
ALKU in r1, =kbd
jneg r1, LOPPU
mod r1, =2
jump HYPPYTAULU(r1)
LOPPU svc sp, =halt
CASE0 add r1, =10
out r1, =crt
jump LOPPU
CASE1 mul r1, =10
out r1, =crt
jump LOPPU
Ja tässä sama koodi kommentein varustettuna:
jump ALKU
; Alustetaan taulukko HYPPYTAULU. Yllä on hypätty jo näiden yli, joten ne voidaan alustaa tässä.
HYPPYTAULU jump CASE0
jump CASE1 ; Näppärää: aiempi rivi määrittyy sanan HYPPYTAULU mukaan,
; joten seuraava rivi on osoitteessa HYPPYTAULU + 1.
; Luetaan näppäimistöltä luku.
; Negatiivisella luvulla hypätään lopetukseen.
; Otetaan jakojäännös, jotta saadaan luvun parillisuus selville
; Jos jakojäännös 0, hypätään kohtaan HYPPYTAULU + 0. Jos 1, hypätään HYPPYTAULU + 1.
ALKU in r1, =kbd ; Luetaan luku näppäimistöltä.
jneg r1, LOPPU ; Ei hyväksytä negatiivisia arvoja: jos r1 < 0, hypätään LOPPUuun.
mod r1, =2 ; r1:n arvoksi jakojäännös luvulla 2 jaettaessa. r1 on nyt 0 tai 1.
jump HYPPYTAULU(r1) ; Hypätään osoitteeseen HYPPYTAULU + r1
; Tässä välissä voi olla muuta koodia.
LOPPU svc sp, =halt
; Alla olevat koodit ovat täällä lopussa, joten niihin pääsee vain hyppäämällä.
CASE0 add r1, =10
out r1, =crt
jump LOPPU
CASE1 mul r1, =10
out r1, =crt
jump LOPPU
Hyppytaulun edut korostuvat, jos ehtoja olisi enemmän. Jos hyväksytty rekisterin r1 arvo olisi vaikkapa välillä 0..999, riittää tarkistaa, ettei luku ole negatiivinen eikä lukua 999 suurempi, ja sen jälkeen vain hyppää JUMP HYPPYTAULU(r1)
. Ilman hyppytaulun hyödyntämistä olisi tehtävä tuhat COMP -vertailua ja JEQU -hyppyä!
Taulukot ovat ohjelmoinnissa tärkeä tietorakenne. Seuraavaksi esitetään miten taulukon alkioiden arvot saa asetettua tiettyyn arvoon eli taulukon alustaminen ja sen jälkeen käyttäjältä kysyttävien lukujen lisääminen taulukkoon.
Havainnollistetaan indeksirekisterin käyttöä taulukon alustamiseen. Seuraavassa koodissa luodaan 100-alkioinen taulukko ja alustetaan se muuttujan X arvolla.
ARVOSANAT DS 100 ; Tämä varaa taulukolle tilaa 100 muistipaikkaa alkaen osoitteesta ARVOSANAT.
N EQU 100 ; Tämä luo vakion N, joka on arvoltaan 100 eli taulukon koko.
X DC 0 ; Muuttujan X varaaminen ja sen alustaminen arvoon 0.
LOAD R1, =0 ; Käytetään rekisteriä R1 indeksirekisterinä alkaen indeksistä 0.
LOAD R2, X ; Ladataan rekisteriin R2 muuttujan X arvo.
Alusta STORE R2, ARVOSANAT(R1) ; Talletetaan rekisterin R2 arvo osoitteeseen ARVOSANAT + R1
ADD R1, =1 ; Lisätään indeksirekisterin R1 arvoa yhdellä
COMP R1, =N ; Verrataan R1 arvoa lukuun 100 (vakio N) eli taulukon kokoon.
JLES Alusta ; Jos edellisen rivin vertailun tulos oli pienempi, eli
; että pätee R1 < N, hypätään kohtaan Alusta.
SVC SP, =HALT ; Lopetetaan ohjelma.
Ohjelmassa alustetaan taulukko ARVOSANAT muuttujan X arvolla. Huomatkaa, että indeksointi menee kuten vaikkapa Javan int[] -taulukoissa, eli taulukon ensimmäinen alkio on indeksissä 0 ja viimeinen indeksissä N-1. Kun N = 100, indeksit ovat 0,1,2,...,99. Lisäksi kannattaa muistaa, että STORE -käskyllä voi tallettaa vain rekisterissä olevia arvoja johonkin muistipaikkaan. Jos siis muuttujan X arvo halutaan tallettaa muistiosoitteeseen ARVOSANAT+R2, on ensin muuttujan X arvo ladattava (LOAD R2, X
) johonkin rekisteriin ja sen jälkeen konekäskyllä STORE R2, ARVOSANAT(R1)
saa sen arvon sitten talletettua haluttuun muistiosoitteeseen.
Tässä esimerkissä käyttäjältä pyydetään 10 positiivista lukua jotka talletetaan taulukkoon LUVUT. Jos käyttäjä syöttää ei-positiivisen luvun, tai kun kaikki luvut on saatu, ohjelma tulostaa annetut luvut viimeisenä annetusta alkaen ja ohjelma lopetetaan. Jos yhtään positiivista lukua ei ole annettu, mitään ei tulosteta ja ohjelma lopetetaan.
LUVUT DS 10 ; Varataan tilaa taulukolle LUVUT
; Javassa: int[] LUVUT = new int[10]
LOAD R1, =0 ; R1 = 0. Tätä käytetään indeksirekisterinä.
Anna IN R2, =KBD ; R2 = näppäimistöltä luettu luku.
JNPOS R2, Elsa ; Jos R2<=0, hyppää kohtaan Elsa.
STORE R2, LUVUT(R1) ; Tallenna R2:n arvo taulukkoon kohtaan R1 eli
; muistiosoitteeseen R1:n arvo + taulukon
; LUVUT osoite. Vertaa: LUVUT[R1] = R2;
ADD R1, =1 ; Lisätään indeksirekisterin arvoa yhdellä.
COMP R1, =10 ; Verrataan sitä 10:een.
JLES Anna ; Jos R1<10, hyppää kohtaan Anna
Elsa JZER R1, LetItGo ; Jos R1 == 0 (yhtään lukua ei ole annettu),
; hypätään kohtaan LetItGo. Jos R1 ei ole 0,
; R1 on positiivinen, koska R1:n arvoa on
; kasvatettu koodissa. Näin ollen ei tarvitse
; hypätä mihinkään, vaan sen sijaan jatketaan
; suoritusta alta seuraavasta konekäskystä.
; Tulostetaan luvut viimeksi annetusta alkaen:
Snowgies SUB R1, =1 ; Vähennetään indeksirekisteristä 1
LOAD R2, LUVUT(R1) ; R2 = LUVUT[R1]
OUT R2, =CRT ; Tulostetaan luku
JPOS R1, Snowgies ; Jos R1 > 0, hyppää kohtaan Snowgies
LetItGo SVC SP, =HALT ; Ohjelman lopetus.
Esimerkissä toteutettiin
❄️ lukujen lisääminen taulukkoon
❄️ lukujen lukeminen taulukosta
❄️ ehdollinen hyppy suuruusjärjestysvertailukäskyn COMP
avulla
❄️ ehdollisia hyppyjä rekisterin sisällön merkin (ei-positiivinen, nolla, positiivinen) perusteella
❄️ näytettiin kaksi suuntaa miten käydä läpi luuppia (0...9 ja 9...0).
Konekäskyn yhteydessä voi käyttää osoitusmuotoina välitöntä, suoraa ja epäsuoraa osoitusta. Konekäskyn kääntämisen jälkeen nämä ilmaistaan kahtena bittinä konekäskyn binääriesityksessä kuten esitettiin aiemmin täällä.
; Alustetaan ensin muuttujat.
X DC 42 ; int x = 42;
PX DC 0 ; X:n osoitinmuuttuja
PPX DC 0 ; PX:n osoitinmuuttuja
; Määritellään osoitinmuuttujien arvot
LOAD R1, =X ; Ladataan R1:een X:n OSOITE.
STORE R1, PX ; Talletetaan X:n osoite PX:n arvoksi.
; Nyt PX osoittaa X:n arvoon.
LOAD R1, =PX ; Ladataan R1:een PX:n OSOITE
STORE R1, PPX ; Talletetaan PX:n osoite PPX:n arvoksi.
; Käytetään noita
LOAD R1, X ; R1 = haeArvoOsoitteesta(X)
LOAD R1, @PX ; Ladataan R1:een PX:n osoittama arvo
; R1 = haeArvoOsoitteesta(haeArvoOsoitteesta(PX))
; R1:ssä on nyt X:n arvo.
LOAD R1, @PPX ; Ladataan R1:een PPX:N osoittama arvo
; R1 = haeArvoOsoitteesta(haeArvoOsoitteesta(PPX))
; R1:ssä on nyt PX:n arvo (jonka arvo on X:n osoite)
Välitön osoitustapa on se, missä konekäskyn osoiteosassa eli oikeanpuoleisimpina 16 bittinä on jo valmiiksi eli välittömästi se arvo mitä on tarkoitus käyttää konekäskyn suorituksessa. Ohjelmoidessa välitön osoitus ilmaistaan = -merkillä. Esimerkiksi konekäskyssä LOAD R1, =100
on kyse välittömästä osoitustavasta. Prosessori havaitsee välittömän osoitustavan konekäskyssä olevista kahdesta bitistä jotka kertovat muistinoutojen määrän. Välittömässä niiden bittien arvo on 0, joten mitään ei tarvitse hakea muistista ja prosessori voi ryhtyä suorittamaan käskyä.
Välitön osoitus: Esimerkiksi
LOAD R1, =100
lataa rekisteriin R1 luvun 100.
Suorassa osoituksessa konekäskyn osoiteosassa eli oikeanpuoleisimpina 16 bittinä on muistiosoite, josta operaatiota varten tarvittava tieto on haettava. Ohjelmoidessa suora osoitus ilmaistaan ilman mitään erikoismerkkejä.
Esimerkki: Oletetaan että muuttujan X muistiosoite on 25, ja oletetaan että muuttujan X arvo on 1000. Tällöin konekäsky LOAD R1, X
on kääntämisen jälkeen on muodossa LOAD R1, 25
.
Käskyn nouto- ja suoritussyklissä asia tapahtuu näin:
- Käsky on noudettu
LOAD R1, 25
käskyrekisteriin (IR, Instruction Register). - Prosessori hoitaa käskyn purkamisen osiin erittäin tehokkaasti. Käskyrekisteristä osoiteosan (oikeanpuoleisimmat 16 bittiä) sisältö (tässä luku 25) siirtyy nopeiden johtimien avulla heti väliaikaisrekisteriin (TR, Temporary Register, "temppari").
- Prosessori tarkistaa osoitustavan osoitustapaa osoittavista biteistä. Koska tässä on kyse suorasta osoituksesta, niiden bittien arvo on 1. Siitä prosessori tietää, että tarvitaan yksi muistinouto lisää.
- Seuraavaksi haetaan muistiosoitteesta 25 sen sisältö, joka on yllä oletettu olevan 1000, ja se talletetaan väliaikaisrekisteriin.
- Ja sitten käsky suoritetaan: rekisterin R1 arvoksi tulee väliaikaisrekisterin arvo 1000.
Suora osoitus: esimerkiksi
LOAD R1, X
lataa rekisteriin R1 muistiosoitteen X sisällön eli toisin sanottuna muuttujan X arvon.
Epäsuora osoitus ilmaistaan ohjelmoidessa @ -merkillä. Esimerkiksi LOAD R1, @PX
lataa rekisteriin R1 muuttujan PX arvon mukaisen muistipaikan sisällön. Toisella tavalla sanoen se konekäsky lataa rekisteriin R1 muuttujan PX osoittaman arvon.
Osoitustapaa ilmaisevien bittien arvo on epäsuorassa osoituksessa 2, joten prosessori tekee 2 muistinoutoa.
Oletetaan seuraavat 4 asiaa
- Muuttujan X arvo eli muistiosoitteen X sisältö on 1000.
- Muuttujan X muistiosoite on 25.
- Muuttujan PX muistiosoite on 30.
- Muuttujan PX arvo, eli muistiosoitteen PX sisältö on X:n osoite eli 25.
Jos muuttujan PX arvo on muuttujan X osoite, sanotaan että muuttuja PX osoittaa muuttujan X arvoon. Tällöin PX on muuttujan X osoitinmuuttuja (pointer = osoitin).
Esimerkiksi LOAD R1, @PX
käskyä suorittaessa tapahtuu käskyjen nouto- ja suoritussyklissä seuraavaa:
- Käsky
LOAD R1, @30
on noudettu käskyrekisteriin.
- @30 eikä PX, sillä kääntämisen jälkeen koodissa ei ole enää muuttujia vaan pelkästään niiden muistiosoitteet. Yllä on oletettu että PX:n osoite on 30.
- Käsky on purettu. Nyt luku 30 eli muuttujan PX osoite on väliaikaisrekisterissä (TR, Temporary Register).
- Osoitustapa näkyy käskyn binääriosoituksen biteistä. Epäsuorassa sen arvo on 2. Prosessori tietää, että on siksi tehtävä 2 muistinoutoa.
- Noudetaan muistista muistipaikan 30 sisältö eli muuttujan PX arvo (joka on nyt tässä muuttujan X osoite 25) ja talletaan se TR:ään.
- Tehdään toinen muistinouto. TR:ssä on luku 25, joten haetaan sen mukaisen muistiosoitteen sisältö ja talletetaan se TR:ään. Nyt TR:ssä on muistipaikan 25 sisältö (muuttujan X arvo), siis luku 1000.
- Ja sitten käsky suoritetaan: Rekisteriin R1 siirtyy TR:ssä oleva luku 1000.
Käydään tässä ensiksi läpi parametrien välittäminen aliohjelmalle ja aliohjelman toteuttamisen runko ja syvennytään asiaan tarkemmin alempana. Aliohjelmien kutsumisessa ja toteutuksessa on tärkeää, että parametreja luetaan aliohjelmassa oikeassa järjestyksessä. Tässä esimerkissä välitetään aliohjelmalle sekä viiteparametri (muuttujan tai taulukon osoite) että arvoparametreja (lukuja). Seuraavassa koodissa kutsutaan aliohjelmaa ALUSTA, joka alustaa sille parametrina annetun taulukon arvolla, joka annetaan sille parametrina. Taulukkoa käsiteltäessä on annettava myös taulukon koko, sillä taulukko ei tiedä omaa kokoaan (toisin kuin Javassa jossa taulukon koon saa selville length-metodilla).
taulukko ds 20
koko dc 20
arvo dc 1729
push sp, =taulukko
push sp, koko
push sp, arvo
call sp, alusta
svc sp, =halt
Parametrien lukemista helpottamaan luo EQU-vakiot SAMASSA JÄRJESTYKSESSÄ KUIN OLET PUSKENUT NE PINOON, ja numeroi ne siten että alin on -2, toiseksi alin -3 jne, näin:
aliTaulukko EQU -4
aliKoko EQU -3
aliArvo EQU -2
Yllä määritellyt luvut kertovat, mikä on parametreina annettujen viitteiden (osoitteiden) ja arvojen (lukujen) sijainti kehysosoittimen (FP, Frame Pointer) arvosta laskettuna. Yllä oleva call sp, alusta
-konekäsky asettaa pinoon PC:n ja FP:n arvon, asettaa uudeksi FP:n arvoksi pino-osoittimen SP arvon ja asettaa PC:n arvoksi alusta -aliohjelman osoitteen. Varsinaisesti ohjelmoidessa ei tarvitse edes järkeillä, miksi nuo EQU-määrittelyt asetetaan juuri noin suhteessa Frame Pointeriin: riittää muistaa, että nuo EQU -määrittelyt pitää kirjata samassa järjestyksessä kuin parametrit on puskettu pinoon ja numeroida ne juuri tuolla tavalla. Vakioiden nimeämisessä kannattaa käyttää jotain lisäkirjainta tai sanaa, joka erottaa ne pääohjelman nimistä. Ne eivät siis saa olla samat. Yllä eriävä nimeäminen on tehty kirjoittamalla eteen sana ali.
Aliohjelman toteuttamisessa perusrunko on seuraava:
ALUSTA pushr sp
; Tähän kohtaan kirjoitetaan koodi taulukon alustukseen hyödyntäen
; taulukon alkuosoitetta, kokoa ja arvoa jolla halutaan taulukko alustaa.
; Nyt esimerkiksi taulukon osoitteen saa ladattua käskyllä
; LOAD R1, aliTaulukko(fp), koon käskyllä LOAD R2, aliKoko(fp) ja arvon
; käskyllä LOAD R3, aliArvo(fp). Tämän jälkeen taulukon indeksiin 0
; saa talletettua rekisterin R3 arvon konekäskyllä STORE R3, 0(R1).
; Jotta saat talletettua muihinkin indekseihin, laske ensin indeksi
; johonkin vapaaseen rekisteriin (esim. R1), lisää siihen taulukon
; alkuosoite konekäskyllä ADD R1, aliTaulukko(fp) ja talleta halutun
; rekisterin (tässä R3) arvo sitten konekäskyllä STORE R3, 0(R1).
; (Tätä voi optimoida suoritustehon parantamiseksi pitämällä
; alkuosoitetta jossakin rekisterissä tallessa.)
;
popr sp
exit sp, =3
Siis ensiksi kirjoitetaan aliohjelman nimi ja sille riville pushr sp
, joka tallettaa rekisterit R0 - R6 talteen pinoon. Toiseksi sitten tehdään mitä aliohjelman pitää tehdä. Lopuksi popr sp
palauttaa rekisterien arvot ja exit sp, =3
-käsky hoitaa poistumisen aliohjelmasta. Tässä aliohjelmassa ei palauteta mitään arvoa, joten exitissä “poistetaan pinosta” kaikki 3 parametria. Jos paluuarvo tai paluuarvoja pitäisi palauttaa, silloin pinoon olisi ensin puskettu joku arvo paluuarvoa varten ensimmäisenä (mikä vain arvo, vaikkapa 0), ja exit -käskyssä olisi sitten vähemmän pinosta poistettavaa. Tästä on alempana esimerkki.
Kun aliohjelman kutsu tehdään oikein ja aliohjelman runko on oikein, pääsee jo aika pitkälle. Seuraavaksi esitellään miten aliohjelma voi lukea ja muuten hyödyntää noita pinoon puskettuja parametreja.
Esimerkki aliohjelman kutsumisesta ja toteutuksesta. Tässä esimerkissä aliohjelmalle välitetään arvoparametrit aktivaatiotietueen avulla. Aktivaatiotietue on alue muistissa, jolla ratkaistaan aliohjelmien toteutus. Se toimii pinona. Pinoon talletetaan arvoja PUSH
-käskyllä ja sieltä saadaan otettua arvoja POP
-käskyllä.
Esimerkissä on kommentit jotka auttavat ymmärtämään miten aliohjelmien kutsu ja toteutus toimii. Tämä kannattaa kokeilla Titokoneella. On suositeltavaa, että muokkaat sitä haluamallasi tavalla niin opit paremmin. Voit esimerkiksi antaa aliohjelmalle lisää parametreja ja tehdä niillä jotain laskentaa aliohjelmassa.
; Tämä on .k91 -tiedosto, joka on käännettävissä Titokoneella.
x dc 314 ; Määritellään x muuttujaksi ja asetetaan sille alkuarvo 314.
; Huomaa, että tämä ei ole muodossa x dc =314, vaikka muutoin
; käytetään merkkiä = kun esim. ladataan rekisteriin jokin luku
; tähän tyyliin: load r1, =314.
y dc 227 ; Määritellään y muuttujaksi ja asetetaan sille alkuarvo 227
push sp, =0 ; varataan pinoon tilaa paluuarvolle
push sp, x ; pushataan muistiosoitteessa x oleva arvo pinoon (arvoparametri)
push sp, y ; pushataan muistiosoitteessa y oleva arvo pinoon (arvoparametri)
call sp, myLittleSuby ; kutsutaan aliohjelmaa myLittleSuby
; call -käsky siirtää automaattisesti vanhan PC:n ja FP:n (R7:n) arvon pinoon
pop sp, r1 ; Aliohjelmasta on palattu. Popataan sp-pinon päällimmäinen
; arvo rekisteriin r1.
; Näin siis saatiin aliohjelman paluuarvo pääohjelmaan.
out r1, =crt ; Tulostetaan se
svc sp, =halt ; Pääohjelman lopetus
; Aliohjelman toteutus
; Varataan ensin tilaa muuttujille jotta on helpompi muistaa mitä on tekemässä.
; Muuttujien nimeämisessä on muistettava että ne ovat globaaleja eli niiden tulee olla yksilöllisiä.
; Jos siis käytät aliohjelmassa allaolevan sx:n paikalla x (jota käytetään pääohjelmassa), tulee ongelmia.
paluuarvo equ -4
sx equ -3
sy equ -2 ; viimeisenä sp-pinoon laitettu arvo on kohdassa -2(fp) eli siis fp:n arvo - 2
; fp-1 ja fp-0 eli kaksi ylintä ovat pääohjelman PC:tä ja FP:tä varten
; Aliohjelma:
MYLITTLESUBY pushr sp ; pushr sp laittaa rekisterit r0-r6 (r6=sp) pinoon jotta
; voit käyttää vapaasti rekistereitä aliohjelmassa
load r1, sx(fp) ; Lataa r1:een osoitteessa fp-3 oleva arvo
load r2, sy(fp) ; Lataa r2:een osoitteessa fp-2 oleva arvo
mul r1, r2 ; Tehdään pikku laskutoimitus r1 = r1 * r2 jotta tämä
; aliohjelma tekee edes jotain.
store r1, paluuarvo(fp) ; Nyt sp-pinossa on osoitteessa fp-4 aliohjelman paluuarvo
popr sp ; Popataan pinosta siihen pushr sp:llä laitetut rekisterit
exit sp, =2 ; Palataan pääohjelmaan siten että 2 pinon päällimmäistä arvoa
; poistetaan pinosta (nuo mitkä laitettiin push sp, x ja
; push sp, y aliohjelmaa kutsuttaessa.
; Pinon päällimmäisenä on nyt ennen aliohjelman kutsua
; ensiksi laitettu arvo eli kohdassa fp - 4 oleva arvo,
; jota aliohjelma on muuttanut.
Viiteparametreja käytettäessä aliohjelmalle välitetään muuttujan tai taulukon osoite. Muuttujan osoitteen välittämisen etuna on se, että aliohjelma voi lukea ja muuttaa annetun muuttujan arvoja. Taulukon osoitteen välittämisen hyöty on se, että aliohjelma voi lukea ja muuttaa taulukon sisältöä.
; Tämä on .k91 -tiedosto, joka on käännettävissä Titokoneella.
x dc 42 ; Määritellään x muuttujaksi ja asetetaan sille alkuarvo 42.
; Huomaa, että tämä ei ole muodossa x dc =42, vaikka muutoin
; käytetään merkkiä = kun esim. ladataan rekisteriin jokin luku
; tähän tyyliin: load r1, =42.
push sp, =x ; pushataan muuttujan x osoite pinoon (viiteparametri)
y dc 33 ; Koska aliohjelma on toteutettu siten, että siihen halutaan viiteparametrit,
; alustetaan nyt muuttuja jota voi käyttää viiteparametrina.
; Tällainen muuttujan alustus voidaan siis tehdä keskellä koodia.
; Sitä ei ole pakko tehdä aivan lähdekoodin alussa.
push sp, =y ; pushataan muuttujan y osoite pinoon (viiteparametri)
call sp, SWAP ; kutsutaan aliohjelmaa SWAP, joka vaihtaa x:n ja y:n arvot keskenään
; call -käsky siirtää automaattisesti vanhan PC:n ja FP:n (R7:n) arvon pinoon
; Nyt tässä ei ole mitään erillistä paluuarvoa, sillä aliohjelma muutti suoraan
; viiteparametreina annettujen muuttujien arvoja. Paluuarvon välittämistä
; on käsitelty yllä olevassa esimerkissä.
load r1, x ; Aliohjelmasta on palattu. Ladataan aliohjelman muuttama x:n arvo rekisteriin r1.
load r2, y ; Ja samoin y:n arvo.
out r1, =crt ; Tulostetaan x:n arvo, joka on nyt 33 eli entinen y:n arvo.
out r2, =crt ; Tulostetaan y:n arvo, joka on nyt 42 eli entinen x:n arvo.
svc sp, =halt ; Pääohjelman lopetus
; Aliohjelman toteutus
; Varataan ensin tilaa muuttujille jotta on helpompi muistaa mitä on tekemässä.
; Muuttujien nimeämisessä on muistettava että ne ovat globaaleja eli niiden tulee olla yksilöllisiä.
; Jos siis käytät aliohjelmassa allaolevan sx:n paikalla x (jota käytetään pääohjelmassa), tulee ongelmia.
; Aliohjelma swap vastaa korkean tason kielen komentoa public static void swap(Luku a, Luku b).
; Se siis vaihtaa parametreina annettujen muuttujien arvot vähän samantyyppisesti kuin olioiden
; sisäistä tilaa voidaan muuttaa Javassa. Tässä tapauksessa aliohjelmalle on annettava
; parametreina muuttujien osoitteet (push sp, =muuttujannimi. Erillistä paluuarvoa ei nyt välitetä.
sx equ -3
sy equ -2 ; viimeisenä sp-pinoon laitettu arvo on kohdassa -2(fp) eli siis fp:n arvo - 2
; fp-1 ja fp-0 eli kaksi ylintä ovat pääohjelman PC:tä ja FP:tä varten
; Aliohjelma:
SWAP pushr sp ; pushr sp laittaa rekisterit r0-r6 (r6=sp) pinoon jotta
; voit käyttää vapaasti rekistereitä aliohjelmassa
load r1, @sx(fp) ; Lataa r1:een arvo, johon osoitteessa fp-3 oleva arvo viittaa
load r2, @sy(fp) ; Lataa r2:een arvo, johon osoitteessa fp-2 oleva arvo viittaa
store r1, @sy(fp) ; Talletetaan r1:stä arvo osoitteeseen johon sy(fp) eli fp-2 viittaa
store r2, @sx(fp) ; Talletetaan r2:sta arvo osoitteeseen johon sx(fp) eli fp-3 viittaa
; Nyt voidaan lähteä pois aliohjelmasta. Viiteparametreina annettujen muuttujien arvot on vaihdettu.
popr sp ; Poprataan pinosta siihen pushr sp:llä laitetut rekisterit. Tämä käsky
; asettaa aliohjelmaa kutsuneen rutiinin rekisterien arvot takaisin.
exit sp, =2 ; Palataan pääohjelmaan. Käsky asettaa FP:n ja PC:n arvon ennalleen ja
; poistaa pinosta 2 päällimmäistä arvoa, eli nuo mitkä laitettiin
; push sp, =x ja push sp, =y käskyillä aliohjelmaa kutsuttaessa.
; Pino-osoitin SP osoittaa nyt siihen mihin kuuluukin. Aliohjelmalle
; parametreina annetut osoitteet eivät ole enää pinossa.
; (Tarkalleen ottaen ne ovat edelleen muistissa siellä
; mihin ne laitettiinkin, mutta niihin ei enää pääse
; käsiksi POP -käskyllä.)
Funktio F(a, b) palauttaa arvonaan lausekkeen 2a+3b+1 arvon. Esimerkiksi lausekkeen x = F(1,1) suorituksen jälkeen muuttujan x arvo on 6. Parametri a on arvoparametri ja parametri b on viiteparametri. Muuttuja x on määritelty pääohjelmatasolla.
i. [3 p] Toteuta ttk-91 symbolisella konekielellä (funktiota F kutsumalla) lauseke x = F(x, 3).
ii. [4 p] Toteuta ttk-91 symbolisella konekielellä funktio F.
Noudata suositusten mukaista aliohjelmien (funktioiden) kutsumekanismia.
(Lähde: Tietokoneen toiminta -kurssin koe 7.5.2018., tehtävä 4 d. Kokeen laatinut Teemu Kerola.)
Muuttuja x on tehtävänannon mukaan määritelty pääohjelmatasolla (eli ei aliohjelmassa). Sitä ei tarvitse siten tässä enää määritellä. Koska a on arvoparametri, välitetään aliohjelmalle sen arvo. Koska b on viiteparametri, välitetään aliohjelmalle sen osoite.
Ohjelman koodi | Selite |
---|---|
b dc 3 |
Alustetaan muuttuja b ja annetaan sen arvoksi 3. |
push sp, =0 |
Varataan pinosta tilaa paluuarvoa varten puskemalla pinoon joku luku. |
push sp, x |
Pusketaan pinoon x:n arvo, eli aliohjelman (funktion) F ensimmäinen parametri. Välitettiin arvo, koska ensimmäinen parametri on arvoparametri. |
push sp, =b |
Pusketaan pinoon muuttujan b osoite, eli aliohjelman (funktion) F toisen parametrin osoite. Osoite on viite muuttujaan. Käytettiin siis viiteparametria. |
call sp, F |
Kutsutaan aliohjelmaa F |
pop sp, r1 |
Popataan pinosta aliohjelman sinne laskema arvo. |
store r1, x |
Talletaan se arvo muuttujan x arvoksi eli osoitteeseen x. |
Valmista! Nyt x = F(x,3).
Toteutetaan funktio F(a,b) = 2a + 3b + 1. Esimerkki mitä funktion kuuluu palauttaa. Olkoon a=10 ja b=5. Tällöin F(a,b) = F(10,5) = 2*10 + 3*5 + 1 = 20+15+1 = 36.
Ohjelman koodi | Selite |
---|---|
paluuarvo equ -4 |
Paluuarvon sijainti suhteessa rekisteriin FP on -4. |
arvoA equ -3 |
Ensimmäisen parametrin sijainti suhteessa rekisteriin FP on -3. |
viiteB equ -2 |
Toisen parametrin sijainti suhteessa rekisteriin FP on -2. |
Yllä on määritelty vakiot auttamaan viittaamisessa. Alla on varsinainen aliohjelma. | |
F pushr sp |
Aliohjelma alkaa. Aliohjelman nimi F määritellään tässä. Rivillä on oltava koodia. PUSHR SP Tallettaa rekisterit talteen. |
load r1, arvoA(fp) |
Ladataan r1:een ensimmäisen parametrin arvo. |
load r2, @viiteB(fp) |
Ladataan r2:een toisen parametrin osoittama arvo. |
mul r1, =2 |
r1 = 2a |
mul r2, =3 |
r2 = 3b |
add r1, r2 |
r1 = r1 + r2 = 2a + 3b |
add r1, =1 |
r1 = r1 + 1, joten nyt rekisterissä r1 on 2a+3b+1 |
store r1, paluuarvo(fp) |
Talletetaan palautettava arvo pinoon kohtaan paluuarvo + fp. Tätä merkitään koodissa paluuarvo(fp). Vakion paluuarvo arvo on -4, joten tämän voisi kirjoittaa myös -4(fp). |
popr sp |
Palautetaan rekisterien arvot, jotka oli pushrattu aliohjelman alussa. |
exit sp, =2 |
Poistetaan 2 ylintä arvoa pinosta, jolloin ylimpänä on paluuarvo, ja poistutaan aliohjelmasta. |
Valmista
Huomaa, että parametrien sijainnin numerointi alkaa -2:sta, koska kohdissa -1 ja 0 on vanha PC ja vanha FP. Huomaa, että nämä equ-vakiot eivät saa olla nimetty samoilla nimillä kuin muualla koodissa!!!
(Lähde: Tietokoneen toiminta -kurssin 7.5.2018., tehtävän 4 d malliratkaisu. Malliratkaisun laatinut Harri Kähkönen.)