石原 博の覚書

電子工作に関する日々の覚書を記載します

KBC-Z84015EMが動いた

2021-11-23 16:24:09 | 日記

デジット閉店セールで売られていたジャンク基板「KBC-Z84015EM」が電脳伝説さんのサイト「https://vintagechips.wordpress.com/2021/06/08/kbc-z84015s/」で解説されておりBASICが動くようだ。そこで私も手持ちのEEPROMで動かしてみた。

手持ちのEEPROMはW27C512で64KB。ROM領域は32KBだが32KBと64KBROMの差はpin1で、32KBROMではVpp、64KBROMではA15となっている。ボード上ではpin1はV+に接続されているため、W27C512でも後半の8000-FFFFにデータを書き込めば使える。

有り難いことに電脳伝説さんのところにHexファイルがあるので、TL866Aを使って書き込み。

TO Buffer Start Addrを 08000としてMBKBC.HEXをロード


書き込み

 

 

動いた。バックスペースはコントロールH

 

 


FlashForthで割り込み

2021-11-19 20:42:13 | 日記

前回SC1602のコントロールを行ったので、応用として時計を作ってみた。タイマーを利用した割り込みを使用している。「https://sourceforge.net/p/flashforthforarduino/wiki/Timer/」を参考にさせていただいた。

割り込みにはTimer2を利用している。本当は1秒毎の割り込みが欲しかったが8ビットタイマのためプリスケーラを最大利用(1024)しても、16MHzクロックでは最大(1/16MHz)*1024*256=16msecにしかならない。そこで156(0x9c)をセットして9.984msec毎に割り込みを行い、100回カウントすることで1秒作成することにした。

その上で、' start is turnkey とすることで起動時に自動的に時計が動くようにしている。

-clock

marker -clock

variable data 6 allot ( HHMMSS )
: inch ( -- ) data    dup c@ 23 = if      0 else dup c@ 1+ then swap c! ;
: incm ( -- ) data 1+ dup c@ 59 = if inch 0 else dup c@ 1+ then swap c! ; 
: incs ( -- ) data 2+ dup c@ 59 = if incm 0 else dup c@ 1+ then swap c! ; 

: disp2 ( pos -- )
    dup 3 * >r data + c@ #10 u/mod $30 + swap $30 +
    r@ 1+ wrch r@ wrch r> ; 
: disp ( -- )
    0 disp2 2 + [char] : swap wrch
    1 disp2 2 + [char] : swap wrch
    2 disp2 drop ;

: set ( HH MM SS -- )
    data 2+ c! data 1+ c! data c! ;

variable count
: t2_compa_isr ( 10msec interrupt )
  1 count +!
  count @ 99 > if
    0 count !
    incs
    disp
  then
;i

$b0 constant TCCR2A
$b1 constant TCCR2B
$b3 constant OCR2A
$70 constant TIMSK2
#8 constant OC2Aaddr

: start
  #15 ms
  init \ lcd initialize

  0 TCCR2B c!  \ stop timer2
  $9c OCR2A c! \ $9c=156 (1/16MHz)*1024*156=9.984msec
  ['] t2_compa_isr OC2Aaddr int!

  0 count !
  2 TCCR2A c! \ ctc mode
  7 TCCR2B c! \ prescaler = 1024
  2 TIMSK2 c! \ interrupt enable
;


FlashForthでSC1602をコントロール

2021-11-14 17:43:34 | 日記

良くあるSC1602をコントロールしてみる。後から調べると「ff5-tutorial-guide-2014-05-12.pdf」のp.41に、同様のものがあったがpic用。こちらはavr(arduino pro mini)なので、なにかの参考になるかと思う。

ソース記載のとおり4ビットパラレル。waitは適当。読み出しは行っていない。CGRAMへ書き込みも簡単に試してみた。

=========================
-lcd
marker -lcd

\ SC1602(HD44780)
\ portb0..3 -> DB4..7
\ portb4 -> R/W
\ portb5 -> RS
\ portc0 -> E

$24 constant ddrb
$25 constant portb
$27 constant ddrc
$28 constant portc

: b2n ( d7..d0 -- d3..d0 d7..d4 ) dup $0f and swap 4 rshift ;
: pbwr ( n -- )
    portb c! 1 ms 1 portc mset 1 portc mclr ; ( write portb, pulse portc1 )
: wrd ( n -- )  b2n $20 or pbwr $20 or pbwr ; ( write data $20 RS_HIGH )
: wrc ( n -- )  b2n pbwr pbwr ; ( write command $20 RS_LOW )
: clr ( -- )       $01 wrc ;
: home ( -- )      $02 wrc ;
: setddpos ( pos -- ) b2n $8 or pbwr pbwr ; 
: setcgpos ( pos -- ) b2n $4 or pbwr pbwr ; 
: setcg ( x0 x1 x2 x3 x4 x5 x6 n -- ) ( write CG RAM n=address x0..x6=pattern )
    3 lshift 7 for dup r@ + setcgpos swap wrd next drop ;

: init ( -- )
   $3f ddrb c! ( RS, R/W, DB7, DB6, DB5, DB4 )
   $01 ddrc c! ( E )

   $3 pbwr 5 ms
   $3 pbwr 1 ms
   $3 pbwr
   $2 pbwr
   $28 wrc ( DL=0 4bit  N=1 2line  F=0 5x10dot )
   $0c wrc ( Display ON/OFF )
           ( D=1 display on  C=0 block cursor off  B=0 blink off )
   clr     ( Clear Display )
   $06 wrc ( Entry Mode Set I/D=1 右シフト  S=0 表示シフトOFF )
;

: wrch ( c pos[0..31] -- )
    dup 32 < if
      dup 15 > if 48 + then setddpos wrd
    then ;

init
$00 $1f $11 $11 $11 $1f $00 0 setcg ( square )
$00 $04 $0a $11 $0a $04 $00 1 setcg ( rhombus )

0 0 wrch
1 1 wrch
$41 2 wrch
$42 3 wrch

 


PL/Mでコンパイル

2021-11-14 14:26:50 | 日記

もう少しPL/M80で遊んで見る。

オークションで「PL/Mマイクロコンピュータプログラミング」を入手してみたが若干異なる。手続きにREENTRANTをつければ再帰出来るとあるが、うまく動かない。

PLMLANG.DOCを確認すると予約語にないし、また「Procedures may not be recursive. 」とある。8080でスタック上の操作は大変だし、サポートしていないんでしょう。

予約後は以下のとおり。
>The following are all reserved words, and may not be used as
>identifier names:
>
>  ADDRESS        DATA           EOF            LABEL          PROCEDURE
>  AND            DECLARE        GO             LITERALLY      RETURN
>  BASED          DISABLE        GOTO           MINUS          THEN
>  BY             DO             HALT           MOD            TO
>  BYTE           ELSE           IF             NOT            WHILE
>  CALL           ENABLE         INITIAL        OR             XOR
>  CASE           END            INTERRUPT      PLUS

本当はCP/MUG030のBASIC.PLMを動かそうと思っていたのだが、BASIC.PLMで使われている$INCLUDEやEXTERNALをサポートしていないようなので諦める。


PL/M80ではどんなコードが出るか興味があったので、簡単なプログラムをコンパイルしてみた。
===ソース======
100H:
DECLARE BDOS LITERALLY '05H';
DECLARE GA8 BYTE;
DECLARE GA16 ADDRESS;

MON1: PROCEDURE(F,A);
    DECLARE F BYTE,
    A ADDRESS;
    GO TO BDOS;
    END MON1;

DECLARE
    CR LITERALLY '13',
    LF LITERALLY '10',
    TRUE LITERALLY '1',             
    FALSE LITERALLY '0';

PRINTCHAR: PROCEDURE(CHAR);
    DECLARE CHAR BYTE;
    CALL MON1(2,CHAR);
    END PRINTCHAR;

CRLF: PROCEDURE;
    CALL PRINTCHAR(CR);
    CALL PRINTCHAR(LF);
    END CRLF;

PRINT: PROCEDURE(A);
    DECLARE A ADDRESS;
    CALL MON1(9,A);
    END PRINT;

RET$CPM: PROCEDURE;
    CALL MON1(0,0);
    END RESET;

PRINTVALUE: PROCEDURE(V);
    DECLARE (D, ZERO) BYTE,
            (K, V) ADDRESS;
    K = 10000;
    ZERO = FALSE;
    DO WHILE K <> 0;
        D = LOW(V / K); V = V MOD K;
        K = K / 10;
        IF ZERO OR D <> 0 THEN
            DO;
                ZERO = TRUE;
                CALL PRINTCHAR('0' + D);
            END;
        ELSE
            CALL PRINTCHAR(' ');
    END;
    END PRINTVALUE;

ARITH:
    DECLARE N LITERALLY '4';
    DECLARE LA8 BYTE;
    DECLARE LA16 ADDRESS;
    DO;
        GA8 = 100;
        GA16 = 200;
        LA8 = 3;
        LA16 = 4;

/* ADD8 */
        CALL PRINT(.'ARITH_ADD8=$');
        CALL PRINTVALUE(GA8 + LA8 + 5);
        CALL CRLF;
/* ADD16 */
        CALL PRINT(.'ARITH_ADD16=$');
        CALL PRINTVALUE(GA16 + LA16 + 6);
        CALL CRLF;
/* SUB8 */
        CALL PRINT(.'ARITH_SUB8=$');
        CALL PRINTVALUE(GA8 - LA8 - 7);
        CALL CRLF;
/* SUB16 */
        CALL PRINT(.'ARITH_SUB16=$');
        CALL PRINTVALUE(GA16 - LA16 - 8);
        CALL CRLF;
/* MUL8 */
        CALL PRINT(.'ARITH_MUL8=$');
        CALL PRINTVALUE(GA8 * LA8 * 9);
        CALL CRLF;
/* MUL16 */
        CALL PRINT(.'ARITH_MUL16=$');
        CALL PRINTVALUE(GA16 * LA16 * 10);
        CALL CRLF;
/* DIV8 */
        CALL PRINT(.'ARITH_DIV8=$');
        CALL PRINTVALUE(GA8 / LA8 / 11);
        CALL CRLF;
/* MUL16 */
        CALL PRINT(.'ARITH_DIV16=$');
        CALL PRINTVALUE(GA16 / LA16 / 12);
        CALL CRLF;

        CALL RET$CPM; 
    END ARITH;                                                                      
EOF
=======================

実行結果と逆アセンブル結果(わかりやすくするため対応するソースを記載)
=======================
>DDT ARITH.HEX
DDT VERS 1.4
NEXT  PC
03BF 0000
-G100
ARITH ADD8=  108
ARITH ADD16=  210
ARITH SUB8=   90
ARITH SUB16=  188
ARITH MUL8= 2700
ARITH MUL16= 8000
ARITH DIV8=    3
ARITH DIV16=    4

RunCPM Version 5.3 (CP/M 2.2 60K)

>DDT ARITH.HEX
DDT VERS 1.4
NEXT  PC
03BF 0000
-L100,3BE
  0100  LXI  SP,03E6
  0103  JMP  01F2
/*
MON1: PROCEDURE(F,A);
    DECLARE F BYTE,
    A ADDRESS;
    GO TO BDOS;
    END MON1;
*/
  0106  LXI  H,03EF
  0109  MOV  M,C
  010A  INR  L
  010B  MOV  M,E
  010C  INX  H
  010D  MOV  M,D
  010E  JMP  0005
  0111  RET  
/*
PRINTCHAR: PROCEDURE(CHAR);
    DECLARE CHAR BYTE;
    CALL MON1(2,CHAR);
    END PRINTCHAR;
*/
  0112  LXI  H,03F3
  0115  MOV  M,C
  0116  MVI  C,02
  0118  MOV  E,M
  0119  MVI  D,00
  011B  CALL 0106
  011E  RET  
/*
CRLF: PROCEDURE;
    CALL PRINTCHAR(CR);
    CALL PRINTCHAR(LF);
    END CRLF;
*/
  011F  MVI  C,0D
  0121  CALL 0112
  0124  MVI  C,0A
  0126  CALL 0112
  0129  RET  
/*
PRINT: PROCEDURE(A);
    DECLARE A ADDRESS;
    CALL MON1(9,A);
    END PRINT;
*/
  012A  LXI  H,03F4
  012D  MOV  M,C
  012E  INX  H
  012F  MOV  M,B
  0130  MVI  C,09
  0132  DCR  L
  0133  MOV  E,M
  0134  INR  L
  0135  MOV  D,M
  0136  CALL 0106
  0139  RET  
/*
RET$CPM: PROCEDURE; 
    CALL MON1(0,0);
    END RESET;
*/
  013A  MVI  C,00
  013C  MVI  E,00
  013E  MVI  D,00
  0140  CALL 0106
  0143  RET  
/*
PRINTVALUE: PROCEDURE(V);
    DECLARE (D, ZERO) BYTE,
            (K, V) ADDRESS;
    K = 10000;
    ZERO = FALSE;
    DO WHILE K <> 0;
        D = LOW(V / K); V = V MOD K;
        K = K / 10;
        IF ZERO OR D <> 0 THEN
            DO;
                ZERO = TRUE;
                CALL PRINTCHAR('0' + D);
            END;
        ELSE
            CALL PRINTCHAR(' ');
    END;
    END PRINTVALUE;
*/
  0144  LXI  H,03F6
  0147  MOV  M,C  /* V */
  0148  INX  H
  0149  MOV  M,B
  014A  MVI  L,FA  /* K = 10000 */
  014C  MVI  M,10
  014E  INX  H
  014F  MVI  M,27
  0151  MVI  L,F9  /* ZERO = FALSE */
  0153  MVI  M,00
  0155  LXI  H,03FA  /* K */
  0158  MOV  A,M
  0159  INR  L
  015A  MOV  B,M
  015B  SUI  00
  015D  MOV  C,A
  015E  MOV  A,B
  015F  SBI  00
  0161  ORA  C
  0162  JZ   01F1  /* K == 0 */
  0165  LXI  H,03FA  /* K */
  0168  MOV  E,M
  0169  INR  L
  016A  MOV  D,M
  016B  MVI  L,F6  /* V */
  016D  MOV  C,M
  016E  INR  L
  016F  MOV  B,M
  0170  JMP  01A0
/*
  DIV BC = BC / DE
*/
  0173  MOV  A,D
  0174  CMA  
  0175  MOV  D,A
  0176  MOV  A,E
  0177  CMA  
  0178  MOV  E,A
  0179  INX  D
  017A  LXI  H,0000
  017D  MVI  A,11
  017F  PUSH H
  0180  DAD  D
  0181  JNC  0185
  0184  XTHL 
  0185  POP  H
  0186  PUSH PSW  
  0187  MOV  A,C
  0188  RAL  
  0189  MOV  C,A
  018A  MOV  A,B
  018B  RAL  
  018C  MOV  B,A
  018D  MOV  A,L
  018E  RAL  
  018F  MOV  L,A
  0190  MOV  A,H
  0191  RAL  
  0192  MOV  H,A
  0193  POP  PSW  
  0194  DCR  A
  0195  JNZ  017F
  0198  ORA  A
  0199  MOV  A,H
  019A  RAR  
  019B  MOV  D,A
  019C  MOV  A,L
  019D  RAR  
  019E  MOV  E,A
  019F  RET  
/*
*/
  01A0  CALL 0173  /* DIV   V/K */
  01A3  LXI  H,03F8  /* D */
  01A6  MOV  M,C
  01A7  LXI  H,03FA  /* K */
  01AA  MOV  E,M
  01AB  INR  L
  01AC  MOV  D,M
  01AD  MVI  L,F6  /* V */
  01AF  MOV  C,M
  01B0  INR  L
  01B1  MOV  B,M
  01B2  CALL 0173  /* DIV */
  01B5  LXI  H,03F6  /* V */
  01B8  MOV  M,E
  01B9  INX  H
  01BA  MOV  M,D
  01BB  MVI  E,0A  /* 10 */
  01BD  MVI  D,00
  01BF  LXI  H,03FA  /* K */
  01C2  MOV  C,M
  01C3  INR  L
  01C4  MOV  B,M
  01C5  CALL 0173  /* DIV */
  01C8  LXI  H,03FA  /* K */
  01CB  MOV  M,C
  01CC  INX  H
  01CD  MOV  M,B
  01CE  MVI  L,F8  /* D */
  01D0  MOV  A,M
  01D1  SUI  00
  01D3  ADI  FF
  01D5  SBB  A
  01D6  INR  L
  01D7  ORA  M
  01D8  RRC  
  01D9  JNC  01E9
  01DC  MVI  M,01
  01DE  DCR  L
  01DF  MOV  A,M
  01E0  ADI  30  /* ' ' */
  01E2  MOV  C,A
  01E3  CALL 0112
  01E6  JMP  0155
  01E9  MVI  C,20
  01EB  CALL 0112
  01EE  JMP  0155
  01F1  RET  
/*
        GA8 = 100;
        GA16 = 200;
        LA8 = 3;
        LA16 = 4;
*/
  01F2  LXI  H,03EB /* GA8 = 100(64H) */
  01F5  MVI  M,64
  01F7  INR  L
  01F8  MVI  M,C8  /* GA16 = 200(0C8H) */
  01FA  INX  H
  01FB  MVI  M,00
  01FD  MVI  L,FD  /* LA8 = 3 */
  01FF  MVI  M,03
  0201  INR  L
  0202  MVI  M,04  /* LA16 = 4 */
  0204  INX  H
  0205  MVI  M,00
  0207  JMP  0216
/*
        CALL PRINT(.'ARITH_ADD8=');
        CALL PRINTVALUE(GA8 + LA8 + 5);
        CALL CRLF;
*/
  020A  MOV  B,C
  020B  MOV  D,D
  020C  MOV  C,C
  020D  MOV  D,H
  020E  MOV  C,B
  020F  ??=  20  /* _はスペースになる */
  0210  MOV  B,C
  0211  MOV  B,H
  0212  MOV  B,H
  0213  ??=  38
  0214  DCR  A
  0215  INR  H
/*
*/
  0216  LXI  B,020A
  0219  CALL 012A
  021C  LXI  H,03FD  /* LA8 */
  021F  MOV  A,M
  0220  MVI  L,EB  /* GA8 */
  0222  ADD  M
  0223  ADI  05  /* 5 */
  0225  MVI  B,00
  0227  MOV  C,A
  0228  CALL 0144
  022B  CALL 011F
  022E  JMP  023E
/*
        CALL PRINT(.'ARITH_ADD16=');
        CALL PRINTVALUE(GA16 + LA16 + 6);
        CALL CRLF;
*/
  0231  MOV  B,C
  0232  MOV  D,D
  0233  MOV  C,C
  0234  MOV  D,H
  0235  MOV  C,B
  0236  ??=  20
  0237  MOV  B,C
  0238  MOV  B,H
  0239  MOV  B,H
  023A  LXI  SP,3D36
  023D  INR  H
/*
*/
  023E  LXI  B,0231
  0241  CALL 012A
  0244  LXI  H,03EC  /* GA16 */
  0247  MOV  C,M
  0248  INR  L
  0249  MOV  B,M
  024A  LHLD 03FE  /* LA16 */
  024D  DAD  B
  024E  XCHG 
  024F  LXI  B,0006  /* 6 */
  0252  XCHG 
  0253  DAD  B
  0254  XCHG 
  0255  MOV  C,E
  0256  MOV  B,D
  0257  CALL 0144
  025A  CALL 011F
  025D  JMP  026C
/*
        CALL PRINT(.'ARITH_SUB8=');
        CALL PRINTVALUE(GA8 - LA8 - 7);
        CALL CRLF;
*/
  0260  MOV  B,C
  0261  MOV  D,D
  0262  MOV  C,C
  0263  MOV  D,H
  0264  MOV  C,B
  0265  ??=  20
  0266  MOV  D,E
  0267  MOV  D,L
  0268  MOV  B,D
  0269  ??=  38
  026A  DCR  A
  026B  INR  H
/*
*/
  026C  LXI  B,0260
  026F  CALL 012A
  0272  LXI  H,03EB  /* GA8 */
  0275  MOV  A,M
  0276  MVI  L,FD  /* LA8 */
  0278  SUB  M
  0279  SUI  07  /* 7 */
  027B  MVI  B,00
  027D  MOV  C,A
  027E  CALL 0144
  0281  CALL 011F
  0284  JMP  0294
/*
        CALL PRINT(.'ARITH_SUB16=');
        CALL PRINTVALUE(GA16 - LA16 - 8);
        CALL CRLF;
*/
  0287  MOV  B,C
  0288  MOV  D,D
  0289  MOV  C,C
  028A  MOV  D,H
  028B  MOV  C,B
  028C  ??=  20
  028D  MOV  D,E
  028E  MOV  D,L
  028F  MOV  B,D
  0290  LXI  SP,3D36
  0293  INR  H
/*
*/
  0294  LXI  B,0287
  0297  CALL 012A
  029A  LXI  H,03EC  /* GA16 */
  029D  MOV  A,M
  029E  INR  L
  029F  MOV  B,M
  02A0  MVI  L,FE  /* LA16 */
  02A2  SUB  M
  02A3  INR  L
  02A4  MOV  C,A
  02A5  MOV  A,B
  02A6  SBB  M
  02A7  MOV  B,A
  02A8  MOV  A,C
  02A9  SUI  08  /* 8 */
  02AB  MOV  C,A
  02AC  MOV  A,B
  02AD  SBI  00
  02AF  MOV  B,A
  02B0  CALL 0144
  02B3  CALL 011F
  02B6  JMP  02C5
/*
        CALL PRINT(.'ARITH_MUL8=');
        CALL PRINTVALUE(GA8 * LA8 * 9);
        CALL CRLF;
*/
  02B9  MOV  B,C
  02BA  MOV  D,D
  02BB  MOV  C,C
  02BC  MOV  D,H
  02BD  MOV  C,B
  02BE  ??=  20
  02BF  MOV  C,L
  02C0  MOV  D,L
  02C1  MOV  C,H
  02C2  ??=  38
  02C3  DCR  A
  02C4  INR  H
/*
*/
  02C5  LXI  B,02B9
  02C8  CALL 012A
  02CB  LXI  H,03FD  /* GA8 */
  02CE  MOV  E,M
  02CF  MVI  D,00
  02D1  MVI  L,EB  /* LA8 */
  02D3  MOV  C,M
  02D4  MVI  B,00
  02D6  JMP  02FC
/*
  MUL  DE = BC * DE
*/
  02D9  MOV  A,C
  02DA  SUB  E
  02DB  MOV  A,B
  02DC  SBB  D
  02DD  JP   02E5
  02E0  MOV  H,B
  02E1  MOV  L,C
  02E2  XCHG 
  02E3  MOV  B,H
  02E4  MOV  C,L
  02E5  LXI  H,0000
  02E8  XCHG 
  02E9  MOV  A,B
  02EA  ORA  C
  02EB  RZ   
  02EC  XCHG 
  02ED  MOV  A,B
  02EE  RAR  
  02EF  MOV  B,A
  02F0  MOV  A,C
  02F1  RAR  
  02F2  MOV  C,A
  02F3  JNC  02F7
  02F6  DAD  D
  02F7  XCHG 
  02F8  DAD  H
  02F9  JMP  02E9
/*
*/
  02FC  CALL 02D9  /* MUL (LA8 * GA8) */
  02FF  LXI  H,03E8  /* WK */
  0302  MOV  M,E
  0303  INR  L
  0304  MOV  M,D
  0305  MVI  E,09  /* 9 */
  0307  MVI  D,00
  0309  LXI  H,03E8  /* WK */
  030C  MOV  C,M
  030D  INR  L
  030E  MOV  B,M
  030F  CALL 02D9  /* MUL (WK * 9) */
  0312  MOV  C,E
  0313  MOV  B,D
  0314  CALL 0144
  0317  CALL 011F
  031A  JMP  032A
/*
        CALL PRINT(.'ARITH_MUL16=');
        CALL PRINTVALUE(GA16 * LA16 * 10);
        CALL CRLF;
*/
  031D  MOV  B,C
  031E  MOV  D,D
  031F  MOV  C,C
  0320  MOV  D,H
  0321  MOV  C,B
  0322  ??=  20
  0323  MOV  C,L
  0324  MOV  D,L
  0325  MOV  C,H
  0326  LXI  SP,3D36
  0329  INR  H
/*
*/
  032A  LXI  B,031D
  032D  CALL 012A
  0330  LXI  H,03FE  /* LA16 */
  0333  MOV  E,M
  0334  INR  L
  0335  MOV  D,M
  0336  MVI  L,EC  /* GA16 */
  0338  MOV  C,M
  0339  INR  L
  033A  MOV  B,M
  033B  CALL 02D9  /* MUL (GA16 * LA16) */
  033E  LXI  H,03E6 /* WK */
  0341  MOV  M,E
  0342  INR  L
  0343  MOV  M,D
  0344  MVI  E,0A  /* 10 */ 
  0346  MVI  D,00
  0348  LXI  H,03E6  /* WK */
  034B  MOV  C,M
  034C  INR  L
  034D  MOV  B,M
  034E  CALL 02D9  /* MUL (WK * 10) */
  0351  MOV  C,E
  0352  MOV  B,D
  0353  CALL 0144
  0356  CALL 011F
  0359  JMP  0368
/*
        CALL PRINT(.'ARITH_DIV8=');
        CALL PRINTVALUE(GA8 / LA8 / 11);
        CALL CRLF;
*/
  035C  MOV  B,C
  035D  MOV  D,D
  035E  MOV  C,C
  035F  MOV  D,H
  0360  MOV  C,B
  0361  ??=  20
  0362  MOV  B,H
  0363  MOV  C,C
  0364  MOV  D,M
  0365  ??=  38
  0366  DCR  A
  0367  INR  H
/*
*/
  0368  LXI  B,035C
  036B  CALL 012A
  036E  LXI  H,03FD  /* LA8 */
  0371  MOV  E,M
  0372  MVI  D,00
  0374  MVI  L,EB  /* GA8 */
  0376  MOV  C,M
  0377  MVI  B,00
  0379  CALL 0173  /* DIV (GA8 / LA8) */
  037C  MVI  E,0B  /* 11 */
  037E  MVI  D,00
  0380  CALL 0173  /* DIV (WK / 11) */
  0383  CALL 0144
  0386  CALL 011F
  0389  JMP  0399
/*
        CALL PRINT(.'ARITH_DIV16=');
        CALL PRINTVALUE(GA16 / LA16 / 12);
        CALL CRLF;
*/
  038C  MOV  B,C
  038D  MOV  D,D
  038E  MOV  C,C
  038F  MOV  D,H
  0390  MOV  C,B
  0391  ??=  20
  0392  MOV  B,H
  0393  MOV  C,C
  0394  MOV  D,M
  0395  LXI  SP,3D36
  0398  INR  H
/*
*/
  0399  LXI  B,038C
  039C  CALL 012A
  039F  LXI  H,03FE  /* LA16 */
  03A2  MOV  E,M
  03A3  INR  L
  03A4  MOV  D,M
  03A5  MVI  L,EC  /* GA16 */
  03A7  MOV  C,M
  03A8  INR  L
  03A9  MOV  B,M
  03AA  CALL 0173  /* DIV (GA16 / LA16) */
  03AD  MVI  E,0C  /* 12 */
  03AF  MVI  D,00
  03B1  CALL 0173  /* DIV (WK / 12) */
  03B4  CALL 0144
  03B7  CALL 011F
  03BA  CALL 013A
  03BD  EI   
  03BE  HLT  
  03BF  
-


特徴的なのは、掛け算・割り算のコードがランタイムではなく、コンパイル時に埋め込まれる点。掛け算や割り算が使われた時点でルーチンが埋め込まれ、他の場所からも使用されている点である。
ローカルの変数も静的に領域を確保しており再帰は無理。最適化なのか普通なら INX H となるところに INR L を使っているのではあるが、CALLのあとのRETとかJMPのあとのRETとか、人だったら書かないだろうなあというコードではある。


PL/MでHELLO WORLD

2021-11-06 18:13:41 | 日記

せっかく動いたPL/M。もう少し遊んで見る。

・よくあるHello worldの例

Unofficial CP/M(http://www.cpm.z80.de/source.html)の「EARLY CP/M SOURCE」内のBDOS.PLMやCCP.PLMを参考に作成

===========================================

100H:                              /* [1] */
DECLARE BDOS LITERALLY '05H';

MON1: PROCEDURE(F,A);              /* [2] */
    DECLARE F BYTE,
    A ADDRESS;
    GO TO BDOS;
    END MON1;

DECLARE
    CR LITERALLY '13',
    LF LITERALLY '10';

PRINTCHAR: PROCEDURE(CHAR);
    DECLARE CHAR BYTE;
    CALL MON1(2,CHAR);
    END PRINTCHAR;

CRLF: PROCEDURE;
    CALL PRINTCHAR(CR);
    CALL PRINTCHAR(LF);
    END CRLF;

PRINT: PROCEDURE(A);
    DECLARE A ADDRESS;
    CALL MON1(9,A);
    END PRINT;

RET$CPM: PROCEDURE;                /* [3] */
    CALL MON1(0,0);
    END RESET;

HELLO:
DO;
    CALL PRINT(.'HELLO WORLD$');   /* [4] */
    CALL CRLF;
    CALL RET$CPM;                  /* [3] */
END HELLO;                                                                      
EOF                                /* [5] */

=========================================

・注意点
  [1]スタートアドレスが必要。ないと0スタート
  [2]入出力は自前でBDOSを呼ぶ
   昔からCP/MのBDOS Callが、CとDEにパラメタを入れるのが不思議だったのだが、PL/Mはそれに合ったコードを出す
   (初期のCP/MはPL/Mで書かれていたようなので、卵が先か鶏が先かなのかもしれない)
  [3]コンパイルしたコードの最後は、EI HALTになる
   これはCP/Mから実行する場合は困る。そこでRET$CPMというCP/Mに戻るコードを加えた。
  [4]文字はUpperCaseのみ。例え文字列であってもLowerCaseはだめ(Hello worldとは書けない)
  [5]最後にEOFが必要
    [6]コンパイル時のエラーは、fort.12の最後を見ること。「1 PROGRAM ERROR」とか出ている。

・コンパイル結果と実行結果

>ddt HELLO.HEX
DDT VERS 1.4
NEXT  PC
0161 0000
-l100,160
  0100  LXI  SP,01F8
  0103  JMP  0144
  0106  LXI  H,01F9             ;MON1 固定なので再帰は出来ない
  0109  MOV  M,C                ;第1引数はBYTEで1バイト
  010A  INR  L         ;INX HではないのはHLは奇数なことを見越した最適化かな
  010B  MOV  M,E                ;第2引数はADDRESSで2バイト
  010C  INX  H
  010D  MOV  M,D
  010E  JMP  0005               ;BDOS CALL
  0111  RET  
  0112  LXI  H,01FD             ;PRINTCHAR
  0115  MOV  M,C                ;第1引数はBYTEで1バイト  メモリ経由でEレジスタへ
  0116  MVI  C,02
  0118  MOV  E,M
  0119  MVI  D,00
  011B  CALL 0106
  011E  RET  
  011F  MVI  C,0D               ;CRLF
  0121  CALL 0112
  0124  MVI  C,0A
  0126  CALL 0112
  0129  RET  
  012A  LXI  H,01FE             ;PRINT
  012D  MOV  M,C                ;第1引数はADRESSで2バイト メモリ経由でDEへ
  012E  INX  H                  ;HLは偶数なのでINX H ?
  012F  MOV  M,B
  0130  MVI  C,09
  0132  DCR  L
  0133  MOV  E,M
  0134  INR  L
  0135  MOV  D,M
  0136  CALL 0106
  0139  RET  
  013A  MVI  C,00               ;RET$CPM
  013C  MVI  E,00
  013E  MVI  D,00
  0140  CALL 0106
  0143  RET  
  0144  JMP  0153               ;メインルーチンへ
  0147  MOV  C,B
  0148  MOV  B,L
  0149  MOV  C,H
  014A  MOV  C,H
  014B  MOV  C,A
  014C  ??=  20
  014D  MOV  D,A
  014E  MOV  C,A
  014F  MOV  D,D
  0150  MOV  C,H
  0151  MOV  B,H
  0152  INR  H
  0153  LXI  B,0147             ;メインルーチン 0x147は DB 'HELLO WORLD$'のアドレス。
  0156  CALL 012A               ;PRINT
  0159  CALL 011F               ;CRLF
  015C  CALL 013A               ;RET$CPM
  015F  EI                      ;ここへは戻らない
  0160  HLT  
  0161  
-

 

>load HELLO.HEX

FIRST ADDRESS 0100
LAST  ADDRESS 0160
BYTES READ    0061
RECORDS WRITTEN 01


E>HELLO
HELLO WORLD

RunCPM Version 5.3 (CP/M 2.2 60K)