[Date Prev][Date Next][Thread Prev][Thread Next][Date Index][Thread Index]
Re: [Qemu-ppc] [Qemu-devel] [PATCH for-2.11] hw/net/vmxnet3: Fix code to
From: |
Thomas Huth |
Subject: |
Re: [Qemu-ppc] [Qemu-devel] [PATCH for-2.11] hw/net/vmxnet3: Fix code to work on big endian hosts, too |
Date: |
Tue, 14 Nov 2017 09:21:41 +0100 |
User-agent: |
Mozilla/5.0 (X11; Linux x86_64; rv:52.0) Gecko/20100101 Thunderbird/52.4.0 |
On 13.11.2017 23:38, Philippe Mathieu-Daudé wrote:
> Hi Thomas,
>
> On 11/13/2017 01:57 PM, Thomas Huth wrote:
>> Since commit ab06ec43577177a442e8 we test the vmxnet3 device in the
>> pxe-tester, too (when running "make check SPEED=slow"). This now
>> revealed that the vmxnet3 code is not working right if the host is a big
>> endian machine (for example ppc64 or s390x) - "make check SPEED=slow"
>> is now failing on such hosts.
>>
>> The vmxnet3 code lacks endianess conversions in a couple of places.
>> Interestingly, the bitfields in the structs in vmxnet3.h already tried to
>> take care of the *bit* endianess of the C compilers - but the code missed
>> to change the *byte* endianess when reading or writing the corresponding
>> structs. So the bitfields are now wrapped into unions which allow to change
>> the byte endianess during runtime with the non-bitfield member of the union.
>> With these changes, "make check SPEED=slow" now properly works on big endian
>> hosts, too.
>>
>> Reported-by: David Gibson <address@hidden>
>> Signed-off-by: Thomas Huth <address@hidden>
>> ---
[...]
>>
>> @@ -535,7 +535,8 @@ static void vmxnet3_complete_packet(VMXNET3State *s, int
>> qidx, uint32_t tx_ridx)
>> memset(&txcq_descr, 0, sizeof(txcq_descr));
>> txcq_descr.txdIdx = tx_ridx;
>> txcq_descr.gen = vmxnet3_ring_curr_gen(&s->txq_descr[qidx].comp_ring);
>> -
>> + txcq_descr.val1 = cpu_to_le32(txcq_descr.val1);
>> + txcq_descr.val2 = cpu_to_le32(txcq_descr.val2);
>> vmxnet3_ring_write_curr_cell(d, &s->txq_descr[qidx].comp_ring,
>> &txcq_descr);
>
> What about using inline functions:
>
> - vmxnet3_rxdesc_pci_to_cpu(struct Vmxnet3_RxDesc *rxd)
> - vmxnet3_txdesc_pci_to_cpu(...)
> - vmxnet3_rxcompdesc_pci_to_cpu(struct Vmxnet3_RxCompDesc *rxcd)
> - vmxnet3_txcompdesc_pci_to_cpu(...)
Most of the structure byte-swapping is only done in one place for each
type of structure, so for adding these two or three new lines to the
code, it's not really justified to put them into separate functions that
add at least six or seven new lines of code. This is only useful if the
same struct is byte-swapped at two or more places in the code.
>> /* Flush changes in TX descriptor before changing the counter value */
>> @@ -695,11 +696,17 @@ vmxnet3_pop_next_tx_descr(VMXNET3State *s,
>> PCIDevice *d = PCI_DEVICE(s);
>>
>> vmxnet3_ring_read_curr_cell(d, ring, txd);
>> + txd->addr = le64_to_cpu(txd->addr);
>> + txd->val1 = le32_to_cpu(txd->val1);
>> + txd->val2 = le32_to_cpu(txd->val2);
>
> ^ probably not useful
At least the txd->gen (via val1) is required one line below! ... and I
wanted to be sure that the caller of vmxnet3_pop_next_tx_descr() always
gets a consistent descriptor, even if the function returns false, so
let's better be safe than sorry and always swap all fields of the struct.
>> if (txd->gen == vmxnet3_ring_curr_gen(ring)) {
>> /* Only read after generation field verification */
>> smp_rmb();
>> /* Re-read to be sure we got the latest version */
>> vmxnet3_ring_read_curr_cell(d, ring, txd);
>> + txd->addr = le64_to_cpu(txd->addr);
>> + txd->val1 = le32_to_cpu(txd->val1);
>> + txd->val2 = le32_to_cpu(txd->val2);
>
> Using inlined func:
>
> vmxnet3_txdesc_pci_to_cpu(txd);
Ok, so that's one of the few spots where a struct is byteswapped not
only in one place, but in two. I guess I could add a inline function for
this...
>> VMXNET3_RING_DUMP(VMW_RIPRN, "TX", qidx, ring);
>> *descr_idx = vmxnet3_ring_curr_cell_idx(ring);
>> vmxnet3_inc_tx_consumption_counter(s, qidx);
>> @@ -749,7 +756,7 @@ static void vmxnet3_process_tx_queue(VMXNET3State *s,
>> int qidx)
>>
>> if (!s->skip_current_tx_pkt) {
>> data_len = (txd.len > 0) ? txd.len : VMXNET3_MAX_TX_BUF_SIZE;
>> - data_pa = le64_to_cpu(txd.addr);
>> + data_pa = txd.addr;
>>
>> if (!net_tx_pkt_add_raw_fragment(s->tx_pkt,
>> data_pa,
>> @@ -792,6 +799,9 @@ vmxnet3_read_next_rx_descr(VMXNET3State *s, int qidx,
>> int ridx,
>> Vmxnet3Ring *ring = &s->rxq_descr[qidx].rx_ring[ridx];
>> *didx = vmxnet3_ring_curr_cell_idx(ring);
>> vmxnet3_ring_read_curr_cell(d, ring, dbuf);
>> + dbuf->addr = le64_to_cpu(dbuf->addr);
>> + dbuf->val1 = le32_to_cpu(dbuf->val1);
>> + dbuf->ext1 = le32_to_cpu(dbuf->ext1);
>
> again:
>
> vmxnet3_txdesc_pci_to_cpu(dbuf);
That function reads an RX descriptor, not a TX descriptor! And this
descriptor type is only byte-swapped in this single place. And the whole
function body is only 6 lines. So an additional inline function is
really not justified here!
>> }
>>
>> static inline uint8_t
>> @@ -811,6 +821,9 @@ vmxnet3_pop_rxc_descr(VMXNET3State *s, int qidx,
>> uint32_t *descr_gen)
>>
>> pci_dma_read(PCI_DEVICE(s),
>> daddr, &rxcd, sizeof(struct Vmxnet3_RxCompDesc));
>> + rxcd.val1 = le32_to_cpu(rxcd.val1);
>> + rxcd.val2 = le32_to_cpu(rxcd.val2);
>> + rxcd.val3 = le32_to_cpu(rxcd.val3);
>
> vmxnet3_txcompdesc_pci_to_cpu(&rxcd);
>
>> ring_gen = vmxnet3_ring_curr_gen(&s->rxq_descr[qidx].comp_ring);
>>
>> if (rxcd.gen != ring_gen) {
>> @@ -1098,14 +1111,16 @@ vmxnet3_indicate_packet(VMXNET3State *s)
>> }
>>
>> chunk_size = MIN(bytes_left, rxd.len);
>> - vmxnet3_pci_dma_writev(d, data, bytes_copied,
>> - le64_to_cpu(rxd.addr), chunk_size);
>> + vmxnet3_pci_dma_writev(d, data, bytes_copied, rxd.addr, chunk_size);
>> bytes_copied += chunk_size;
>> bytes_left -= chunk_size;
>>
>> vmxnet3_dump_rx_descr(&rxd);
>
> ^ this dump is incorrect (not host swapped yet).
I think you're wrong, the endianess should be fine here. rxd is ready
via vmxnet3_get_next_rx_descr() which eventually calls
vmxnet3_read_next_rx_descr() and that function is taking care of the
byte-swapping.
> vmxnet3_rxcompdesc_pci_to_cpu(&rxcd);
>
>>
>> if (ready_rxcd_pa != 0) {
>> + rxcd.val1 = le32_to_cpu(rxcd.val1);
>> + rxcd.val2 = le32_to_cpu(rxcd.val2);
>> + rxcd.val3 = le32_to_cpu(rxcd.val3);
>
> (out of if)
Why?
>> pci_dma_write(d, ready_rxcd_pa, &rxcd, sizeof(rxcd));
>> }
>>
>> @@ -1138,6 +1153,9 @@ vmxnet3_indicate_packet(VMXNET3State *s)
>> rxcd.eop = 1;
>> rxcd.err = (bytes_left != 0);
>>
>> + rxcd.val1 = le32_to_cpu(rxcd.val1);
>> + rxcd.val2 = le32_to_cpu(rxcd.val2);
>> + rxcd.val3 = le32_to_cpu(rxcd.val3);
>
> vmxnet3_rxcompdesc_pci_to_cpu(&rxcd);
>
>> pci_dma_write(d, ready_rxcd_pa, &rxcd, sizeof(rxcd));
One of the few spots where a struct is written twice. I guess I could
indeed add an inline wrapper for this ... not sure whether it's worth
the effort...
Thomas