[<prev] [next>] [<thread-prev] [thread-next>] [day] [month] [year] [list]
Message-ID: <CAKgT0UdY=7w1Pkt99mzuwc+uFmWe5qOVJWJmMRyhDW-o7Fao9A@mail.gmail.com>
Date: Mon, 13 Feb 2023 07:14:34 -0800
From: Alexander Duyck <alexander.duyck@...il.com>
To: Geoff Levand <geoff@...radead.org>
Cc: netdev@...r.kernel.org, Jakub Kicinski <kuba@...nel.org>,
"David S. Miller" <davem@...emloft.net>
Subject: Re: [PATCH net v4 1/2] net/ps3_gelic_net: Fix RX sk_buff length
On Sun, Feb 12, 2023 at 9:06 AM Geoff Levand <geoff@...radead.org> wrote:
>
> Hi Alexander,
>
> Thanks for the review.
>
> On 2/6/23 08:25, Alexander H Duyck wrote:
> > On Sun, 2023-02-05 at 22:10 +0000, Geoff Levand wrote:
> >> The Gelic Ethernet device needs to have the RX sk_buffs aligned to
> >> GELIC_NET_RXBUF_ALIGN and the length of the RX sk_buffs must be a multiple of
> >> GELIC_NET_RXBUF_ALIGN.
> >>
>
> >> static int gelic_descr_prepare_rx(struct gelic_card *card,
> >> struct gelic_descr *descr)
> >> {
> >> - int offset;
> >> - unsigned int bufsize;
> >> + struct device *dev = ctodev(card);
> >> + struct {
> >> + unsigned int total_bytes;
> >> + unsigned int offset;
> >> + } aligned_buf;
> >> + dma_addr_t cpu_addr;
> >>
> >> if (gelic_descr_get_status(descr) != GELIC_DESCR_DMA_NOT_IN_USE)
> >> dev_info(ctodev(card), "%s: ERROR status\n", __func__);
> >> - /* we need to round up the buffer size to a multiple of 128 */
> >> - bufsize = ALIGN(GELIC_NET_MAX_MTU, GELIC_NET_RXBUF_ALIGN);
> >>
> >> - /* and we need to have it 128 byte aligned, therefore we allocate a
> >> - * bit more */
> >> - descr->skb = dev_alloc_skb(bufsize + GELIC_NET_RXBUF_ALIGN - 1);
> >> + aligned_buf.total_bytes = (GELIC_NET_RXBUF_ALIGN - 1) +
> >> + GELIC_NET_MAX_MTU + (GELIC_NET_RXBUF_ALIGN - 1);
> >> +
> >
> > This value isn't aligned to anything as there have been no steps taken
> > to align it. In fact it is guaranteed to be off by 2. Did you maybe
> > mean to use an "&" somewhere?
>
> total_bytes here means the total number of bytes to allocate that will
> allow for the desired alignment. This value a bit too much though since
> we really just need it to end on a GELIC_NET_RXBUF_ALIGN boundary, so
> adding ALIGN(GELIC_NET_MAX_MTU, GELIC_NET_RXBUF_ALIGN) should be enough.
> I'll fix that in the next patch version.
>
> >> + descr->skb = dev_alloc_skb(aligned_buf.total_bytes);
> >> +
> >> if (!descr->skb) {
> >> - descr->buf_addr = 0; /* tell DMAC don't touch memory */
> >> + descr->buf_addr = 0;
> >> return -ENOMEM;
> >
> > Why remove this comment?
>
> If we return -ENOMEM this descriptor shouldn't be used.
Right. But the comment is essentially saying that. The question is why
remove the documentation?
> >> }
> >> - descr->buf_size = cpu_to_be32(bufsize);
> >> +
> >> + aligned_buf.offset =
> >> + PTR_ALIGN(descr->skb->data, GELIC_NET_RXBUF_ALIGN) -
> >> + descr->skb->data;
> >> +
> >> + descr->buf_size = ALIGN(GELIC_NET_MAX_MTU, GELIC_NET_RXBUF_ALIGN);
> >
> > Originally this was being written using cpu_to_be32. WIth this you are
> > writing it raw w/ the cpu endianness. Is there a byte ordering issue
> > here?
>
> No. The PS3 has a big endian CPU, so we really don't need any
> of the endian conversions.
This doesn't introduce an ordering error, but it introduces a type
error. The bufsize variable was defined as a CPU ordered variable
whereas buf_size is a __be32. If you enable sparse checking it should
return an error.
I would recommend keeping the cpu_to_be32. If your architecture is big
endian it will do nothing and add no overhead. If at some point in the
future someone were to plug this IP into another architecture it would
take care of sorting out the byte ordering for you.
> >
> >> descr->dmac_cmd_status = 0;
> >> descr->result_size = 0;
> >> descr->valid_size = 0;
> >> descr->data_error = 0;
> >>
> >> - offset = ((unsigned long)descr->skb->data) &
> >> - (GELIC_NET_RXBUF_ALIGN - 1);
> >> - if (offset)
> >> - skb_reserve(descr->skb, GELIC_NET_RXBUF_ALIGN - offset);
> >
> > Rather than messing with all this it might be easier to just drop
> > offset in favor of NET_SKB_PAD since that should be offset in all cases
> > where dev_alloc_skb is being used. With that the reserve could just be
> > a constant.
>
> GELIC_NET_RXBUF_ALIGN is a property of the gelic hardware device. I
> would think if NET_SKB_PAD would work it would just be by coincidence.
NET_SKB_PAD is the alignment generated when you use dev_alloc_skb.
What I was getting at is that "offset" could probably be removed in
favor of just using NET_SKB_PAD.
Powered by blists - more mailing lists