This is a really brain-dead script to brute-force convert M6800 source code to non-optimal M6809 source code, without any optimization.
- #!/usr/bin/env perl
- # Script to convert M6800 source code to non-optimal M6809 source code.
- # By Joel Matthew Rees, Amagasaki, Japan, September 2018.
- # Copyright 2018 Joel Matthew Rees
- #
- # Permission to use current version for personal research, entertainment,
- # and other non-commercial purposes hereby granted,
- # on condition that authorship and copyright notice are left intact.
- # For other uses, contact the author on social media.
- use v5.010.000;
- use warnings;
- use strict;
- # print $ARGV[ 1 ];
- # Including stuff we don't need, in case I get ambitious.
- # Not including 6801 at this point.
- # Branches are completely unchanged going from 6800 to 6809:
- my $branchlist = "BCC|BCS|BEQ|BGE|BGT|BHI|BLE|BLS|BLT|BMI|BNE|BPL|BRA|BVC|BVS";
- # These implicit ops are unchanged from 6800 to 6809:
- my $implist = "DAA|NOP|RTS";
- # 16-bit pseudo-binary, unchanged from 6800 to 6809:
- my $jumplist ="JMP|JSR";
- # 8-bit binary ops are unchanged going from 6800 to 6809,
- # but we'll eliminate optional space for sociability:
- my $binop8list = "ADC|ADD|AND|BIT|CMP|EOR|SBC|SUB";
- # 8-bit unary ops are unchanged going from 6800to 6809,
- # but we'll eliminate optional space for sociability:
- my $unoplist = "ASL|ASR|CLR|COM|DEC|INC|LSR|LSL|NEG|ROL|ROR|TST";
- # These binary 8-bits have one too many As from 6800 to 6809:
- my $binopLDORST8list = "LDA|ORA|STA";
- # These loads and stores are unchanged from 6800 to 6809,
- # except for optional space:
- my $binopLDST16list = "LDS|LDX|STS|STX";
- # Form changes to CMPX
- my $binop16list = "CPX";
- # Push and pop (pull) are generalized:
- # (6800 had no pshx! -- But fixed in 6801.)
- my $pushmepullyoulist = "PSH|PUL";
- # Transfers are generalized:
- my $transferlist = "TAB|TAP|TBA|TPA|TSX|TXS";
- # These convert to LEA instructions:
- my $lealist = "DES|DEX|INS|INX";
- # Processor status bit handling is generalized:
- # my $flaghandlerlist = "CLC|CLI|CLV|SEC|SEI|SEV";
- my $flaghandleroplist = "CL|SE";
- my $flaghandlerbitlist = "[CIV]";
- # Special handling for inter-accumulator:
- my $b2alist = "ABA|CBA|SBA";
- # Interrupt stuff, form remains the same, different register set,
- # flag working on by hand:
- my $interruptstufflist = "RTI|SWI";
- # Interrupt wait, form changes, semantics change,
- # flag for working on by hand:
- my $waitstufflist = "WAI";
- while ( my $line = <> )
- {
- if ( $line =~ m/^(\w*)\s+FCC\s+(\d+),(.*)$/ )
- {
- my $label = $1;
- my $symlength = $2;
- my $strfield = $3;
- my $symbol = substr( $strfield, 0, $symlength );
- my $leftovers = substr( $strfield, $symlength );
- my $strlength = length( $symbol );
- if ( $strlength < $symlength )
- {
- print "$label\tFCC error\t'$symbol' not complete to "
- . "$symlength characters (only $strlength). ****error***";
- }
- else
- {
- my $fullsymbol = $symbol;
- if ( $leftovers =~ m/^(\S+)(.*)$/ )
- {
- $fullsymbol .= $1;
- $leftovers = $2;
- }
- print "$label\tFCC\t'$symbol'\t; '$fullsymbol'";
- }
- if ( length( $leftovers ) > 0 )
- {
- print " : $leftovers";
- }
- print "\n";
- }
- elsif ( $line =~ m/^(\w*)\s+(${pushmepullyoulist})\s*(A|B)\s*(.*)$/ )
- {
- my $label = $1;
- my $operator = $2;
- my $operand = $3;
- my $comments = $4;
- print "$label\t${operator}S $operand\t; $comments\n";
- }
- elsif ( $line =~ m/^(\w*)\s+(${binopLDORST8list})\s*(A|B)\s+(.*)$/ )
- {
- my $label = $1;
- my $operator = $2;
- my $operand = $3;
- my $comments = $4; # Fudging, comments includes memory operand.
- my $op2letter = substr( $operator, 0, 2 );
- print "$label\t${op2letter}$operand $comments\n";
- }
- elsif ( $line =~ m/^(\w*)\s+(${lealist})(.*)$/ )
- {
- my $label = $1;
- my $operator = $2;
- my $comments = $3;
- if ( $operator =~ m/IN(\w)/ )
- { $operator = "LEA$1 1,$1";
- }
- elsif ( $operator =~ m/DE(\w)/ )
- { $operator = "LEA$1 -1,$1";
- }
- print "$label\t${operator}\t; $comments\n";
- }
- elsif ( $line =~ m/^(\w*)\s+(${unoplist})\s*(A|B)(.*)$/ )
- {
- my $label = $1;
- my $operator = $2;
- my $operand = $3;
- my $comments = $4;
- print "$label\t$operator$operand\t;$comments\n";
- }
- elsif ( $line =~ m/^(\w*)\s+(${unoplist})\s+(.*)$/ )
- {
- my $label = $1;
- my $operator = $2;
- my $operand = $3; # Fudging, operand includes any comments.
- print "$label\t$operator $operand\n";
- }
- elsif ( $line =~ m/^(\w*)\s+(${b2alist})(.*)$/ )
- {
- my $label = $1;
- my $operator = $2;
- my $comments = $3;
- my $realoperator = $operator;
- print "$label\tPSHS B\t; ** emulating $operator:\n";
- if ( $operator eq "ABA" )
- { $realoperator = "ADDA";
- }
- elsif ( $operator eq "CBA" )
- { $realoperator = "CMPA";
- }
- elsif ( $operator eq "SBA" )
- { $realoperator = "SUBA";
- }
- print "\t$realoperator ,S+\t; $comments\n";
- }
- elsif ( $line =~ m/^(\w*)\s+(${binop8list})\s*(A|B)\s+(.+)$/ )
- {
- my $label = $1;
- my $operator = $2;
- my $operand = $3;
- my $comments = $4; # Fudging, comments includes memory operand.
- print "$label\t${operator}$operand $comments\n";
- }
- elsif ( $line =~ m/^(\w*)\s+(${transferlist})(.*)$/ )
- {
- my $label = $1;
- my $operator = $2;
- my $comments = $3;
- my $source = substr( $operator, 1, 1 );
- my $destination = substr( $operator, 2, 1 );
- if ( $source eq "P" )
- { $source = "CCR";
- }
- if ( $destination eq "P" )
- { $destination = "CCR";
- }
- print "$label\tTFR $source,$destination\t; $operator : $comments\n";
- }
- elsif ( $line =~ m/^(\w*)\s+(${flaghandleroplist})($flaghandlerbitlist)(.*)$/ )
- { # Bits: EFHINZVC, thus I is 0x10, V is 0x02, and C is 0x01.
- my $label = $1;
- my $operator = $2;
- my $bit = $3;
- my $comments = $4;
- my $realbits =
- ( $bit eq "C" )
- ? "\$01"
- : ( $bit eq "V" )
- ? "\$02"
- : ( $bit eq "I" )
- ? "\$10"
- : $bit;
- my $realoperator = $operator;
- if ( $operator eq "CL" )
- { $realoperator = "ANDCC";
- $realbits = "~" . $realbits;
- }
- elsif ( $operator eq "SE" )
- { $realoperator = "ORCC"
- }
- print "$label\t$realoperator #$realbits\t; ${operator}${bit} : $comments\n";
- }
- elsif ( $line =~ m/^(\w*)\s+($binop16list)\s+(.+)$/ )
- {
- my $label = $1;
- my $operator = $2;
- my $operand = $3; # Fudging, operand includes any comments.
- my $reg16bit = substr( $operator, 2, 1 );
- print "$label\tCMP$reg16bit\t$operand\n";
- }
- elsif ( $line =~ m/^(\w*)\s+(${branchlist})\s+(\*[+-]\$?[0-9A-Fa-f]+)(.*)$/ )
- {
- my $label = $1;
- my $operator = $2;
- my $operand = $3;
- my $comments = $4;
- print "$label\t$operator $operand\t; $comments\n"
- . "\t****WARNING**** HARD OFFSET: $operand ****\n";
- }
- elsif ( $line =~ m/^(\w*)\s+(${interruptstufflist})(.*)$/ )
- {
- my $label = $1;
- my $operator = $2;
- my $operand = $3; # Fudging, operand includes any comments.
- print "$label\t$operator$operand\n"
- . "\t****WARNING**** Interrupt routines must change!! ****\n";
- }
- elsif ( $line =~ m/^(\w*)\s+(${waitstufflist})(.*)$/ )
- {
- my $label = $1;
- my $operator = $2;
- my $operand = $3; # Fudging, operand includes any comments.
- print "$label\tC$operator #\$EF\t; $operand\n"
- . "\t****WARNING**** WAI must change to CWAI!! ****\n";
- }
- else
- {
- print $line;
- }
- }