RE: [PATCH 2/2] tools: ffs-test: convert to new descriptor format fixing compilation error

From: Krzysztof Opasiak
Date: Thu Jun 05 2014 - 09:49:49 EST


Hi,

> -----Original Message-----
> From: Michal Nazarewicz [mailto:mpn@xxxxxxxxxx] On Behalf Of Michal
> Nazarewicz
> Sent: Thursday, June 05, 2014 1:56 PM
> To: Krzysztof Opasiak; 'Felipe Balbi'
> Cc: Krzysztof Opasiak; linux-usb@xxxxxxxxxxxxxxx; linux-
> kernel@xxxxxxxxxxxxxxx
> Subject: Re: [PATCH 2/2] tools: ffs-test: convert to new descriptor
> format fixing compilation error
>
> On Thu, Jun 05 2014, Krzysztof Opasiak <k.opasiak@xxxxxxxxxxx>
> wrote:
> > I would suggest adding a suitable structure as you described in
> > previous discussion[1]. Writing first 3 fields in each userspace
> > program doesn't look quite good. Using:
> >
> > #ifndef USE_LEGACY_DESC_HEAD
> > struct {
> > struct usb_functionfs_desc_head2 header;
> > __le32 fs_count
> > (... and rest according to flags ...)
> > } __attribute__((packed)) header;
> > #else ...
> >
> > Would be shorter, more flexible and user friendly. Moreover it
> gives
> > less places for mistake (writing fields in wrong order).
>
> For ffs-test with USE_LEGACY_DESC_HEAD support it would be longer.
> Compare:

(... snip ...)

>
> The second one uses an unreadable hack to match length of the first
> one
> and the third one is two lines longer. On top of that, the second
> and
> third produce different structures, use more complicated
> preprocessing
> directives, and repeat value of fs_count and hs_count twice.
>
> Without legacy support, it would indeed be shorter (by two lines),

Yes and I was talking about apps which use only new API without compatibility layer.

I agree that when providing compatibility layer it is more complicated but as always this is a cost of compatibility layer.


> but
> would lead to awkward structures:
>
> ----------- >8 ----------------------------------------------------
> -----
> struct {
> struct usb_functionfs_descs_head2 header;
> /* Why aren't the two below fields inside of a header? */
> __le32 fs_count;
> __le32 hs_count;
> };
>
> struct {
> struct {
> /* Why is there a header.header field? */
> struct usb_functionfs_descs_head2 header;
> __le32 fs_count;
> __le32 hs_count;
> };
> };
> ----------- >8 ----------------------------------------------------
> -----
>

Please consider this:

struct {
struct usb_functionfs_descs_head2 header;
__le32 fs_count;
__le32 hs_count;
struct {
(...)
} fs_desc;
struct {
(...) /* Other than previous */
} hs_desc;
} = {
.header = (...),
.fs_count = X,
.fs_desc = {
(...)
},
.hs_count = Y,
.hs_desc = {
(...)
},
}

This approach looks quite reasonable for me and don't cause any confusion. Numbers of fs and hs descriptors don't go to headers because they are more connected with fs_desc and hs_desc than with header. Using this allows you to initialize them as close to fs_desc and hs_desc as possible. BTW yes I see that when we would like to have backward compatibility this will be few lines longer.

> And I don't see how it would be more flexible. If anything, it
> would be
> less.

When I was writing about flexibility I was talking about use case described above.

>
> I understand, and share, your sentiment, but I don't think adding
> usb_functionfs_descs_head2 structure with magic, flags and length
> would
> improve the situation.
>
> Something I thought about shortly was:
>
> ----------- >8 ----------------------------------------------------
> -----
> #define USB_FFS_DESCS_HEAD(_flags) struct { \
> __le32 magic, length, flags; \
> __le32 data[!!(_flags & FUNCTIONFS_HAS_FS_DESC) + \
> !!(_flags & FUNCTIONFS_HAS_HS_DESC) + \
> !!(_flags & FUNCTIONFS_HAS_SS_DESC)]; \
> } __attribute__((packed))
>
> #define USB_FFS_DESCS_HEAD_INIT(_flags, length, ...) {
> \
> .magic = cpu_to_le32(FUNCTIONFS_DESCRIPTORS_MAGIC_V2),
> \
> .flags = cpu_to_le32(flags), \
> .langth = cpu_to_le32(length), \
> .data = { __VA_ARGS__ } \
> }
>
> static const struct {
> USB_FFS_DESCS_HEAD(FUNCTIONFS_HAS_FS_DESC |
> FUNCTIONFS_HAS_HS_DESC),
> /* â */
> } __attribute__((packed)) descriptors = {
> USB_FFS_DESCS_HEAD_INIT(FUNCTIONFS_HAS_FS_DESC |
> FUNCTIONFS_HAS_HS_DESC,
> sizeof(descriptors), 3, 3),
> /* â */
> };
> ----------- >8 ----------------------------------------------------
> -----
>
> But whether this is nicer is arguable as well.

In my opinion those macros are quite nice and should appear in functionfs.h. Maybe it would be suitable to provide ifdef __USB_FFS_USE_LEGACY_API and two versions of those macros? This will make backward compatibility to be handled by one switch before including a header instead of directly by user with ifndefs. Then the whole code with binary backward compatibility would be as simple as:

#ifdef MY_USE_LEGACY_FLAG
#define __USB_FFS_USE_LEGACY_API
#endif

#include <linux/usb/functionfs.h>

#define FFS_FLAGS (FUNCTIONFS_HAS_FS_DESC | FUNCTIONFS_HAS_HS_DESC)

struct {
USB_FFS_DESCS_HEAD(FFS_FLAGS) header;
struct {
(...) /* descriptors */
} fs_desc, hs_desc;
} __attribute__((packed)) my_desc = {
.header = USB_FFS_DESCS_HEAD_INIT(FFS_FLAGS, sizeof(my_desc), 3, 3),
(...) /* rest of initialization */
};

Of course two more suitable ifndef will be required for ss_desc. Taking all into account this approach looks very good to me and needs only minimal user action to provide support for both legacy and new API. This approach have also some disadvantage because we assume that developer has new kernel headers which provide such macros and only place where such app run has old kernel. I'm also considering approach with two macros because this would allow app to detect in runtime whether current kernel has new ffs version and use legacy macro as fallback procedure.


--
BR's
Krzysztof Opasiak
Samsung R&D Institute Poland
Samsung Electronics
k.opasiak@xxxxxxxxxxx





--
To unsubscribe from this list: send the line "unsubscribe linux-kernel" in
the body of a message to majordomo@xxxxxxxxxxxxxxx
More majordomo info at http://vger.kernel.org/majordomo-info.html
Please read the FAQ at http://www.tux.org/lkml/