I am following the A totorial in the book using 6502 assembly language by randy hyde there is a part in chapter 14, section 7 where he writes "jsr incrtn" the problem is that he has not created a subroutine called incrtn here is the full code,
    PRTSTR:
     STA ASAVE
     STY YSAVE
     PLA ;GET RETURN ADDRESS FROM
     STA RTNADR ;THE 6502 STACK
     PLA
     STA RTNADR+$1
     ;
     JSR INCRTN ;INCREMEOT THE RETURN ADDRESS
     LDY #$0
     LDA (RTNADR),Y ;GET L.O. ADDRESS OF STRING
     STA ZPAGE
     INY
     LDA (RTNADR),Y ;GET H.O.ADDRESS OF STRING
     STA ZPAGE+$1
     ;
     JSR INCRTN ;MOVE RTNADR PAST THE ADDRESS
     JSR INCRTN ;BYTES
     ;
     ;
     ; AT THIS POINT, ZPAGE POINTS TO THE STRING WHICH
     ; IS SUPPOSED TO BE OUTPUT
     ;
     DEY ;RESET Y REG TO ZERO
     LDA (ZPAGE),Y ;GET THE LENGTH OF THE STRING
     STA LENGTH ;AND STORE IT IN "LENGTH"
     PRTS1 INY ;MOVE TO THE NEXT CHARACTER
     CPY LENGTH ;ARE WE THROUGH YET?
     BEQ PRTS2
     ;
     LDA (ZPAGE),Y ;GET THIS CHARACTER
     JSR COUT ;AND OUTPUT
     JMP PRTS1 ;MOVE TO NEXT CHAR AND REPEAT
     ;
     PRTS2 LDA ASAVE ;RESTORE THE REGISTERS
     LDY YSAVE
     JMP (RTNADR) ;SIMULATE AN RTS
     ;
     ;
     ;
     ASAVE EPZ $0 ;ZERO PAGE WORKSPACE
     YSAVE EPZ ASAVE+$1
     ZPAGE EPZ YSAVE+$1
     RTNADR EPZ ZPAGE+$2
     COUT EQU $FDED ;COUT ROUTINE
     END
can anyone help me?
Update, if anyone wants to know how to print text the shorter way here is some working code
           LDX #$0
  LOOP     INX 
           LDA STRING,X 
           JSR $FDF0 
           CPX STRING 
           BLT LOOP 
           RTS  
  STRING   STR "hello world"
         END
 
                        
Before processors had a stack or sufficient registers, parameters were often passed via "inline" parameter passing, which is to pass parameters after the
JSRor call-type instruction, by placing values or pointers directly succeeding the call instruction, with the requirement that the callee skip over them when returning to resume the caller.As you know, a call instruction captures a return address (somehow, somewhere: maybe on the stack or maybe the code memory of the subroutine!), and that return address can be used to fetch parameters that are placed after the call instruction. The return address is incremented to obtain successive parameters, and when done, needs to be incremented (by 1 or 2) one more time to return to actual code of the caller instead of returning to the inline parameters, which are, of course, data, not code.
The style Hyde is using for this
PRTSTRis this inline parameter mechanism.Here is the usage of this
PRTSTR:Given that the 6502 had both a call stack and registers, it is a bit of a throwback to use the inline mechanism for calling, but certainly was done. (The inline parameter passing mechanism predates call stacks and lots of registers as we have in today's modern processors.)
With a stack but few registers, some calls with numerous parameters would have been done placing parameters onto the stack, while others were done by passing in available registers.
In pure assembly code, you can create a custom calling convention for each function. Only the advent of C and other high level languages creates the requirement for canonical, regular, and predictable calling conventions that can be codified in the compiler.
I have to admit that I don't understand the first of the three
JSR INCRTNinstructions. (I understand the last two — they skip the pointer to string (the inline parameter following theJSR), and since pointers are two bytes, the address needs to increment by 2 bytes — but the very first of the three looks erroneous to me.) There is noADRopcode on 6502, so I have to assume it is an assembler pseudo opcode that forms a 2 byte pointer to the label.My assumption is somewhat confirmed by Hyde in 14-6 where it says:
that the above is a 7 byte sequence (i.e. 1 for the
JSRopcode, 2 forSASIGN(the operand ofJSR), and 2 each forADR DESTandADR SOURCE).Thus, the 16-bit variable at
RTNADRandRTNADR + 1needs to be incremented by one when the code doesJSR INCRTN.I believe this code sequence could be much improved by using an index register more directly as a pointer, instead of using memory indirect along with an integer in index register Y.
Further, as I cannot fathom the first
JSR INCRTN, and the second is called twice, it might as well simply increment by 2 instead of by 1, twice.