2022-04-28 16:55:47 -04:00
//
// PerformImplementation.hpp
// Clock Signal
//
// Created by Thomas Harte on 28/04/2022.
// Copyright © 2022 Thomas Harte. All rights reserved.
//
# ifndef InstructionSets_M68k_PerformImplementation_h
# define InstructionSets_M68k_PerformImplementation_h
2023-10-25 22:21:23 -04:00
# include "../../../Numeric/Carry.hpp"
2022-05-11 07:00:35 -04:00
# include "../ExceptionVectors.hpp"
2022-10-18 12:17:55 -04:00
# include <algorithm>
2022-04-28 16:55:47 -04:00
# include <cassert>
# include <cmath>
2023-05-10 16:02:18 -05:00
namespace InstructionSet : : M68k {
2022-04-28 16:55:47 -04:00
2022-10-17 15:37:13 -04:00
/// Sign-extend @c x to 32 bits and return as an unsigned 32-bit int.
inline uint32_t u_extend16 ( uint16_t x ) { return uint32_t ( int16_t ( x ) ) ; }
/// Sign-extend @c x to 32 bits and return as a signed 32-bit int.
inline int32_t s_extend16 ( uint16_t x ) { return int32_t ( int16_t ( x ) ) ; }
2022-04-28 16:55:47 -04:00
2022-10-10 22:21:13 -04:00
namespace Primitive {
/// Performs an add or subtract (as per @c is_add) between @c source and @c destination,
/// updating @c status. @c is_extend indicates whether this is an extend operation (e.g. ADDX)
/// or a plain one (e.g. ADD).
template < bool is_add , bool is_extend , typename IntT >
static void add_sub ( IntT source , IntT & destination , Status & status ) {
static_assert ( ! std : : numeric_limits < IntT > : : is_signed ) ;
2022-10-19 22:17:51 -04:00
IntT result = is_add ?
destination + source :
destination - source ;
status . carry_flag = is_add ? result < destination : result > destination ;
// If this is an extend operation, there's a second opportunity to create carry,
// which requires a second test.
if ( is_extend & & status . extend_flag ) {
if constexpr ( is_add ) {
+ + result ;
status . carry_flag | = result = = 0 ;
} else {
status . carry_flag | = result = = 0 ;
- - result ;
}
}
status . extend_flag = status . carry_flag ;
2022-10-10 22:21:13 -04:00
// Extend operations can reset the zero flag only; non-extend operations
// can either set it or reset it. Which in the reverse-logic world of
// zero_result means ORing or storing.
if constexpr ( is_extend ) {
2022-10-11 10:33:28 -04:00
status . zero_result | = Status : : FlagT ( result ) ;
2022-10-10 22:21:13 -04:00
} else {
2022-10-11 10:33:28 -04:00
status . zero_result = Status : : FlagT ( result ) ;
2022-10-10 22:21:13 -04:00
}
2022-10-18 14:51:51 -04:00
status . set_negative ( result ) ;
2023-10-25 22:21:23 -04:00
status . overflow_flag = Numeric : : overflow < is_add > ( destination , source , result ) ;
2022-10-11 10:33:28 -04:00
destination = result ;
2022-10-10 22:21:13 -04:00
}
2022-10-17 15:37:13 -04:00
/// Perform an SBCD of @c lhs - @c rhs, storing the result to @c destination and updating @c status.
///
/// @discussion The slightly awkward abandonment of source, destination permits the use of this for both
/// SBCD and NBCD.
2022-10-17 15:12:38 -04:00
inline void sbcd ( uint8_t rhs , uint8_t lhs , uint8_t & destination , Status & status ) {
const int extend = ( status . extend_flag ? 1 : 0 ) ;
const int unadjusted_result = lhs - rhs - extend ;
const int top = ( lhs & 0xf0 ) - ( rhs & 0xf0 ) - ( 0x60 & ( unadjusted_result > > 4 ) ) ;
int result = ( lhs & 0xf ) - ( rhs & 0xf ) - extend ;
const int low_adjustment = 0x06 & ( result > > 4 ) ;
status . extend_flag = status . carry_flag = Status : : FlagT (
( unadjusted_result - low_adjustment ) & 0x300
) ;
result = result + top - low_adjustment ;
/* Store the result. */
destination = uint8_t ( result ) ;
/* Set all remaining flags essentially as if this were normal subtraction. */
status . zero_result | = destination ;
2022-10-18 14:51:51 -04:00
status . set_negative ( destination ) ;
2022-10-17 15:12:38 -04:00
status . overflow_flag = unadjusted_result & ~ result & 0x80 ;
}
2022-10-17 15:37:13 -04:00
/// Perform the bitwise operation defined by @c operation on @c source and @c destination and update @c status.
/// Bitwise operations are any of the byte, word or long versions of AND, OR and EOR.
2022-10-17 15:21:54 -04:00
template < Operation operation , typename IntT >
void bitwise ( IntT source , IntT & destination , Status & status ) {
static_assert (
operation = = Operation : : ANDb | | operation = = Operation : : ANDw | | operation = = Operation : : ANDl | |
operation = = Operation : : ORb | | operation = = Operation : : ORw | | operation = = Operation : : ORl | |
operation = = Operation : : EORb | | operation = = Operation : : EORw | | operation = = Operation : : EORl
) ;
switch ( operation ) {
case Operation : : ANDb : case Operation : : ANDw : case Operation : : ANDl :
destination & = source ;
break ;
case Operation : : ORb : case Operation : : ORw : case Operation : : ORl :
destination | = source ;
break ;
case Operation : : EORb : case Operation : : EORw : case Operation : : EORl :
destination ^ = source ;
break ;
}
status . overflow_flag = status . carry_flag = 0 ;
2022-10-18 14:51:51 -04:00
status . set_neg_zero ( destination ) ;
2022-10-17 15:21:54 -04:00
}
2022-10-17 15:37:13 -04:00
/// Compare of @c source to @c destination, setting zero, carry, negative and overflow flags.
2022-10-11 12:57:02 -04:00
template < typename IntT >
2022-10-17 15:21:54 -04:00
void compare ( IntT source , IntT destination , Status & status ) {
2022-10-11 12:57:02 -04:00
const IntT result = destination - source ;
status . carry_flag = result > destination ;
2022-10-18 14:51:51 -04:00
status . set_neg_zero ( result ) ;
2023-10-25 22:21:23 -04:00
status . overflow_flag = Numeric : : overflow < false > ( destination , source , result ) ;
2022-10-11 12:57:02 -04:00
}
2022-10-11 11:22:34 -04:00
/// @returns the name of the bit to be used as a mask for BCLR, BCHG, BSET or BTST for
/// @c instruction given @c source.
2022-10-11 10:47:55 -04:00
inline uint32_t mask_bit ( const Preinstruction & instruction , uint32_t source ) {
return source & ( instruction . mode < 1 > ( ) = = AddressingMode : : DataRegisterDirect ? 31 : 7 ) ;
}
2022-10-17 15:37:13 -04:00
/// Perform a BCLR, BCHG or BSET as specified by @c operation and described by @c instruction, @c source and @c destination, updating @c destination and @c status.
2022-10-11 11:22:34 -04:00
/// Also makes an appropriate notification to the @c flow_controller.
2022-10-11 10:47:55 -04:00
template < Operation operation , typename FlowController >
void bit_manipulate ( const Preinstruction & instruction , uint32_t source , uint32_t & destination , Status & status , FlowController & flow_controller ) {
2022-10-11 15:53:11 -04:00
static_assert (
operation = = Operation : : BCLR | |
operation = = Operation : : BCHG | |
operation = = Operation : : BSET ) ;
2022-10-11 10:47:55 -04:00
const auto bit = mask_bit ( instruction , source ) ;
status . zero_result = destination & ( 1 < < bit ) ;
switch ( operation ) {
case Operation : : BCLR : destination & = ~ ( 1 < < bit ) ; break ;
case Operation : : BCHG : destination ^ = ( 1 < < bit ) ; break ;
case Operation : : BSET : destination | = ( 1 < < bit ) ; break ;
}
flow_controller . did_bit_op ( int ( bit ) ) ;
}
2022-10-11 11:22:34 -04:00
/// Sets @c destination to 0, clears the overflow, carry and negative flags, sets the zero flag.
template < typename IntT > void clear ( IntT & destination , Status & status ) {
destination = 0 ;
status . negative_flag = status . overflow_flag = status . carry_flag = status . zero_result = 0 ;
}
2022-10-17 15:37:13 -04:00
/// Perform an ANDI, EORI or ORI to either SR or CCR, notifying @c flow_controller if appropriate.
2022-10-11 15:53:11 -04:00
template < Operation operation , typename FlowController >
void apply_sr_ccr ( uint16_t source , Status & status , FlowController & flow_controller ) {
static_assert (
operation = = Operation : : ANDItoSR | | operation = = Operation : : ANDItoCCR | |
operation = = Operation : : EORItoSR | | operation = = Operation : : EORItoCCR | |
operation = = Operation : : ORItoSR | | operation = = Operation : : ORItoCCR
) ;
auto sr = status . status ( ) ;
switch ( operation ) {
case Operation : : ANDItoSR : case Operation : : ANDItoCCR :
sr & = source ;
break ;
case Operation : : EORItoSR : case Operation : : EORItoCCR :
sr ^ = source ;
break ;
case Operation : : ORItoSR : case Operation : : ORItoCCR :
sr | = source ;
break ;
}
switch ( operation ) {
case Operation : : ANDItoSR :
case Operation : : EORItoSR :
case Operation : : ORItoSR :
status . set_status ( sr ) ;
flow_controller . did_update_status ( ) ;
break ;
case Operation : : ANDItoCCR :
case Operation : : EORItoCCR :
case Operation : : ORItoCCR :
status . set_ccr ( sr ) ;
break ;
}
}
2022-10-17 15:37:13 -04:00
/// Perform a MULU or MULS between @c source and @c destination, updating @c status and notifying @c flow_controller.
2022-10-11 16:02:20 -04:00
template < bool is_mulu , typename FlowController >
void multiply ( uint16_t source , uint32_t & destination , Status & status , FlowController & flow_controller ) {
if constexpr ( is_mulu ) {
destination = source * uint16_t ( destination ) ;
} else {
destination = u_extend16 ( source ) * u_extend16 ( uint16_t ( destination ) ) ;
}
status . carry_flag = status . overflow_flag = 0 ;
2022-10-18 14:51:51 -04:00
status . set_neg_zero ( destination ) ;
2022-10-11 16:02:20 -04:00
if constexpr ( is_mulu ) {
flow_controller . did_mulu ( source ) ;
} else {
flow_controller . did_muls ( source ) ;
}
}
2022-10-17 15:37:13 -04:00
/// Announce a DIVU or DIVS to @c flow_controller.
2022-10-11 16:16:53 -04:00
template < bool is_divu , bool did_overflow , typename IntT , typename FlowController >
void did_divide ( IntT dividend , IntT divisor , FlowController & flow_controller ) {
if constexpr ( is_divu ) {
flow_controller . template did_divu < did_overflow > ( dividend , divisor ) ;
} else {
flow_controller . template did_divs < did_overflow > ( dividend , divisor ) ;
}
}
2022-10-17 15:37:13 -04:00
/// Perform a DIVU or DIVS between @c source and @c destination, updating @c status and notifying @c flow_controller.
2022-10-11 16:16:53 -04:00
template < bool is_divu , typename Int16 , typename Int32 , typename FlowController >
void divide ( uint16_t source , uint32_t & destination , Status & status , FlowController & flow_controller ) {
status . carry_flag = 0 ;
const auto dividend = Int32 ( destination ) ;
const auto divisor = Int32 ( Int16 ( source ) ) ;
if ( ! divisor ) {
status . negative_flag = status . overflow_flag = 0 ;
status . zero_result = 1 ;
flow_controller . raise_exception ( Exception : : IntegerDivideByZero ) ;
did_divide < is_divu , false > ( dividend , divisor , flow_controller ) ;
return ;
}
const auto quotient = int64_t ( dividend ) / int64_t ( divisor ) ;
if ( quotient ! = Int32 ( Int16 ( quotient ) ) ) {
status . overflow_flag = 1 ;
did_divide < is_divu , true > ( dividend , divisor , flow_controller ) ;
return ;
}
const auto remainder = Int16 ( dividend % divisor ) ;
destination = uint32_t ( ( uint32_t ( remainder ) < < 16 ) | uint16_t ( quotient ) ) ;
status . overflow_flag = 0 ;
status . zero_result = Status : : FlagT ( quotient ) ;
2022-10-18 14:51:51 -04:00
status . set_negative ( uint16_t ( quotient ) ) ;
2022-10-11 16:16:53 -04:00
did_divide < is_divu , false > ( dividend , divisor , flow_controller ) ;
}
2022-10-17 15:37:13 -04:00
/// Move @c source to @c destination, updating @c status.
2022-10-11 21:27:18 -04:00
template < typename IntT > void move ( IntT source , IntT & destination , Status & status ) {
destination = source ;
2022-10-18 14:51:51 -04:00
status . set_neg_zero ( destination ) ;
2022-10-11 21:27:18 -04:00
status . overflow_flag = status . carry_flag = 0 ;
}
2022-10-17 15:37:13 -04:00
/// Perform NEG.[b/l/w] on @c source, updating @c status.
2022-10-11 21:27:18 -04:00
template < bool is_extend , typename IntT > void negative ( IntT & source , Status & status ) {
const IntT result = - source - ( is_extend & & status . extend_flag ? 1 : 0 ) ;
if constexpr ( is_extend ) {
status . zero_result | = result ;
} else {
status . zero_result = result ;
}
status . extend_flag = status . carry_flag = result ; // i.e. any value other than 0 will result in carry.
2022-10-18 14:51:51 -04:00
status . set_negative ( result ) ;
2023-10-25 22:21:23 -04:00
status . overflow_flag = Numeric : : overflow < false > ( IntT ( 0 ) , source , result ) ;
2022-10-11 21:27:18 -04:00
source = result ;
}
2022-10-17 15:37:13 -04:00
/// Perform TST.[b/l/w] with @c source, updating @c status.
2022-10-11 21:31:14 -04:00
template < typename IntT > void test ( IntT source , Status & status ) {
status . carry_flag = status . overflow_flag = 0 ;
2022-10-18 14:51:51 -04:00
status . set_neg_zero ( source ) ;
2022-10-11 21:31:14 -04:00
}
2022-10-17 15:37:13 -04:00
/// Decodes the proper shift distance from @c source, notifying the @c flow_controller.
2022-10-16 12:06:28 -04:00
template < typename IntT , typename FlowController > int shift_count ( uint8_t source , FlowController & flow_controller ) {
const int count = source & 63 ;
flow_controller . template did_shift < IntT > ( count ) ;
return count ;
}
2022-10-16 22:21:20 -04:00
/// Perform an arithmetic or logical shift, i.e. any of LSL, LSR, ASL or ASR.
template < Operation operation , typename IntT , typename FlowController > void shift ( uint32_t source , IntT & destination , Status & status , FlowController & flow_controller ) {
static_assert (
operation = = Operation : : ASLb | | operation = = Operation : : ASLw | | operation = = Operation : : ASLl | |
operation = = Operation : : ASRb | | operation = = Operation : : ASRw | | operation = = Operation : : ASRl | |
operation = = Operation : : LSLb | | operation = = Operation : : LSLw | | operation = = Operation : : LSLl | |
operation = = Operation : : LSRb | | operation = = Operation : : LSRw | | operation = = Operation : : LSRl
) ;
2023-10-25 22:21:23 -04:00
constexpr auto size = Numeric : : bit_size < IntT > ( ) ;
2022-10-16 22:21:20 -04:00
const auto shift = shift_count < IntT > ( uint8_t ( source ) , flow_controller ) ;
if ( ! shift ) {
status . carry_flag = status . overflow_flag = 0 ;
} else {
enum class Type {
ASL , LSL , ASR , LSR
} type ;
switch ( operation ) {
case Operation : : ASLb : case Operation : : ASLw : case Operation : : ASLl :
type = Type : : ASL ;
break ;
case Operation : : LSLb : case Operation : : LSLw : case Operation : : LSLl :
type = Type : : LSL ;
break ;
case Operation : : ASRb : case Operation : : ASRw : case Operation : : ASRl :
type = Type : : ASR ;
break ;
case Operation : : LSRb : case Operation : : LSRw : case Operation : : LSRl :
type = Type : : LSR ;
break ;
}
switch ( type ) {
case Type : : ASL :
case Type : : LSL :
2022-10-18 11:30:40 -04:00
if ( shift > size ) {
2022-10-16 22:21:20 -04:00
status . carry_flag = status . extend_flag = 0 ;
2022-10-18 11:30:40 -04:00
} else {
2023-10-25 22:21:23 -04:00
status . carry_flag = status . extend_flag = ( destination < < ( shift - 1 ) ) & Numeric : : top_bit < IntT > ( ) ;
2022-10-18 11:30:40 -04:00
}
if ( type = = Type : : LSL ) {
status . overflow_flag = 0 ;
} else {
2022-10-18 11:47:36 -04:00
// Overflow records whether the top bit changed at any point during the operation.
2022-10-18 11:30:40 -04:00
if ( shift > = size ) {
2022-10-18 11:47:36 -04:00
// The result is going to be all bits evacuated through the top giving a net
// result of 0, so overflow is set if any bit was originally set.
status . overflow_flag = destination ;
2022-10-18 11:30:40 -04:00
} else {
2022-10-19 14:40:29 -04:00
// For a shift of n places, overflow will be set if the top n+1 bits were not
2022-10-18 11:47:36 -04:00
// all the same value.
2022-10-18 22:43:17 -04:00
const auto affected_bits = IntT (
2023-10-25 22:21:23 -04:00
~ ( ( Numeric : : top_bit < IntT > ( ) > > shift ) - 1 )
2022-10-19 14:40:29 -04:00
) ; // e.g. shift = 1 => ~((0x80 >> 1) - 1) = ~(0x40 - 1) = ~0x3f = 0xc0, i.e. if shift is
// 1 then the top two bits are relevant to whether there was overflow. If they have the
// same value, i.e. are both 0 or are both 1, then there wasn't. Otherwise there was.
2022-10-18 11:47:36 -04:00
status . overflow_flag = ( destination & affected_bits ) & & ( destination & affected_bits ) ! = affected_bits ;
2022-10-18 11:30:40 -04:00
}
}
if ( shift > = size ) {
destination = 0 ;
} else {
destination < < = shift ;
2022-10-16 22:21:20 -04:00
}
break ;
case Type : : ASR :
case Type : : LSR : {
2022-10-18 11:30:40 -04:00
if ( shift > size ) {
status . carry_flag = status . extend_flag = 0 ;
} else {
status . carry_flag = status . extend_flag = ( destination > > ( shift - 1 ) ) & 1 ;
}
2022-10-18 11:47:36 -04:00
status . overflow_flag = 0 ; // The top bit can't change during an ASR, and LSR always clears overflow.
2022-10-18 11:30:40 -04:00
2022-10-16 22:21:20 -04:00
const IntT sign_word =
type = = Type : : LSR ?
2023-10-25 22:21:23 -04:00
0 : ( destination & Numeric : : top_bit < IntT > ( ) ? IntT ( ~ 0 ) : 0 ) ;
2022-10-16 22:21:20 -04:00
2022-10-18 11:30:40 -04:00
if ( shift > = size ) {
2022-10-16 22:21:20 -04:00
destination = sign_word ;
2022-10-18 11:30:40 -04:00
} else {
destination = IntT ( ( destination > > shift ) | ( sign_word < < ( size - shift ) ) ) ;
2022-10-16 22:21:20 -04:00
}
} break ;
}
}
2022-10-18 14:51:51 -04:00
status . set_neg_zero ( destination ) ;
2022-10-16 22:21:20 -04:00
}
2022-10-16 12:19:09 -04:00
/// Perform a rotate without extend, i.e. any of RO[L/R].[b/w/l].
template < Operation operation , typename IntT , typename FlowController > void rotate ( uint32_t source , IntT & destination , Status & status , FlowController & flow_controller ) {
static_assert (
operation = = Operation : : ROLb | | operation = = Operation : : ROLw | | operation = = Operation : : ROLl | |
operation = = Operation : : RORb | | operation = = Operation : : RORw | | operation = = Operation : : RORl
) ;
2023-10-25 22:21:23 -04:00
constexpr auto size = Numeric : : bit_size < IntT > ( ) ;
2022-10-17 22:57:21 -04:00
auto shift = shift_count < IntT > ( uint8_t ( source ) , flow_controller ) ;
2022-10-16 12:19:09 -04:00
if ( ! shift ) {
status . carry_flag = 0 ;
} else {
2022-10-17 22:57:21 -04:00
shift & = size - 1 ;
2022-10-16 12:19:09 -04:00
switch ( operation ) {
case Operation : : ROLb : case Operation : : ROLw : case Operation : : ROLl :
2022-10-18 11:30:40 -04:00
if ( shift ) {
destination = IntT (
( destination < < shift ) |
( destination > > ( size - shift ) )
) ;
}
2022-10-16 21:52:00 -04:00
status . carry_flag = Status : : FlagT ( destination & 1 ) ;
2022-10-16 12:19:09 -04:00
break ;
case Operation : : RORb : case Operation : : RORw : case Operation : : RORl :
2022-10-18 11:30:40 -04:00
if ( shift ) {
destination = IntT (
( destination > > shift ) |
( destination < < ( size - shift ) )
) ;
}
2023-10-25 22:21:23 -04:00
status . carry_flag = Status : : FlagT ( destination & Numeric : : top_bit < IntT > ( ) ) ;
2022-10-16 12:19:09 -04:00
break ;
}
}
2022-10-18 14:51:51 -04:00
status . set_neg_zero ( destination ) ;
2022-10-16 21:52:00 -04:00
status . overflow_flag = 0 ;
2022-10-16 12:19:09 -04:00
}
2022-10-16 12:06:28 -04:00
/// Perform a rotate-through-extend, i.e. any of ROX[L/R].[b/w/l].
template < Operation operation , typename IntT , typename FlowController > void rox ( uint32_t source , IntT & destination , Status & status , FlowController & flow_controller ) {
static_assert (
operation = = Operation : : ROXLb | | operation = = Operation : : ROXLw | | operation = = Operation : : ROXLl | |
operation = = Operation : : ROXRb | | operation = = Operation : : ROXRw | | operation = = Operation : : ROXRl
) ;
2023-10-25 22:21:23 -04:00
constexpr auto size = Numeric : : bit_size < IntT > ( ) ;
2022-10-16 21:52:00 -04:00
auto shift = shift_count < IntT > ( uint8_t ( source ) , flow_controller ) % ( size + 1 ) ;
2022-10-16 12:06:28 -04:00
2022-10-16 21:52:00 -04:00
if ( ! shift ) {
// When shift is zero, extend is unaffected but is copied to carry.
status . carry_flag = status . extend_flag ;
} else {
2023-05-12 14:14:45 -04:00
switch ( operation ) {
2022-10-16 21:52:00 -04:00
case Operation : : ROXLb : case Operation : : ROXLw : case Operation : : ROXLl :
2022-10-17 22:19:35 -04:00
status . carry_flag = Status : : FlagT ( ( destination > > ( size - shift ) ) & 1 ) ;
2022-10-18 11:30:40 -04:00
2023-10-25 22:21:23 -04:00
if ( shift = = Numeric : : bit_size < IntT > ( ) ) {
2022-10-18 11:30:40 -04:00
destination = IntT (
2023-10-25 22:21:23 -04:00
( status . extend_flag ? Numeric : : top_bit < IntT > ( ) : 0 ) |
2022-10-18 11:30:40 -04:00
( destination > > 1 )
) ;
} else if ( shift = = 1 ) {
destination = IntT (
( destination < < 1 ) |
IntT ( status . extend_flag ? 1 : 0 )
) ;
} else {
destination = IntT (
( destination < < shift ) |
( IntT ( status . extend_flag ? 1 : 0 ) < < ( shift - 1 ) ) |
( destination > > ( size + 1 - shift ) )
) ;
}
2022-10-17 22:19:35 -04:00
status . extend_flag = status . carry_flag ;
2022-10-16 21:52:00 -04:00
break ;
case Operation : : ROXRb : case Operation : : ROXRw : case Operation : : ROXRl :
2022-10-17 22:19:35 -04:00
status . carry_flag = Status : : FlagT ( destination & ( 1 < < ( shift - 1 ) ) ) ;
2022-10-18 11:30:40 -04:00
2023-10-25 22:21:23 -04:00
if ( shift = = Numeric : : bit_size < IntT > ( ) ) {
2022-10-18 11:30:40 -04:00
destination = IntT (
( status . extend_flag ? 1 : 0 ) |
( destination < < 1 )
) ;
} else if ( shift = = 1 ) {
destination = IntT (
( destination > > 1 ) |
2023-10-25 22:21:23 -04:00
( status . extend_flag ? Numeric : : top_bit < IntT > ( ) : 0 )
2022-10-18 11:30:40 -04:00
) ;
} else {
destination = IntT (
( destination > > shift ) |
2023-10-25 22:21:23 -04:00
( ( status . extend_flag ? Numeric : : top_bit < IntT > ( ) : 0 ) > > ( shift - 1 ) ) |
2022-10-18 11:30:40 -04:00
( destination < < ( size + 1 - shift ) )
) ;
}
2022-10-17 22:19:35 -04:00
status . extend_flag = status . carry_flag ;
2022-10-16 21:52:00 -04:00
break ;
}
2022-10-16 12:06:28 -04:00
}
2022-10-18 14:51:51 -04:00
status . set_neg_zero ( destination ) ;
2022-10-16 21:52:00 -04:00
status . overflow_flag = 0 ;
2022-10-16 12:06:28 -04:00
}
2022-10-10 22:21:13 -04:00
}
2022-04-28 16:55:47 -04:00
template <
Model model ,
2022-05-01 17:39:56 -04:00
typename FlowController ,
Operation operation = Operation : : Undefined
2022-04-30 14:08:51 -04:00
> void perform ( Preinstruction instruction , CPU : : SlicedInt32 & src , CPU : : SlicedInt32 & dest , Status & status , FlowController & flow_controller ) {
2022-04-28 16:55:47 -04:00
2022-05-01 17:39:56 -04:00
switch ( ( operation ! = Operation : : Undefined ) ? operation : instruction . operation ) {
2022-04-28 16:55:47 -04:00
/*
2022-04-29 07:57:02 -04:00
ABCD adds the lowest bytes from the source and destination using BCD arithmetic ,
2022-04-28 16:55:47 -04:00
obeying the extend flag .
*/
case Operation : : ABCD : {
// Pull out the two halves, for simplicity.
2022-04-29 04:51:02 -04:00
const uint8_t source = src . b ;
const uint8_t destination = dest . b ;
2022-05-12 15:18:03 -04:00
const int extend = ( status . extend_flag ? 1 : 0 ) ;
2022-04-28 16:55:47 -04:00
// Perform the BCD add by evaluating the two nibbles separately.
2022-05-12 15:18:03 -04:00
const int unadjusted_result = destination + source + extend ;
int result = ( destination & 0xf ) + ( source & 0xf ) + extend ;
2022-05-12 15:25:01 -04:00
result + =
( destination & 0xf0 ) +
( source & 0xf0 ) +
2022-05-12 16:21:36 -04:00
( ( ( 9 - result ) > > 4 ) & 0x06 ) ; // i.e. ((result > 0x09) ? 0x06 : 0x00)
2022-05-12 15:25:01 -04:00
result + = ( ( 0x9f - result ) > > 4 ) & 0x60 ; // i.e. ((result > 0x9f) ? 0x60 : 0x00)
2022-04-28 16:55:47 -04:00
// Set all flags essentially as if this were normal addition.
2022-05-12 08:19:41 -04:00
status . zero_result | = result & 0xff ;
status . extend_flag = status . carry_flag = uint_fast32_t ( result & ~ 0xff ) ;
2022-10-18 14:51:51 -04:00
status . set_negative ( uint8_t ( result ) ) ;
2022-05-12 08:19:41 -04:00
status . overflow_flag = ~ unadjusted_result & result & 0x80 ;
2022-04-28 16:55:47 -04:00
// Store the result.
2022-04-29 04:51:02 -04:00
dest . b = uint8_t ( result ) ;
2022-04-28 16:55:47 -04:00
} break ;
// ADD and ADDA add two quantities, the latter sign extending and without setting any flags;
// ADDQ and SUBQ act as ADD and SUB, but taking the second argument from the instruction code.
2022-10-10 22:21:13 -04:00
case Operation : : ADDb : Primitive : : add_sub < true , false > ( src . b , dest . b , status ) ; break ;
case Operation : : SUBb : Primitive : : add_sub < false , false > ( src . b , dest . b , status ) ; break ;
case Operation : : ADDXb : Primitive : : add_sub < true , true > ( src . b , dest . b , status ) ; break ;
case Operation : : SUBXb : Primitive : : add_sub < false , true > ( src . b , dest . b , status ) ; break ;
case Operation : : ADDw : Primitive : : add_sub < true , false > ( src . w , dest . w , status ) ; break ;
case Operation : : SUBw : Primitive : : add_sub < false , false > ( src . w , dest . w , status ) ; break ;
case Operation : : ADDXw : Primitive : : add_sub < true , true > ( src . w , dest . w , status ) ; break ;
case Operation : : SUBXw : Primitive : : add_sub < false , true > ( src . w , dest . w , status ) ; break ;
case Operation : : ADDl : Primitive : : add_sub < true , false > ( src . l , dest . l , status ) ; break ;
case Operation : : SUBl : Primitive : : add_sub < false , false > ( src . l , dest . l , status ) ; break ;
case Operation : : ADDXl : Primitive : : add_sub < true , true > ( src . l , dest . l , status ) ; break ;
case Operation : : SUBXl : Primitive : : add_sub < false , true > ( src . l , dest . l , status ) ; break ;
case Operation : : ADDAw : dest . l + = u_extend16 ( src . w ) ; break ;
case Operation : : ADDAl : dest . l + = src . l ; break ;
case Operation : : SUBAw : dest . l - = u_extend16 ( src . w ) ; break ;
case Operation : : SUBAl : dest . l - = src . l ; break ;
2022-04-28 16:55:47 -04:00
2022-05-04 20:57:22 -04:00
// BTST/BCLR/etc: modulo for the mask depends on whether memory or a data register is the target.
2022-10-11 10:47:55 -04:00
case Operation : : BTST :
status . zero_result = dest . l & ( 1 < < Primitive : : mask_bit ( instruction , src . l ) ) ;
break ;
case Operation : : BCLR : Primitive : : bit_manipulate < Operation : : BCLR > ( instruction , src . l , dest . l , status , flow_controller ) ; break ;
case Operation : : BCHG : Primitive : : bit_manipulate < Operation : : BCHG > ( instruction , src . l , dest . l , status , flow_controller ) ; break ;
case Operation : : BSET : Primitive : : bit_manipulate < Operation : : BSET > ( instruction , src . l , dest . l , status , flow_controller ) ; break ;
2022-05-13 15:08:15 -04:00
2022-04-29 17:12:06 -04:00
case Operation : : Bccb :
2022-05-09 16:19:25 -04:00
flow_controller . template complete_bcc < int8_t > (
status . evaluate_condition ( instruction . condition ( ) ) ,
2022-05-24 11:05:24 -04:00
int8_t ( src . b ) ) ;
2022-05-03 14:45:49 -04:00
break ;
2022-04-29 17:12:06 -04:00
case Operation : : Bccw :
2022-05-09 16:19:25 -04:00
flow_controller . template complete_bcc < int16_t > (
status . evaluate_condition ( instruction . condition ( ) ) ,
2022-05-24 11:05:24 -04:00
int16_t ( src . w ) ) ;
2022-05-03 14:45:49 -04:00
break ;
2022-04-29 17:12:06 -04:00
2022-05-03 14:45:49 -04:00
case Operation : : Bccl :
2022-05-09 16:19:25 -04:00
flow_controller . template complete_bcc < int32_t > (
status . evaluate_condition ( instruction . condition ( ) ) ,
2022-05-24 11:05:24 -04:00
int32_t ( src . l ) ) ;
2022-05-03 14:45:49 -04:00
break ;
2022-04-28 16:55:47 -04:00
2022-05-03 15:32:54 -04:00
case Operation : : BSRb :
2022-05-24 11:05:24 -04:00
flow_controller . bsr ( uint32_t ( int8_t ( src . b ) ) ) ;
2022-05-03 15:32:54 -04:00
break ;
case Operation : : BSRw :
2022-05-24 11:05:24 -04:00
flow_controller . bsr ( uint32_t ( int16_t ( src . w ) ) ) ;
2022-05-03 15:32:54 -04:00
break ;
case Operation : : BSRl :
2022-05-20 12:40:09 -04:00
flow_controller . bsr ( src . l ) ;
2022-05-03 15:32:54 -04:00
break ;
2022-05-09 16:19:25 -04:00
case Operation : : DBcc : {
const bool matched_condition = status . evaluate_condition ( instruction . condition ( ) ) ;
bool overflowed = false ;
2022-04-29 17:12:06 -04:00
2022-05-09 16:19:25 -04:00
// Classify the dbcc.
if ( ! matched_condition ) {
- - src . w ;
overflowed = src . w = = 0xffff ;
2022-04-29 17:12:06 -04:00
}
2022-05-09 16:19:25 -04:00
// Take the branch.
flow_controller . complete_dbcc (
matched_condition ,
overflowed ,
int16_t ( dest . w ) ) ;
} break ;
2022-04-29 17:12:06 -04:00
2022-05-20 11:19:16 -04:00
case Operation : : Scc : {
const bool condition = status . evaluate_condition ( instruction . condition ( ) ) ;
src . b = condition ? 0xff : 0x00 ;
flow_controller . did_scc ( condition ) ;
} break ;
2022-04-29 17:12:06 -04:00
2022-04-28 16:55:47 -04:00
/*
CLRs : store 0 to the destination , set the zero flag , and clear
negative , overflow and carry .
*/
2022-10-11 11:22:34 -04:00
case Operation : : CLRb : Primitive : : clear ( src . b , status ) ; break ;
case Operation : : CLRw : Primitive : : clear ( src . w , status ) ; break ;
case Operation : : CLRl : Primitive : : clear ( src . l , status ) ; break ;
2022-04-28 16:55:47 -04:00
/*
CMP . b , CMP . l and CMP . w : sets the condition flags ( other than extend ) based on a subtraction
of the source from the destination ; the result of the subtraction is not stored .
*/
2022-10-11 12:57:02 -04:00
case Operation : : CMPb : Primitive : : compare ( src . b , dest . b , status ) ; break ;
case Operation : : CMPw : Primitive : : compare ( src . w , dest . w , status ) ; break ;
case Operation : : CMPAw : Primitive : : compare ( u_extend16 ( src . w ) , dest . l , status ) ; break ;
2022-05-03 09:20:02 -04:00
case Operation : : CMPAl :
2022-10-11 12:57:02 -04:00
case Operation : : CMPl : Primitive : : compare ( src . l , dest . l , status ) ; break ;
2022-04-28 16:55:47 -04:00
// JMP: copies EA(0) to the program counter.
2022-04-29 17:12:06 -04:00
case Operation : : JMP :
2022-05-09 16:20:15 -04:00
flow_controller . jmp ( src . l ) ;
2022-04-29 17:12:06 -04:00
break ;
2022-04-28 16:55:47 -04:00
2022-05-03 15:49:55 -04:00
// JSR: jump to EA(0), pushing the current PC to the stack.
case Operation : : JSR :
flow_controller . jsr ( src . l ) ;
break ;
2022-04-28 16:55:47 -04:00
/*
MOVE . b , MOVE . l and MOVE . w : move the least significant byte or word , or the entire long word ,
and set negative , zero , overflow and carry as appropriate .
*/
2022-10-11 21:27:18 -04:00
case Operation : : MOVEb : Primitive : : move ( src . b , dest . b , status ) ; break ;
case Operation : : MOVEw : Primitive : : move ( src . w , dest . w , status ) ; break ;
case Operation : : MOVEl : Primitive : : move ( src . l , dest . l , status ) ; break ;
2022-04-28 16:55:47 -04:00
/*
MOVEA . l : move the entire long word ;
MOVEA . w : move the least significant word and sign extend it .
Neither sets any flags .
*/
case Operation : : MOVEAw :
2022-04-29 04:51:02 -04:00
dest . l = u_extend16 ( src . w ) ;
2022-04-28 16:55:47 -04:00
break ;
case Operation : : MOVEAl :
2022-04-29 04:51:02 -04:00
dest . l = src . l ;
2022-04-28 16:55:47 -04:00
break ;
2022-04-29 17:12:06 -04:00
case Operation : : LEA :
2022-04-29 20:30:48 -04:00
dest . l = src . l ;
2022-04-29 17:12:06 -04:00
break ;
2022-05-09 16:19:25 -04:00
case Operation : : PEA :
flow_controller . pea ( src . l ) ;
break ;
2022-04-28 16:55:47 -04:00
/*
Status word moves and manipulations .
*/
case Operation : : MOVEtoSR :
2022-04-29 04:51:02 -04:00
status . set_status ( src . w ) ;
2022-05-07 21:29:12 -04:00
flow_controller . did_update_status ( ) ;
2022-04-28 16:55:47 -04:00
break ;
case Operation : : MOVEfromSR :
2022-05-06 11:33:57 -04:00
src . w = status . status ( ) ;
2022-04-28 16:55:47 -04:00
break ;
case Operation : : MOVEtoCCR :
2022-04-29 04:51:02 -04:00
status . set_ccr ( src . w ) ;
2022-04-28 16:55:47 -04:00
break ;
2022-05-11 07:52:23 -04:00
case Operation : : MOVEtoUSP :
flow_controller . move_to_usp ( src . l ) ;
break ;
case Operation : : MOVEfromUSP :
flow_controller . move_from_usp ( src . l ) ;
break ;
2022-04-28 16:55:47 -04:00
case Operation : : EXTbtow :
2022-05-03 20:17:36 -04:00
src . w = uint16_t ( int8_t ( src . b ) ) ;
2022-05-12 08:19:41 -04:00
status . overflow_flag = status . carry_flag = 0 ;
2022-10-18 14:51:51 -04:00
status . set_neg_zero ( src . w ) ;
2022-04-28 16:55:47 -04:00
break ;
case Operation : : EXTwtol :
2022-05-03 20:17:36 -04:00
src . l = u_extend16 ( src . w ) ;
2022-05-12 08:19:41 -04:00
status . overflow_flag = status . carry_flag = 0 ;
2022-10-18 14:51:51 -04:00
status . set_neg_zero ( src . l ) ;
2022-04-28 16:55:47 -04:00
break ;
2022-10-11 15:53:11 -04:00
case Operation : : ANDItoSR : Primitive : : apply_sr_ccr < Operation : : ANDItoSR > ( src . w , status , flow_controller ) ; break ;
case Operation : : EORItoSR : Primitive : : apply_sr_ccr < Operation : : EORItoSR > ( src . w , status , flow_controller ) ; break ;
case Operation : : ORItoSR : Primitive : : apply_sr_ccr < Operation : : ORItoSR > ( src . w , status , flow_controller ) ; break ;
case Operation : : ANDItoCCR : Primitive : : apply_sr_ccr < Operation : : ANDItoCCR > ( src . w , status , flow_controller ) ; break ;
case Operation : : EORItoCCR : Primitive : : apply_sr_ccr < Operation : : EORItoCCR > ( src . w , status , flow_controller ) ; break ;
case Operation : : ORItoCCR : Primitive : : apply_sr_ccr < Operation : : ORItoCCR > ( src . w , status , flow_controller ) ; break ;
2022-04-28 16:55:47 -04:00
/*
Multiplications .
*/
2022-10-22 15:20:30 -04:00
case Operation : : MULUw : Primitive : : multiply < true > ( src . w , dest . l , status , flow_controller ) ; break ;
case Operation : : MULSw : Primitive : : multiply < false > ( src . w , dest . l , status , flow_controller ) ; break ;
2022-04-28 16:55:47 -04:00
/*
Divisions .
*/
2022-10-22 15:20:30 -04:00
case Operation : : DIVUw : Primitive : : divide < true , uint16_t , uint32_t > ( src . w , dest . l , status , flow_controller ) ; break ;
case Operation : : DIVSw : Primitive : : divide < false , int16_t , int32_t > ( src . w , dest . l , status , flow_controller ) ; break ;
2022-04-28 16:55:47 -04:00
// TRAP, which is a nicer form of ILLEGAL.
2022-04-29 14:43:30 -04:00
case Operation : : TRAP :
2022-05-24 11:05:24 -04:00
flow_controller . template raise_exception < false > ( int ( src . l + Exception : : TrapBase ) ) ;
2022-04-29 14:43:30 -04:00
break ;
2022-04-28 16:55:47 -04:00
2022-04-29 14:43:30 -04:00
case Operation : : TRAPV : {
2022-05-12 08:19:41 -04:00
if ( status . overflow_flag ) {
2022-05-11 07:00:35 -04:00
flow_controller . template raise_exception < false > ( Exception : : TRAPV ) ;
2022-04-29 14:43:30 -04:00
}
} break ;
2022-04-28 16:55:47 -04:00
2022-10-22 15:20:30 -04:00
case Operation : : CHKw : {
2022-04-29 04:51:02 -04:00
const bool is_under = s_extend16 ( dest . w ) < 0 ;
const bool is_over = s_extend16 ( dest . w ) > s_extend16 ( src . w ) ;
2022-04-28 16:55:47 -04:00
2022-05-12 08:19:41 -04:00
status . overflow_flag = status . carry_flag = 0 ;
status . zero_result = dest . w ;
2022-04-28 16:55:47 -04:00
// Test applied for N:
//
// if Dn < 0, set negative flag;
// otherwise, if Dn > <ea>, reset negative flag.
2022-05-12 08:19:41 -04:00
if ( is_over ) status . negative_flag = 0 ;
if ( is_under ) status . negative_flag = 1 ;
2022-04-28 16:55:47 -04:00
// No exception is the default course of action; deviate only if an
// exception is necessary.
2022-05-09 20:58:51 -04:00
flow_controller . did_chk ( is_under , is_over ) ;
2022-04-28 16:55:47 -04:00
if ( is_under | | is_over ) {
2022-05-11 07:00:35 -04:00
flow_controller . template raise_exception < false > ( Exception : : CHK ) ;
2022-04-28 16:55:47 -04:00
}
} break ;
/*
NEGs : negatives the destination , setting the zero ,
negative , overflow and carry flags appropriate , and extend .
NB : since the same logic as SUB is used to calculate overflow ,
and SUB calculates ` destination - source ` , the NEGs deliberately
label ' source ' and ' destination ' differently from Motorola .
*/
2022-10-11 21:27:18 -04:00
case Operation : : NEGb : Primitive : : negative < false > ( src . b , status ) ; break ;
case Operation : : NEGw : Primitive : : negative < false > ( src . w , status ) ; break ;
case Operation : : NEGl : Primitive : : negative < false > ( src . l , status ) ; break ;
2022-04-28 16:55:47 -04:00
/*
NEGXs : NEG , with extend .
*/
2022-10-11 21:27:18 -04:00
case Operation : : NEGXb : Primitive : : negative < true > ( src . b , status ) ; break ;
case Operation : : NEGXw : Primitive : : negative < true > ( src . w , status ) ; break ;
case Operation : : NEGXl : Primitive : : negative < true > ( src . l , status ) ; break ;
2022-04-28 16:55:47 -04:00
/*
The no - op .
*/
2022-04-29 17:12:06 -04:00
case Operation : : NOP : break ;
2022-04-28 16:55:47 -04:00
/*
LINK and UNLINK help with stack frames , allowing a certain
amount of stack space to be allocated or deallocated .
*/
2022-05-04 08:26:11 -04:00
case Operation : : LINKw :
2022-05-24 11:05:24 -04:00
flow_controller . link ( instruction , uint32_t ( int16_t ( dest . w ) ) ) ;
2022-05-04 08:26:11 -04:00
break ;
2022-04-29 14:43:30 -04:00
2022-05-04 08:26:11 -04:00
case Operation : : UNLINK :
flow_controller . unlink ( src . l ) ;
break ;
2022-04-28 16:55:47 -04:00
/*
2022-10-18 12:17:55 -04:00
TAS : requiring a specialised bus cycle , just kick this out to the flow controller .
2022-04-28 16:55:47 -04:00
*/
case Operation : : TAS :
2022-05-06 12:23:04 -04:00
flow_controller . tas ( instruction , src . l ) ;
2022-04-28 16:55:47 -04:00
break ;
/*
Bitwise operators : AND , OR and EOR . All three clear the overflow and carry flags ,
and set zero and negative appropriately .
*/
2022-10-17 15:21:54 -04:00
case Operation : : ANDb : Primitive : : bitwise < Operation : : ANDb > ( src . b , dest . b , status ) ; break ;
case Operation : : ANDw : Primitive : : bitwise < Operation : : ANDw > ( src . w , dest . w , status ) ; break ;
case Operation : : ANDl : Primitive : : bitwise < Operation : : ANDl > ( src . l , dest . l , status ) ; break ;
case Operation : : ORb : Primitive : : bitwise < Operation : : ORb > ( src . b , dest . b , status ) ; break ;
case Operation : : ORw : Primitive : : bitwise < Operation : : ORw > ( src . w , dest . w , status ) ; break ;
case Operation : : ORl : Primitive : : bitwise < Operation : : ORl > ( src . l , dest . l , status ) ; break ;
case Operation : : EORb : Primitive : : bitwise < Operation : : EORb > ( src . b , dest . b , status ) ; break ;
case Operation : : EORw : Primitive : : bitwise < Operation : : EORw > ( src . w , dest . w , status ) ; break ;
case Operation : : EORl : Primitive : : bitwise < Operation : : EORl > ( src . l , dest . l , status ) ; break ;
2022-04-28 16:55:47 -04:00
2022-10-18 12:17:55 -04:00
case Operation : : NOTb : Primitive : : bitwise < Operation : : EORb > ( uint8_t ( ~ 0 ) , src . b , status ) ; break ;
case Operation : : NOTw : Primitive : : bitwise < Operation : : EORw > ( uint16_t ( ~ 0 ) , src . w , status ) ; break ;
case Operation : : NOTl : Primitive : : bitwise < Operation : : EORl > ( uint32_t ( ~ 0 ) , src . l , status ) ; break ;
2022-04-28 16:55:47 -04:00
/*
SBCD subtracts the lowest byte of the source from that of the destination using
BCD arithmetic , obeying the extend flag .
*/
2022-10-17 15:12:38 -04:00
case Operation : : SBCD :
Primitive : : sbcd ( src . b , dest . b , dest . b , status ) ;
break ;
2022-04-28 16:55:47 -04:00
/*
2022-05-02 12:57:45 -04:00
NBCD is like SBCD except that the result is 0 - source rather than
2022-04-28 16:55:47 -04:00
destination - source .
*/
2022-10-17 15:12:38 -04:00
case Operation : : NBCD :
Primitive : : sbcd ( src . b , 0 , src . b , status ) ;
break ;
2022-05-02 07:45:07 -04:00
2022-04-28 16:55:47 -04:00
// EXG and SWAP exchange/swap words or long words.
case Operation : : EXG : {
2022-04-29 04:51:02 -04:00
const auto temporary = src . l ;
src . l = dest . l ;
dest . l = temporary ;
2022-04-28 16:55:47 -04:00
} break ;
case Operation : : SWAP : {
2022-05-03 20:17:36 -04:00
uint16_t * const words = reinterpret_cast < uint16_t * > ( & src . l ) ;
2022-10-18 12:17:55 -04:00
std : : swap ( words [ 0 ] , words [ 1 ] ) ;
2022-10-18 14:51:51 -04:00
status . set_neg_zero ( src . l ) ;
2022-05-12 08:19:41 -04:00
status . overflow_flag = status . carry_flag = 0 ;
2022-04-28 16:55:47 -04:00
} break ;
/*
Shifts and rotates .
*/
2022-10-17 11:18:10 -04:00
case Operation : : ASLm :
2023-10-25 22:21:23 -04:00
status . extend_flag = status . carry_flag = src . w & Numeric : : top_bit < uint16_t > ( ) ;
status . overflow_flag = ( src . w ^ ( src . w < < 1 ) ) & Numeric : : top_bit < uint16_t > ( ) ;
2022-10-17 11:18:10 -04:00
src . w < < = 1 ;
2022-10-18 14:51:51 -04:00
status . set_neg_zero ( src . w ) ;
2022-10-17 11:18:10 -04:00
break ;
case Operation : : LSLm :
2023-10-25 22:21:23 -04:00
status . extend_flag = status . carry_flag = src . w & Numeric : : top_bit < uint16_t > ( ) ;
2022-10-17 11:18:10 -04:00
status . overflow_flag = 0 ;
src . w < < = 1 ;
2022-10-18 14:51:51 -04:00
status . set_neg_zero ( src . w ) ;
2022-10-17 11:18:10 -04:00
break ;
case Operation : : ASRm :
status . extend_flag = status . carry_flag = src . w & 1 ;
status . overflow_flag = 0 ;
2023-10-25 22:21:23 -04:00
src . w = ( src . w & Numeric : : top_bit < uint16_t > ( ) ) | ( src . w > > 1 ) ;
2022-10-18 14:51:51 -04:00
status . set_neg_zero ( src . w ) ;
2022-10-17 11:18:10 -04:00
break ;
case Operation : : LSRm :
status . extend_flag = status . carry_flag = src . w & 1 ;
status . overflow_flag = 0 ;
src . w > > = 1 ;
2022-10-18 14:51:51 -04:00
status . set_neg_zero ( src . w ) ;
2022-10-17 11:18:10 -04:00
break ;
2022-10-17 11:27:04 -04:00
case Operation : : ROLm :
2022-10-17 15:04:28 -04:00
src . w = uint16_t ( ( src . w < < 1 ) | ( src . w > > 15 ) ) ;
2022-10-17 11:27:04 -04:00
status . carry_flag = src . w & 0x0001 ;
status . overflow_flag = 0 ;
2022-10-18 14:51:51 -04:00
status . set_neg_zero ( src . w ) ;
2022-10-17 11:27:04 -04:00
break ;
2022-04-28 16:55:47 -04:00
2022-10-17 11:27:04 -04:00
case Operation : : RORm :
2022-10-17 15:04:28 -04:00
src . w = uint16_t ( ( src . w > > 1 ) | ( src . w < < 15 ) ) ;
2023-10-25 22:21:23 -04:00
status . carry_flag = src . w & Numeric : : top_bit < uint16_t > ( ) ;
2022-10-17 11:27:04 -04:00
status . overflow_flag = 0 ;
2022-10-18 14:51:51 -04:00
status . set_neg_zero ( src . w ) ;
2022-10-17 11:27:04 -04:00
break ;
2022-04-28 16:55:47 -04:00
2022-10-17 11:27:04 -04:00
case Operation : : ROXLm :
2023-10-25 22:21:23 -04:00
status . carry_flag = src . w & Numeric : : top_bit < uint16_t > ( ) ;
2022-10-17 15:04:28 -04:00
src . w = uint16_t ( ( src . w < < 1 ) | ( status . extend_flag ? 0x0001 : 0x0000 ) ) ;
2022-10-17 11:27:04 -04:00
status . extend_flag = status . carry_flag ;
status . overflow_flag = 0 ;
2022-10-18 14:51:51 -04:00
status . set_neg_zero ( src . w ) ;
2022-10-17 11:27:04 -04:00
break ;
2022-04-28 16:55:47 -04:00
2022-10-17 11:27:04 -04:00
case Operation : : ROXRm :
status . carry_flag = src . w & 0x0001 ;
2022-10-17 15:04:28 -04:00
src . w = uint16_t ( ( src . w > > 1 ) | ( status . extend_flag ? 0x8000 : 0x0000 ) ) ;
2022-10-17 11:27:04 -04:00
status . extend_flag = status . carry_flag ;
status . overflow_flag = 0 ;
2022-10-18 14:51:51 -04:00
status . set_neg_zero ( src . w ) ;
2022-10-17 11:27:04 -04:00
break ;
2022-04-28 16:55:47 -04:00
2022-10-17 15:05:23 -04:00
case Operation : : ASLb : Primitive : : shift < Operation : : ASLb > ( src . l , dest . b , status , flow_controller ) ; break ;
case Operation : : ASLw : Primitive : : shift < Operation : : ASLw > ( src . l , dest . w , status , flow_controller ) ; break ;
case Operation : : ASLl : Primitive : : shift < Operation : : ASLl > ( src . l , dest . l , status , flow_controller ) ; break ;
2022-04-28 16:55:47 -04:00
2022-10-17 15:05:23 -04:00
case Operation : : ASRb : Primitive : : shift < Operation : : ASRb > ( src . l , dest . b , status , flow_controller ) ; break ;
case Operation : : ASRw : Primitive : : shift < Operation : : ASRw > ( src . l , dest . w , status , flow_controller ) ; break ;
case Operation : : ASRl : Primitive : : shift < Operation : : ASRl > ( src . l , dest . l , status , flow_controller ) ; break ;
2022-04-28 16:55:47 -04:00
2022-10-17 15:05:23 -04:00
case Operation : : LSLb : Primitive : : shift < Operation : : LSLb > ( src . l , dest . b , status , flow_controller ) ; break ;
case Operation : : LSLw : Primitive : : shift < Operation : : LSLw > ( src . l , dest . w , status , flow_controller ) ; break ;
case Operation : : LSLl : Primitive : : shift < Operation : : LSLl > ( src . l , dest . l , status , flow_controller ) ; break ;
2022-04-28 16:55:47 -04:00
2022-10-17 15:05:23 -04:00
case Operation : : LSRb : Primitive : : shift < Operation : : LSRb > ( src . l , dest . b , status , flow_controller ) ; break ;
case Operation : : LSRw : Primitive : : shift < Operation : : LSRw > ( src . l , dest . w , status , flow_controller ) ; break ;
case Operation : : LSRl : Primitive : : shift < Operation : : LSRl > ( src . l , dest . l , status , flow_controller ) ; break ;
2022-04-28 16:55:47 -04:00
2022-10-17 15:05:23 -04:00
case Operation : : ROLb : Primitive : : rotate < Operation : : ROLb > ( src . l , dest . b , status , flow_controller ) ; break ;
case Operation : : ROLw : Primitive : : rotate < Operation : : ROLw > ( src . l , dest . w , status , flow_controller ) ; break ;
case Operation : : ROLl : Primitive : : rotate < Operation : : ROLl > ( src . l , dest . l , status , flow_controller ) ; break ;
2022-04-28 16:55:47 -04:00
2022-10-17 15:05:23 -04:00
case Operation : : RORb : Primitive : : rotate < Operation : : RORb > ( src . l , dest . b , status , flow_controller ) ; break ;
case Operation : : RORw : Primitive : : rotate < Operation : : RORw > ( src . l , dest . w , status , flow_controller ) ; break ;
case Operation : : RORl : Primitive : : rotate < Operation : : RORl > ( src . l , dest . l , status , flow_controller ) ; break ;
2022-04-28 16:55:47 -04:00
2022-10-16 12:06:28 -04:00
case Operation : : ROXLb : Primitive : : rox < Operation : : ROXLb > ( src . l , dest . b , status , flow_controller ) ; break ;
case Operation : : ROXLw : Primitive : : rox < Operation : : ROXLw > ( src . l , dest . w , status , flow_controller ) ; break ;
case Operation : : ROXLl : Primitive : : rox < Operation : : ROXLl > ( src . l , dest . l , status , flow_controller ) ; break ;
2022-04-28 16:55:47 -04:00
2022-10-16 12:06:28 -04:00
case Operation : : ROXRb : Primitive : : rox < Operation : : ROXRb > ( src . l , dest . b , status , flow_controller ) ; break ;
case Operation : : ROXRw : Primitive : : rox < Operation : : ROXRw > ( src . l , dest . w , status , flow_controller ) ; break ;
case Operation : : ROXRl : Primitive : : rox < Operation : : ROXRl > ( src . l , dest . l , status , flow_controller ) ; break ;
2022-04-28 16:55:47 -04:00
2022-05-05 09:00:33 -04:00
case Operation : : MOVEPl :
2022-05-05 12:27:36 -04:00
flow_controller . template movep < uint32_t > ( instruction , src . l , dest . l ) ;
2022-05-05 09:00:33 -04:00
break ;
case Operation : : MOVEPw :
2022-05-05 12:27:36 -04:00
flow_controller . template movep < uint16_t > ( instruction , src . l , dest . l ) ;
2022-05-05 09:00:33 -04:00
break ;
2022-05-06 09:45:06 -04:00
case Operation : : MOVEMtoRl :
flow_controller . template movem_toR < uint32_t > ( instruction , src . l , dest . l ) ;
2022-05-05 12:42:57 -04:00
break ;
2022-05-06 09:45:06 -04:00
case Operation : : MOVEMtoMl :
flow_controller . template movem_toM < uint32_t > ( instruction , src . l , dest . l ) ;
break ;
case Operation : : MOVEMtoRw :
flow_controller . template movem_toR < uint16_t > ( instruction , src . l , dest . l ) ;
break ;
case Operation : : MOVEMtoMw :
flow_controller . template movem_toM < uint16_t > ( instruction , src . l , dest . l ) ;
2022-05-05 12:42:57 -04:00
break ;
2022-04-28 16:55:47 -04:00
/*
2022-10-18 12:17:55 -04:00
RTE , RTR and RTS defer to the flow controller .
2022-04-28 16:55:47 -04:00
*/
2022-10-18 12:17:55 -04:00
case Operation : : RTR : flow_controller . rtr ( ) ; break ;
case Operation : : RTE : flow_controller . rte ( ) ; break ;
case Operation : : RTS : flow_controller . rts ( ) ; break ;
2022-04-28 16:55:47 -04:00
/*
TSTs : compare to zero .
*/
2022-10-11 21:31:14 -04:00
case Operation : : TSTb : Primitive : : test ( src . b , status ) ; break ;
case Operation : : TSTw : Primitive : : test ( src . w , status ) ; break ;
case Operation : : TSTl : Primitive : : test ( src . l , status ) ; break ;
2022-04-28 16:55:47 -04:00
2022-04-29 14:43:30 -04:00
case Operation : : STOP :
status . set_status ( src . w ) ;
2022-05-07 21:29:12 -04:00
flow_controller . did_update_status ( ) ;
2022-04-29 14:43:30 -04:00
flow_controller . stop ( ) ;
break ;
2022-04-28 16:55:47 -04:00
2022-05-11 07:52:23 -04:00
case Operation : : RESET :
flow_controller . reset ( ) ;
break ;
2022-04-28 16:55:47 -04:00
/*
Development period debugging .
*/
default :
assert ( false ) ;
break ;
}
}
}
# endif /* InstructionSets_M68k_PerformImplementation_h */