Received: by 2002:ad5:474a:0:0:0:0:0 with SMTP id i10csp1550557imu; Sat, 5 Jan 2019 01:34:36 -0800 (PST) X-Google-Smtp-Source: ALg8bN6g+q462nXMmQlgSmQAZw+hZ1fAbvv7/EvA5C79tD8vlsiLYkVwDt9zlANtyWeYCO92AFYr X-Received: by 2002:a17:902:5601:: with SMTP id h1mr55395611pli.160.1546680876771; Sat, 05 Jan 2019 01:34:36 -0800 (PST) ARC-Seal: i=1; a=rsa-sha256; t=1546680876; cv=none; d=google.com; s=arc-20160816; b=g6dZF0/ReXdHRYTC0SfGn2eRltb+Jtvl8NKWmDVaBGNsW98Jgkfi05pI1pK87f8wV3 CI8WXtZNrJ7zPaEK1gpa3hHg+RC0EHvSiQfUgMi68WhCCzSRLvcd5ARChd8EBp0bdPhU gsI86j7S9PY61WbOcAqFPuDqfT9ZeW3WhvIwee55RjhVagFgwIy7/9DW4BZBVruq8TCC b5c30+yVYkz8Gis6LAY1TcZZhrSCTBCdvlBq4jJfYQj8BZ3TTxGtdAsa9rSTf5KKsI7A DR7ddbUFS0n6tKnPGwpEODKdrqwvnhQuRAWnRKWGUULuk56xSxjF/GkD7n/swaZ6gyU6 EM5w== ARC-Message-Signature: i=1; a=rsa-sha256; c=relaxed/relaxed; d=google.com; s=arc-20160816; h=list-id:precedence:sender:content-transfer-encoding :content-language:in-reply-to:mime-version:user-agent:date :message-id:from:references:cc:to:subject:dkim-signature; bh=VnuicCqlM9GWDJcMv+9khaahhhyaduHL3lXYfWVRhWM=; b=NDo5OIK9hoJqe1UJ9cpSaO43vTMlsHCmAvHP9cwOuFaGEbFBQLrj9n8JOM7AHST/fJ L1Fw5sfZ9kPEE0cLfgBYSbISlyUO6K8MHRNYuweRPFB89QdNImM/GEDDX4Hecs2FJn4A JD8vnCJzZdtjwHClAzLm1gjk0j9G+w+Y4DwnF/a3/IQ7hRfgzNaYe1La+IMB+vhb1QEu WAgp4ReQ1i3QhmfbDAdcvgyYBJy+qZS1eA6DiH6vTgSZZnGLEOom65Gzrs8YQsLqJYLR AEDvyS0O2LS3CfsNJo++e/yDO3RjrQsLNzXXnXPxvinWUnXIHW/kYnOwUOl88V7jGVfs fMHA== ARC-Authentication-Results: i=1; mx.google.com; dkim=fail header.i=@gmail.com header.s=20161025 header.b=nFsMagjn; spf=pass (google.com: best guess record for domain of linux-kernel-owner@vger.kernel.org designates 209.132.180.67 as permitted sender) smtp.mailfrom=linux-kernel-owner@vger.kernel.org; dmarc=fail (p=NONE sp=QUARANTINE dis=NONE) header.from=gmail.com Return-Path: Received: from vger.kernel.org (vger.kernel.org. [209.132.180.67]) by mx.google.com with ESMTP id t12si54205092plr.311.2019.01.05.01.34.08; Sat, 05 Jan 2019 01:34:36 -0800 (PST) Received-SPF: pass (google.com: best guess record for domain of linux-kernel-owner@vger.kernel.org designates 209.132.180.67 as permitted sender) client-ip=209.132.180.67; Authentication-Results: mx.google.com; dkim=fail header.i=@gmail.com header.s=20161025 header.b=nFsMagjn; spf=pass (google.com: best guess record for domain of linux-kernel-owner@vger.kernel.org designates 209.132.180.67 as permitted sender) smtp.mailfrom=linux-kernel-owner@vger.kernel.org; dmarc=fail (p=NONE sp=QUARANTINE dis=NONE) header.from=gmail.com Received: (majordomo@vger.kernel.org) by vger.kernel.org via listexpand id S1726246AbfAEJac (ORCPT + 99 others); Sat, 5 Jan 2019 04:30:32 -0500 Received: from mail-pg1-f196.google.com ([209.85.215.196]:34236 "EHLO mail-pg1-f196.google.com" rhost-flags-OK-OK-OK-OK) by vger.kernel.org with ESMTP id S1726140AbfAEJab (ORCPT ); Sat, 5 Jan 2019 04:30:31 -0500 Received: by mail-pg1-f196.google.com with SMTP id j10so18531716pga.1; Sat, 05 Jan 2019 01:30:29 -0800 (PST) DKIM-Signature: v=1; a=rsa-sha256; c=relaxed/relaxed; d=gmail.com; s=20161025; h=sender:subject:to:cc:references:from:message-id:date:user-agent :mime-version:in-reply-to:content-language:content-transfer-encoding; bh=VnuicCqlM9GWDJcMv+9khaahhhyaduHL3lXYfWVRhWM=; b=nFsMagjne/sraBhzy55IZQNNK046IWNDlw2Lc7k2vKT5iZcGmJ/pwaTIP39MCnGvFe 0MxYEp6jAgyn/i9o+WmEkmR2AoVeFa0rolplCyJQMQY7S1sTNz4TOgOTog/+CVA5bUHI M9fb/iTDMUTfvlAF3KAEDFQdBApZToM3wzhtgBMR8zg+CaMtXJ8RFHUHGtooV2e2oqZV zGaOHZPnRD/ffo6p3tuF7D1UqoGoU6RJvVaqbgMPIBi/8SxIOVWnMkx3rJb44f70BzOT jqTLq751Djkq8frucvjO5oOWqQkE03fudGZJ1O0gYHeprJpGIYqsY9JUxcizUE4wOGOn OFfQ== X-Google-DKIM-Signature: v=1; a=rsa-sha256; c=relaxed/relaxed; d=1e100.net; s=20161025; h=x-gm-message-state:sender:subject:to:cc:references:from:message-id :date:user-agent:mime-version:in-reply-to:content-language :content-transfer-encoding; bh=VnuicCqlM9GWDJcMv+9khaahhhyaduHL3lXYfWVRhWM=; b=Ydt3N/XuvgI2NHznhHiRLc6k0T6MPfoYzw8zNbgpfydVgr1GtBteb71ig8V2SQzSn3 62ggHINwWB8+LzjqI8lzstaWy3tfgRiRjtU/ru4pPcX/CMe/1n37qsu2N/aL6NTBNgku E9icCoSYb8BPtqNUKiRbYGSc4fJa8pqFFcMjO1m4UM7uCNo0ILaNZIOa1RkX3NtGZU9U cwiCvG+2Lo8XyFKUFECMxZRdLspqCJgv55MmSJ0887PmbAZRANUxmAaGk8NiQF+7DCjw WzizPlTyrEZNPBqjNAxRIM2Zu9yKS5LP/2Fcd3+HQOA0rWAzXm4y4TJDry0XdcIrn3fJ tMsw== X-Gm-Message-State: AA+aEWazRNCun/1omEb+aKf6nUdzYP5vhk4xW4RZ7V3MwADS/er47t+j PJSp3SLBpaiWJqkrNNDQ+GA= X-Received: by 2002:a62:33c1:: with SMTP id z184mr55129621pfz.104.1546680628639; Sat, 05 Jan 2019 01:30:28 -0800 (PST) Received: from [192.168.1.140] (i121-116-192-145.s41.a020.ap.plala.or.jp. [121.116.192.145]) by smtp.gmail.com with ESMTPSA id n68sm103390551pfb.62.2019.01.05.01.30.24 (version=TLS1_2 cipher=ECDHE-RSA-AES128-GCM-SHA256 bits=128/128); Sat, 05 Jan 2019 01:30:27 -0800 (PST) Subject: Re: [PATCH lora-next 3/5] net: lora: sx130x: Add PicoCell serdev driver To: =?UTF-8?Q?Andreas_F=c3=a4rber?= , linux-lpwan@lists.infradead.org, linux-serial@vger.kernel.org Cc: Rob Herring , devicetree@vger.kernel.org, netdev@vger.kernel.org, linux-usb@vger.kernel.org, linux-kernel@vger.kernel.org, Johan Hovold , "David S. Miller" References: <20190104112131.14451-1-afaerber@suse.de> <20190104112131.14451-4-afaerber@suse.de> From: Ben Whitten Message-ID: <0e1aed6e-d076-d5b9-4d05-2a3fb3f5e8af@gmail.com> Date: Sat, 5 Jan 2019 18:18:22 +0900 User-Agent: Mozilla/5.0 (X11; Linux x86_64; rv:60.0) Gecko/20100101 Thunderbird/60.2.1 MIME-Version: 1.0 In-Reply-To: <20190104112131.14451-4-afaerber@suse.de> Content-Type: text/plain; charset=utf-8; format=flowed Content-Language: en-GB Content-Transfer-Encoding: 8bit Sender: linux-kernel-owner@vger.kernel.org Precedence: bulk List-ID: X-Mailing-List: linux-kernel@vger.kernel.org Hi, On 04/01/2019 20:21, Andreas Färber wrote: > The picoGW reference MCU firmware implements a USB CDC or UART interface > with a set of serial commands. It can be found on multiple mPCIe cards > as well as USB adapters. > > https://github.com/Lora-net/picoGW_mcu > > That MCU design superseded earlier attempts of using FTDI chips (MPSSE) > for controlling the SPI based chip directly from the host. > > This commit adds a serdev driver implementing another regmap_bus to > abstract the register access and reuses our existing regmap driver on top. > Thereby we have an early proof of concept that we can drive both types > of modules/cards with a single driver core! > > It assumes there is only one SX130x (with its radios) accessible, whereas > some new cards reportedly also have an SX127x on-board. So the DT/driver > design may need to be reconsidered once such a card or documentation > becomes available. > It also assumes SX1255/1258 are fully compatible with "semtech,sx1257". > > Currently there's still some bugs to be investigated, with communication > stalling on one device and another reading the radio versions wrong > (07 / 1f instead of 21, also observed on spi once). > > Signed-off-by: Andreas Färber > --- > drivers/net/lora/Makefile | 2 + > drivers/net/lora/sx130x_picogw.c | 466 +++++++++++++++++++++++++++++++++++++++ We are missing a Kconfig select or depend, compilation fails if CONFIG_SERIAL_DEV_BUS is not selected. > 2 files changed, 468 insertions(+) > create mode 100644 drivers/net/lora/sx130x_picogw.c > > diff --git a/drivers/net/lora/Makefile b/drivers/net/lora/Makefile > index c6a0410f80ce..5fef38abf5ed 100644 > --- a/drivers/net/lora/Makefile > +++ b/drivers/net/lora/Makefile > @@ -25,6 +25,8 @@ lora-sx127x-y := sx127x.o > obj-$(CONFIG_LORA_SX130X) += lora-sx130x.o > lora-sx130x-y := sx130x.o > lora-sx130x-y += sx130x_radio.o > +obj-$(CONFIG_LORA_SX130X) += lora-sx130x-picogw.o > +lora-sx130x-picogw-y := sx130x_picogw.o > > obj-$(CONFIG_LORA_USI) += lora-usi.o > lora-usi-y := usi.o > diff --git a/drivers/net/lora/sx130x_picogw.c b/drivers/net/lora/sx130x_picogw.c > new file mode 100644 > index 000000000000..577f9fb71d46 > --- /dev/null > +++ b/drivers/net/lora/sx130x_picogw.c > @@ -0,0 +1,466 @@ > +// SPDX-License-Identifier: GPL-2.0-or-later > +/* > + * Semtech SX1301/1308 PicoCell gateway serial MCU interface > + * > + * Copyright (c) 2018-2019 Andreas Färber > + */ > +#include > +#include > +#include > +#include > +#include > +#include > +#include > +#include > + > +struct picogw_device { > + struct serdev_device *serdev; > + > + u8 rx_buf[1024]; > + int rx_len; > + > + struct completion answer_comp; > + struct completion answer_read_comp; > +}; > + > +static inline struct picogw_device *picogw_get_drvdata(struct serdev_device *sdev) > +{ > + return sx130x_get_drvdata(&sdev->dev); > +} > + > +static bool picogw_valid_cmd(char ch) > +{ > + switch (ch) { > + case 'k': /* invalid command error */ > + case 'r': > + case 'w': > + case 'l': > + return true; > + default: > + return false; > + } > +} > + > +static int picogw_send_cmd(struct picogw_device *picodev, char cmd, > + u8 addr, const void *data, u16 data_len) > +{ > + struct serdev_device *sdev = picodev->serdev; > + u8 buf[4]; > + int i, ret; > + > + buf[0] = cmd; > + buf[1] = (data_len >> 8) & 0xff; > + buf[2] = (data_len >> 0) & 0xff; > + buf[3] = addr; > + > + dev_dbg(&sdev->dev, "%s: %c %02x %02x %02x\n", __func__, buf[0], > + (unsigned int)buf[1], (unsigned int)buf[2], (unsigned int)buf[3]); > + for (i = 0; i < data_len; i++) { > + dev_dbg(&sdev->dev, "%s: data %02x\n", __func__, (unsigned int)((const u8*)data)[i]); > + } > + > + ret = serdev_device_write_buf(sdev, buf, 4); > + if (ret < 0) > + return ret; > + if (ret != 4) > + return -EIO; > + > + if (data_len) { > + ret = serdev_device_write_buf(sdev, data, data_len); > + if (ret < 0) > + return ret; > + if (ret != data_len) > + return -EIO; > + } > + > + return 0; > +} > + > +static int picogw_recv_answer(struct picogw_device *picodev, > + char *cmd, bool *ack, void *buf, int buf_len, > + unsigned long timeout) > +{ > + int len; > + > + timeout = wait_for_completion_timeout(&picodev->answer_comp, timeout); > + if (!timeout) > + return -ETIMEDOUT; > + > + if (cmd) > + *cmd = picodev->rx_buf[0]; > + > + if (ack) > + *ack = (picodev->rx_buf[3] == 1); > + > + len = min(picodev->rx_len - 4, buf_len); > + if (buf) > + memcpy(buf, picodev->rx_buf + 4, len); > + > + reinit_completion(&picodev->answer_comp); > + complete(&picodev->answer_read_comp); > + > + return len; > +} > + > +static int picogw_reg_read(struct picogw_device *picodev, u8 addr, u8 *val, unsigned long timeout) > +{ > + const u8 dummy = 0; > + char cmd; > + bool ack; > + int ret; > + > + ret = picogw_send_cmd(picodev, 'r', addr, &dummy, false ? 1 : 0); > + if (ret) > + return ret; > + > + ret = picogw_recv_answer(picodev, &cmd, &ack, val, 1, timeout); > + if (ret < 0) > + return ret; > + if (cmd != 'r') > + return -EIO; > + if (!ack || ret != 1) > + return -EIO; > + > + return 0; > +} > + > +static int picogw_reg_write(struct picogw_device *picodev, u8 addr, u8 val, unsigned long timeout) > +{ > + char cmd; > + bool ack; > + int ret; > + > + ret = picogw_send_cmd(picodev, 'w', addr, &val, 1); > + if (ret) > + return ret; > + > + ret = picogw_recv_answer(picodev, &cmd, &ack, NULL, 0, timeout); > + if (ret < 0) > + return ret; > + if (cmd != 'w') > + return -EIO; > + if (!ack || ret != 0) > + return -EIO; > + > + return 0; > +} > + > +static int picogw_mcu_fw_check(struct picogw_device *picodev, > + u32 fw_version, u8 *id, unsigned long timeout) > +{ > + char cmd; > + bool ack; > + int ret; > + > + fw_version = cpu_to_be32(fw_version); > + ret = picogw_send_cmd(picodev, 'l', 0, &fw_version, sizeof(fw_version)); > + if (ret) > + return ret; > + > + ret = picogw_recv_answer(picodev, &cmd, &ack, id, id ? 8 : 0, timeout); > + if (ret < 0) > + return ret; > + if (cmd != 'l') > + return -EIO; > + if (id && ret != 8) > + return -EIO; > + > + return ack ? 0 : -ENOTSUPP; > +} > + > +static int picogw_regmap_gather_write(void *context, > + const void *reg_buf, size_t reg_size, const void *val_buf, size_t val_size) > +{ > + struct picogw_device *picodev = context; > + const u8 *addr_buf = reg_buf; > + const u8 *val = val_buf; > + u8 addr; > + int ret; > + > + //dev_dbg(&picodev->serdev->dev, "%s: 0x%x (reg_size %zu) 0x%x (val_size %zu)\n", > + // __func__, (unsigned int)addr_buf[0], reg_size, (unsigned int)val[0], val_size); > + > + if (reg_size != 1 || val_size > 0xffff) > + return -EINVAL; > + > + addr = addr_buf[0] & ~BIT(7); > + if (val_size == 1) { > + ret = picogw_reg_write(picodev, addr, val[0], HZ); > + if (ret) > + return ret; > + return 0; > + } else { > + /* TODO burst mode */ > + dev_err(&picodev->serdev->dev, "burst mode write not yet implemented\n"); > + return -ENOTSUPP; > + } > +} > + > +static int picogw_regmap_write(void *context, > + const void *data_buf, size_t count) > +{ > + const u8 *data = data_buf; > + if (count < 1) > + return -EINVAL; > + > + return picogw_regmap_gather_write(context, data, 1, data + 1, count - 1); > +} > + > +static int picogw_regmap_read(void *context, > + const void *reg_buf, size_t reg_size, void *val_buf, size_t val_size) > +{ > + struct picogw_device *picodev = context; > + const u8 *addr_buf = reg_buf; > + u8 addr; > + int ret; > + > + //dev_dbg(&picodev->serdev->dev, "%s: 0x%x (reg_size %zu) (val_size %zu)\n", __func__, (unsigned int)addr_buf[0], reg_size, val_size); > + > + if (reg_size != 1 || val_size > 0xffff) > + return -EINVAL; > + > + addr = addr_buf[0] & ~BIT(7); > + if (val_size == 1) { > + ret = picogw_reg_read(picodev, addr, val_buf, HZ); > + if (ret) > + return ret; > + return 0; > + } else { > + /* TODO burst mode */ > + dev_err(&picodev->serdev->dev, "burst mode read not yet implemented\n"); > + return -ENOTSUPP; > + } > +} > + > +static const struct regmap_bus picogw_regmap_bus = { > + .write = picogw_regmap_write, > + .gather_write = picogw_regmap_gather_write, > + .read = picogw_regmap_read, > + > + .max_raw_write = 0xffff, > + .max_raw_read = 0xffff, > +}; > + > +static int picogw_handle_answer(struct picogw_device *picodev) > +{ > + struct device *dev = &picodev->serdev->dev; > + unsigned int data_len = ((u16)picodev->rx_buf[1] << 8) | picodev->rx_buf[2]; > + int cmd_len = 4 + data_len; > + int i, ret; > + > + if (picodev->rx_len < 4) > + return 0; > + > + if (cmd_len > sizeof(picodev->rx_buf)) { > + dev_warn(dev, "answer too long (%u)\n", data_len); > + return 0; > + } > + > + if (picodev->rx_len < cmd_len) { > + dev_dbg(dev, "got %u, need %u bytes\n", picodev->rx_len, cmd_len); > + return 0; > + } > + > + dev_dbg(dev, "Answer %c =%u %s (%u)\n", picodev->rx_buf[0], > + (unsigned int)picodev->rx_buf[3], > + (picodev->rx_buf[3] == 1) ? "OK" : "K0", > + data_len); > + for (i = 0; i < data_len; i++) { > + //dev_dbg(dev, "%s: %02x\n", __func__, (unsigned int)picodev->rx_buf[4 + i]); > + } > + > + complete(&picodev->answer_comp); > + ret = wait_for_completion_interruptible_timeout(&picodev->answer_read_comp, HZ / 2); > + if (ret < 0) > + return 0; > + > + reinit_completion(&picodev->answer_read_comp); > + > + return cmd_len; > +} > + > +static int picogw_receive_buf(struct serdev_device *sdev, const u8 *data, size_t count) > +{ > + struct picogw_device *picodev = picogw_get_drvdata(sdev); > + size_t i; > + int len = 0; > + > + dev_dbg(&sdev->dev, "Receive (%zu)\n", count); > + > + if (completion_done(&picodev->answer_comp)) { > + dev_info(&sdev->dev, "RX waiting on completion\n"); > + return 0; > + } > + if (picodev->rx_len == sizeof(picodev->rx_buf)) { > + dev_warn(&sdev->dev, "RX buffer full\n"); > + return 0; > + } > + > + i = min(count, sizeof(picodev->rx_buf) - picodev->rx_len); > + if (i > 0) { > + memcpy(&picodev->rx_buf[picodev->rx_len], data, i); > + picodev->rx_len += i; > + len += i; > + } > + > + while (picodev->rx_len > 0) { > + /* search for valid beginning */ > + for (i = 0; i < picodev->rx_len; i++) { > + if (picogw_valid_cmd(picodev->rx_buf[i])) > + break; > + } > + if (i > 0) { > + dev_dbg(&sdev->dev, "skipping %zu bytes of garbage\n", i); > + if (i < picodev->rx_len) { > + memmove(picodev->rx_buf, &picodev->rx_buf[i], picodev->rx_len - i); > + picodev->rx_len -= i; > + } else > + picodev->rx_len = 0; > + } > + > + i = picogw_handle_answer(picodev); > + if (i == 0) > + break; > + > + if (i % 64 == 0) { > + dev_info(&sdev->dev, "skipping padding byte\n"); > + i++; > + } > + if (picodev->rx_len > i) > + memmove(picodev->rx_buf, &picodev->rx_buf[i], picodev->rx_len - i); > + if (picodev->rx_len >= i) > + picodev->rx_len -= i; > + } > + > + return len; > +} > + > +static const struct serdev_device_ops picogw_serdev_client_ops = { > + .receive_buf = picogw_receive_buf, > + .write_wakeup = serdev_device_write_wakeup, > +}; > + > +static int picogw_serdev_probe(struct serdev_device *sdev) > +{ > + struct picogw_device *picodev; > + struct regmap *regmap; > + u32 fw_version = 0x010a0006; Worth moving the version to a define? > + u8 mac[8];here > + int ret; > + > + //dev_info(&sdev->dev, "Probing\n"); > + > + picodev = devm_kzalloc(&sdev->dev, sizeof(*picodev), GFP_KERNEL); > + if (!picodev) > + return -ENOMEM; > + > + picodev->serdev = sdev; > + init_completion(&picodev->answer_comp); > + init_completion(&picodev->answer_read_comp); > + > + ret = serdev_device_open(sdev); > + if (ret) { > + dev_err(&sdev->dev, "Failed to open (%d)\n", ret); > + return ret; > + } > + > + serdev_device_set_baudrate(sdev, 115200); > + serdev_device_set_parity(sdev, SERDEV_PARITY_NONE); > + serdev_device_set_flow_control(sdev, true); > + > + regmap = devm_regmap_init(&sdev->dev, &picogw_regmap_bus, picodev, &sx130x_regmap_config); > + if (IS_ERR(regmap)) { > + ret = PTR_ERR(regmap); > + dev_err(&sdev->dev, "error initializing regmap (%d)\n", ret); > + serdev_device_close(sdev); > + return ret; > + } > + > + ret = sx130x_early_probe(regmap, NULL); > + if (ret) { > + serdev_device_close(sdev); > + return ret; > + } > + > + sx130x_set_drvdata(&sdev->dev, picodev); > + serdev_device_set_client_ops(sdev, &picogw_serdev_client_ops); > + > + //msleep(1000); > + ret = picogw_mcu_fw_check(picodev, fw_version, mac, HZ); > + if (!ret || ret == -ENOTSUPP) > + dev_info(&sdev->dev, "ID = %02x%02x%02x%02x%02x%02x%02x%02x\n", > + (unsigned int)mac[0], > + (unsigned int)mac[1], > + (unsigned int)mac[2], > + (unsigned int)mac[3], > + (unsigned int)mac[4], > + (unsigned int)mac[5], > + (unsigned int)mac[6], > + (unsigned int)mac[7]); > + while (ret == -ENOTSUPP && ((fw_version & 0xff) > 4)) { > + ret = picogw_mcu_fw_check(picodev, --fw_version, NULL, HZ); > + } > + if (ret == -ENOTSUPP) { > + dev_warn(&sdev->dev, "firmware check failed (%08x)\n", fw_version); > + } else { > + dev_err(&sdev->dev, "ID retrieval failed (%d)\n", ret); > + serdev_device_close(sdev); > + return ret; > + } > + > + ret = sx130x_probe(&sdev->dev); > + if (ret) { > + serdev_device_close(sdev); > + return ret; > + } > + > + //dev_info(&sdev->dev, "Done.\n"); > + > + return 0; > +} > + > +static void picogw_serdev_remove(struct serdev_device *sdev) > +{ > + sx130x_remove(&sdev->dev); > + > + serdev_device_close(sdev); > + > + //dev_info(&sdev->dev, "Removed\n"); > +} > + > +static const struct of_device_id picogw_serdev_of_match[] = { > + { .compatible = "semtech,lora-picocell" }, lora-picocell or lora-picogw... picocell has mobile connotations, should we match the module name? > + {} > +}; > +MODULE_DEVICE_TABLE(of, picogw_serdev_of_match); > + > +static struct serdev_device_driver picogw_serdev_driver = { > + .probe = picogw_serdev_probe, > + .remove = picogw_serdev_remove, > + .driver = { > + .name = "lora-picogw", > + .of_match_table = picogw_serdev_of_match, > + }, > +}; > + > +static int __init picogw_serdev_init(void) > +{ > + int ret; > + > + ret = serdev_device_driver_register(&picogw_serdev_driver); > + if (ret) { > + pr_err("serdev_device_driver_register failed (%d)", ret); > + return ret; > + } > + > + return 0; > +} > +module_init(picogw_serdev_init); > + > +static void __exit picogw_serdev_exit(void) > +{ > + serdev_device_driver_unregister(&picogw_serdev_driver); > +} > +module_exit(picogw_serdev_exit); > + > +MODULE_LICENSE("GPL"); > Thanks, Ben Whitten