The Program Counter (PC) and Stack Pointer (SP) Module implements two unrelated registers. There are no interconnections or dependencies between the PC and SP, so they are co-located only because each design is simple and both could be implemented together on a single board.

Program Counter and Stack Pointer

Program Counter

The Program Counter (PC) is implemented using two 74LS161 4-bit counters. It increments on any clock cycle that has the Program Increment (PI) signal asserted.

New values are clocked into the PC from the data bus when the Write Program (WP) signal is asserted. The WP signal is produced by the register selects to perform an unconditional jump.

The value of the PC is read to the bus when the Read Program (RP) signal is asserted. This operation is done as the first step of every instruction to read the value of the PC into the MAR.

The PC is reset back to zero whenever RST is asserted. After a reset, program execution starts at address zero.

Stack Pointer

The Stack Pointer (SP) is an 8-bit register implemented using a pair of 74LS193 4-bit up/down counters. New values are clocked into the SP from the data bus when the Write Stack (WS) signal is asserted. The value of the SP is read to the bus when the Read Stack (RS) signal is asserted.

The SP enables new instructions to push and pull registers (PHA, PLA) and to call and return from subroutines (JSR, RTS). There are also instructions to load the SP to and from the A register using the TAS and TSA instructions. If no stack operations are needed, the SP can also be used as a general-purpose register.

The CLR signal is not used to reset the SP at system reset, so the SP always starts in an unknown state. This isn’t a problem, because any programs that use the stack will want to initialize it to point to a free memory area before any stack operations are performed.

Program Counter and Stack Pointer Schematic

Stack Pointer implementation

Unlike the 74LS161, the 74LS193 counters do not have a master clock or a count enable control. Counting is performed by pulsing the UP or DOWN pins and loading a new value is performed by pulsing the LOAD signal. Additional logic is used to AND these signals with the master system clock. To load a new value, the CLK and WS signals are combined to produce a pulse for the LOAD pin.

Spare bus and microcode signals are in short supply. Rather than adding two new signals for the SP count up and down operations, a single Stack Count (SC) signal was added. This is used in conjunction with the shared CX carry control signal to control counting in the SP. Asserting both SC and CX will cause the SP to count down and SC without CX will to cause it to count up. The SC and CX signals are combined with the CLK signal to produce the pulses for the DOWN and UP lines.

The shared use of the CX signal means that the carry flag cannot be modified in the same microinstruction step where the SP is counting. The normal use of the CX signal will not cause the SP to count because the SC signal will not be asserted. Note that the CX signals is also shared by the RAM, so SP and RAM access operations also cannot happen in a single microinstruction step.

The combined SC and CX signals are also used to drive the UP and DN indicator LEDs when SC is active. This is preferable to just adding LEDs to the raw SC and CX signals because CX would give a misleading indication on the Stack Pointer Module when the CX signal was used elsewhere in the system.

Stack Pointer microcode

The SP counts up when pushing values to the stack and counts down when pulling values. There is no overflow protection, so the counter will quietly wrap if it hits the top or bottom of memory. The push operation uses a post-increment, so the value is stored and then the SP is bumped to point to the next available stack address. In other words, the SP always points to the address after the location of the value on the top of the stack. The pull operations pre-decrement, so the SP is moved to point to the value and then the value is retrieved.

Note that the Jump to Subroutine (JSR) uses a few tricks in the microcode. The value of the PC is needed to retrieve the subroutine address, but the PC also needs to be saved onto the stack before being overwritten with the new address. The ALU B register is used for temporary storage to save the subroutine address. The B register is not directly accessible by the user, so it is a good candidate for the microcode to use. The example below shows a JSR instruction at memory address 20.

Address Contents
20 JSR opcode
21 JSR address
22 next opcode

After the instruction fetch, the PC will have the value 21 and JSR microcode performs the following steps:

  1. move the PC value to the MAR and increment the PC. MAR contains 21 and PC contains 22.
  2. read the subroutine address from RAM[21] and place it in B for temp storage
  3. move the SP value into the MAR and increment the SP.
  4. store the PC value (which points to the next instruction) in memory, i.e. push the JSR return address on the stack.
  5. move the B register value into the PC, effectively jumping to the subroutine

Note that the B register is needed because the PC value that is pushed to the stack is one address past the location that holds the subroutine address. If the PC value was incremented and pushed before fetching the subroutine address, there would be no way to fetch the address because the PC has already moved past it.

An alternate approach that would not require temporary storage would be to push the PC while it was still pointing to the subroutine address instead of pointing to the next opcode. The RTS instruction would then need to increment the PC after fetching it from the stack. The downside to this is that the PC address on the stack would not be the true return address.

Bill of Materials

  • 74HCT00 quad 2-input NAND gate(1)
  • 74HCT32 quad 2-input OR gate(1)
  • 74LS161 4-bit counter (2)
  • 74LS193 4-bit up/down counter (2)
  • 74HCT245 8-bit bus transceiver (2)

Updated: