1 #include <linux/errno.h>
2 #include <linux/sched.h>
3 #include <linux/syscalls.h>
10 #include <linux/stat.h>
11 #include <linux/mman.h>
12 #include <linux/file.h>
13 #include <linux/utsname.h>
14 #include <linux/personality.h>
16 #include <asm/uaccess.h>
20 * sys_pipe() is the normal C calling standard for creating
21 * a pipe. It's not the way Unix traditionally does this, though.
23 asmlinkage long sys_pipe(int __user *fildes)
30 if (copy_to_user(fildes, fd, 2*sizeof(int)))
36 asmlinkage long sys_mmap(unsigned long addr, unsigned long len, unsigned long prot, unsigned long flags,
37 unsigned long fd, unsigned long off)
48 flags &= ~(MAP_EXECUTABLE | MAP_DENYWRITE);
49 if (!(flags & MAP_ANONYMOUS)) {
54 down_write(¤t->mm->mmap_sem);
55 error = do_mmap_pgoff(file, addr, len, prot, flags, off >> PAGE_SHIFT);
56 up_write(¤t->mm->mmap_sem);
64 static void find_start_end(unsigned long flags, unsigned long *begin,
67 if (!test_thread_flag(TIF_IA32) && (flags & MAP_32BIT)) {
68 /* This is usually used needed to map code in small
69 model, so it needs to be in the first 31bit. Limit
70 it to that. This means we need to move the
71 unmapped base down for this case. This can give
72 conflicts with the heap, but we assume that glibc
73 malloc knows how to fall back to mmap. Give it 1GB
74 of playground for now. -AK */
78 *begin = TASK_UNMAPPED_BASE;
84 arch_get_unmapped_area(struct file *filp, unsigned long addr,
85 unsigned long len, unsigned long pgoff, unsigned long flags)
87 struct mm_struct *mm = current->mm;
88 struct vm_area_struct *vma;
89 unsigned long start_addr;
90 unsigned long begin, end;
92 if (flags & MAP_FIXED)
95 find_start_end(flags, &begin, &end);
101 addr = PAGE_ALIGN(addr);
102 vma = find_vma(mm, addr);
103 if (end - len >= addr &&
104 (!vma || addr + len <= vma->vm_start))
107 if (((flags & MAP_32BIT) || test_thread_flag(TIF_IA32))
108 && len <= mm->cached_hole_size) {
109 mm->cached_hole_size = 0;
110 mm->free_area_cache = begin;
112 addr = mm->free_area_cache;
118 for (vma = find_vma(mm, addr); ; vma = vma->vm_next) {
119 /* At this point: (!vma || addr < vma->vm_end). */
120 if (end - len < addr) {
122 * Start a new search - just in case we missed
125 if (start_addr != begin) {
126 start_addr = addr = begin;
127 mm->cached_hole_size = 0;
132 if (!vma || addr + len <= vma->vm_start) {
134 * Remember the place where we stopped the search:
136 mm->free_area_cache = addr + len;
139 if (addr + mm->cached_hole_size < vma->vm_start)
140 mm->cached_hole_size = vma->vm_start - addr;
146 asmlinkage long sys_uname(struct new_utsname __user * name)
150 err = copy_to_user(name, utsname(), sizeof (*name));
152 if (personality(current->personality) == PER_LINUX32)
153 err |= copy_to_user(&name->machine, "i686", 5);
154 return err ? -EFAULT : 0;