2014-04-09 00:15:46 +01:00
// Copyright 2014 Citra Emulator Project
// Licensed under GPLv2
// Refer to the license.txt file included.
2014-04-05 06:23:51 +01:00
# pragma once
2014-08-03 15:00:52 +01:00
# include <cstddef>
2014-04-09 01:15:08 +01:00
# include "common/common_types.h"
2014-05-31 23:08:00 +01:00
# include "common/bit_field.h"
2014-04-05 06:23:51 +01:00
2014-05-17 21:50:33 +01:00
namespace GPU {
2014-04-05 06:23:51 +01:00
2014-06-06 05:06:33 +01:00
static const u32 kFrameCycles = 268123480 / 60 ; ///< 268MHz / 60 frames per second
static const u32 kFrameTicks = kFrameCycles / 3 ; ///< Approximate number of instructions/frame
2014-05-29 02:19:13 +01:00
2014-08-03 15:00:52 +01:00
// Returns index corresponding to the Regs member labeled by field_name
// TODO: Due to Visual studio bug 209229, offsetof does not return constant expressions
// when used with array elements (e.g. GPU_REG_INDEX(memory_fill_config[0])).
// For details cf. https://connect.microsoft.com/VisualStudio/feedback/details/209229/offsetof-does-not-produce-a-constant-expression-for-array-members
// Hopefully, this will be fixed sometime in the future.
// For lack of better alternatives, we currently hardcode the offsets when constant
// expressions are needed via GPU_REG_INDEX_WORKAROUND (on sane compilers, static_asserts
// will then make sure the offsets indeed match the automatically calculated ones).
# define GPU_REG_INDEX(field_name) (offsetof(GPU::Regs, field_name) / sizeof(u32))
# if defined(_MSC_VER)
# define GPU_REG_INDEX_WORKAROUND(field_name, backup_workaround_index) (backup_workaround_index)
# else
// NOTE: Yeah, hacking in a static_assert here just to workaround the lacking MSVC compiler
// really is this annoying. This macro just forwards its first argument to GPU_REG_INDEX
// and then performs a (no-op) cast to size_t iff the second argument matches the expected
// field offset. Otherwise, the compiler will fail to compile this code.
# define GPU_REG_INDEX_WORKAROUND(field_name, backup_workaround_index) \
( ( typename std : : enable_if < backup_workaround_index = = GPU_REG_INDEX ( field_name ) , size_t > : : type ) GPU_REG_INDEX ( field_name ) )
# endif
2014-07-16 10:24:09 +01:00
// MMIO region 0x1EFxxxxx
struct Regs {
2014-08-03 15:00:52 +01:00
// helper macro to properly align structure members.
// Calling INSERT_PADDING_WORDS will add a new member variable with a name like "pad121",
// depending on the current source line to make sure variable names are unique.
# define INSERT_PADDING_WORDS_HELPER1(x, y) x ## y
# define INSERT_PADDING_WORDS_HELPER2(x, y) INSERT_PADDING_WORDS_HELPER1(x, y)
2014-08-23 12:42:34 +01:00
# define INSERT_PADDING_WORDS(num_words) u32 INSERT_PADDING_WORDS_HELPER2(pad, __LINE__)[(num_words)]
2014-08-03 15:00:52 +01:00
// helper macro to make sure the defined structures are of the expected size.
# if defined(_MSC_VER)
// TODO: MSVC does not support using sizeof() on non-static data members even though this
// is technically allowed since C++11. This macro should be enabled once MSVC adds
// support for that.
# define ASSERT_MEMBER_SIZE(name, size_in_bytes)
# else
# define ASSERT_MEMBER_SIZE(name, size_in_bytes) \
static_assert ( sizeof ( name ) = = size_in_bytes , \
2014-08-23 12:42:34 +01:00
" Structure size and register block length don't match " )
2014-08-03 15:00:52 +01:00
# endif
2014-07-16 10:24:09 +01:00
2014-07-11 18:10:08 +01:00
enum class FramebufferFormat : u32 {
RGBA8 = 0 ,
RGB8 = 1 ,
RGB565 = 2 ,
RGB5A1 = 3 ,
RGBA4 = 4 ,
} ;
2014-07-16 10:24:09 +01:00
2014-08-03 15:00:52 +01:00
INSERT_PADDING_WORDS ( 0x4 ) ;
2014-07-16 10:27:58 +01:00
2014-08-03 15:00:52 +01:00
struct {
u32 address_start ;
u32 address_end ; // ?
u32 size ;
u32 value ; // ?
2014-07-16 10:27:58 +01:00
2014-08-03 15:00:52 +01:00
inline u32 GetStartAddress ( ) const {
return DecodeAddressRegister ( address_start ) ;
}
2014-06-04 17:30:23 +01:00
2014-08-03 15:00:52 +01:00
inline u32 GetEndAddress ( ) const {
return DecodeAddressRegister ( address_end ) ;
}
} memory_fill_config [ 2 ] ;
ASSERT_MEMBER_SIZE ( memory_fill_config [ 0 ] , 0x10 ) ;
2014-05-17 21:07:06 +01:00
2014-08-03 15:00:52 +01:00
INSERT_PADDING_WORDS ( 0x10b ) ;
2014-07-11 18:10:08 +01:00
2014-08-03 15:00:52 +01:00
struct {
using Format = Regs : : FramebufferFormat ;
2014-07-11 18:10:08 +01:00
2014-08-03 15:00:52 +01:00
union {
u32 size ;
2014-07-16 10:24:09 +01:00
2014-08-03 15:00:52 +01:00
BitField < 0 , 16 , u32 > width ;
BitField < 16 , 16 , u32 > height ;
} ;
2014-07-16 10:24:09 +01:00
2014-08-03 15:00:52 +01:00
INSERT_PADDING_WORDS ( 0x2 ) ;
2014-07-11 18:10:08 +01:00
2014-08-03 15:00:52 +01:00
u32 address_left1 ;
u32 address_left2 ;
2014-07-11 18:10:08 +01:00
2014-08-03 15:00:52 +01:00
union {
u32 format ;
2014-07-16 10:24:09 +01:00
2014-08-03 15:00:52 +01:00
BitField < 0 , 3 , Format > color_format ;
} ;
2014-07-11 18:10:08 +01:00
2014-08-03 15:00:52 +01:00
INSERT_PADDING_WORDS ( 0x1 ) ;
2014-07-11 18:10:08 +01:00
2014-08-03 15:00:52 +01:00
union {
u32 active_fb ;
2014-07-16 10:24:09 +01:00
2014-08-03 15:00:52 +01:00
// 0: Use parameters ending with "1"
// 1: Use parameters ending with "2"
BitField < 0 , 1 , u32 > second_fb_active ;
} ;
2014-07-16 10:24:09 +01:00
2014-08-03 15:00:52 +01:00
INSERT_PADDING_WORDS ( 0x5 ) ;
2014-07-16 10:27:58 +01:00
2014-08-03 15:00:52 +01:00
// Distance between two pixel rows, in bytes
u32 stride ;
2014-07-16 10:24:09 +01:00
2014-08-03 15:00:52 +01:00
u32 address_right1 ;
u32 address_right2 ;
2014-07-11 18:10:08 +01:00
2014-08-03 15:00:52 +01:00
INSERT_PADDING_WORDS ( 0x30 ) ;
} framebuffer_config [ 2 ] ;
ASSERT_MEMBER_SIZE ( framebuffer_config [ 0 ] , 0x100 ) ;
2014-05-31 23:08:00 +01:00
2014-08-03 15:00:52 +01:00
INSERT_PADDING_WORDS ( 0x169 ) ;
2014-05-31 23:08:00 +01:00
2014-08-03 15:00:52 +01:00
struct {
using Format = Regs : : FramebufferFormat ;
2014-05-31 23:08:00 +01:00
2014-08-03 15:00:52 +01:00
u32 input_address ;
u32 output_address ;
2014-05-31 23:08:00 +01:00
2014-08-03 15:00:52 +01:00
inline u32 GetPhysicalInputAddress ( ) const {
return DecodeAddressRegister ( input_address ) ;
}
2014-05-31 23:08:00 +01:00
2014-08-03 15:00:52 +01:00
inline u32 GetPhysicalOutputAddress ( ) const {
return DecodeAddressRegister ( output_address ) ;
}
2014-05-31 23:08:00 +01:00
2014-08-03 15:00:52 +01:00
union {
u32 output_size ;
2014-05-31 23:08:00 +01:00
2014-08-03 15:00:52 +01:00
BitField < 0 , 16 , u32 > output_width ;
BitField < 16 , 16 , u32 > output_height ;
} ;
2014-05-31 23:08:00 +01:00
2014-08-03 15:00:52 +01:00
union {
u32 input_size ;
2014-05-31 23:08:00 +01:00
2014-08-03 15:00:52 +01:00
BitField < 0 , 16 , u32 > input_width ;
BitField < 16 , 16 , u32 > input_height ;
} ;
2014-07-22 12:21:57 +01:00
2014-08-03 15:00:52 +01:00
union {
u32 flags ;
2014-05-31 23:08:00 +01:00
2014-08-03 15:00:52 +01:00
BitField < 0 , 1 , u32 > flip_data ; // flips input data horizontally (TODO) if true
BitField < 8 , 3 , Format > input_format ;
BitField < 12 , 3 , Format > output_format ;
BitField < 16 , 1 , u32 > output_tiled ; // stores output in a tiled format
} ;
2014-07-22 12:21:57 +01:00
2014-08-03 15:00:52 +01:00
INSERT_PADDING_WORDS ( 0x1 ) ;
2014-07-22 12:21:57 +01:00
2014-08-03 15:00:52 +01:00
// it seems that writing to this field triggers the display transfer
u32 trigger ;
} display_transfer_config ;
ASSERT_MEMBER_SIZE ( display_transfer_config , 0x1c ) ;
2014-07-22 12:21:57 +01:00
2014-08-03 15:00:52 +01:00
INSERT_PADDING_WORDS ( 0x331 ) ;
2014-07-22 12:21:57 +01:00
2014-08-03 15:00:52 +01:00
struct {
// command list size
u32 size ;
INSERT_PADDING_WORDS ( 0x1 ) ;
// command list address
u32 address ;
INSERT_PADDING_WORDS ( 0x1 ) ;
// it seems that writing to this field triggers command list processing
u32 trigger ;
2014-07-16 10:24:09 +01:00
2014-08-03 15:00:52 +01:00
inline u32 GetPhysicalAddress ( ) const {
return DecodeAddressRegister ( address ) ;
}
} command_processor_config ;
ASSERT_MEMBER_SIZE ( command_processor_config , 0x14 ) ;
2014-04-27 17:39:57 +01:00
2014-08-03 15:00:52 +01:00
INSERT_PADDING_WORDS ( 0x9c3 ) ;
# undef INSERT_PADDING_WORDS_HELPER1
# undef INSERT_PADDING_WORDS_HELPER2
# undef INSERT_PADDING_WORDS
2014-09-28 16:20:06 +01:00
static inline size_t NumIds ( ) {
2014-08-03 15:00:52 +01:00
return sizeof ( Regs ) / sizeof ( u32 ) ;
}
u32 & operator [ ] ( int index ) const {
u32 * content = ( u32 * ) this ;
return content [ index ] ;
}
u32 & operator [ ] ( int index ) {
u32 * content = ( u32 * ) this ;
return content [ index ] ;
}
private :
/*
* Most physical addresses which GPU registers refer to are 8 - byte aligned .
* This function should be used to get the address from a raw register value .
*/
static inline u32 DecodeAddressRegister ( u32 register_value ) {
return register_value * 8 ;
}
} ;
static_assert ( std : : is_standard_layout < Regs > : : value , " Structure does not use standard layout " ) ;
// TODO: MSVC does not support using offsetof() on non-static data members even though this
// is technically allowed since C++11. This macro should be enabled once MSVC adds
// support for that.
# ifndef _MSC_VER
# define ASSERT_REG_POSITION(field_name, position) \
static_assert ( offsetof ( Regs , field_name ) = = position * 4 , \
" Field " # field_name " has invalid position " )
ASSERT_REG_POSITION ( memory_fill_config [ 0 ] , 0x00004 ) ;
ASSERT_REG_POSITION ( memory_fill_config [ 1 ] , 0x00008 ) ;
ASSERT_REG_POSITION ( framebuffer_config [ 0 ] , 0x00117 ) ;
ASSERT_REG_POSITION ( framebuffer_config [ 1 ] , 0x00157 ) ;
ASSERT_REG_POSITION ( display_transfer_config , 0x00300 ) ;
ASSERT_REG_POSITION ( command_processor_config , 0x00638 ) ;
# undef ASSERT_REG_POSITION
# endif // !defined(_MSC_VER)
// The total number of registers is chosen arbitrarily, but let's make sure it's not some odd value anyway.
static_assert ( sizeof ( Regs ) = = 0x1000 * sizeof ( u32 ) , " Invalid total size of register set " ) ;
extern Regs g_regs ;
2014-04-27 17:39:57 +01:00
2014-04-05 06:23:51 +01:00
template < typename T >
2014-07-22 12:49:25 +01:00
void Read ( T & var , const u32 addr ) ;
2014-04-05 06:23:51 +01:00
template < typename T >
2014-07-22 12:49:25 +01:00
void Write ( u32 addr , const T data ) ;
2014-04-05 06:23:51 +01:00
/// Update hardware
void Update ( ) ;
/// Initialize hardware
void Init ( ) ;
/// Shutdown hardware
void Shutdown ( ) ;
} // namespace