← back to syllabus ← back to notes
At its core the fundamental principle of a computer is that it performs computation, specifically arithmetic operations.
Let’s emphasize that arithmetic operations are a core function of the computer’s Arithmetic and Logic Unit (ALU). It would explain that all other components of the computer system exist primarily to bring data to the ALU for processing and to handle the results, like storing them and using them again in another computation.
According to von Neumann’s “stored program concept,” number representation is crucial for understanding how arithmetic is performed.
Integers represent the set of whole numbers in a CPU, using the binary format limited to the maximum bit width of the operands supported by the ALU. In our MIPS computer, that width is 32 bits.
Binary Numbers: The fundamental way computers represent numbers using strings of bits.
Two’s Complement Representation: We have already covered this standard method for representing signed integers, especially covering its advantages for arithmetic operations.
Floating-point represents the set of real numbers using a format similar to scientific notation in binary. The IEEE 754 standard for floating-point representation defines the format for single-precision and double-precision floating-point numbers, including the sign bit, exponent, and significand (or mantissa). With IEEE 754 computers can adequately handle computating involving normalized and denormalized numbers, as well as special values like infinity and NaN (not a number), all defined in the IEEE 754 standard.
We will cover:
It’s interesting how different architectures handle integer arithmetic with operands of varying sizes. Some programming languages allow two’s complement integer arithmetic on variables declared byte and halfword, whereas MIPS only has integer arithmetic operations on full words.
There’s a key difference between some programming languages’ capabilities and the fundamental integer arithmetic operations in the basic MIPS architecture.
Operand Sizes and Arithmetic:
Some programming languages, particularly at lower levels or when interacting closely with hardware, allow you to declare integer variables as bytes (8 bits) or halfwords (16 bits) and perform direct two’s complement arithmetic on these sizes. This implies that the underlying architecture has instructions that can operate directly on these smaller data types.
Regarding MIPS Architecture and Operand Sizes:
In contrast, the base MIPS architecture primarily performs integer arithmetic operations on full 32-bit words held in its general-purpose registers (GPRs). While MIPS has load and store instructions (lb
, lbu
, lh
, lhu
, sb
, sh
) that can move byte and halfword data between memory and registers, when these smaller data types are loaded into a 32-bit register, they are either sign-extended (lb
, lh
) to preserve their signed value or zero-extended (lbu
, ldhu
) to treat them as unsigned non-negative integers.
The primary arithmetic instructions (add
, sub
, mul
, div
, and
, or
, xor
, etc.) in MIPS typically operate on these 32-bit register values.
There are “word” versions of some arithmetic instructions in MIPS (e.g., addw
, subw
, mulw
, divw
, remw
). These instructions operate on the lower 32 bits of registers, and the result is a 32-bit sign-extended value. This is particularly relevant in 64-bit MIPS implementations or when explicitly dealing with 32-bit sub-sections of larger registers, but the fundamental unit for general integer arithmetic remains the 32-bit word.
Comparison with Other Architectures (x86 and ARM):
Other architectures like Intel’s x86 and ARM have more flexibility in handling different data sizes for arithmetic operations.
add AL
, imm8
for 8-bit addition).Why the Difference?
The design philosophy of MIPS, particularly as a RISC (Reduced Instruction Set Computer) architecture, emphasizes simplicity and regularity. By focusing most integer arithmetic on a single, fixed word size (32 bits in the base MIPS), the instruction set and the underlying hardware can be made simpler and potentially faster for common operations. Operations on smaller data types are handled primarily through load/store with extension to fit the 32-bit register format for subsequent arithmetic.
Implications for Programming:
When programming in a language that allows direct byte and halfword arithmetic and then targeting a MIPS architecture, the compiler must handle these operations by:
In summary, while some programming languages offer the abstraction of performing two’s complement integer arithmetic directly on byte and halfword variables, the fundamental design of the base MIPS architecture primarily focuses its integer arithmetic operations on 32-bit words in registers. Operations on smaller data sizes in MIPS involve loading and extending them to 32 bits before performing arithmetic. Architectures like x86 and ARM provide more direct hardware support for arithmetic on byte and halfword operands.
In MIPS, the detection of overflow in integer arithmetic depends on whether the operation is treated as signed or unsigned. MIPS provides separate sets of instructions for these cases, and their behavior with respect to overflow is different.
Here’s how overflow detection works in MIPS integer arithmetic:
Signed Arithmetic:
add
, addi
, and sub
are used for signed integer arithmetic.PC
) is copied to the Exception Program Counter (EPC
) register, and a code indicating the source of the exception (0x30
for arithmetic overflow) is stored in the Cause
register. The processor then jumps to the exception handler at a predefined memory address (0x80000180
). Check where on the memory map this address lives.Unsigned Arithmetic:
addu
, addiu
, and subu
for unsigned integer arithmetic.addu
and related instructions do not automatically signal this overflow in a way that triggers an exception.add r4, r2, r3
, you can check if the unsigned sum in r4
is less than either of the unsigned operands (r2
or r3
). If it is, a carry
(and thus overflow in the unsigned context) has occurred. The bltu
(branch if less than unsigned) instruction can be used for this purpose. Similarly, for unsigned subtraction, a carry
(borrow) can be detected if the minuend is less than the subtrahend, which can be checked using bltu
.Carry Flag (C Flag
):
While the base MIPS architecture doesn’t have explicit condition code flags (like a dedicated carry or overflow flag) that are automatically set by arithmetic instructions in the same way as some other architectures, the concept of carry is still relevant for implementing multi-precision arithmetic or for explicitly checking for unsigned overflow as mentioned above.
In summary, MIPS detects overflow automatically for signed integer arithmetic operations (add
, addi
, sub
) by raising an exception. For unsigned integer arithmetic operations (addu
, addiu
, subu
), overflow is not automatically detected by an exception; programmers need to implement explicit checks if overflow detection is required. The distinction between signed and unsigned instructions is crucial in how overflow is handled in MIPS.
In MIPS, an exception is an unexpected event that disrupts the normal flow of program execution, requiring a change in the control flow. Exceptions can arise from various sources, both within the CPU (like arithmetic overflow, undefined opcode, system calls) and from external devices (interrupts from I/O controllers). The term “exception” in MIPS often encompasses both hardware-initiated interrupts and software-initiated deviations from normal execution.
When an exception occurs in MIPS, the processor needs to handle it in a structured manner to maintain system stability and allow for potential recovery or graceful termination of the program. This is where the exception handler comes into play. An exception handler is a specific piece of code, typically part of the operating system (kernel program), that is designed to determine the cause of the exception and take appropriate action.
In MIPS:
Exception Program Counter
(EPC
).Cause
register. Different codes in the Cause
register indicate different types of exceptions (e.g., 0x30
for arithmetic overflow, 0x28
for undefined instruction).0x80000180
.Cause
register. 1. Based on the cause, the exception handler will perform the necessary actions. This might involve:
EPC
register back into the PC
.jr $k0
, where $k0
is often used to hold the return address from EPC
, or a specific exception return instruction if available in an extended architecture) to resume the program from where it was interrupted.Here is a simple example of MIPS assembly code that will intentionally cause an arithmetic overflow exception:
Explanation of the Example:
0x7FFFFFFF
).add $t2, $t0, $t1
instruction.add
instruction (or the next one) into the EPC
register.Cause
register to indicate an arithmetic overflow.0x80000180
(in the .kseg0
kernel segment).Cause
register to identify the type of exception.EPC
register and uses the jr
(jump register) instruction to return to the interrupted program (in this simple example, it would re-attempt the add instruction or proceed to the next instruction, depending on the exact behavior).In a real operating system, the exception handler would be significantly more sophisticated, potentially handling the error more robustly or terminating the program if necessary. This example serves to illustrate the basic concept of an exception being triggered and the control being transferred to a designated handler. It also demonstrates the use of the EPC and Cause registers to manage the exception process
Integer multiplication in the MIPS CPU involves dedicated digital hardware within the Arithmetic Logic Unit (ALU). Due to the fact that multiplying two n-bit numbers can result in a product of up to 2n bits, MIPS uses a special mechanism to handle the potentially larger result.
Here’s a breakdown of how integer multiplication is handled:
HI
and LO
to store the result of a multiplication operation. These registers are separate from the 32 general-purpose registers.
LO
register holds the least significant 32 bits of the product.HI
register holds the most significant 32 bits of the product.mult rs, rt
: This instruction multiplies the 32-bit integer values in source registers rs
and rt
. The 64-bit product is then stored in the HI
and LO
registers. The HI
register receives the upper 32 bits, and the LO
register receives the lower 32 bits of the product.multu rs, rt
: This is similar to mult
, but it treats the operands as unsigned integers.mflo rd
(Move From LO
): After a mult
or multu
instruction, this instruction is used to move the value from the LO
register into a general-purpose destination register rd
. This is typically how you access the lower 32 bits of the multiplication result for further use.mfhi rd (Move From
HI)
: This instruction is used to move the value from the HI
register into a general-purpose destination register rd
. This allows you to access the upper 32 bits of the product, which is crucial for detecting overflow or for working with the full 64-bit result.mul rd, rs, rt
: This instruction performs the multiplication of the values in rs
and rt
and directly places the least significant 32 bits of the product into the destination register rd
. This instruction is useful when you know that the product will fit within 32 bits or when you are only interested in the lower 32 bits.mult
operation (for signed numbers), you can check the HI
register to detect if an overflow occurred. If the HI
register contains only 0s (for positive results) or only 1s (as an extension of the sign bit for negative results in the LO
register), then no overflow occurred within the 32-bit range. If the HI
register contains any other pattern, it indicates that the result exceeded 32 bits and an overflow happened.multu
, if the HI
register is non-zero, it signifies that the product is larger than 32 bits.mulo rd, rs, rt
(multiply with overflow check) which multiplies rs
and rt
, stores the result in rd
, and raises an exception if an overflow occurs.mul
or div
operations.In summary, MIPS handles integer multiplication by using the mult
(and multu
for unsigned) instruction to generate a 64-bit product stored in the special HI
and LO
registers. The mflo
and mfhi
instructions are then used to move the lower and upper halves of the result to general-purpose registers. Overflow in signed multiplication can be detected by examining the HI
register, and the mulo
pseudo-instruction provides built-in overflow detection with an exception. The mul
instruction offers a direct way to get the lower 32 bits of the product when the full 64-bit result is not needed.
Integer division in the MIPS CPU is handled using dedicated instructions and a pair of special-purpose registers, similar to integer multiplication.
Here’s a breakdown of how it works:
HI
and LO
to store the results of integer division. These registers are distinct from the 32 general-purpose registers available for other operations.
LO
register.HI
register.div rs, rt
: This instruction divides the 32-bit signed integer value in source register rs
(the dividend) by the 32-bit signed integer value in source register rt
(the divisor). After the operation, the integer quotient is placed in the LO
register, and the remainder is placed in the HI
register. The result is truncated towards zero, which is typical for integer division.divu rs, rt
: This instruction performs the same operation as div
, but it treats both the dividend (rs
) and the divisor (rt
) as unsigned integers. The unsigned quotient is stored in LO
, and the unsigned remainder is stored in HI
.HI
and LO
registers to general-purpose registers using the following instructions:
mflo rd
(Move From LO
): This instruction copies the value from the LO
register into the specified general-purpose destination register rd
. This is how you retrieve the quotient.mfhi rd
(Move From HI
): This instruction copies the value from the HI
register into the specified general-purpose destination register rd
. This is how you retrieve the remainder.div
and divu
instructions in MIPS do not have built-in hardware mechanisms to detect division by zero or overflow (where the quotient might be too large to fit in the LO
register). If a program attempts to divide by zero, the behavior is undefined and may lead to an exception, but it’s not a guaranteed hardware check. Similarly, if the quotient exceeds 32 bits (though this is less common with integer division of two 32-bit numbers within their typical ranges), the result in LO
might not be accurate. Therefore, programmers must explicitly include checks in their code if they need to handle these conditions. For example, a program might check if the divisor is zero before executing a division instruction.rem
(remainder) pseudo-operator to simplify obtaining the remainder. For example, rem rd, rs, rt
calculates rs % rt
and stores the result in rd
. This pseudo-instruction is typically translated into a sequence of actual MIPS instructions involving div
and mfhi
.div
, divu
, mflo
, and mfhi
instructions.In summary, MIPS integer division is performed using the div (for signed) and divu (for unsigned) instructions. The quotient is stored in the LO register, and the remainder is stored in the HI register. Programmers then use mflo and mfhi to move these results to general-purpose registers. It is crucial to remember that hardware-level checks for division by zero or overflow are not provided by these basic instructions, requiring software-level handling if necessary.
We will cover:
Let’s cover these in the CODmips textbook’s Chapter 3 presentation deck