2 * The PCI Library -- Portable interface to pread() and pwrite()
4 * Copyright (c) 1997--2003 Martin Mares <mj@ucw.cz>
6 * Can be freely distributed and used under the terms of the GNU GPL.
10 * We'd like to use pread/pwrite for configuration space accesses, but
11 * unfortunately it isn't simple at all since all libc's until glibc 2.1
15 #if defined(__GLIBC__) && __GLIBC__ == 2 && __GLIBC_MINOR__ > 0
16 /* glibc 2.1 or newer -> pread/pwrite supported automatically */
18 #elif defined(i386) && defined(__GLIBC__)
19 /* glibc 2.0 on i386 -> call syscalls directly */
20 #include <asm/unistd.h>
21 #include <syscall-list.h>
25 static int pread(unsigned int fd, void *buf, size_t size, loff_t where)
26 { return syscall(SYS_pread, fd, buf, size, where); }
28 #define SYS_pwrite 181
30 static int pwrite(unsigned int fd, void *buf, size_t size, loff_t where)
31 { return syscall(SYS_pwrite, fd, buf, size, where); }
34 /* old libc on i386 -> call syscalls directly the old way */
35 #include <asm/unistd.h>
36 static _syscall5(int, pread, unsigned int, fd, void *, buf, size_t, size, u32, where_lo, u32, where_hi);
37 static _syscall5(int, pwrite, unsigned int, fd, void *, buf, size_t, size, u32, where_lo, u32, where_hi);
38 static int do_read(struct pci_dev *d UNUSED, int fd, void *buf, size_t size, int where) { return pread(fd, buf, size, where, 0); }
39 static int do_write(struct pci_dev *d UNUSED, int fd, void *buf, size_t size, int where) { return pwrite(fd, buf, size, where, 0); }
40 #define PCI_HAVE_DO_READ
43 /* In all other cases we use lseek/read/write instead to be safe */
44 #define make_rw_glue(op) \
45 static int do_##op(struct pci_dev *d, int fd, void *buf, size_t size, int where) \
47 struct pci_access *a = d->access; \
49 if (a->fd_pos != where && lseek(fd, where, SEEK_SET) < 0) \
51 r = op(fd, buf, size); \
55 a->fd_pos = where + r; \
60 #define PCI_HAVE_DO_READ
63 #ifndef PCI_HAVE_DO_READ
64 #define do_read(d,f,b,l,p) pread(f,b,l,p)
65 #define do_write(d,f,b,l,p) pwrite(f,b,l,p)