Blame view

arch/s390/kernel/sys_s390.c 6.3 KB
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
  /*
   *  arch/s390/kernel/sys_s390.c
   *
   *  S390 version
   *    Copyright (C) 1999,2000 IBM Deutschland Entwicklung GmbH, IBM Corporation
   *    Author(s): Martin Schwidefsky (schwidefsky@de.ibm.com),
   *               Thomas Spatzier (tspat@de.ibm.com)
   *
   *  Derived from "arch/i386/kernel/sys_i386.c"
   *
   *  This file contains various random system calls that
   *  have a non-standard calling sequence on the Linux/s390
   *  platform.
   */
  
  #include <linux/errno.h>
  #include <linux/sched.h>
  #include <linux/mm.h>
  #include <linux/smp.h>
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
20
21
22
23
24
25
26
27
  #include <linux/sem.h>
  #include <linux/msg.h>
  #include <linux/shm.h>
  #include <linux/stat.h>
  #include <linux/syscalls.h>
  #include <linux/mman.h>
  #include <linux/file.h>
  #include <linux/utsname.h>
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
28
  #include <linux/personality.h>
fe74290d5   Arnd Bergmann   [PATCH] provide k...
29
  #include <linux/unistd.h>
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
  
  #include <asm/uaccess.h>
  #include <asm/ipc.h>
  
  /*
   * sys_pipe() is the normal C calling standard for creating
   * a pipe. It's not the way Unix traditionally does this, though.
   */
  asmlinkage long sys_pipe(unsigned long __user *fildes)
  {
  	int fd[2];
  	int error;
  
  	error = do_pipe(fd);
  	if (!error) {
  		if (copy_to_user(fildes, fd, 2*sizeof(int)))
  			error = -EFAULT;
  	}
  	return error;
  }
  
  /* common code for old and new mmaps */
  static inline long do_mmap2(
  	unsigned long addr, unsigned long len,
  	unsigned long prot, unsigned long flags,
  	unsigned long fd, unsigned long pgoff)
  {
  	long error = -EBADF;
  	struct file * file = NULL;
  
  	flags &= ~(MAP_EXECUTABLE | MAP_DENYWRITE);
  	if (!(flags & MAP_ANONYMOUS)) {
  		file = fget(fd);
  		if (!file)
  			goto out;
  	}
  
  	down_write(&current->mm->mmap_sem);
  	error = do_mmap_pgoff(file, addr, len, prot, flags, pgoff);
  	up_write(&current->mm->mmap_sem);
  
  	if (file)
  		fput(file);
  out:
  	return error;
  }
  
  /*
   * Perform the select(nd, in, out, ex, tv) and mmap() system
   * calls. Linux for S/390 isn't able to handle more than 5
   * system call parameters, so these system calls used a memory
   * block for parameter passing..
   */
  
  struct mmap_arg_struct {
  	unsigned long addr;
  	unsigned long len;
  	unsigned long prot;
  	unsigned long flags;
  	unsigned long fd;
  	unsigned long offset;
  };
  
  asmlinkage long sys_mmap2(struct mmap_arg_struct __user  *arg)
  {
  	struct mmap_arg_struct a;
  	int error = -EFAULT;
  
  	if (copy_from_user(&a, arg, sizeof(a)))
  		goto out;
  	error = do_mmap2(a.addr, a.len, a.prot, a.flags, a.fd, a.offset);
  out:
  	return error;
  }
  
  asmlinkage long old_mmap(struct mmap_arg_struct __user *arg)
  {
  	struct mmap_arg_struct a;
  	long error = -EFAULT;
  
  	if (copy_from_user(&a, arg, sizeof(a)))
  		goto out;
  
  	error = -EINVAL;
  	if (a.offset & ~PAGE_MASK)
  		goto out;
  
  	error = do_mmap2(a.addr, a.len, a.prot, a.flags, a.fd, a.offset >> PAGE_SHIFT);
  out:
  	return error;
  }
347a8dc3b   Martin Schwidefsky   [PATCH] s390: cle...
121
  #ifndef CONFIG_64BIT
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
122
123
  struct sel_arg_struct {
  	unsigned long n;
793af2440   Al Viro   [PATCH] s390 misc...
124
125
  	fd_set __user *inp, *outp, *exp;
  	struct timeval __user *tvp;
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
126
127
128
129
130
131
132
133
134
135
136
137
  };
  
  asmlinkage long old_select(struct sel_arg_struct __user *arg)
  {
  	struct sel_arg_struct a;
  
  	if (copy_from_user(&a, arg, sizeof(a)))
  		return -EFAULT;
  	/* sys_select() does the appropriate kernel locking */
  	return sys_select(a.n, a.inp, a.outp, a.exp, a.tvp);
  
  }
347a8dc3b   Martin Schwidefsky   [PATCH] s390: cle...
138
  #endif /* CONFIG_64BIT */
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
  
  /*
   * sys_ipc() is the de-multiplexer for the SysV IPC calls..
   *
   * This is really horribly ugly.
   */
  asmlinkage long sys_ipc(uint call, int first, unsigned long second,
  				  unsigned long third, void __user *ptr)
  {
          struct ipc_kludge tmp;
  	int ret;
  
          switch (call) {
          case SEMOP:
  		return sys_semtimedop(first, (struct sembuf __user *)ptr,
  				       (unsigned)second, NULL);
  	case SEMTIMEDOP:
  		return sys_semtimedop(first, (struct sembuf __user *)ptr,
  				       (unsigned)second,
  				       (const struct timespec __user *) third);
          case SEMGET:
                  return sys_semget(first, (int)second, third);
          case SEMCTL: {
                  union semun fourth;
                  if (!ptr)
                          return -EINVAL;
                  if (get_user(fourth.__pad, (void __user * __user *) ptr))
                          return -EFAULT;
                  return sys_semctl(first, (int)second, third, fourth);
          }
          case MSGSND:
  		return sys_msgsnd (first, (struct msgbuf __user *) ptr,
                                     (size_t)second, third);
  		break;
          case MSGRCV:
                  if (!ptr)
                          return -EINVAL;
                  if (copy_from_user (&tmp, (struct ipc_kludge __user *) ptr,
                                      sizeof (struct ipc_kludge)))
                          return -EFAULT;
                  return sys_msgrcv (first, tmp.msgp,
                                     (size_t)second, tmp.msgtyp, third);
          case MSGGET:
                  return sys_msgget((key_t)first, (int)second);
          case MSGCTL:
                  return sys_msgctl(first, (int)second,
  				   (struct msqid_ds __user *)ptr);
  
  	case SHMAT: {
  		ulong raddr;
  		ret = do_shmat(first, (char __user *)ptr,
  				(int)second, &raddr);
  		if (ret)
  			return ret;
  		return put_user (raddr, (ulong __user *) third);
  		break;
          }
  	case SHMDT:
  		return sys_shmdt ((char __user *)ptr);
  	case SHMGET:
  		return sys_shmget(first, (size_t)second, third);
  	case SHMCTL:
  		return sys_shmctl(first, (int)second,
                                     (struct shmid_ds __user *) ptr);
  	default:
  		return -ENOSYS;
  
  	}
  
  	return -EINVAL;
  }
347a8dc3b   Martin Schwidefsky   [PATCH] s390: cle...
210
  #ifdef CONFIG_64BIT
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
  asmlinkage long s390x_newuname(struct new_utsname __user *name)
  {
  	int ret = sys_newuname(name);
  
  	if (current->personality == PER_LINUX32 && !ret) {
  		ret = copy_to_user(name->machine, "s390\0\0\0\0", 8);
  		if (ret) ret = -EFAULT;
  	}
  	return ret;
  }
  
  asmlinkage long s390x_personality(unsigned long personality)
  {
  	int ret;
  
  	if (current->personality == PER_LINUX32 && personality == PER_LINUX)
  		personality = PER_LINUX32;
  	ret = sys_personality(personality);
  	if (ret == PER_LINUX32)
  		ret = PER_LINUX;
  
  	return ret;
  }
347a8dc3b   Martin Schwidefsky   [PATCH] s390: cle...
234
  #endif /* CONFIG_64BIT */
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
235
236
237
238
  
  /*
   * Wrapper function for sys_fadvise64/fadvise64_64
   */
347a8dc3b   Martin Schwidefsky   [PATCH] s390: cle...
239
  #ifndef CONFIG_64BIT
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
  
  asmlinkage long
  s390_fadvise64(int fd, u32 offset_high, u32 offset_low, size_t len, int advice)
  {
  	return sys_fadvise64(fd, (u64) offset_high << 32 | offset_low,
  			len, advice);
  }
  
  #endif
  
  struct fadvise64_64_args {
  	int fd;
  	long long offset;
  	long long len;
  	int advice;
  };
  
  asmlinkage long
  s390_fadvise64_64(struct fadvise64_64_args __user *args)
  {
  	struct fadvise64_64_args a;
  
  	if ( copy_from_user(&a, args, sizeof(a)) )
  		return -EFAULT;
  	return sys_fadvise64_64(a.fd, a.offset, a.len, a.advice);
  }