Date: Mon, 31 May 2010 10:32:28 +1200 From: Andrew Thompson <thompsa@FreeBSD.org> To: Kostik Belousov <kostikbel@gmail.com> Cc: freebsd-hackers@freebsd.org, Hans Petter Selasky <hselasky@c2i.net> Subject: Re: [FreeBSD 8/9] [64-bit IOCTL] Using the USB stack from a 32-bit application under a 64-bit kernel. Message-ID: <AANLkTikjHeZowHu2WUSCcxoaixY4sAH2B_KZNDF2wlW4@mail.gmail.com> In-Reply-To: <20100530221955.GN83316@deviant.kiev.zoral.com.ua> References: <201005302150.15232.hselasky@c2i.net> <20100530221955.GN83316@deviant.kiev.zoral.com.ua>
next in thread | previous in thread | raw e-mail | index | archive | help
On 31 May 2010 10:19, Kostik Belousov <kostikbel@gmail.com> wrote: > On Sun, May 30, 2010 at 09:50:15PM +0200, Hans Petter Selasky wrote: >> Hi, >> >> The USB team in FreeBSD has discussed this issue internally and would li= ke >> some advice how to best resolve the 32 bit to 64 bit IOCTL conversion is= sue. >> >> Sometimes IOCTL requests contain userland pointers of type "void *". Whe= n >> compiled on a 64-bit OS, sizeof(void *) is 8 bytes and when compiled on = a 32- >> bit OS sizeof(void *) is 4 bytes. This size difference makes it impossib= le to >> forward IOCTLs 1:1 from a 32-bit application running under a 64-bit kern= el. >> >> This issue does not only apply to the USB subsystem, but also other kern= el >> IOCTL interfaces. I suggest a more general solution: >> >> typedef uint64_t pvoid64; >> >> When an IOCTL needs to reference pointers in userspace, then they should >> always pad the pointer to 64-bit and use the pvoid64, which could have b= een a >> compiler type, so that we can easily convert to "void *" without using c= asts. >> >> >> When converting 32-bit userland pointers into 64-bit ones, there is no p= ointer >> conversion except for the unsigned expansion and resulting zero-padding.= I >> assume this assumption will always be valid. >> >> >> Please find attached an example patch to make the USB stack in 8 and 9 f= ully >> 64-bit compatible. >> >> >> Any comments are welcome! >> >> >> --HPS >> --- src/sys/dev/usb/usb_ioctl.h =A0 =A0 =A0 2010-02-14 12:03:51.00000000= 0 0000 >> +++ src/sys/dev/usb/usb_ioctl.h =A0 =A0 =A0 2010-02-14 12:03:51.00000000= 0 0000 >> @@ -131,9 +131,10 @@ >> =A0 =A0 =A0 =A0* NOTE: isochronous USB transfer only use one buffer, but= can have >> =A0 =A0 =A0 =A0* multiple frame lengths ! >> =A0 =A0 =A0 =A0*/ >> - =A0 =A0 void =A0**ppBuffer; =A0 =A0 =A0 =A0 =A0 =A0 =A0 /* pointer to = userland buffers */ >> - =A0 =A0 uint32_t *pLength; =A0 =A0 =A0 =A0 =A0 =A0 =A0/* pointer to fr= ame lengths, updated >> - =A0 =A0 =A0 =A0 =A0 =A0 =A0 =A0 =A0 =A0 =A0 =A0 =A0 =A0 =A0 =A0 =A0 = =A0 =A0* to actual length */ >> + =A0 =A0 uint64_t ppBuffer; =A0 =A0 =A0 =A0 =A0 =A0 =A0/* pointer to 64= -bit userland buffer pointers */ >> + =A0 =A0 uint64_t pLength; =A0 =A0 =A0 =A0 =A0 =A0 =A0 /* pointer to 32= -bit frame lengths, which >> + =A0 =A0 =A0 =A0 =A0 =A0 =A0 =A0 =A0 =A0 =A0 =A0 =A0 =A0 =A0 =A0 =A0 = =A0 =A0* get updated to the actual length after >> + =A0 =A0 =A0 =A0 =A0 =A0 =A0 =A0 =A0 =A0 =A0 =A0 =A0 =A0 =A0 =A0 =A0 = =A0 =A0* the transfer is complete. */ >> =A0 =A0 =A0 uint32_t nFrames; =A0 =A0 =A0 =A0 =A0 =A0 =A0 /* number of f= rames */ >> =A0 =A0 =A0 uint32_t aFrames; =A0 =A0 =A0 =A0 =A0 =A0 =A0 /* actual numb= er of frames */ >> =A0 =A0 =A0 uint16_t flags; > > Doesn't this change the existing ABI for 32bit platforms ? > > You may take a look at the sys/net/bpf.c, where the similar > issue is handled for bpf ioctls. To keep the ABI intact, you > would need to define the 32bit ABI structures and define > compat ioctls, then handle the ioctls by converting the structures > and calling the native handler. BIOCSRTIMEOUT32 is a good example. This has been done for other usb ioctls but the above struct has a pointer to an array of pointers in userland which makes it difficult. It isnt copied in with the ioctl so doesnt get the chance to be fixed up. so far, http://people.freebsd.org/~thompsa/linux_usb_amd64_2.diff Andrew
Want to link to this message? Use this URL: <https://mail-archive.FreeBSD.org/cgi/mid.cgi?AANLkTikjHeZowHu2WUSCcxoaixY4sAH2B_KZNDF2wlW4>