1 /* SPDX-License-Identifier: GPL-2.0 */
3 * arch/arm/mach-s3c2410/include/mach/io.h
4 * from arch/arm/mach-rpc/include/mach/io.h
6 * Copyright (C) 1997 Russell King
7 * (C) 2003 Simtec Electronics
10 #ifndef __ASM_ARM_ARCH_IO_H
11 #define __ASM_ARM_ARCH_IO_H
13 #include <mach/hardware.h>
15 #define IO_SPACE_LIMIT 0xffffffff
18 * We use two different types of addressing - PC style addresses, and ARM
19 * addresses. PC style accesses the PC hardware with the normal PC IO
20 * addresses, eg 0x3f8 for serial#1. ARM addresses are above A28
21 * and are translated to the start of IO. Note that all addresses are
25 #define __PORT_PCIO(x) ((x) < (1<<28))
27 #define PCIO_BASE (S3C24XX_VA_ISA_WORD)
28 #define PCIO_BASE_b (S3C24XX_VA_ISA_BYTE)
29 #define PCIO_BASE_w (S3C24XX_VA_ISA_WORD)
30 #define PCIO_BASE_l (S3C24XX_VA_ISA_WORD)
32 * Dynamic IO functions - let the compiler
33 * optimize the expressions
36 #define DECLARE_DYN_OUT(sz,fnsuffix,instr) \
37 static inline void __out##fnsuffix (unsigned int val, unsigned int port) \
40 __asm__ __volatile__( \
41 "cmp %2, #(1<<28)\n\t" \
43 "addcc %0, %0, %3\n\t" \
44 "str" instr " %1, [%0, #0 ] @ out" #fnsuffix \
46 : "r" (val), "r" (port), "Ir" (PCIO_BASE_##fnsuffix) \
51 #define DECLARE_DYN_IN(sz,fnsuffix,instr) \
52 static inline unsigned sz __in##fnsuffix (unsigned int port) \
54 unsigned long temp, value; \
55 __asm__ __volatile__( \
56 "cmp %2, #(1<<28)\n\t" \
58 "addcc %0, %0, %3\n\t" \
59 "ldr" instr " %1, [%0, #0 ] @ in" #fnsuffix \
60 : "=&r" (temp), "=r" (value) \
61 : "r" (port), "Ir" (PCIO_BASE_##fnsuffix) \
63 return (unsigned sz)value; \
66 static inline void __iomem
*__ioaddr (unsigned long port
)
68 return __PORT_PCIO(port
) ? (PCIO_BASE
+ port
) : (void __iomem
*)port
;
71 #define DECLARE_IO(sz,fnsuffix,instr) \
72 DECLARE_DYN_IN(sz,fnsuffix,instr) \
73 DECLARE_DYN_OUT(sz,fnsuffix,instr)
75 DECLARE_IO(char,b
,"b")
76 DECLARE_IO(short,w
,"h")
83 * Constant address IO functions
85 * These have to be macros for the 'J' constraint to work -
86 * +/-4096 immediate operand.
88 #define __outbc(value,port) \
90 if (__PORT_PCIO((port))) \
91 __asm__ __volatile__( \
92 "strb %0, [%1, %2] @ outbc" \
93 : : "r" (value), "r" (PCIO_BASE), "Jr" ((port))); \
95 __asm__ __volatile__( \
96 "strb %0, [%1, #0] @ outbc" \
97 : : "r" (value), "r" ((port))); \
100 #define __inbc(port) \
102 unsigned char result; \
103 if (__PORT_PCIO((port))) \
104 __asm__ __volatile__( \
105 "ldrb %0, [%1, %2] @ inbc" \
106 : "=r" (result) : "r" (PCIO_BASE), "Jr" ((port))); \
108 __asm__ __volatile__( \
109 "ldrb %0, [%1, #0] @ inbc" \
110 : "=r" (result) : "r" ((port))); \
114 #define __outwc(value,port) \
116 unsigned long v = value; \
117 if (__PORT_PCIO((port))) { \
118 if ((port) < 256 && (port) > -256) \
119 __asm__ __volatile__( \
120 "strh %0, [%1, %2] @ outwc" \
121 : : "r" (v), "r" (PCIO_BASE), "Jr" ((port))); \
122 else if ((port) > 0) \
123 __asm__ __volatile__( \
124 "strh %0, [%1, %2] @ outwc" \
126 "r" (PCIO_BASE + ((port) & ~0xff)), \
127 "Jr" (((port) & 0xff))); \
129 __asm__ __volatile__( \
130 "strh %0, [%1, #0] @ outwc" \
132 "r" (PCIO_BASE + (port))); \
134 __asm__ __volatile__( \
135 "strh %0, [%1, #0] @ outwc" \
136 : : "r" (v), "r" ((port))); \
139 #define __inwc(port) \
141 unsigned short result; \
142 if (__PORT_PCIO((port))) { \
143 if ((port) < 256 && (port) > -256 ) \
144 __asm__ __volatile__( \
145 "ldrh %0, [%1, %2] @ inwc" \
149 else if ((port) > 0) \
150 __asm__ __volatile__( \
151 "ldrh %0, [%1, %2] @ inwc" \
153 : "r" (PCIO_BASE + ((port) & ~0xff)), \
154 "Jr" (((port) & 0xff))); \
156 __asm__ __volatile__( \
157 "ldrh %0, [%1, #0] @ inwc" \
159 : "r" (PCIO_BASE + ((port)))); \
161 __asm__ __volatile__( \
162 "ldrh %0, [%1, #0] @ inwc" \
163 : "=r" (result) : "r" ((port))); \
167 #define __outlc(value,port) \
169 unsigned long v = value; \
170 if (__PORT_PCIO((port))) \
171 __asm__ __volatile__( \
172 "str %0, [%1, %2] @ outlc" \
173 : : "r" (v), "r" (PCIO_BASE), "Jr" ((port))); \
175 __asm__ __volatile__( \
176 "str %0, [%1, #0] @ outlc" \
177 : : "r" (v), "r" ((port))); \
180 #define __inlc(port) \
182 unsigned long result; \
183 if (__PORT_PCIO((port))) \
184 __asm__ __volatile__( \
185 "ldr %0, [%1, %2] @ inlc" \
186 : "=r" (result) : "r" (PCIO_BASE), "Jr" ((port))); \
188 __asm__ __volatile__( \
189 "ldr %0, [%1, #0] @ inlc" \
190 : "=r" (result) : "r" ((port))); \
194 #define __ioaddrc(port) ((__PORT_PCIO(port) ? PCIO_BASE + (port) : (void __iomem *)0 + (port)))
196 #define inb(p) (__builtin_constant_p((p)) ? __inbc(p) : __inb(p))
197 #define inw(p) (__builtin_constant_p((p)) ? __inwc(p) : __inw(p))
198 #define inl(p) (__builtin_constant_p((p)) ? __inlc(p) : __inl(p))
199 #define outb(v,p) (__builtin_constant_p((p)) ? __outbc(v,p) : __outb(v,p))
200 #define outw(v,p) (__builtin_constant_p((p)) ? __outwc(v,p) : __outw(v,p))
201 #define outl(v,p) (__builtin_constant_p((p)) ? __outlc(v,p) : __outl(v,p))
202 #define __ioaddr(p) (__builtin_constant_p((p)) ? __ioaddr(p) : __ioaddrc(p))
204 #define insb(p,d,l) __raw_readsb(__ioaddr(p),d,l)
205 #define insw(p,d,l) __raw_readsw(__ioaddr(p),d,l)
206 #define insl(p,d,l) __raw_readsl(__ioaddr(p),d,l)
208 #define outsb(p,d,l) __raw_writesb(__ioaddr(p),d,l)
209 #define outsw(p,d,l) __raw_writesw(__ioaddr(p),d,l)
210 #define outsl(p,d,l) __raw_writesl(__ioaddr(p),d,l)