back to main

Introduction

In Yoshi's Cookie for the Super Nintendo, the VS mode can be played against a second player or a CPU-controlled opponent. This document is my brief analysis of the implementation of the CPU-controlled opponent. I am writing with an assumption that the reader is familiar with playing the VS mode of Yoshi's Cookie. Additionally, if you are familiar with assembly language programming, there are some details that you may find interesting.

In my descriptions of the implementation, sometimes it's more convenient for me to use hexadecimal values rather than decimal values. When a number is shown in fixed-width font, it's a hexadecimal value. Otherwise, it's a decimal value. For example, if I wrote 11, that would equal a decimal value of seventeen.

Implementation Structure

The code that controls the CPU opponent is divided into subroutines, each of which represent a step in its operation. For each of these steps, I will describe it in terms of the conditions when the step starts and ends. When I say "conditions," this includes both the visible state of the game board, as well as internal memory values that are used by later steps. If there are any helper subroutines used in the step, I will try to give some details about those as well.

Top level structure and jump table

The top-level subroutine calls the subroutine for the current step. The jump address for each step is stored in a data table immediately following the top-level subroutine. The addresses are shown in the following table. Note that I start numbering the steps at 1, but the value stored for the step starts at 0 for easier data storage.

Step Start address from jump table
1 $DF0A
2 $DF5C
3 $DF9B
4 $DFB9
5 $E027
6 $E16C
7 $E2CD
8 $E2FB
9 $E339

At the beginning of the game, the CPU is at step 1. The steps do not always proceed in order, so I will make note of what step changes can occur.

Step one

This step is responsible for determining which of the two game boards will be used: the one belonging to player one, or the one belonging to player two.

That is quite a few conditions that must be met in order to go for a reversal attack! It is interesting to note that there are safeguards against going for it: when player one is only one point away from winning, or when player two's fuse is low. However, the amount of time remaining before player one's attack changes is not factored in here. This means that the CPU player may start going for a reversal, then stop before actually performing the reversal. In narrowly timed situations, the attack may change while the last Yoshi cookie is moving into place, usually thwarting the attempted reversal.

After choosing which board to control, a delay timer is set that will cause the next step to exit early until it counts down to zero. The delay is shorter on hard mode, and can be different at different steps. Most steps after this will one start by waiting for the delay counter and setting the delay counter when the step changes.

Time units

If a player's timer hits zero, they immediately lose the round. Time is tracked with a pair of variables: the main timer and the sub-timer. The sub-timer decrements by one every frame. When it hits zero, the main timer decrements by one, and the sub-timer is reset to its starting value. The starting value for the sub-timer is equal to the character's LIMIT value times the game speed value. Mario, Yoshi, and Peach all have a LIMIT value of 2. Bowser has a LIMIT value of 1. The game speed value is 9 on slow, 7 on medium, and 5 on fast. When the player clears cookies, the main timer resets to 80.

Step two

Never trying to use Yoshi cookies as the second target is interesting. This implementation can't employ a tactic of trying to make a double that uses only four Yoshi cookies in order to get an attack without needing a fifth Yoshi cookie.

Step three

This is a sensible step that most players will also attempt to take. If a row or column already has three of a target cookie, it usually doesn't take long to put the two other cookies into place.

Step four

The implementation will always minimize the number of moves it has to make. So, this will move the cursor at most two spaces.

Step five

Get a glass of water before this one. It's a bit more complex.

This step checks some values that haven't been explained yet.

Cursor sub-position

What is a cursor sub-position? A cursor's horizontal and vertical positions are stored in one byte each. The upper bits store the board position, and the lower bits store the sub-position. For example, a value of 2A means that the cursor is at board position 2 and sub-position A. The board position corresponds to an actual row or column on the board, so its value will always be between 0 (the leftmost column and bottommost row) and 4 (the rightmost column and topmost row), inclusive. The sub-position is what lets movement happen gradually as a directional input is held down. Each frame that a directional input is held down, 2 is added to (for moving right or up) or subtracted from (for moving left or down) from the position. If no directional inputs are held down, the sub-position is set to 8. So, a value of 18 represents the cursor at rest in row or column 1.

It would take eight frames of continuous input for the cursor to move from one "rest point" to the other. But, the cursor's board position will change sooner than that! Consider a cursor currently at rest. When moving up or right, the board position will move to the next value after four addition operations (for example, 18, 1A, 1C, 1E, 20). When moving down or left, the board position will move to the next value after five subtraction operations (for example, 18, 16, 14, 12, 10, 0E). Once the board position has changed, the directional input can be released, and the sub-position will snap to the rest value of 8.

Double preference

The CPU-controlled opponent doesn't always go for a double. To go for a double, a byte from the random number generator has to be less than the character's double preference value.

Character Double preference Approximate percentage
Mario 33 19.9%
Yoshi 19 9.8%
Peach 55 33.2%
Bowser 55 33.2%

Peach and Bowser go for a double about 1/3 of the time. Mario goes for a double about 1/5 of the time, and Yoshi a mere 1/10 of the time. These values are not affected by difficulty.

Choices about bad attacks

The different decisions made by the characters when targeting Yoshi cookies with a bad attack available creates interesting play behaviors. The ever-patient Peach is willing to wait for the attack to change. This works well with her characteristic of the attacks changing more frequently. Mario demonstrates some patience, but will sometimes get tired of waiting and decide to try matching something else instead. Bowser is more prone to frustration, charging forward recklessly when he can't see. Yoshi has no sense of danger, never waiting for the attack to change. This fits with the slower attack change speed and higher defense: even if Yoshi ends up stumbling, it's not as much trouble as it would be for the other characters.

Step six

After all the complex checks of step five, the step that actually puts the cookies into place is refreshingly simple. This strategy keeps the cursor on the build line at all times, moving the cursor to a place where the target cookie can be pulled in. The build line slides so that the cursor is on top of a non-target cookie, and the target cookie is pulled in.

Steps seven, eight, and nine

These steps are discussed together because they fulfill similar purposes: they keep the controller inputs the same until a certain condition is met.

In all cases, when the condition is met, zero out the controller inputs and go back to step 5.

Discussion

The implementation opts for simplicity over optimality. By trying to keep the cursor in the build line, it means that there are really only four types of inputs:

There's nothing in here that tries to calculate what is most optimal from a given board state. But, although this isn't optimal, it's still reasonably good. And, of course, it doesn't make any input mistakes: given these four types of movement, it will always use the least amount of time it can.

Honestly, I'm not too sure what else to put here. I don't think I could change this for a more optimal result without dramatically increasing the implementation complexity. I think it's a useful reminder for both programming and playing Yoshi's Cookie that while optimal may be difficult and intricate, you don't always need to be perfectly optimal. A merely good solution that is much simpler to write is often good enough to meet your needs.

back to main