From: Aaron Ma <aaron.ma@canonical.com>
To: Even Xu <even.xu@intel.com>,
jikos@kernel.org, bentiss@kernel.org, corbet@lwn.net,
bagasdotme@gmail.com
Cc: linux-input@vger.kernel.org, linux-kernel@vger.kernel.org,
linux-doc@vger.kernel.org, Xinpeng Sun <xinpeng.sun@intel.com>,
Rui Zhang <rui1.zhang@intel.com>,
Srinivas Pandruvada <srinivas.pandruvada@linux.intel.com>
Subject: Re: [PATCH v2 04/22] HID: intel-thc-hid: intel-thc: Add THC PIO operation APIs
Date: Wed, 4 Dec 2024 13:37:25 +0800 [thread overview]
Message-ID: <157e94a2-bc03-42c1-8c4f-d6042032a500@canonical.com> (raw)
In-Reply-To: <20241114053416.4085715-5-even.xu@intel.com>
On 11/14/24 1:33 PM, Even Xu wrote:
> From: Xinpeng Sun <xinpeng.sun@intel.com>
>
> THC PIO (programmed I/O) operations are very similar with general
> SPI/I2C read/write operation to access external slave device on the bus
> through internal FIFO.
>
> THC PIO operations are split into 4 steps:
> 1. prepare: configure hardware with correct opcode, slave address, and
> fill the PIO FIFO
> 2. start: set start bit to issue a bus send/receive
> 3. wait: wait for bus sending/receiving completion
> 4. complete: check send/receive data in FIFO and return
>
> Co-developed-by: Even Xu <even.xu@intel.com>
> Signed-off-by: Even Xu <even.xu@intel.com>
> Signed-off-by: Xinpeng Sun <xinpeng.sun@intel.com>
> Tested-by: Rui Zhang <rui1.zhang@intel.com>
> Reviewed-by: Srinivas Pandruvada <srinivas.pandruvada@linux.intel.com>
> ---
> .../intel-thc-hid/intel-thc/intel-thc-dev.c | 245 ++++++++++++++++++
> .../intel-thc-hid/intel-thc/intel-thc-dev.h | 24 ++
> .../intel-thc-hid/intel-thc/intel-thc-hw.h | 23 ++
> 3 files changed, 292 insertions(+)
>
> diff --git a/drivers/hid/intel-thc-hid/intel-thc/intel-thc-dev.c b/drivers/hid/intel-thc-hid/intel-thc/intel-thc-dev.c
> index d77603e4c01e..0571b8270716 100644
> --- a/drivers/hid/intel-thc-hid/intel-thc/intel-thc-dev.c
> +++ b/drivers/hid/intel-thc-hid/intel-thc/intel-thc-dev.c
> @@ -1,9 +1,11 @@
> /* SPDX-License-Identifier: GPL-2.0 */
> /* Copyright (c) 2024 Intel Corporation */
>
> +#include <linux/bitfield.h>
> #include <linux/regmap.h>
>
> #include "intel-thc-dev.h"
> +#include "intel-thc-hw.h"
>
> static int thc_regmap_read(void *context, unsigned int reg,
> unsigned int *val)
> @@ -76,10 +78,253 @@ struct thc_device *thc_dev_init(struct device *device, void __iomem *mem_addr)
> return ERR_PTR(ret);
> }
>
> + mutex_init(&thc_dev->thc_bus_lock);
> +
> return thc_dev;
> }
> EXPORT_SYMBOL_NS_GPL(thc_dev_init, INTEL_THC);
Since the commit in 6.13-rc1+ change the symbol namespace:
cdd30ebb1b9f ("module: Convert symbol namespace to string literal")
This should be changed to
EXPORT_SYMBOL_NS_GPL(thc_dev_init, "INTEL_THC");
There are many of them in this driver, better to define a helper macro in header file.
And group them consistently.
Aaron
>
> +static int prepare_pio(const struct thc_device *dev, const u8 pio_op,
> + const u32 address, const u32 size)
> +{
> + u32 sts, ctrl, addr, mask;
> +
> + regmap_read(dev->thc_regmap, THC_M_PRT_SW_SEQ_STS_OFFSET, &sts);
> +
> + /* Check if THC previous PIO still in progress */
> + if (sts & THC_M_PRT_SW_SEQ_STS_THC_SS_CIP) {
> + dev_err_once(dev->dev, "THC PIO is still busy!\n");
> + return -EBUSY;
> + }
> +
> + /* Clear error bit and complete bit in state register */
> + sts |= THC_M_PRT_SW_SEQ_STS_THC_SS_ERR |
> + THC_M_PRT_SW_SEQ_STS_TSSDONE;
> + regmap_write(dev->thc_regmap, THC_M_PRT_SW_SEQ_STS_OFFSET, sts);
> +
> + /* Set PIO data size, opcode and interrupt capability */
> + ctrl = FIELD_PREP(THC_M_PRT_SW_SEQ_CNTRL_THC_SS_BC, size) |
> + FIELD_PREP(THC_M_PRT_SW_SEQ_CNTRL_THC_SS_CMD, pio_op);
> + if (dev->pio_int_supported)
> + ctrl |= THC_M_PRT_SW_SEQ_CNTRL_THC_SS_CD_IE;
> +
> + mask = THC_M_PRT_SW_SEQ_CNTRL_THC_SS_BC |
> + THC_M_PRT_SW_SEQ_CNTRL_THC_SS_CMD |
> + THC_M_PRT_SW_SEQ_CNTRL_THC_SS_CD_IE;
> + regmap_write_bits(dev->thc_regmap,
> + THC_M_PRT_SW_SEQ_CNTRL_OFFSET, mask, ctrl);
> +
> + /* Set PIO target address */
> + addr = FIELD_PREP(THC_M_PRT_SW_SEQ_DATA0_ADDR_THC_SW_SEQ_DATA0_ADDR, address);
> + mask = THC_M_PRT_SW_SEQ_DATA0_ADDR_THC_SW_SEQ_DATA0_ADDR;
> + regmap_write_bits(dev->thc_regmap,
> + THC_M_PRT_SW_SEQ_DATA0_ADDR_OFFSET, mask, addr);
> + return 0;
> +}
> +
> +static void pio_start(const struct thc_device *dev,
> + u32 size_in_bytes, const u32 *buffer)
> +{
> + if (size_in_bytes && buffer)
> + regmap_bulk_write(dev->thc_regmap, THC_M_PRT_SW_SEQ_DATA1_OFFSET,
> + buffer, size_in_bytes / sizeof(u32));
> +
> + /* Enable Start bit */
> + regmap_write_bits(dev->thc_regmap,
> + THC_M_PRT_SW_SEQ_CNTRL_OFFSET,
> + THC_M_PRT_SW_SEQ_CNTRL_TSSGO,
> + THC_M_PRT_SW_SEQ_CNTRL_TSSGO);
> +}
> +
> +static int pio_complete(const struct thc_device *dev,
> + u32 *buffer, u32 *size)
> +{
> + u32 sts, ctrl;
> +
> + regmap_read(dev->thc_regmap, THC_M_PRT_SW_SEQ_STS_OFFSET, &sts);
> + if (sts & THC_M_PRT_SW_SEQ_STS_THC_SS_ERR) {
> + dev_err_once(dev->dev, "PIO operation error\n");
> + return -EBUSY;
> + }
> +
> + if (buffer && size) {
> + regmap_read(dev->thc_regmap, THC_M_PRT_SW_SEQ_CNTRL_OFFSET, &ctrl);
> + *size = FIELD_GET(THC_M_PRT_SW_SEQ_CNTRL_THC_SS_BC, ctrl);
> +
> + regmap_bulk_read(dev->thc_regmap, THC_M_PRT_SW_SEQ_DATA1_OFFSET,
> + buffer, *size / sizeof(u32));
> + }
> +
> + sts |= THC_M_PRT_SW_SEQ_STS_THC_SS_ERR | THC_M_PRT_SW_SEQ_STS_TSSDONE;
> + regmap_write(dev->thc_regmap, THC_M_PRT_SW_SEQ_STS_OFFSET, sts);
> + return 0;
> +}
> +
> +static int pio_wait(const struct thc_device *dev)
> +{
> + u32 sts = 0;
> + int ret;
> +
> + ret = regmap_read_poll_timeout(dev->thc_regmap, THC_M_PRT_SW_SEQ_STS_OFFSET, sts,
> + !(sts & THC_M_PRT_SW_SEQ_STS_THC_SS_CIP ||
> + !(sts & THC_M_PRT_SW_SEQ_STS_TSSDONE)),
> + THC_REGMAP_POLLING_INTERVAL_US, THC_PIO_DONE_TIMEOUT_US);
> + if (ret)
> + dev_err_once(dev->dev, "Timeout while polling PIO operation done\n");
> +
> + return ret;
> +}
> +
> +/**
> + * thc_tic_pio_read - Read data from touch device by PIO
> + *
> + * @dev: The pointer of THC private device context
> + * @address: Slave address for the PIO operation
> + * @size: Expected read data size
> + * @actual_size: The pointer of the actual data size read from touch device
> + * @buffer: The pointer of data buffer to store the data read from touch device
> + *
> + * Return: 0 on success, other error codes on failed.
> + */
> +int thc_tic_pio_read(struct thc_device *dev, const u32 address,
> + const u32 size, u32 *actual_size, u32 *buffer)
> +{
> + u8 opcode;
> + int ret;
> +
> + if (size <= 0 || !actual_size || !buffer) {
> + dev_err(dev->dev, "Invalid input parameters, size %u, actual_size %p, buffer %p\n",
> + size, actual_size, buffer);
> + return -EINVAL;
> + }
> +
> + if (mutex_lock_interruptible(&dev->thc_bus_lock))
> + return -EINTR;
> +
> + opcode = (dev->port_type == THC_PORT_TYPE_SPI) ?
> + THC_PIO_OP_SPI_TIC_READ : THC_PIO_OP_I2C_TIC_READ;
> +
> + ret = prepare_pio(dev, opcode, address, size);
> + if (ret < 0)
> + goto end;
> +
> + pio_start(dev, 0, NULL);
> +
> + ret = pio_wait(dev);
> + if (ret < 0)
> + goto end;
> +
> + ret = pio_complete(dev, buffer, actual_size);
> +
> +end:
> + mutex_unlock(&dev->thc_bus_lock);
> + return ret;
> +}
> +EXPORT_SYMBOL_NS_GPL(thc_tic_pio_read, INTEL_THC);
> +
> +/**
> + * thc_tic_pio_write - Write data to touch device by PIO
> + *
> + * @dev: The pointer of THC private device context
> + * @address: Slave address for the PIO operation
> + * @size: PIO write data size
> + * @buffer: The pointer of the write data buffer
> + *
> + * Return: 0 on success, other error codes on failed.
> + */
> +int thc_tic_pio_write(struct thc_device *dev, const u32 address,
> + const u32 size, const u32 *buffer)
> +{
> + u8 opcode;
> + int ret;
> +
> + if (size <= 0 || !buffer) {
> + dev_err(dev->dev, "Invalid input parameters, size %u, buffer %p\n",
> + size, buffer);
> + return -EINVAL;
> + }
> +
> + if (mutex_lock_interruptible(&dev->thc_bus_lock))
> + return -EINTR;
> +
> + opcode = (dev->port_type == THC_PORT_TYPE_SPI) ?
> + THC_PIO_OP_SPI_TIC_WRITE : THC_PIO_OP_I2C_TIC_WRITE;
> +
> + ret = prepare_pio(dev, opcode, address, size);
> + if (ret < 0)
> + goto end;
> +
> + pio_start(dev, size, buffer);
> +
> + ret = pio_wait(dev);
> + if (ret < 0)
> + goto end;
> +
> + ret = pio_complete(dev, NULL, NULL);
> +
> +end:
> + mutex_unlock(&dev->thc_bus_lock);
> + return ret;
> +}
> +EXPORT_SYMBOL_NS_GPL(thc_tic_pio_write, INTEL_THC);
> +
> +/**
> + * thc_tic_pio_write_and_read - Write data followed by read data by PIO
> + *
> + * @dev: The pointer of THC private device context
> + * @address: Slave address for the PIO operation
> + * @write_size: PIO write data size
> + * @write_buffer: The pointer of the write data buffer
> + * @read_size: Expected PIO read data size
> + * @actual_size: The pointer of the actual read data size
> + * @read_buffer: The pointer of PIO read data buffer
> + *
> + * Return: 0 on success, other error codes on failed.
> + */
> +int thc_tic_pio_write_and_read(struct thc_device *dev, const u32 address,
> + const u32 write_size, const u32 *write_buffer,
> + const u32 read_size, u32 *actual_size, u32 *read_buffer)
> +{
> + u32 i2c_ctrl, mask;
> + int ret;
> +
> + if (dev->port_type == THC_PORT_TYPE_SPI) {
> + dev_err(dev->dev, "SPI port type doesn't support pio write and read!");
> + return -EINVAL;
> + }
> +
> + if (mutex_lock_interruptible(&dev->thc_bus_lock))
> + return -EINTR;
> +
> + /* Config i2c PIO write and read sequence */
> + i2c_ctrl = FIELD_PREP(THC_M_PRT_SW_SEQ_I2C_WR_CNTRL_THC_PIO_I2C_WBC, write_size);
> + mask = THC_M_PRT_SW_SEQ_I2C_WR_CNTRL_THC_PIO_I2C_WBC;
> +
> + regmap_write_bits(dev->thc_regmap, THC_M_PRT_SW_SEQ_I2C_WR_CNTRL_OFFSET,
> + mask, i2c_ctrl);
> +
> + regmap_write_bits(dev->thc_regmap, THC_M_PRT_SW_SEQ_I2C_WR_CNTRL_OFFSET,
> + THC_M_PRT_SW_SEQ_I2C_WR_CNTRL_THC_I2C_RW_PIO_EN,
> + THC_M_PRT_SW_SEQ_I2C_WR_CNTRL_THC_I2C_RW_PIO_EN);
> +
> + ret = prepare_pio(dev, THC_PIO_OP_I2C_TIC_WRITE_AND_READ, address, read_size);
> + if (ret < 0)
> + goto end;
> +
> + pio_start(dev, write_size, write_buffer);
> +
> + ret = pio_wait(dev);
> + if (ret < 0)
> + goto end;
> +
> + ret = pio_complete(dev, read_buffer, actual_size);
> +
> +end:
> + mutex_unlock(&dev->thc_bus_lock);
> + return ret;
> +}
> +EXPORT_SYMBOL_NS_GPL(thc_tic_pio_write_and_read, INTEL_THC);
> +
> MODULE_AUTHOR("Xinpeng Sun <xinpeng.sun@intel.com>");
> MODULE_AUTHOR("Even Xu <even.xu@intel.com>");
>
> diff --git a/drivers/hid/intel-thc-hid/intel-thc/intel-thc-dev.h b/drivers/hid/intel-thc-hid/intel-thc/intel-thc-dev.h
> index 7060f0a36cbd..88a9f606a6a9 100644
> --- a/drivers/hid/intel-thc-hid/intel-thc/intel-thc-dev.h
> +++ b/drivers/hid/intel-thc-hid/intel-thc/intel-thc-dev.h
> @@ -5,21 +5,45 @@
> #define _INTEL_THC_DEV_H_
>
> #include <linux/cdev.h>
> +#include <linux/mutex.h>
>
> #define THC_REGMAP_COMMON_OFFSET 0x10
> #define THC_REGMAP_MMIO_OFFSET 0x1000
>
> +/*
> + * THC Port type
> + * @THC_PORT_TYPE_SPI: This port is used for HIDSPI
> + * @THC_PORT_TYPE_I2C: This port is used for HIDI2C
> + */
> +enum thc_port_type {
> + THC_PORT_TYPE_SPI = 0,
> + THC_PORT_TYPE_I2C = 1,
> +};
> +
> /**
> * struct thc_device - THC private device struct
> * @thc_regmap: MMIO regmap structure for accessing THC registers
> * @mmio_addr: MMIO registers address
> + * @thc_bus_lock: mutex locker for THC config
> + * @port_type: port type of THC port instance
> + * @pio_int_supported: PIO interrupt supported flag
> */
> struct thc_device {
> struct device *dev;
> struct regmap *thc_regmap;
> void __iomem *mmio_addr;
> + struct mutex thc_bus_lock;
> + enum thc_port_type port_type;
> + bool pio_int_supported;
> };
>
> struct thc_device *thc_dev_init(struct device *device, void __iomem *mem_addr);
> +int thc_tic_pio_read(struct thc_device *dev, const u32 address,
> + const u32 size, u32 *actual_size, u32 *buffer);
> +int thc_tic_pio_write(struct thc_device *dev, const u32 address,
> + const u32 size, const u32 *buffer);
> +int thc_tic_pio_write_and_read(struct thc_device *dev, const u32 address,
> + const u32 write_size, const u32 *write_buffer,
> + const u32 read_size, u32 *actual_size, u32 *read_buffer);
>
> #endif /* _INTEL_THC_DEV_H_ */
> diff --git a/drivers/hid/intel-thc-hid/intel-thc/intel-thc-hw.h b/drivers/hid/intel-thc-hid/intel-thc/intel-thc-hw.h
> index 77b275a400d5..9f5f39dcedcd 100644
> --- a/drivers/hid/intel-thc-hid/intel-thc/intel-thc-hw.h
> +++ b/drivers/hid/intel-thc-hid/intel-thc/intel-thc-hw.h
> @@ -633,4 +633,27 @@
>
> #define THC_M_PRT_SW_DMA_PRD_TABLE_LEN_THC_M_PRT_SW_DMA_PRD_TABLE_LEN GENMASK(23, 0)
>
> +#define THC_REGMAP_POLLING_INTERVAL_US 10 /* 10us */
> +#define THC_PIO_DONE_TIMEOUT_US USEC_PER_SEC /* 1s */
> +
> +/*
> + * THC PIO opcode default value
> + * @THC_PIO_OP_SPI_TIC_READ: THC opcode for SPI PIO read
> + * @THC_PIO_OP_SPI_TIC_WRITE: THC opcode for SPI PIO write
> + * @THC_PIO_OP_I2C_SUBSYSTEM_READ: THC opcode for read I2C subsystem registers
> + * @THC_PIO_OP_I2C_SUBSYSTEM_WRITE: THC opcode for write I2C subsystem registers
> + * @THC_PIO_OP_I2C_TIC_READ: THC opcode for read I2C device
> + * @THC_PIO_OP_I2C_TIC_WRITE: THC opcode for write I2C device
> + * @THC_PIO_OP_I2C_TIC_WRITE_AND_READ: THC opcode for write followed by read I2C device
> + */
> +enum thc_pio_opcode {
> + THC_PIO_OP_SPI_TIC_READ = 0x4,
> + THC_PIO_OP_SPI_TIC_WRITE = 0x6,
> + THC_PIO_OP_I2C_SUBSYSTEM_READ = 0x12,
> + THC_PIO_OP_I2C_SUBSYSTEM_WRITE = 0x13,
> + THC_PIO_OP_I2C_TIC_READ = 0x14,
> + THC_PIO_OP_I2C_TIC_WRITE = 0x18,
> + THC_PIO_OP_I2C_TIC_WRITE_AND_READ = 0x1C,
> +};
> +
> #endif /* _INTEL_THC_HW_H_ */
next prev parent reply other threads:[~2024-12-04 5:45 UTC|newest]
Thread overview: 37+ messages / expand[flat|nested] mbox.gz Atom feed top
2024-11-14 5:33 [PATCH v2 00/22] Add Intel Touch Host Controller drivers Even Xu
2024-11-14 5:33 ` [PATCH v2 01/22] HID: THC: Add documentation Even Xu
2024-11-14 13:16 ` Srinivas Pandruvada
2024-11-15 0:39 ` Xu, Even
2024-11-15 4:04 ` Bagas Sanjaya
2024-11-15 5:10 ` Xu, Even
2024-11-15 11:38 ` Bagas Sanjaya
2024-11-18 0:34 ` Xu, Even
2024-11-14 5:33 ` [PATCH v2 02/22] HID: intel-thc-hid: Add basic THC driver skeleton Even Xu
2024-11-14 5:41 ` Randy Dunlap
2024-11-14 5:56 ` Xu, Even
2024-11-14 5:33 ` [PATCH v2 03/22] HID: intel-thc-hid: intel-thc: Add THC registers definition Even Xu
2024-11-14 5:33 ` [PATCH v2 04/22] HID: intel-thc-hid: intel-thc: Add THC PIO operation APIs Even Xu
2024-12-04 5:37 ` Aaron Ma [this message]
2024-12-04 6:00 ` Xu, Even
2024-11-14 5:33 ` [PATCH v2 05/22] HID: intel-thc-hid: intel-thc: Add APIs for interrupt Even Xu
2024-11-14 5:34 ` [PATCH v2 06/22] HID: intel-thc-hid: intel-thc: Add THC DMA interfaces Even Xu
2024-11-14 5:34 ` [PATCH v2 07/22] HID: intel-thc-hid: intel-thc: Add THC LTR interfaces Even Xu
2024-11-14 5:34 ` [PATCH v2 08/22] HID: intel-thc-hid: intel-thc: Add THC interrupt handler Even Xu
2024-11-14 5:34 ` [PATCH v2 09/22] HID: intel-thc-hid: intel-thc: Add THC SPI config interfaces Even Xu
2024-11-14 5:34 ` [PATCH v2 10/22] HID: intel-thc-hid: intel-thc: Add THC I2C " Even Xu
2024-11-14 5:34 ` [PATCH v2 11/22] HID: intel-thc-hid: intel-quickspi: Add THC QuickSPI driver skeleton Even Xu
2024-11-14 5:44 ` Randy Dunlap
2024-11-14 5:59 ` Xu, Even
2024-11-14 5:34 ` [PATCH v2 12/22] HID: intel-thc-hid: intel-quickspi: Add THC QuickSPI driver hid layer Even Xu
2024-11-14 5:34 ` [PATCH v2 13/22] HID: intel-thc-hid: intel-quickspi: Add THC QuickSPI ACPI interfaces Even Xu
2024-11-14 5:34 ` [PATCH v2 14/22] HID: intel-thc-hid: intel-quickspi: Add HIDSPI protocol implementation Even Xu
2024-11-14 5:34 ` [PATCH v2 15/22] HID: intel-thc-hid: intel-quickspi: Complete THC QuickSPI driver Even Xu
2024-11-14 5:34 ` [PATCH v2 16/22] HID: intel-thc-hid: intel-quickspi: Add PM implementation Even Xu
2024-11-14 5:34 ` [PATCH v2 17/22] HID: intel-thc-hid: intel-quicki2c: Add THC QuickI2C driver skeleton Even Xu
2024-11-14 5:45 ` Randy Dunlap
2024-11-14 6:01 ` Xu, Even
2024-11-14 5:34 ` [PATCH v2 18/22] HID: intel-thc-hid: intel-quicki2c: Add THC QuickI2C driver hid layer Even Xu
2024-11-14 5:34 ` [PATCH v2 19/22] HID: intel-thc-hid: intel-quicki2c: Add THC QuickI2C ACPI interfaces Even Xu
2024-11-14 5:34 ` [PATCH v2 20/22] HID: intel-thc-hid: intel-quicki2c: Add HIDI2C protocol implementation Even Xu
2024-11-14 5:34 ` [PATCH v2 21/22] HID: intel-thc-hid: intel-quicki2c: Complete THC QuickI2C driver Even Xu
2024-11-14 5:34 ` [PATCH v2 22/22] HID: intel-thc-hid: intel-quicki2c: Add PM implementation Even Xu
Reply instructions:
You may reply publicly to this message via plain-text email
using any one of the following methods:
* Save the following mbox file, import it into your mail client,
and reply-to-all from there: mbox
Avoid top-posting and favor interleaved quoting:
https://en.wikipedia.org/wiki/Posting_style#Interleaved_style
* Reply using the --to, --cc, and --in-reply-to
switches of git-send-email(1):
git send-email \
--in-reply-to=157e94a2-bc03-42c1-8c4f-d6042032a500@canonical.com \
--to=aaron.ma@canonical.com \
--cc=bagasdotme@gmail.com \
--cc=bentiss@kernel.org \
--cc=corbet@lwn.net \
--cc=even.xu@intel.com \
--cc=jikos@kernel.org \
--cc=linux-doc@vger.kernel.org \
--cc=linux-input@vger.kernel.org \
--cc=linux-kernel@vger.kernel.org \
--cc=rui1.zhang@intel.com \
--cc=srinivas.pandruvada@linux.intel.com \
--cc=xinpeng.sun@intel.com \
/path/to/YOUR_REPLY
https://kernel.org/pub/software/scm/git/docs/git-send-email.html
* If your mail client supports setting the In-Reply-To header
via mailto: links, try the mailto: link
Be sure your reply has a Subject: header at the top and a blank line
before the message body.
This is a public inbox, see mirroring instructions
for how to clone and mirror all data and code used for this inbox;
as well as URLs for NNTP newsgroup(s).