Skip to content

Commit 2412638

Browse files
committed
add further ioctl time64 fallback conversions
this commit covers all remaining ioctls I'm aware of that use time_t-derived types in their interfaces. it may still be incomplete, and has undergone only minimal testing for a few commands used in audio playback. the SNDRV_PCM_IOCTL_SYNC_PTR command is special-cased because, rather than the whole structure expanding, it has two substructures each padded to 64 bytes that expand within their own 64-byte reserved zone. as long as it's the only one of its type, it doesn't really make sense to make a general framework for it, but the existing table framework is still used for the substructures in the special-case. one of the substructures, snd_pcm_mmap_status, has a snd_pcm_uframes_t member which is not a timestamp but is expanded just like one, to match the 64-bit-arch version of the structure. this is handled just like a timestamp at offset 8, and is the motivation for the conversions table holding offsets of individual values to be expanded rather than timespec/timeval type pairs. for some of the types, the size to which they expand is dependent on whether the arch's ABI aligns 8-byte types on 8-byte boundaries. new_req entries in the table need to reflect this size to get the right ioctl request number that will match what callers pass, but we don't have access to the actual structure type definitions here and duplicating them would be cumbersome. instead, the new_misaligned macro introduced here constructs an artificial object whose size is the result of expanding a misaligned timespec/timeval to 64-bit and imposing the arch's alignment on the result, which can be passed to the _IO{R,W,WR} macros.
1 parent 64d0e86 commit 2412638

File tree

1 file changed

+49
-1
lines changed

1 file changed

+49
-1
lines changed

src/misc/ioctl.c

Lines changed: 49 additions & 1 deletion
Original file line numberDiff line numberDiff line change
@@ -23,9 +23,49 @@ struct ioctl_compat_map {
2323
#define COUNT(...) NINTH(__VA_ARGS__,8,7,6,5,4,3,2,1,0)
2424
#define OFFS(...) COUNT(__VA_ARGS__), { __VA_ARGS__ }
2525

26+
/* yields a type for a struct with original size n, with a misaligned
27+
* timeval/timespec expanded from 32- to 64-bit. for use with ioctl
28+
* number producing macros; only size of result is meaningful. */
29+
#define new_misaligned(n) struct { int i; time_t t; char c[(n)-4]; }
30+
2631
static const struct ioctl_compat_map compat_map[] = {
2732
{ SIOCGSTAMP, SIOCGSTAMP_OLD, 8, R, 0, OFFS(0, 4) },
2833
{ SIOCGSTAMPNS, SIOCGSTAMPNS_OLD, 8, R, 0, OFFS(0, 4) },
34+
35+
/* SNDRV_TIMER_IOCTL_STATUS */
36+
{ _IOR('T', 0x14, char[96]), _IOR('T', 0x14, 88), 88, R, 0, OFFS(0,4) },
37+
38+
/* SNDRV_PCM_IOCTL_STATUS[_EXT] */
39+
{ _IOR('A', 0x20, char[128]), _IOR('A', 0x20, char[108]), 108, R, 1, OFFS(4,8,12,16,52,56,60,64) },
40+
{ _IOWR('A', 0x24, char[128]), _IOWR('A', 0x24, char[108]), 108, WR, 1, OFFS(4,8,12,16,52,56,60,64) },
41+
42+
/* SNDRV_RAWMIDI_IOCTL_STATUS */
43+
{ _IOWR('W', 0x20, char[48]), _IOWR('W', 0x20, char[36]), 36, WR, 1, OFFS(4,8) },
44+
45+
/* SNDRV_PCM_IOCTL_SYNC_PTR - with 3 subtables */
46+
{ _IOWR('A', 0x23, char[136]), _IOWR('A', 0x23, char[132]), 0, WR, 1, 0 },
47+
{ 0, 0, 4, WR, 1, 0 }, /* snd_pcm_sync_ptr (flags only) */
48+
{ 0, 0, 32, WR, 1, OFFS(8,12,16,24,28) }, /* snd_pcm_mmap_status */
49+
{ 0, 0, 8, WR, 1, OFFS(0,4) }, /* snd_pcm_mmap_control */
50+
51+
/* VIDIOC_QUERYBUF, VIDIOC_QBUF, VIDIOC_DQBUF, VIDIOC_PREPARE_BUF */
52+
{ _IOWR('V', 9, new_misaligned(72)), _IOWR('V', 9, char[72]), 72, WR, 0, OFFS(20) },
53+
{ _IOWR('V', 15, new_misaligned(72)), _IOWR('V', 15, char[72]), 72, WR, 0, OFFS(20) },
54+
{ _IOWR('V', 17, new_misaligned(72)), _IOWR('V', 17, char[72]), 72, WR, 0, OFFS(20) },
55+
{ _IOWR('V', 93, new_misaligned(72)), _IOWR('V', 93, char[72]), 72, WR, 0, OFFS(20) },
56+
57+
/* VIDIOC_DQEVENT */
58+
{ _IOR('V', 89, new_misaligned(96)), _IOR('V', 89, char[96]), 96, R, 0, OFFS(76,80) },
59+
60+
/* PPPIOCGIDLE */
61+
{ _IOR('t', 63, char[16]), _IOR('t', 63, char[8]), 8, R, 0, OFFS(0,4) },
62+
63+
/* PPGETTIME, PPSETTIME */
64+
{ _IOR('p', 0x95, char[16]), _IOR('p', 0x95, char[8]), 8, R, 0, OFFS(0,4) },
65+
{ _IOW('p', 0x96, char[16]), _IOW('p', 0x96, char[8]), 8, W, 0, OFFS(0,4) },
66+
67+
/* LPSETTIMEOUT */
68+
{ _IOW(0x6, 0xf, char[16]), 0x060f, 8, W, 0, OFFS(0,4) },
2969
};
3070

3171
static void convert_ioctl_struct(const struct ioctl_compat_map *map, char *old, char *new, int dir)
@@ -34,6 +74,14 @@ static void convert_ioctl_struct(const struct ioctl_compat_map *map, char *old,
3474
int old_offset = 0;
3575
int old_size = map->old_size;
3676
if (!(dir & map->dir)) return;
77+
if (!map->old_size) {
78+
/* offsets hard-coded for SNDRV_PCM_IOCTL_SYNC_PTR;
79+
* if another exception appears this needs changing. */
80+
convert_ioctl_struct(map+1, old, new, dir);
81+
convert_ioctl_struct(map+2, old+4, new+8, dir);
82+
convert_ioctl_struct(map+3, old+68, new+72, dir);
83+
return;
84+
}
3785
for (int i=0; i < map->noffs; i++) {
3886
int ts_offset = map->offsets[i];
3987
int len = ts_offset-old_offset;
@@ -69,7 +117,7 @@ int ioctl(int fd, int req, ...)
69117
arg = va_arg(ap, void *);
70118
va_end(ap);
71119
int r = __syscall(SYS_ioctl, fd, req, arg);
72-
if (r==-ENOTTY) {
120+
if (req && r==-ENOTTY) {
73121
for (int i=0; i<sizeof compat_map/sizeof *compat_map; i++) {
74122
if (compat_map[i].new_req != req) continue;
75123
union {

0 commit comments

Comments
 (0)