Fedora kernel-2.6.17-1.2142_FC4 patched with stable patch-2.6.17.4-vs2.0.2-rc26.diff
[linux-2.6.git] / arch / sparc64 / solaris / misc.c
index cea38c0..036d5f2 100644 (file)
@@ -17,6 +17,7 @@
 #include <linux/timex.h>
 #include <linux/major.h>
 #include <linux/compat.h>
+#include <linux/vs_cvirt.h>
 
 #include <asm/uaccess.h>
 #include <asm/string.h>
@@ -90,7 +91,7 @@ static u32 do_solaris_mmap(u32 addr, u32 len, u32 prot, u32 flags, u32 fd, u64 o
        len = PAGE_ALIGN(len);
        if(!(flags & MAP_FIXED))
                addr = 0;
-       else if (len > 0xf0000000UL || addr > 0xf0000000UL - len)
+       else if (len > STACK_TOP32 || addr > STACK_TOP32 - len)
                goto out_putf;
        ret_type = flags & _MAP_NEW;
        flags &= ~_MAP_NEW;
@@ -102,7 +103,7 @@ static u32 do_solaris_mmap(u32 addr, u32 len, u32 prot, u32 flags, u32 fd, u64 o
                         (unsigned long) prot, (unsigned long) flags, off);
        up_write(&current->mm->mmap_sem);
        if(!ret_type)
-               retval = ((retval < 0xf0000000) ? 0 : retval);
+               retval = ((retval < STACK_TOP32) ? 0 : retval);
                                
 out_putf:
        if (file)
@@ -121,10 +122,10 @@ asmlinkage u32 solaris_mmap64(struct pt_regs *regs, u32 len, u32 prot, u32 flags
        u32 offlo;
        
        if (regs->u_regs[UREG_G1]) {
-               if (get_user (offlo, (u32 *)(long)((u32)regs->u_regs[UREG_I6] + 0x5c)))
+               if (get_user (offlo, (u32 __user *)(long)((u32)regs->u_regs[UREG_I6] + 0x5c)))
                        return -EFAULT;
        } else {
-               if (get_user (offlo, (u32 *)(long)((u32)regs->u_regs[UREG_I6] + 0x60)))
+               if (get_user (offlo, (u32 __user *)(long)((u32)regs->u_regs[UREG_I6] + 0x60)))
                        return -EFAULT;
        }
        return do_solaris_mmap((u32)regs->u_regs[UREG_I0], len, prot, flags, fd, (((u64)offhi)<<32)|offlo);
@@ -137,21 +138,34 @@ asmlinkage int solaris_brk(u32 brk)
        return sunos_brk(brk);
 }
 
-#define set_utsfield(to, from, dotchop, countfrom) {                   \
-       char *p;                                                        \
-       int i, len = (countfrom) ?                                      \
-               ((sizeof(to) > sizeof(from) ?                           \
-                       sizeof(from) : sizeof(to))) : sizeof(to);       \
-       if (copy_to_user(to, from, len))                                \
-               return -EFAULT;                                         \
-       if (dotchop)                                                    \
-               for (p=from,i=0; *p && *p != '.' && --len; p++,i++);    \
-       else                                                            \
-               i = len - 1;                                            \
-       if (__put_user('\0', (char *)(to+i)))                           \
-               return -EFAULT;                                         \
+static int __set_utsfield(char __user *to, int to_size,
+                         const char *from, int from_size,
+                         int dotchop, int countfrom)
+{
+       int len = countfrom ? (to_size > from_size ?
+                              from_size : to_size) : to_size;
+       int off;
+
+       if (copy_to_user(to, from, len))
+               return -EFAULT;
+
+       off = len < to_size? len: len - 1;
+       if (dotchop) {
+               const char *p = strnchr(from, len, '.');
+               if (p) off = p - from;
+       }
+
+       if (__put_user('\0', to + off))
+               return -EFAULT;
+
+       return 0;
 }
 
+#define set_utsfield(to, from, dotchop, countfrom) \
+       __set_utsfield((to), sizeof(to), \
+                      (from), sizeof(from), \
+                      (dotchop), (countfrom))
+
 struct sol_uname {
        char sysname[9];
        char nodename[9];
@@ -218,22 +232,21 @@ static char *serial(char *buffer)
 
 asmlinkage int solaris_utssys(u32 buf, u32 flags, int which, u32 buf2)
 {
+       struct sol_uname __user *v = A(buf);
+       int err;
+
        switch (which) {
        case 0: /* old uname */
                /* Let's cheat */
-               set_utsfield(((struct sol_uname *)A(buf))->sysname, 
-                       "SunOS", 1, 0);
+               err  = set_utsfield(v->sysname, "SunOS", 1, 0);
                down_read(&uts_sem);
-               set_utsfield(((struct sol_uname *)A(buf))->nodename, 
-                       system_utsname.nodename, 1, 1);
+               err |= set_utsfield(v->nodename, vx_new_uts(nodename),
+                                   1, 1);
                up_read(&uts_sem);
-               set_utsfield(((struct sol_uname *)A(buf))->release, 
-                       "2.6", 0, 0);
-               set_utsfield(((struct sol_uname *)A(buf))->version, 
-                       "Generic", 0, 0);
-               set_utsfield(((struct sol_uname *)A(buf))->machine, 
-                       machine(), 0, 0);
-               return 0;
+               err |= set_utsfield(v->release, "2.6", 0, 0);
+               err |= set_utsfield(v->version, "Generic", 0, 0);
+               err |= set_utsfield(v->machine, machine(), 0, 0);
+               return (err ? -EFAULT : 0);
        case 2: /* ustat */
                return -ENOSYS;
        case 3: /* fusers */
@@ -245,20 +258,19 @@ asmlinkage int solaris_utssys(u32 buf, u32 flags, int which, u32 buf2)
 
 asmlinkage int solaris_utsname(u32 buf)
 {
+       struct sol_utsname __user *v = A(buf);
+       int err;
+
        /* Why should we not lie a bit? */
        down_read(&uts_sem);
-       set_utsfield(((struct sol_utsname *)A(buf))->sysname, 
-                       "SunOS", 0, 0);
-       set_utsfield(((struct sol_utsname *)A(buf))->nodename, 
-                       system_utsname.nodename, 1, 1);
-       set_utsfield(((struct sol_utsname *)A(buf))->release, 
-                       "5.6", 0, 0);
-       set_utsfield(((struct sol_utsname *)A(buf))->version, 
-                       "Generic", 0, 0);
-       set_utsfield(((struct sol_utsname *)A(buf))->machine, 
-                       machine(), 0, 0);
+       err  = set_utsfield(v->sysname, "SunOS", 0, 0);
+       err |= set_utsfield(v->nodename, vx_new_uts(nodename), 1, 1);
+       err |= set_utsfield(v->release, "5.6", 0, 0);
+       err |= set_utsfield(v->version, "Generic", 0, 0);
+       err |= set_utsfield(v->machine, machine(), 0, 0);
        up_read(&uts_sem);
-       return 0;
+
+       return (err ? -EFAULT : 0);
 }
 
 #define SI_SYSNAME             1       /* return name of operating system */
@@ -284,7 +296,7 @@ asmlinkage int solaris_sysinfo(int cmd, u32 buf, s32 count)
        case SI_HOSTNAME:
                r = buffer + 256;
                down_read(&uts_sem);
-               for (p = system_utsname.nodename, q = buffer; 
+               for (p = vx_new_uts(nodename), q = buffer;
                     q < r && *p && *p != '.'; *q++ = *p++);
                up_read(&uts_sem);
                *q = 0;
@@ -302,11 +314,11 @@ asmlinkage int solaris_sysinfo(int cmd, u32 buf, s32 count)
        }
        len = strlen(r) + 1;
        if (count < len) {
-               if (copy_to_user((char *)A(buf), r, count - 1) ||
-                   __put_user(0, (char *)A(buf) + count - 1))
+               if (copy_to_user(A(buf), r, count - 1) ||
+                   __put_user(0, (char __user *)A(buf) + count - 1))
                        return -EFAULT;
        } else {
-               if (copy_to_user((char *)A(buf), r, len))
+               if (copy_to_user(A(buf), r, len))
                        return -EFAULT;
        }
        return len;
@@ -342,7 +354,7 @@ asmlinkage int solaris_sysconf(int id)
 {
        switch (id) {
        case SOLARIS_CONFIG_NGROUPS:    return NGROUPS_MAX;
-       case SOLARIS_CONFIG_CHILD_MAX:  return CHILD_MAX;
+       case SOLARIS_CONFIG_CHILD_MAX:  return -1; /* no limit */
        case SOLARIS_CONFIG_OPEN_FILES: return OPEN_MAX;
        case SOLARIS_CONFIG_POSIX_VER:  return 199309;
        case SOLARIS_CONFIG_PAGESIZE:   return PAGE_SIZE;
@@ -453,7 +465,7 @@ struct rlimit32 {
        u32     rlim_max;
 };
 
-asmlinkage int solaris_getrlimit(unsigned int resource, struct rlimit32 *rlim)
+asmlinkage int solaris_getrlimit(unsigned int resource, struct rlimit32 __user *rlim)
 {
        struct rlimit r;
        int ret;
@@ -486,15 +498,15 @@ asmlinkage int solaris_getrlimit(unsigned int resource, struct rlimit32 *rlim)
        return ret;
 }
 
-asmlinkage int solaris_setrlimit(unsigned int resource, struct rlimit32 *rlim)
+asmlinkage int solaris_setrlimit(unsigned int resource, struct rlimit32 __user *rlim)
 {
        struct rlimit r, rold;
        int ret;
        mm_segment_t old_fs = get_fs ();
-       int (*sys_getrlimit)(unsigned int, struct rlimit *) =
-               (int (*)(unsigned int, struct rlimit *))SYS(getrlimit);
-       int (*sys_setrlimit)(unsigned int, struct rlimit *) =
-               (int (*)(unsigned int, struct rlimit *))SYS(setrlimit);
+       int (*sys_getrlimit)(unsigned int, struct rlimit __user *) =
+               (int (*)(unsigned int, struct rlimit __user *))SYS(getrlimit);
+       int (*sys_setrlimit)(unsigned int, struct rlimit __user *) =
+               (int (*)(unsigned int, struct rlimit __user *))SYS(setrlimit);
 
        if (resource > RLIMIT_SOL_VMEM)
                return -EINVAL; 
@@ -527,13 +539,13 @@ asmlinkage int solaris_setrlimit(unsigned int resource, struct rlimit32 *rlim)
        return ret;
 }
 
-asmlinkage int solaris_getrlimit64(unsigned int resource, struct rlimit *rlim)
+asmlinkage int solaris_getrlimit64(unsigned int resource, struct rlimit __user *rlim)
 {
        struct rlimit r;
        int ret;
        mm_segment_t old_fs = get_fs ();
-       int (*sys_getrlimit)(unsigned int, struct rlimit *) =
-               (int (*)(unsigned int, struct rlimit *))SYS(getrlimit);
+       int (*sys_getrlimit)(unsigned int, struct rlimit __user *) =
+               (int (*)(unsigned int, struct rlimit __user *))SYS(getrlimit);
 
        if (resource > RLIMIT_SOL_VMEM)
                return -EINVAL; 
@@ -556,15 +568,15 @@ asmlinkage int solaris_getrlimit64(unsigned int resource, struct rlimit *rlim)
        return ret;
 }
 
-asmlinkage int solaris_setrlimit64(unsigned int resource, struct rlimit *rlim)
+asmlinkage int solaris_setrlimit64(unsigned int resource, struct rlimit __user *rlim)
 {
        struct rlimit r, rold;
        int ret;
        mm_segment_t old_fs = get_fs ();
-       int (*sys_getrlimit)(unsigned int, struct rlimit *) =
-               (int (*)(unsigned int, struct rlimit *))SYS(getrlimit);
-       int (*sys_setrlimit)(unsigned int, struct rlimit *) =
-               (int (*)(unsigned int, struct rlimit *))SYS(setrlimit);
+       int (*sys_getrlimit)(unsigned int, struct rlimit __user *) =
+               (int (*)(unsigned int, struct rlimit __user *))SYS(getrlimit);
+       int (*sys_setrlimit)(unsigned int, struct rlimit __user *) =
+               (int (*)(unsigned int, struct rlimit __user *))SYS(setrlimit);
 
        if (resource > RLIMIT_SOL_VMEM)
                return -EINVAL; 
@@ -623,10 +635,10 @@ struct sol_timex {
        s32 stbcnt;
 };
 
-asmlinkage int solaris_ntp_gettime(struct sol_ntptimeval *ntp)
+asmlinkage int solaris_ntp_gettime(struct sol_ntptimeval __user *ntp)
 {
-       int (*sys_adjtimex)(struct timex *) =
-               (int (*)(struct timex *))SYS(adjtimex);
+       int (*sys_adjtimex)(struct timex __user *) =
+               (int (*)(struct timex __user *))SYS(adjtimex);
        struct timex t;
        int ret;
        mm_segment_t old_fs = get_fs();
@@ -644,10 +656,10 @@ asmlinkage int solaris_ntp_gettime(struct sol_ntptimeval *ntp)
        return ret;                             
 }
 
-asmlinkage int solaris_ntp_adjtime(struct sol_timex *txp)
+asmlinkage int solaris_ntp_adjtime(struct sol_timex __user *txp)
 {
-       int (*sys_adjtimex)(struct timex *) =
-               (int (*)(struct timex *))SYS(adjtimex);
+       int (*sys_adjtimex)(struct timex __user *) =
+               (int (*)(struct timex __user *))SYS(adjtimex);
        struct timex t;
        int ret, err;
        mm_segment_t old_fs = get_fs();
@@ -726,7 +738,8 @@ MODULE_LICENSE("GPL");
 extern u32 tl0_solaris[8];
 #define update_ttable(x)                                                                               \
        tl0_solaris[3] = (((long)(x) - (long)tl0_solaris - 3) >> 2) | 0x40000000;                       \
-       __asm__ __volatile__ ("membar #StoreStore; flush %0" : : "r" (&tl0_solaris[3]))
+       wmb();          \
+       __asm__ __volatile__ ("flush %0" : : "r" (&tl0_solaris[3]))
 #else
 #endif 
 
@@ -750,7 +763,8 @@ int init_module(void)
        entry64_personality_patch |=
                (offsetof(struct task_struct, personality) +
                 (sizeof(unsigned long) - 1));
-       __asm__ __volatile__("membar #StoreStore; flush %0"
+       wmb();
+       __asm__ __volatile__("flush %0"
                             : : "r" (&entry64_personality_patch));
        return 0;
 }