RE: [PATCH v8] can: usb: IXXAT USB-to-CAN adapters drivers

[Date Prev][Date Next][Thread Prev][Thread Next][Date Index][Thread Index]

 



Hi,

> If you do not have the hardware specification (or at least the documentation of ixxat_canbtp->tdo) then it will be hard for you to do anything here.
The tdo field is documented in our windows drivers manual[1]. The socketcan driver seems to use CAN_BTMODE_RAW(IXXAT_USB_BTMODE_NAT), so the hardware specific number of CAN clock cycles is expected here.

>From the manual:
wTDO:
Offset to the transceiver delay (or Secondary Sample Point SSP) that is
automatically determined by the controller. Value is only relevant with fast data
bit rate. If in field dwMode the bit CAN_BTMODE_RAW is set, the hardware
specific number of CAN clock cycles is expected here. If not, the value defines
the Secondary Sample Point (SSP) in relation to the total number of time quanta
per bit (example: if wTS1+wTS2=100 and wTDO=65 the SSP is 65% of a bit time).
Value 0 deactivates the SSP.

[1]: https://www.ixxat.com/docs/librariesprovider8/ixxat-english-new/pc-can-interfaces/02---windows-drivers/vci-v4.0.1133.0-windows-11-10.zip?sfvrsn=2d1dfdd7_20
See "./VCI V4.0.1133.0 - Windows 11, 10/Manuals/VCI V4 C-CanFD Software Design Guide English.pdf"
Please note that for example the simplified SSP positioning (setting tdo to 0XFFFF) described in the document is handled in the windows driver and not in the device firmware, so it cannot be used here.

Best regards
Markus

-----Original Message-----
From: Vincent Mailhol <vincent.mailhol@xxxxxxxxx> 
Sent: Wednesday, May 24, 2023 7:31 AM
To: Marc Kleine-Budde <mkl@xxxxxxxxxxxxxx>
Cc: Peter Seiderer <ps.report@xxxxxxx>; linux-can@xxxxxxxxxxxxxxx; Florian Ferg <flfe@xxxxxxxxxxxxxxx>; socketcan <socketcan@xxxxxxxxxxxxxxx>
Subject: Re: [PATCH v8] can: usb: IXXAT USB-to-CAN adapters drivers

On Tue. 23 May 2023 at 05:06, Marc Kleine-Budde <mkl@xxxxxxxxxxxxxx> wrote:
> On 22.05.2023 21:53:54, Peter Seiderer wrote:
> > > > +static int ixxat_usb_init_ctrl(struct ixxat_usb_device *dev) {
> > > > +       const struct can_bittiming *bt = &dev->can.bittiming;
> > > > +       const u16 port = dev->ctrl_index;
> > > > +       int err;
> > > > +       struct ixxat_usb_init_cl1_cmd *cmd;
> > > > +       const u32 rcv_size = sizeof(cmd->res);
> > > > +       const u32 snd_size = sizeof(*cmd);
> > >
> > > Remove those size variables and directly use sizeof(cmd->res) and
> > > sizeof(*cmd) in the code.
> >
> > O.k.
> >
> > >
> > > > +       u8 opmode = IXXAT_USB_OPMODE_EXTENDED | IXXAT_USB_OPMODE_STANDARD;
> > > > +       u8 btr0 = ((bt->brp - 1) & 0x3f) | (((bt->sjw - 1) & 
> > > > + 0x3) << 6);
> > >
> > > Add a macro definition for the 0x3f  and 0x3 masks using GENMASK() 
> > > and then calculate the value using FIELD_PREP().
> >
> > O.k., define names o.k.?
>
> ACK
>
> > +#define IXXAT_USB_CAN_BTR0_BRP_MASK GENMASK(5, 0) #define 
> > +IXXAT_USB_CAN_BTR0_SJW_MASK GENMASK(7, 6)
>
> Now use:
>         btr0 = FIELD_PREP(IXXAT_USB_CAN_BTR0_BRP_MASK, bt->brp - 1) |
>                 FIELD_PREP(IXXAT_USB_CAN_BTR0_SJW_MASK, bt->sjw - 1);
>
> [...]
>
> > > > +static void ixxat_usb_update_ts_now(struct ixxat_usb_device 
> > > > +*dev, u32 ts_now) {
> > > > +       u32 *ts_dev = &dev->time_ref.ts_dev_0;
> > > > +       ktime_t *kt_host = &dev->time_ref.kt_host_0;
> > > > +       u64 timebase = (u64)0x00000000FFFFFFFF - (u64)(*ts_dev) 
> > > > ++ (u64)ts_now;
> > >
> > > Replace 0x00000000FFFFFFFF by U32_MAX from linux/limits.h
> >
> > O.k.
>
> Can you read the free running timer register a USB transfer? If so, 
> it's better to use the cyclecounter/timecounter framework.
>
> [...]
>
> > > > +static void ixxat_usb_decode_buf(struct urb *urb) {
> > > > +       struct ixxat_usb_device *dev = urb->context;
> > > > +       struct net_device *netdev = dev->netdev;
> > > > +       struct ixxat_can_msg *can_msg;
> > > > +       int err = 0;
> > > > +       u32 pos = 0;
> > > > +       u8 *data = urb->transfer_buffer;
> > > > +
> > > > +       while (pos < urb->actual_length) {
> > > > +               u32 time;
> > > > +               u8 size;
> > > > +               u8 type;
> > > > +
> > > > +               can_msg = (struct ixxat_can_msg *)&data[pos];
> > > > +               if (!can_msg || !can_msg->base.size) {
> > > > +                       err = -ENOTSUPP;
> >
> > checkpatch.pl warns about this one:
> >
> >       WARNING: ENOTSUPP is not a SUSV4 error code, prefer EOPNOTSUPP
> >       #1045: FILE: drivers/net/can/usb/ixxat_usb/ixxat_usb_core.c:644:
> >       +                       err = -ENOTSUPP;
> >
> > Is EOPNOTSUPP appropriate (or EBADMSG as below)?
>
> EBADMSG sounds good.

These being used within your driver, I think it is fine to use what you think is convenient. At the end of the day, if there is an issue, the error code will be reported to the maintainer (you?). Of course, it is always better to silence the checkpatch.pl warnings.

EBADMSG is good. I also like EMSGSIZE. While EMSGSIZE is normally reserved for "Message[s] too long":

  https://elixir.bootlin.com/linux/latest/source/include/uapi/asm-generic/errno.h#L73

it describes the issue pretty well.

> [...]
>
> > > > +static void ixxat_usb_read_bulk_callback(struct urb *urb) {
> > > > +       struct ixxat_usb_device *dev = urb->context;
> > > > +       const struct ixxat_usb_adapter *adapter = dev->adapter;
> > > > +       struct net_device *netdev = dev->netdev;
> > > > +       struct usb_device *udev = dev->udev;
> > > > +       int err;
> > > > +
> > > > +       if (!netif_device_present(netdev))
> > > > +               return;
> > > > +
> > > > +       switch (urb->status) {
> > > > +       case 0: /* success */
> > > > +               break;
> > > > +       case -EPROTO:
> > > > +       case -EILSEQ:
> > > > +       case -ENOENT:
> > > > +       case -ECONNRESET:
> > > > +       case -ESHUTDOWN:
> > > > +               return;
> > > > +       default:
> > > > +               netdev_err(netdev, "Rx urb aborted /(%d)\n", 
> > > > + urb->status);
> > >
> > > Do not use parenthesis in log messages.
> > >
> > > Ref: 
> > > https://www.kernel.org/doc/html/latest/process/coding-style.html#p
> > > rinting-kernel-messages
> > >
> > >   Printing numbers in parentheses (%d) adds no value and should be avoided.
> > >
> >
> > O.k., seems a common pattern, at least the same logging used in 
> > peak_usb, change it there as well (but seems you answered this in 
> > another thread already '...but bonus points if you send a clean-up 
> > patch to fix the existing log messages...')?
>
> IIRC the %pe is quite new compared to the peak_usb driver. Code 
> evolves and new code should not be based on outdated examples/drivers.

Yes, that was added in October 2019:

  commit 57f5677e535b ("printf: add support for printing symbolic error names")
  Link: https://git.kernel.org/torvalds/c/57f5677e535b

There is no strong rule to use %pe but I think this gives a better user experience, so I encourage any new code to use it. The ban of parentheses (%d) is a way older rule which predates the git history, it is just that humans are humans and that went through the cracks in past reviews.

For the existing code, it is as it is. If someone courageous enough wants to do a big clean-up, it would be great, but there are style discrepancies everywhere in the code base.

> [...]
>
> > > > +/**
> > > > + * struct ixxat_canbtp Bittiming parameters (CL2)
> > > > + * @mode: Operation mode
> > > > + * @bps: Bits per second
> > > > + * @ts1: First time segment
> > > > + * @ts2: Second time segment
> > > > + * @sjw: Synchronization jump width
> > > > + * @tdo: Transmitter delay offset
> > > > + *
> > > > + * Bittiming parameters of a CL2 initialization request  */ 
> > > > +struct ixxat_canbtp {
> > > > +       __le32 mode;
> > > > +       __le32 bps;
> > > > +       __le16 ts1;
> > > > +       __le16 ts2;
> > > > +       __le16 sjw;
> > > > +       __le16 tdo;
> > >
> > > It seems that your device supports TDC. What is the reason to not configure it?
> > >
> > > Please have a look at struct can_tdc:
> > >
> > >   
> > > https://elixir.bootlin.com/linux/v6.2/source/include/linux/can/bit
> > > timing.h#L21
> > >
> > > Please refer to this patch if you want an example of how to use TDC:
> > >
> > >   
> > > https://git.kernel.org/pub/scm/linux/kernel/git/torvalds/linux.git
> > > /commit/?id=1010a8fa9608
> > >
> >
> > Not sure about this one (and I have limited knowledge of the 
> > hardware details), seems I need to find more (spare) time to look 
> > into this one, or maybe something better for an follow-up patch?
>
> We usually need TDC for bit rates > 1..2 MBit/s

If you do not have the hardware specification (or at least the documentation of ixxat_canbtp->tdo) then it will be hard for you to do anything here.

I am fine to skip the TDC features for now and have it as a follow-up patch if you manage to figure it out someday.


Yours sincerely,
Vincent Mailhol




[Index of Archives]     [Automotive Discussions]     [Linux ARM Kernel]     [Linux ARM]     [Linux Omap]     [Fedora ARM]     [IETF Annouce]     [Security]     [Bugtraq]     [Linux]     [Linux OMAP]     [Linux MIPS]     [eCos]     [Asterisk Internet PBX]     [Linux API]     [CAN Bus]

  Powered by Linux