blob: 472a744f0406b34cd7186c74a1a9aa4a0df296b1 [file] [log] [blame]
Elliott Hughesf274a202024-06-17 15:07:25 +00001/*
2 * Copyright (C) 2024 The Android Open Source Project
3 * All rights reserved.
4 *
5 * Redistribution and use in source and binary forms, with or without
6 * modification, are permitted provided that the following conditions
7 * are met:
8 * * Redistributions of source code must retain the above copyright
9 * notice, this list of conditions and the following disclaimer.
10 * * Redistributions in binary form must reproduce the above copyright
11 * notice, this list of conditions and the following disclaimer in
12 * the documentation and/or other materials provided with the
13 * distribution.
14 *
15 * THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS
16 * "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT
17 * LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS
18 * FOR A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE
19 * COPYRIGHT OWNER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT,
20 * INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING,
21 * BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS
22 * OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED
23 * AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY,
24 * OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT
25 * OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF
26 * SUCH DAMAGE.
27 */
28
29#pragma once
30
31/**
32 * @file sys/io.h
Elliott Hughesf32cb142025-02-06 15:25:15 -050033 * @brief The x86/x86-64 I/O port functions.
Elliott Hughesf274a202024-06-17 15:07:25 +000034 */
35
36#include <sys/cdefs.h>
37
38#include <errno.h>
39#include <sys/syscall.h>
40#include <unistd.h>
41
42__BEGIN_DECLS
43
44/**
Elliott Hughesbbd39aa2024-08-13 20:59:16 +000045 * [iopl(2)](https://man7.org/linux/man-pages/man2/iopl.2.html) changes the I/O
Elliott Hughesf274a202024-06-17 15:07:25 +000046 * privilege level for all x86/x8-64 I/O ports, for the calling thread.
47 *
48 * New callers should use ioperm() instead.
49 *
50 * Returns 0 on success, and returns -1 and sets `errno` on failure.
51 *
52 * Only available for x86/x86-64.
53 */
54#if defined(__NR_iopl)
55__attribute__((__deprecated__("use ioperm() instead"))) static __inline int iopl(int __level) {
56 return syscall(__NR_iopl, __level);
57}
58#endif
59
60/**
Elliott Hughesbbd39aa2024-08-13 20:59:16 +000061 * [ioperm(2)](https://man7.org/linux/man-pages/man2/ioperm.2.html) sets the I/O
Elliott Hughesf274a202024-06-17 15:07:25 +000062 * permissions for the given number of x86/x86-64 I/O ports, starting at the
63 * given port.
64 *
65 * Returns 0 on success, and returns -1 and sets `errno` on failure.
66 *
67 * Only available for x86/x86-64.
68 */
69#if defined(__NR_iopl)
70static __inline int ioperm(unsigned long __from, unsigned long __n, int __enabled) {
71 return syscall(__NR_ioperm, __from, __n, __enabled);
72}
73#endif
74
Elliott Hughesf32cb142025-02-06 15:25:15 -050075/**
76 * [inb(2)](https://man7.org/linux/man-pages/man2/inb.2.html)
77 * reads a byte from the given x86/x86-64 I/O port.
78 *
79 * Only available for x86/x86-64.
80 */
81#if defined(__i386__) || defined(__x86_64__)
82static __inline unsigned char inb(unsigned short __port) {
83 unsigned char __value;
84 __asm__ __volatile__("inb %1, %0" : "=a"(__value) : "dN"(__port));
85 return __value;
86}
87#endif
88
89/**
90 * [inw(2)](https://man7.org/linux/man-pages/man2/inw.2.html)
91 * reads a 16-bit "word" from the given x86/x86-64 I/O port.
92 *
93 * Only available for x86/x86-64.
94 */
95#if defined(__i386__) || defined(__x86_64__)
96static __inline unsigned short inw(unsigned short __port) {
97 unsigned short __value;
98 __asm__ __volatile__("inw %1, %0" : "=a"(__value) : "dN"(__port));
99 return __value;
100}
101#endif
102
103/**
104 * [inl(2)](https://man7.org/linux/man-pages/man2/inl.2.html)
105 * reads a 32-bit "long word" from the given x86/x86-64 I/O port.
106 *
107 * Only available for x86/x86-64.
108 */
109#if defined(__i386__) || defined(__x86_64__)
110static __inline unsigned int inl(unsigned short __port) {
111 unsigned int __value;
112 __asm__ __volatile__("inl %1, %0" : "=a"(__value) : "dN"(__port));
113 return __value;
114}
115#endif
116
117/**
118 * [outb(2)](https://man7.org/linux/man-pages/man2/outb.2.html)
119 * writes the given byte to the given x86/x86-64 I/O port.
120 *
121 * Only available for x86/x86-64.
122 */
123#if defined(__i386__) || defined(__x86_64__)
124static __inline void outb(unsigned char __value, unsigned short __port) {
125 __asm__ __volatile__("outb %0, %1" : : "a"(__value), "dN"(__port));
126}
127#endif
128
129/**
130 * [outw(2)](https://man7.org/linux/man-pages/man2/outw.2.html)
131 * writes the given 16-bit "word" to the given x86/x86-64 I/O port.
132 *
133 * Only available for x86/x86-64.
134 */
135#if defined(__i386__) || defined(__x86_64__)
136static __inline void outw(unsigned short __value, unsigned short __port) {
137 __asm__ __volatile__("outw %0, %1" : : "a"(__value), "dN"(__port));
138}
139#endif
140
141/**
142 * [outl(2)](https://man7.org/linux/man-pages/man2/outl.2.html)
143 * writes the given 32-bit "long word" to the given x86/x86-64 I/O port.
144 *
145 * Only available for x86/x86-64.
146 */
147#if defined(__i386__) || defined(__x86_64__)
148static __inline void outl(unsigned int __value, unsigned short __port) {
149 __asm__ __volatile__("outl %0, %1" : : "a"(__value), "dN"(__port));
150}
151#endif
152
Elliott Hughesf274a202024-06-17 15:07:25 +0000153__END_DECLS