On Thu, Aug 28, 2025 at 05:34:51PM +0800, Haixu Cui wrote: > This is the virtio SPI Linux kernel driver.
... > +#include <linux/completion.h> > +#include <linux/interrupt.h> > +#include <linux/io.h> > +#include <linux/module.h> > +#include <linux/spi/spi.h> > +#include <linux/stddef.h> A lot of headers are still missing. See below. ... > +struct virtio_spi_priv { > + /* The virtio device we're associated with */ > + struct virtio_device *vdev; > + /* Pointer to the virtqueue */ > + struct virtqueue *vq; > + /* Copy of config space mode_func_supported */ > + u32 mode_func_supported; uXX (in particular u32) is defined in types.h. > + /* Copy of config space max_freq_hz */ > + u32 max_freq_hz; > +}; ... > +static int virtio_spi_set_delays(struct spi_transfer_head *th, > + struct spi_device *spi, > + struct spi_transfer *xfer) > +{ > + int cs_setup; > + int cs_word_delay_xfer; > + int cs_word_delay_spi; > + int delay; > + int cs_hold; > + int cs_inactive; > + int cs_change_delay; > + > + cs_setup = spi_delay_to_ns(&spi->cs_setup, xfer); > + if (cs_setup < 0) { > + dev_warn(&spi->dev, "Cannot convert cs_setup\n"); dev_warn() et al. are defined in dev_printk.h. > + return cs_setup; > + } > + th->cs_setup_ns = cpu_to_le32(cs_setup); This requires #include <asm/byteorder.h> > + cs_word_delay_xfer = spi_delay_to_ns(&xfer->word_delay, xfer); > + if (cs_word_delay_xfer < 0) { > + dev_warn(&spi->dev, "Cannot convert cs_word_delay_xfer\n"); > + return cs_word_delay_xfer; > + } > + cs_word_delay_spi = spi_delay_to_ns(&spi->word_delay, xfer); > + if (cs_word_delay_spi < 0) { > + dev_warn(&spi->dev, "Cannot convert cs_word_delay_spi\n"); > + return cs_word_delay_spi; > + } > + > + th->word_delay_ns = cpu_to_le32(max(cs_word_delay_spi, > cs_word_delay_xfer)); max() is defined in math.h. > + delay = spi_delay_to_ns(&xfer->delay, xfer); > + if (delay < 0) { > + dev_warn(&spi->dev, "Cannot convert delay\n"); > + return delay; > + } > + cs_hold = spi_delay_to_ns(&spi->cs_hold, xfer); > + if (cs_hold < 0) { > + dev_warn(&spi->dev, "Cannot convert cs_hold\n"); > + return cs_hold; > + } > + th->cs_delay_hold_ns = cpu_to_le32(delay + cs_hold); > + > + cs_inactive = spi_delay_to_ns(&spi->cs_inactive, xfer); > + if (cs_inactive < 0) { > + dev_warn(&spi->dev, "Cannot convert cs_inactive\n"); > + return cs_inactive; > + } > + cs_change_delay = spi_delay_to_ns(&xfer->cs_change_delay, xfer); > + if (cs_change_delay < 0) { > + dev_warn(&spi->dev, "Cannot convert cs_change_delay\n"); > + return cs_change_delay; > + } > + th->cs_change_delay_inactive_ns = > + cpu_to_le32(cs_inactive + cs_change_delay); > + > + return 0; > +} ... > +static int virtio_spi_transfer_one(struct spi_controller *ctrl, > + struct spi_device *spi, > + struct spi_transfer *xfer) > +{ > + struct virtio_spi_priv *priv = spi_controller_get_devdata(ctrl); > + struct virtio_spi_req *spi_req __free(kfree); This is incorrect template. It's one of the exceptions when we mix code and definitions... > + struct spi_transfer_head *th; > + struct scatterlist sg_out_head, sg_out_payload; > + struct scatterlist sg_in_result, sg_in_payload; + scatterlist.h > + struct scatterlist *sgs[4]; > + unsigned int outcnt = 0; > + unsigned int incnt = 0; > + int ret; > + spi_req = kzalloc(sizeof(*spi_req), GFP_KERNEL); ...so this should be struct virtio_spi_req *spi_req __free(kfree) = kzalloc(sizeof(*spi_req), GFP_KERNEL); (or on one line if you are okay with a 100 limit). And do not forget to include cleanup.h (__free() macro) and slab.h (kzalloc() API). > + if (!spi_req) > + return -ENOMEM; + errno.h But since you already have IS_ERR()/PTR_ERR() use, just + err.h > + init_completion(&spi_req->completion); > + > + th = &spi_req->transfer_head; > + > + /* Fill struct spi_transfer_head */ > + th->chip_select_id = spi_get_chipselect(spi, 0); > + th->bits_per_word = spi->bits_per_word; > + th->cs_change = xfer->cs_change; > + th->tx_nbits = xfer->tx_nbits; > + th->rx_nbits = xfer->rx_nbits; > + th->reserved[0] = 0; > + th->reserved[1] = 0; > + th->reserved[2] = 0; > + > + static_assert(VIRTIO_SPI_CPHA == SPI_CPHA, > + "VIRTIO_SPI_CPHA must match SPI_CPHA"); > + static_assert(VIRTIO_SPI_CPOL == SPI_CPOL, > + "VIRTIO_SPI_CPOL must match SPI_CPOL"); > + static_assert(VIRTIO_SPI_CS_HIGH == SPI_CS_HIGH, > + "VIRTIO_SPI_CS_HIGH must match SPI_CS_HIGH"); > + static_assert(VIRTIO_SPI_MODE_LSB_FIRST == SPI_LSB_FIRST, > + "VIRTIO_SPI_MODE_LSB_FIRST must match SPI_LSB_FIRST"); > + > + th->mode = cpu_to_le32(spi->mode & VIRTIO_SPI_MODE_MASK); > + if (spi->mode & SPI_LOOP) > + th->mode |= cpu_to_le32(VIRTIO_SPI_MODE_LOOP); > + > + th->freq = cpu_to_le32(xfer->speed_hz); > + > + ret = virtio_spi_set_delays(th, spi, xfer); > + if (ret) > + goto msg_done; > + > + /* Set buffers */ > + spi_req->tx_buf = xfer->tx_buf; > + spi_req->rx_buf = xfer->rx_buf; > + > + /* Prepare sending of virtio message */ > + init_completion(&spi_req->completion); > + > + sg_init_one(&sg_out_head, th, sizeof(*th)); > + sgs[outcnt] = &sg_out_head; > + outcnt++; > + > + if (spi_req->tx_buf) { > + sg_init_one(&sg_out_payload, spi_req->tx_buf, xfer->len); > + sgs[outcnt] = &sg_out_payload; > + outcnt++; > + } > + > + if (spi_req->rx_buf) { > + sg_init_one(&sg_in_payload, spi_req->rx_buf, xfer->len); > + sgs[outcnt] = &sg_in_payload; > + incnt++; > + } > + > + sg_init_one(&sg_in_result, &spi_req->result, > + sizeof(struct spi_transfer_result)); > + sgs[outcnt + incnt] = &sg_in_result; > + incnt++; > + > + ret = virtqueue_add_sgs(priv->vq, sgs, outcnt, incnt, spi_req, > + GFP_KERNEL); > + if (ret) > + goto msg_done; > + > + /* Simple implementation: There can be only one transfer in flight */ > + virtqueue_kick(priv->vq); > + > + wait_for_completion(&spi_req->completion); > + > + /* Read result from message and translate return code */ > + switch (spi_req->result.result) { > + case VIRTIO_SPI_TRANS_OK: > + break; > + case VIRTIO_SPI_PARAM_ERR: > + ret = -EINVAL; > + break; > + case VIRTIO_SPI_TRANS_ERR: > + ret = -EIO; > + break; > + default: > + ret = -EIO; > + break; > + } > + > +msg_done: > + if (ret) > + ctrl->cur_msg->status = ret; > + > + return ret; > +} ... > +static int virtio_spi_find_vqs(struct virtio_spi_priv *priv) > +{ > + struct virtqueue *vq; > + > + vq = virtio_find_single_vq(priv->vdev, virtio_spi_msg_done, "spi-rq"); > + if (IS_ERR(vq)) > + return PTR_ERR(vq); See above. > + priv->vq = vq; > + return 0; > +} ... > +static int virtio_spi_probe(struct virtio_device *vdev) > +{ > + struct virtio_spi_priv *priv; > + struct spi_controller *ctrl; > + int ret; > + u32 bus_num; > + > + ctrl = devm_spi_alloc_host(&vdev->dev, sizeof(*priv)); > + if (!ctrl) > + return -ENOMEM; > + > + priv = spi_controller_get_devdata(ctrl); > + priv->vdev = vdev; > + vdev->priv = priv; > + device_set_node(&ctrl->dev, dev_fwnode(&vdev->dev)); > + dev_set_drvdata(&vdev->dev, ctrl); > + ret = device_property_read_u32(&vdev->dev, "spi,bus-num", &bus_num); + device.h + property.h > + if (ret || bus_num > S16_MAX) + limits.h > + ctrl->bus_num = -1; > + else > + ctrl->bus_num = bus_num; But why do we need this property at all? And where is it documented in the device tree bindings? > + virtio_spi_read_config(vdev); > + > + ctrl->transfer_one = virtio_spi_transfer_one; > + > + ret = virtio_spi_find_vqs(priv); > + if (ret) > + return dev_err_probe(&vdev->dev, ret, "Cannot setup > virtqueues\n"); > + > + /* Register cleanup for virtqueues using devm */ > + ret = devm_add_action_or_reset(&vdev->dev, virtio_spi_del_vq, vdev); > + if (ret) > + return dev_err_probe(&vdev->dev, ret, "Cannot register > virtqueue cleanup\n"); > + > + /* Use devm version to register controller */ > + ret = devm_spi_register_controller(&vdev->dev, ctrl); > + if (ret) > + return dev_err_probe(&vdev->dev, ret, "Cannot register > controller\n"); > + > + return 0; > +} ... > +static struct virtio_device_id virtio_spi_id_table[] = { The type is or should be defined in mod_devicetable.h. > + { VIRTIO_ID_SPI, VIRTIO_DEV_ANY_ID }, > + {} > +}; ... > +static const struct dev_pm_ops virtio_spi_pm_ops = { The type is defined in pm.h. > + .freeze = pm_sleep_ptr(virtio_spi_freeze), > + .restore = pm_sleep_ptr(virtio_spi_restore), > +}; -- With Best Regards, Andy Shevchenko