mirror of
https://git.kernel.org/pub/scm/linux/kernel/git/stable/linux.git
synced 2025-01-15 01:44:52 +00:00
152194fe9c
The input events use struct timeval to store event time, unfortunately this structure is not y2038 safe and is being replaced in kernel with y2038 safe structures. Because of ABI concerns we can not change the size or the layout of structure input_event, so we opt to re-interpreting the 'seconds' part of timestamp as an unsigned value, effectively doubling the range of values, to year 2106. Newer glibc that has support for 32 bit applications to use 64 bit time_t supplies __USE_TIME_BITS64 define [1], that we can use to present the userspace with updated input_event layout. The updated layout will cause the compile time breakage, alerting applications and distributions maintainers to the issue. Existing 32 binaries will continue working without any changes until 2038. Ultimately userspace applications should switch to using monotonic or boot time clocks, as realtime clock is not very well suited for input event timestamps as it can go backwards (see a80b83b7b8 "Input: evdev - add CLOCK_BOOTTIME support" by by John Stultz). With monotonic clock the practical range of reported times will always fit into the pair of 32 bit values, as we do not expect any system to stay up for a hundred years without a single reboot. [1] https://sourceware.org/glibc/wiki/Y2038ProofnessDesign Suggested-by: Arnd Bergmann <arnd@arndb.de> Signed-off-by: Deepa Dinamani <deepa.kernel@gmail.com> Acked-by: Peter Hutterer <peter.hutterer@who-t.net> Patchwork-Id: 10148083 Signed-off-by: Dmitry Torokhov <dmitry.torokhov@gmail.com>
137 lines
3.3 KiB
C
137 lines
3.3 KiB
C
/*
|
|
* 32bit compatibility wrappers for the input subsystem.
|
|
*
|
|
* Very heavily based on evdev.c - Copyright (c) 1999-2002 Vojtech Pavlik
|
|
*
|
|
* This program is free software; you can redistribute it and/or modify it
|
|
* under the terms of the GNU General Public License version 2 as published by
|
|
* the Free Software Foundation.
|
|
*/
|
|
|
|
#include <linux/export.h>
|
|
#include <linux/uaccess.h>
|
|
#include "input-compat.h"
|
|
|
|
#ifdef CONFIG_COMPAT
|
|
|
|
int input_event_from_user(const char __user *buffer,
|
|
struct input_event *event)
|
|
{
|
|
if (in_compat_syscall() && !COMPAT_USE_64BIT_TIME) {
|
|
struct input_event_compat compat_event;
|
|
|
|
if (copy_from_user(&compat_event, buffer,
|
|
sizeof(struct input_event_compat)))
|
|
return -EFAULT;
|
|
|
|
event->input_event_sec = compat_event.sec;
|
|
event->input_event_usec = compat_event.usec;
|
|
event->type = compat_event.type;
|
|
event->code = compat_event.code;
|
|
event->value = compat_event.value;
|
|
|
|
} else {
|
|
if (copy_from_user(event, buffer, sizeof(struct input_event)))
|
|
return -EFAULT;
|
|
}
|
|
|
|
return 0;
|
|
}
|
|
|
|
int input_event_to_user(char __user *buffer,
|
|
const struct input_event *event)
|
|
{
|
|
if (in_compat_syscall() && !COMPAT_USE_64BIT_TIME) {
|
|
struct input_event_compat compat_event;
|
|
|
|
compat_event.sec = event->input_event_sec;
|
|
compat_event.usec = event->input_event_usec;
|
|
compat_event.type = event->type;
|
|
compat_event.code = event->code;
|
|
compat_event.value = event->value;
|
|
|
|
if (copy_to_user(buffer, &compat_event,
|
|
sizeof(struct input_event_compat)))
|
|
return -EFAULT;
|
|
|
|
} else {
|
|
if (copy_to_user(buffer, event, sizeof(struct input_event)))
|
|
return -EFAULT;
|
|
}
|
|
|
|
return 0;
|
|
}
|
|
|
|
int input_ff_effect_from_user(const char __user *buffer, size_t size,
|
|
struct ff_effect *effect)
|
|
{
|
|
if (in_compat_syscall()) {
|
|
struct ff_effect_compat *compat_effect;
|
|
|
|
if (size != sizeof(struct ff_effect_compat))
|
|
return -EINVAL;
|
|
|
|
/*
|
|
* It so happens that the pointer which needs to be changed
|
|
* is the last field in the structure, so we can retrieve the
|
|
* whole thing and replace just the pointer.
|
|
*/
|
|
compat_effect = (struct ff_effect_compat *)effect;
|
|
|
|
if (copy_from_user(compat_effect, buffer,
|
|
sizeof(struct ff_effect_compat)))
|
|
return -EFAULT;
|
|
|
|
if (compat_effect->type == FF_PERIODIC &&
|
|
compat_effect->u.periodic.waveform == FF_CUSTOM)
|
|
effect->u.periodic.custom_data =
|
|
compat_ptr(compat_effect->u.periodic.custom_data);
|
|
} else {
|
|
if (size != sizeof(struct ff_effect))
|
|
return -EINVAL;
|
|
|
|
if (copy_from_user(effect, buffer, sizeof(struct ff_effect)))
|
|
return -EFAULT;
|
|
}
|
|
|
|
return 0;
|
|
}
|
|
|
|
#else
|
|
|
|
int input_event_from_user(const char __user *buffer,
|
|
struct input_event *event)
|
|
{
|
|
if (copy_from_user(event, buffer, sizeof(struct input_event)))
|
|
return -EFAULT;
|
|
|
|
return 0;
|
|
}
|
|
|
|
int input_event_to_user(char __user *buffer,
|
|
const struct input_event *event)
|
|
{
|
|
if (copy_to_user(buffer, event, sizeof(struct input_event)))
|
|
return -EFAULT;
|
|
|
|
return 0;
|
|
}
|
|
|
|
int input_ff_effect_from_user(const char __user *buffer, size_t size,
|
|
struct ff_effect *effect)
|
|
{
|
|
if (size != sizeof(struct ff_effect))
|
|
return -EINVAL;
|
|
|
|
if (copy_from_user(effect, buffer, sizeof(struct ff_effect)))
|
|
return -EFAULT;
|
|
|
|
return 0;
|
|
}
|
|
|
|
#endif /* CONFIG_COMPAT */
|
|
|
|
EXPORT_SYMBOL_GPL(input_event_from_user);
|
|
EXPORT_SYMBOL_GPL(input_event_to_user);
|
|
EXPORT_SYMBOL_GPL(input_ff_effect_from_user);
|