2 * Copyright (c) 2006 The DragonFly Project. All rights reserved.
4 * This code is derived from software contributed to The DragonFly Project
5 * by Matthew Dillon <dillon@backplane.com>
7 * Redistribution and use in source and binary forms, with or without
8 * modification, are permitted provided that the following conditions
11 * 1. Redistributions of source code must retain the above copyright
12 * notice, this list of conditions and the following disclaimer.
13 * 2. Redistributions in binary form must reproduce the above copyright
14 * notice, this list of conditions and the following disclaimer in
15 * the documentation and/or other materials provided with the
17 * 3. Neither the name of The DragonFly Project nor the names of its
18 * contributors may be used to endorse or promote products derived
19 * from this software without specific, prior written permission.
21 * THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS
22 * ``AS IS'' AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT
23 * LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS
24 * FOR A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE
25 * COPYRIGHT HOLDERS OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT,
26 * INCIDENTAL, SPECIAL, EXEMPLARY OR CONSEQUENTIAL DAMAGES (INCLUDING,
27 * BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES;
28 * LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED
29 * AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY,
30 * OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT
31 * OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF
34 * $DragonFly: src/sys/platform/vkernel/platform/copyio.c,v 1.9 2008/05/09 07:24:47 dillon Exp $
37 #include <sys/types.h>
38 #include <sys/systm.h>
39 #include <sys/sfbuf.h>
40 #include <vm/vm_page.h>
41 #include <vm/vm_extern.h>
47 #include <sys/mplock2.h>
50 * A bcopy that works dring low level boot, before FP is working
53 ovbcopy(const void *src
, void *dst
, size_t len
)
59 bcopyi(const void *src
, void *dst
, size_t len
)
65 copystr(const void *kfaddr
, void *kdaddr
, size_t len
, size_t *lencopied
)
69 for (i
= 0; i
< len
; ++i
) {
70 if ((((char *)kdaddr
)[i
] = ((const char *)kfaddr
)[i
]) == 0) {
76 return (ENAMETOOLONG
);
80 * Copies a NUL-terminated string from user space to kernel space.
81 * The number of bytes copied, including the terminator, is returned in
84 * Returns 0 on success, EFAULT or ENAMETOOLONG on failure.
87 copyinstr(const void *udaddr
, void *kaddr
, size_t len
, size_t *res
)
91 const char *uptr
= udaddr
;
97 n
= PAGE_SIZE
- ((vm_offset_t
)uptr
& PAGE_MASK
);
102 if ((error
= copyin(uptr
, kptr
, n
)) != 0)
116 return(ENAMETOOLONG
);
120 * Copy a binary buffer from user space to kernel space.
122 * NOTE: on a real system copyin/copyout are MP safe, but the current
123 * implementation on a vkernel is not so we get the mp lock.
125 * Returns 0 on success, EFAULT on failure.
128 copyin(const void *udaddr
, void *kaddr
, size_t len
)
130 struct vmspace
*vm
= curproc
->p_vmspace
;
139 m
= vm_fault_page(&vm
->vm_map
, trunc_page((vm_offset_t
)udaddr
),
141 VM_FAULT_NORMAL
, &error
);
144 n
= PAGE_SIZE
- ((vm_offset_t
)udaddr
& PAGE_MASK
);
147 sf
= sf_buf_alloc(m
, SFB_CPUPRIVATE
);
148 bcopy((char *)sf_buf_kva(sf
)+((vm_offset_t
)udaddr
& PAGE_MASK
),
151 udaddr
= (const char *)udaddr
+ n
;
152 kaddr
= (char *)kaddr
+ n
;
161 * Copy a binary buffer from kernel space to user space.
163 * Returns 0 on success, EFAULT on failure.
166 copyout(const void *kaddr
, void *udaddr
, size_t len
)
168 struct vmspace
*vm
= curproc
->p_vmspace
;
177 m
= vm_fault_page(&vm
->vm_map
, trunc_page((vm_offset_t
)udaddr
),
178 VM_PROT_READ
|VM_PROT_WRITE
,
179 VM_FAULT_NORMAL
, &error
);
182 n
= PAGE_SIZE
- ((vm_offset_t
)udaddr
& PAGE_MASK
);
185 sf
= sf_buf_alloc(m
, SFB_CPUPRIVATE
);
186 bcopy(kaddr
, (char *)sf_buf_kva(sf
) +
187 ((vm_offset_t
)udaddr
& PAGE_MASK
), n
);
189 udaddr
= (char *)udaddr
+ n
;
190 kaddr
= (const char *)kaddr
+ n
;
200 * Fetch the byte at the specified user address. Returns -1 on failure.
203 fubyte(const void *base
)
208 if ((error
= copyin(base
, &c
, 1)) == 0)
214 * Store a byte at the specified user address. Returns -1 on failure.
217 subyte (void *base
, int byte
)
219 unsigned char c
= byte
;
222 if ((error
= copyout(&c
, base
, 1)) == 0)
228 * Fetch a word (integer, 32 bits) from user space
231 fuword(const void *base
)
236 if ((error
= copyin(base
, &v
, sizeof(v
))) == 0)
242 * Store a word (integer, 32 bits) to user space
245 suword(void *base
, long word
)
249 if ((error
= copyout(&word
, base
, sizeof(word
))) == 0)
255 * Fetch an short word (16 bits) from user space
260 unsigned short sword
;
263 if ((error
= copyin(base
, &sword
, sizeof(sword
))) == 0)
269 * Store a short word (16 bits) to user space
272 susword (void *base
, int word
)
274 unsigned short sword
= word
;
277 if ((error
= copyout(&sword
, base
, sizeof(sword
))) == 0)