diff options
Diffstat (limited to 'fs/compat.c')
| -rw-r--r-- | fs/compat.c | 37 |
1 files changed, 25 insertions, 12 deletions
diff --git a/fs/compat.c b/fs/compat.c index 70c5af4cc270..a2ba78bdf7f7 100644 --- a/fs/compat.c +++ b/fs/compat.c | |||
| @@ -1751,11 +1751,15 @@ asmlinkage long compat_sys_select(int n, compat_ulong_t __user *inp, | |||
| 1751 | ret = compat_core_sys_select(n, inp, outp, exp, &timeout); | 1751 | ret = compat_core_sys_select(n, inp, outp, exp, &timeout); |
| 1752 | 1752 | ||
| 1753 | if (tvp) { | 1753 | if (tvp) { |
| 1754 | struct compat_timeval rtv; | ||
| 1755 | |||
| 1754 | if (current->personality & STICKY_TIMEOUTS) | 1756 | if (current->personality & STICKY_TIMEOUTS) |
| 1755 | goto sticky; | 1757 | goto sticky; |
| 1756 | tv.tv_usec = jiffies_to_usecs(do_div((*(u64*)&timeout), HZ)); | 1758 | rtv.tv_usec = jiffies_to_usecs(do_div((*(u64*)&timeout), HZ)); |
| 1757 | tv.tv_sec = timeout; | 1759 | rtv.tv_sec = timeout; |
| 1758 | if (copy_to_user(tvp, &tv, sizeof(tv))) { | 1760 | if (compat_timeval_compare(&rtv, &tv) < 0) |
| 1761 | rtv = tv; | ||
| 1762 | if (copy_to_user(tvp, &rtv, sizeof(rtv))) { | ||
| 1759 | sticky: | 1763 | sticky: |
| 1760 | /* | 1764 | /* |
| 1761 | * If an application puts its timeval in read-only | 1765 | * If an application puts its timeval in read-only |
| @@ -1822,13 +1826,17 @@ asmlinkage long compat_sys_pselect7(int n, compat_ulong_t __user *inp, | |||
| 1822 | } while (!ret && !timeout && tsp && (ts.tv_sec || ts.tv_nsec)); | 1826 | } while (!ret && !timeout && tsp && (ts.tv_sec || ts.tv_nsec)); |
| 1823 | 1827 | ||
| 1824 | if (tsp && !(current->personality & STICKY_TIMEOUTS)) { | 1828 | if (tsp && !(current->personality & STICKY_TIMEOUTS)) { |
| 1825 | ts.tv_sec += timeout / HZ; | 1829 | struct compat_timespec rts; |
| 1826 | ts.tv_nsec += (timeout % HZ) * (1000000000/HZ); | 1830 | |
| 1827 | if (ts.tv_nsec >= 1000000000) { | 1831 | rts.tv_sec = timeout / HZ; |
| 1828 | ts.tv_sec++; | 1832 | rts.tv_nsec = (timeout % HZ) * (NSEC_PER_SEC/HZ); |
| 1829 | ts.tv_nsec -= 1000000000; | 1833 | if (rts.tv_nsec >= NSEC_PER_SEC) { |
| 1834 | rts.tv_sec++; | ||
| 1835 | rts.tv_nsec -= NSEC_PER_SEC; | ||
| 1830 | } | 1836 | } |
| 1831 | (void)copy_to_user(tsp, &ts, sizeof(ts)); | 1837 | if (compat_timespec_compare(&rts, &ts) < 0) |
| 1838 | rts = ts; | ||
| 1839 | copy_to_user(tsp, &rts, sizeof(rts)); | ||
| 1832 | } | 1840 | } |
| 1833 | 1841 | ||
| 1834 | if (ret == -ERESTARTNOHAND) { | 1842 | if (ret == -ERESTARTNOHAND) { |
| @@ -1918,12 +1926,17 @@ asmlinkage long compat_sys_ppoll(struct pollfd __user *ufds, | |||
| 1918 | sigprocmask(SIG_SETMASK, &sigsaved, NULL); | 1926 | sigprocmask(SIG_SETMASK, &sigsaved, NULL); |
| 1919 | 1927 | ||
| 1920 | if (tsp && timeout >= 0) { | 1928 | if (tsp && timeout >= 0) { |
| 1929 | struct compat_timespec rts; | ||
| 1930 | |||
| 1921 | if (current->personality & STICKY_TIMEOUTS) | 1931 | if (current->personality & STICKY_TIMEOUTS) |
| 1922 | goto sticky; | 1932 | goto sticky; |
| 1923 | /* Yes, we know it's actually an s64, but it's also positive. */ | 1933 | /* Yes, we know it's actually an s64, but it's also positive. */ |
| 1924 | ts.tv_nsec = jiffies_to_usecs(do_div((*(u64*)&timeout), HZ)) * 1000; | 1934 | rts.tv_nsec = jiffies_to_usecs(do_div((*(u64*)&timeout), HZ)) * |
| 1925 | ts.tv_sec = timeout; | 1935 | 1000; |
| 1926 | if (copy_to_user(tsp, &ts, sizeof(ts))) { | 1936 | rts.tv_sec = timeout; |
| 1937 | if (compat_timespec_compare(&rts, &ts) < 0) | ||
| 1938 | rts = ts; | ||
| 1939 | if (copy_to_user(tsp, &rts, sizeof(rts))) { | ||
| 1927 | sticky: | 1940 | sticky: |
| 1928 | /* | 1941 | /* |
| 1929 | * If an application puts its timeval in read-only | 1942 | * If an application puts its timeval in read-only |
