|
|
/*++
Copyright (c) Microsoft Corporation. All rights reserved.
Module Name:
ioaccess.h
Abstract:
Definitions of function prototypes for accessing I/O ports and memory on I/O adapters from display drivers.
Cloned from parts of nti386.h.
Author:
--*/
//
// Note: IA64 is for 64 bits Merced. Under Merced compiler option, we don't have
// _X86_, instead, we use _IA64_. Same thing, _AXP64_ is for 64 bits compiler
// option for ALPHA
//
#if defined(_MIPS_) || defined(_X86_) || defined(_AMD64_)
//
// Memory barriers on X86 and MIPS are not required since the Io
// Operations are always garanteed to be executed in order
//
#define MEMORY_BARRIER() 0
#elif defined(_IA64_)
//
// Itanium requires memory barriers
//
void __mf();
#define MEMORY_BARRIER() __mf()
#elif defined(_PPC_)
//
// A memory barrier function is provided by the PowerPC Enforce
// In-order Execution of I/O instruction (eieio).
//
#if defined(_M_PPC) && defined(_MSC_VER) && (_MSC_VER>=1000)
void __emit( unsigned const __int32 ); #define __builtin_eieio() __emit( 0x7C0006AC )
#else
void __builtin_eieio(void); #endif
#define MEMORY_BARRIER() __builtin_eieio()
#elif defined(_ALPHA_) || (_AXP64_)
//
// ALPHA requires memory barriers
//
#define MEMORY_BARRIER() __MB()
#endif
#ifndef NO_PORT_MACROS
//
// I/O space read and write macros.
//
// The READ/WRITE_REGISTER_* calls manipulate MEMORY registers.
// (Use x86 move instructions, with LOCK prefix to force correct behavior
// w.r.t. caches and write buffers.)
//
// The READ/WRITE_PORT_* calls manipulate I/O ports.
// (Use x86 in/out instructions.)
//
//
// inp(),inpw(), inpd(), outp(), outpw(), outpd() are X86 specific intrinsic
// inline functions. So for IA64, we have to put READ_PORT_USHORT() etc. back
// to it's supposed to be, defined in sdk\inc\wdm.h
//
#if defined(_IA64_)
#define READ_REGISTER_UCHAR(Register) (*(volatile UCHAR *)(Register))
#define READ_REGISTER_USHORT(Register) (*(volatile USHORT *)(Register))
#define READ_REGISTER_ULONG(Register) (*(volatile ULONG *)(Register))
#define WRITE_REGISTER_UCHAR(Register, Value) (*(volatile UCHAR *)(Register) = (Value))
#define WRITE_REGISTER_USHORT(Register, Value) (*(volatile USHORT *)(Register) = (Value))
#define WRITE_REGISTER_ULONG(Register, Value) (*(volatile ULONG *)(Register) = (Value))
__declspec(dllimport) UCHAR READ_PORT_UCHAR( PVOID Port );
__declspec(dllimport) USHORT READ_PORT_USHORT( PVOID Port );
__declspec(dllimport) ULONG READ_PORT_ULONG( PVOID Port );
//
// All these function prototypes take a ULONG as a parameter so that
// we don't force an extra typecast in the code (which will cause
// the X86 to generate bad code).
//
__declspec(dllimport) VOID WRITE_PORT_UCHAR( PVOID Port, ULONG Value );
__declspec(dllimport) VOID WRITE_PORT_USHORT( PVOID Port, ULONG Value );
__declspec(dllimport) VOID WRITE_PORT_ULONG( PVOID Port, ULONG Value );
#elif defined(_X86_)
#define READ_REGISTER_UCHAR(Register) (*(volatile UCHAR *)(Register))
#define READ_REGISTER_USHORT(Register) (*(volatile USHORT *)(Register))
#define READ_REGISTER_ULONG(Register) (*(volatile ULONG *)(Register))
#define WRITE_REGISTER_UCHAR(Register, Value) (*(volatile UCHAR *)(Register) = (Value))
#define WRITE_REGISTER_USHORT(Register, Value) (*(volatile USHORT *)(Register) = (Value))
#define WRITE_REGISTER_ULONG(Register, Value) (*(volatile ULONG *)(Register) = (Value))
#define READ_PORT_UCHAR(Port) (UCHAR)(inp (Port))
#define READ_PORT_USHORT(Port) (USHORT)(inpw (Port))
#define READ_PORT_ULONG(Port) (ULONG)(inpd (Port))
#define WRITE_PORT_UCHAR(Port, Value) outp ((Port), (Value))
#define WRITE_PORT_USHORT(Port, Value) outpw ((Port), (Value))
#define WRITE_PORT_ULONG(Port, Value) outpd ((Port), (Value))
#elif defined(_PPC_) || defined(_MIPS_)
#define READ_REGISTER_UCHAR(x) (*(volatile UCHAR * const)(x))
#define READ_REGISTER_USHORT(x) (*(volatile USHORT * const)(x))
#define READ_REGISTER_ULONG(x) (*(volatile ULONG * const)(x))
#define WRITE_REGISTER_UCHAR(x, y) (*(volatile UCHAR * const)(x) = (y))
#define WRITE_REGISTER_USHORT(x, y) (*(volatile USHORT * const)(x) = (y))
#define WRITE_REGISTER_ULONG(x, y) (*(volatile ULONG * const)(x) = (y))
#define READ_PORT_UCHAR(x) READ_REGISTER_UCHAR(x)
#define READ_PORT_USHORT(x) READ_REGISTER_USHORT(x)
#define READ_PORT_ULONG(x) READ_REGISTER_ULONG(x)
//
// All these macros take a ULONG as a parameter so that we don't
// force an extra typecast in the code (which will cause the X86 to
// generate bad code).
//
#define WRITE_PORT_UCHAR(x, y) WRITE_REGISTER_UCHAR(x, (UCHAR) (y))
#define WRITE_PORT_USHORT(x, y) WRITE_REGISTER_USHORT(x, (USHORT) (y))
#define WRITE_PORT_ULONG(x, y) WRITE_REGISTER_ULONG(x, (ULONG) (y))
#elif defined(_ALPHA_) || (_AXP64_)
//
// READ/WRITE_PORT/REGISTER_UCHAR_USHORT_ULONG are all functions that
// go to the HAL on ALPHA
//
// So we only put the prototypes here
//
__declspec(dllimport) UCHAR READ_REGISTER_UCHAR( PVOID Register );
__declspec(dllimport) USHORT READ_REGISTER_USHORT( PVOID Register );
__declspec(dllimport) ULONG READ_REGISTER_ULONG( PVOID Register );
__declspec(dllimport) VOID WRITE_REGISTER_UCHAR( PVOID Register, UCHAR Value );
__declspec(dllimport) VOID WRITE_REGISTER_USHORT( PVOID Register, USHORT Value );
__declspec(dllimport) VOID WRITE_REGISTER_ULONG( PVOID Register, ULONG Value );
__declspec(dllimport) UCHAR READ_PORT_UCHAR( PVOID Port );
__declspec(dllimport) USHORT READ_PORT_USHORT( PVOID Port );
__declspec(dllimport) ULONG READ_PORT_ULONG( PVOID Port );
//
// All these function prototypes take a ULONG as a parameter so that
// we don't force an extra typecast in the code (which will cause
// the X86 to generate bad code).
//
__declspec(dllimport) VOID WRITE_PORT_UCHAR( PVOID Port, ULONG Value );
__declspec(dllimport) VOID WRITE_PORT_USHORT( PVOID Port, ULONG Value );
__declspec(dllimport) VOID WRITE_PORT_ULONG( PVOID Port, ULONG Value );
#elif defined(_AMD64_)
UCHAR __inbyte ( IN USHORT Port );
USHORT __inword ( IN USHORT Port );
ULONG __indword ( IN USHORT Port );
VOID __outbyte ( IN USHORT Port, IN UCHAR Data );
VOID __outword ( IN USHORT Port, IN USHORT Data );
VOID __outdword ( IN USHORT Port, IN ULONG Data );
#pragma intrinsic(__inbyte)
#pragma intrinsic(__inword)
#pragma intrinsic(__indword)
#pragma intrinsic(__outbyte)
#pragma intrinsic(__outword)
#pragma intrinsic(__outdword)
LONG _InterlockedOr ( IN OUT LONG volatile *Target, IN LONG Set );
#pragma intrinsic(_InterlockedOr)
__inline UCHAR READ_REGISTER_UCHAR ( PVOID Register ) { return *(UCHAR volatile *)Register; }
__inline USHORT READ_REGISTER_USHORT ( PVOID Register ) { return *(USHORT volatile *)Register; }
__inline ULONG READ_REGISTER_ULONG ( PVOID Register ) { return *(ULONG volatile *)Register; }
__inline VOID WRITE_REGISTER_UCHAR ( PVOID Register, UCHAR Value ) { LONG Synch;
*(UCHAR volatile *)Register = Value; _InterlockedOr(&Synch, 1); return; }
__inline VOID WRITE_REGISTER_USHORT ( PVOID Register, USHORT Value ) { LONG Synch;
*(USHORT volatile *)Register = Value; _InterlockedOr(&Synch, 1); return; }
__inline VOID WRITE_REGISTER_ULONG ( PVOID Register, ULONG Value ) { LONG Synch;
*(ULONG volatile *)Register = Value; _InterlockedOr(&Synch, 1); return; }
__inline UCHAR READ_PORT_UCHAR ( PVOID Port )
{ return __inbyte((USHORT)((ULONG64)Port)); }
__inline USHORT READ_PORT_USHORT ( PVOID Port )
{ return __inword((USHORT)((ULONG64)Port)); }
__inline ULONG READ_PORT_ULONG ( PVOID Port )
{ return __indword((USHORT)((ULONG64)Port)); }
__inline VOID WRITE_PORT_UCHAR ( PVOID Port, UCHAR Value )
{ __outbyte((USHORT)((ULONG64)Port), Value); return; }
__inline VOID WRITE_PORT_USHORT ( PVOID Port, USHORT Value )
{ __outword((USHORT)((ULONG64)Port), Value); return; }
__inline VOID WRITE_PORT_ULONG ( PVOID Port, ULONG Value )
{ __outdword((USHORT)((ULONG64)Port), Value); return; }
#endif // NO_PORT_MACROS
#endif
|