net: Add a WWAN subsystem
This change introduces initial support for a WWAN framework. Given the
complexity and heterogeneity of existing WWAN hardwares and interfaces,
there is no strict definition of what a WWAN device is and how it should
be represented. It's often a collection of multiple devices that perform
the global WWAN feature (netdev, tty, chardev, etc).
One usual way to expose modem controls and configuration is via high
level protocols such as the well known AT command protocol, MBIM or
QMI. The USB modems started to expose them as character devices, and
user daemons such as ModemManager learnt to use them.
This initial version adds the concept of WWAN port, which is a logical
pipe to a modem control protocol. The protocols are rawly exposed to
user via character device, allowing straigthforward support in existing
tools (ModemManager, ofono...). The WWAN core takes care of the generic
part, including character device management, and relies on port driver
operations to receive/submit protocol data.
Since the different devices exposing protocols for a same WWAN hardware
do not necessarily know about each others (e.g. two different USB
interfaces, PCI/MHI channel devices...) and can be created/removed in
different orders, the WWAN core ensures that all WAN ports contributing
to the 'whole' WWAN feature are grouped under the same virtual WWAN
device, relying on the provided parent device (e.g. mhi controller,
USB device). It's a 'trick' I copied from Johannes's earlier WWAN
subsystem proposal.
This initial version is purposely minimalist, it's essentially moving
the generic part of the previously proposed mhi_wwan_ctrl driver inside
a common WWAN framework, but the implementation is open and flexible
enough to allow extension for further drivers.
Signed-off-by: Loic Poulain <loic.poulain@linaro.org>
Signed-off-by: David S. Miller <davem@davemloft.net>
2021-04-16 10:36:33 +02:00
// SPDX-License-Identifier: GPL-2.0-only
/* Copyright (c) 2021, Linaro Ltd <loic.poulain@linaro.org> */
# include <linux/err.h>
# include <linux/errno.h>
# include <linux/fs.h>
# include <linux/init.h>
# include <linux/idr.h>
# include <linux/kernel.h>
# include <linux/module.h>
# include <linux/poll.h>
# include <linux/skbuff.h>
# include <linux/slab.h>
# include <linux/types.h>
2021-06-08 07:02:39 +03:00
# include <linux/termios.h>
net: Add a WWAN subsystem
This change introduces initial support for a WWAN framework. Given the
complexity and heterogeneity of existing WWAN hardwares and interfaces,
there is no strict definition of what a WWAN device is and how it should
be represented. It's often a collection of multiple devices that perform
the global WWAN feature (netdev, tty, chardev, etc).
One usual way to expose modem controls and configuration is via high
level protocols such as the well known AT command protocol, MBIM or
QMI. The USB modems started to expose them as character devices, and
user daemons such as ModemManager learnt to use them.
This initial version adds the concept of WWAN port, which is a logical
pipe to a modem control protocol. The protocols are rawly exposed to
user via character device, allowing straigthforward support in existing
tools (ModemManager, ofono...). The WWAN core takes care of the generic
part, including character device management, and relies on port driver
operations to receive/submit protocol data.
Since the different devices exposing protocols for a same WWAN hardware
do not necessarily know about each others (e.g. two different USB
interfaces, PCI/MHI channel devices...) and can be created/removed in
different orders, the WWAN core ensures that all WAN ports contributing
to the 'whole' WWAN feature are grouped under the same virtual WWAN
device, relying on the provided parent device (e.g. mhi controller,
USB device). It's a 'trick' I copied from Johannes's earlier WWAN
subsystem proposal.
This initial version is purposely minimalist, it's essentially moving
the generic part of the previously proposed mhi_wwan_ctrl driver inside
a common WWAN framework, but the implementation is open and flexible
enough to allow extension for further drivers.
Signed-off-by: Loic Poulain <loic.poulain@linaro.org>
Signed-off-by: David S. Miller <davem@davemloft.net>
2021-04-16 10:36:33 +02:00
# include <linux/wwan.h>
2021-06-12 10:20:56 +02:00
# include <net/rtnetlink.h>
# include <uapi/linux/wwan.h>
net: Add a WWAN subsystem
This change introduces initial support for a WWAN framework. Given the
complexity and heterogeneity of existing WWAN hardwares and interfaces,
there is no strict definition of what a WWAN device is and how it should
be represented. It's often a collection of multiple devices that perform
the global WWAN feature (netdev, tty, chardev, etc).
One usual way to expose modem controls and configuration is via high
level protocols such as the well known AT command protocol, MBIM or
QMI. The USB modems started to expose them as character devices, and
user daemons such as ModemManager learnt to use them.
This initial version adds the concept of WWAN port, which is a logical
pipe to a modem control protocol. The protocols are rawly exposed to
user via character device, allowing straigthforward support in existing
tools (ModemManager, ofono...). The WWAN core takes care of the generic
part, including character device management, and relies on port driver
operations to receive/submit protocol data.
Since the different devices exposing protocols for a same WWAN hardware
do not necessarily know about each others (e.g. two different USB
interfaces, PCI/MHI channel devices...) and can be created/removed in
different orders, the WWAN core ensures that all WAN ports contributing
to the 'whole' WWAN feature are grouped under the same virtual WWAN
device, relying on the provided parent device (e.g. mhi controller,
USB device). It's a 'trick' I copied from Johannes's earlier WWAN
subsystem proposal.
This initial version is purposely minimalist, it's essentially moving
the generic part of the previously proposed mhi_wwan_ctrl driver inside
a common WWAN framework, but the implementation is open and flexible
enough to allow extension for further drivers.
Signed-off-by: Loic Poulain <loic.poulain@linaro.org>
Signed-off-by: David S. Miller <davem@davemloft.net>
2021-04-16 10:36:33 +02:00
2021-06-08 07:02:38 +03:00
/* Maximum number of minors in use */
# define WWAN_MAX_MINORS (1 << MINORBITS)
net: Add a WWAN subsystem
This change introduces initial support for a WWAN framework. Given the
complexity and heterogeneity of existing WWAN hardwares and interfaces,
there is no strict definition of what a WWAN device is and how it should
be represented. It's often a collection of multiple devices that perform
the global WWAN feature (netdev, tty, chardev, etc).
One usual way to expose modem controls and configuration is via high
level protocols such as the well known AT command protocol, MBIM or
QMI. The USB modems started to expose them as character devices, and
user daemons such as ModemManager learnt to use them.
This initial version adds the concept of WWAN port, which is a logical
pipe to a modem control protocol. The protocols are rawly exposed to
user via character device, allowing straigthforward support in existing
tools (ModemManager, ofono...). The WWAN core takes care of the generic
part, including character device management, and relies on port driver
operations to receive/submit protocol data.
Since the different devices exposing protocols for a same WWAN hardware
do not necessarily know about each others (e.g. two different USB
interfaces, PCI/MHI channel devices...) and can be created/removed in
different orders, the WWAN core ensures that all WAN ports contributing
to the 'whole' WWAN feature are grouped under the same virtual WWAN
device, relying on the provided parent device (e.g. mhi controller,
USB device). It's a 'trick' I copied from Johannes's earlier WWAN
subsystem proposal.
This initial version is purposely minimalist, it's essentially moving
the generic part of the previously proposed mhi_wwan_ctrl driver inside
a common WWAN framework, but the implementation is open and flexible
enough to allow extension for further drivers.
Signed-off-by: Loic Poulain <loic.poulain@linaro.org>
Signed-off-by: David S. Miller <davem@davemloft.net>
2021-04-16 10:36:33 +02:00
static DEFINE_MUTEX ( wwan_register_lock ) ; /* WWAN device create|remove lock */
static DEFINE_IDA ( minors ) ; /* minors for WWAN port chardevs */
static DEFINE_IDA ( wwan_dev_ids ) ; /* for unique WWAN device IDs */
static struct class * wwan_class ;
static int wwan_major ;
# define to_wwan_dev(d) container_of(d, struct wwan_device, dev)
# define to_wwan_port(d) container_of(d, struct wwan_port, dev)
/* WWAN port flags */
2021-04-20 21:09:57 +02:00
# define WWAN_PORT_TX_OFF 0
net: Add a WWAN subsystem
This change introduces initial support for a WWAN framework. Given the
complexity and heterogeneity of existing WWAN hardwares and interfaces,
there is no strict definition of what a WWAN device is and how it should
be represented. It's often a collection of multiple devices that perform
the global WWAN feature (netdev, tty, chardev, etc).
One usual way to expose modem controls and configuration is via high
level protocols such as the well known AT command protocol, MBIM or
QMI. The USB modems started to expose them as character devices, and
user daemons such as ModemManager learnt to use them.
This initial version adds the concept of WWAN port, which is a logical
pipe to a modem control protocol. The protocols are rawly exposed to
user via character device, allowing straigthforward support in existing
tools (ModemManager, ofono...). The WWAN core takes care of the generic
part, including character device management, and relies on port driver
operations to receive/submit protocol data.
Since the different devices exposing protocols for a same WWAN hardware
do not necessarily know about each others (e.g. two different USB
interfaces, PCI/MHI channel devices...) and can be created/removed in
different orders, the WWAN core ensures that all WAN ports contributing
to the 'whole' WWAN feature are grouped under the same virtual WWAN
device, relying on the provided parent device (e.g. mhi controller,
USB device). It's a 'trick' I copied from Johannes's earlier WWAN
subsystem proposal.
This initial version is purposely minimalist, it's essentially moving
the generic part of the previously proposed mhi_wwan_ctrl driver inside
a common WWAN framework, but the implementation is open and flexible
enough to allow extension for further drivers.
Signed-off-by: Loic Poulain <loic.poulain@linaro.org>
Signed-off-by: David S. Miller <davem@davemloft.net>
2021-04-16 10:36:33 +02:00
/**
* struct wwan_device - The structure that defines a WWAN device
*
* @ id : WWAN device unique ID .
* @ dev : Underlying device .
2021-06-12 10:20:56 +02:00
* @ port_id : Current available port ID to pick .
* @ ops : wwan device ops
* @ ops_ctxt : context to pass to ops
net: Add a WWAN subsystem
This change introduces initial support for a WWAN framework. Given the
complexity and heterogeneity of existing WWAN hardwares and interfaces,
there is no strict definition of what a WWAN device is and how it should
be represented. It's often a collection of multiple devices that perform
the global WWAN feature (netdev, tty, chardev, etc).
One usual way to expose modem controls and configuration is via high
level protocols such as the well known AT command protocol, MBIM or
QMI. The USB modems started to expose them as character devices, and
user daemons such as ModemManager learnt to use them.
This initial version adds the concept of WWAN port, which is a logical
pipe to a modem control protocol. The protocols are rawly exposed to
user via character device, allowing straigthforward support in existing
tools (ModemManager, ofono...). The WWAN core takes care of the generic
part, including character device management, and relies on port driver
operations to receive/submit protocol data.
Since the different devices exposing protocols for a same WWAN hardware
do not necessarily know about each others (e.g. two different USB
interfaces, PCI/MHI channel devices...) and can be created/removed in
different orders, the WWAN core ensures that all WAN ports contributing
to the 'whole' WWAN feature are grouped under the same virtual WWAN
device, relying on the provided parent device (e.g. mhi controller,
USB device). It's a 'trick' I copied from Johannes's earlier WWAN
subsystem proposal.
This initial version is purposely minimalist, it's essentially moving
the generic part of the previously proposed mhi_wwan_ctrl driver inside
a common WWAN framework, but the implementation is open and flexible
enough to allow extension for further drivers.
Signed-off-by: Loic Poulain <loic.poulain@linaro.org>
Signed-off-by: David S. Miller <davem@davemloft.net>
2021-04-16 10:36:33 +02:00
*/
struct wwan_device {
unsigned int id ;
struct device dev ;
2021-06-12 10:20:56 +02:00
atomic_t port_id ;
const struct wwan_ops * ops ;
void * ops_ctxt ;
net: Add a WWAN subsystem
This change introduces initial support for a WWAN framework. Given the
complexity and heterogeneity of existing WWAN hardwares and interfaces,
there is no strict definition of what a WWAN device is and how it should
be represented. It's often a collection of multiple devices that perform
the global WWAN feature (netdev, tty, chardev, etc).
One usual way to expose modem controls and configuration is via high
level protocols such as the well known AT command protocol, MBIM or
QMI. The USB modems started to expose them as character devices, and
user daemons such as ModemManager learnt to use them.
This initial version adds the concept of WWAN port, which is a logical
pipe to a modem control protocol. The protocols are rawly exposed to
user via character device, allowing straigthforward support in existing
tools (ModemManager, ofono...). The WWAN core takes care of the generic
part, including character device management, and relies on port driver
operations to receive/submit protocol data.
Since the different devices exposing protocols for a same WWAN hardware
do not necessarily know about each others (e.g. two different USB
interfaces, PCI/MHI channel devices...) and can be created/removed in
different orders, the WWAN core ensures that all WAN ports contributing
to the 'whole' WWAN feature are grouped under the same virtual WWAN
device, relying on the provided parent device (e.g. mhi controller,
USB device). It's a 'trick' I copied from Johannes's earlier WWAN
subsystem proposal.
This initial version is purposely minimalist, it's essentially moving
the generic part of the previously proposed mhi_wwan_ctrl driver inside
a common WWAN framework, but the implementation is open and flexible
enough to allow extension for further drivers.
Signed-off-by: Loic Poulain <loic.poulain@linaro.org>
Signed-off-by: David S. Miller <davem@davemloft.net>
2021-04-16 10:36:33 +02:00
} ;
/**
* struct wwan_port - The structure that defines a WWAN port
* @ type : Port type
* @ start_count : Port start counter
* @ flags : Store port state and capabilities
* @ ops : Pointer to WWAN port operations
* @ ops_lock : Protect port ops
* @ dev : Underlying device
* @ rxq : Buffer inbound queue
* @ waitqueue : The waitqueue for port fops ( read / write / poll )
2021-06-08 07:02:40 +03:00
* @ data_lock : Port specific data access serialization
* @ at_data : AT port specific data
net: Add a WWAN subsystem
This change introduces initial support for a WWAN framework. Given the
complexity and heterogeneity of existing WWAN hardwares and interfaces,
there is no strict definition of what a WWAN device is and how it should
be represented. It's often a collection of multiple devices that perform
the global WWAN feature (netdev, tty, chardev, etc).
One usual way to expose modem controls and configuration is via high
level protocols such as the well known AT command protocol, MBIM or
QMI. The USB modems started to expose them as character devices, and
user daemons such as ModemManager learnt to use them.
This initial version adds the concept of WWAN port, which is a logical
pipe to a modem control protocol. The protocols are rawly exposed to
user via character device, allowing straigthforward support in existing
tools (ModemManager, ofono...). The WWAN core takes care of the generic
part, including character device management, and relies on port driver
operations to receive/submit protocol data.
Since the different devices exposing protocols for a same WWAN hardware
do not necessarily know about each others (e.g. two different USB
interfaces, PCI/MHI channel devices...) and can be created/removed in
different orders, the WWAN core ensures that all WAN ports contributing
to the 'whole' WWAN feature are grouped under the same virtual WWAN
device, relying on the provided parent device (e.g. mhi controller,
USB device). It's a 'trick' I copied from Johannes's earlier WWAN
subsystem proposal.
This initial version is purposely minimalist, it's essentially moving
the generic part of the previously proposed mhi_wwan_ctrl driver inside
a common WWAN framework, but the implementation is open and flexible
enough to allow extension for further drivers.
Signed-off-by: Loic Poulain <loic.poulain@linaro.org>
Signed-off-by: David S. Miller <davem@davemloft.net>
2021-04-16 10:36:33 +02:00
*/
struct wwan_port {
enum wwan_port_type type ;
unsigned int start_count ;
unsigned long flags ;
const struct wwan_port_ops * ops ;
struct mutex ops_lock ; /* Serialize ops + protect against removal */
struct device dev ;
struct sk_buff_head rxq ;
wait_queue_head_t waitqueue ;
2021-06-08 07:02:40 +03:00
struct mutex data_lock ; /* Port specific data access serialization */
union {
struct {
struct ktermios termios ;
int mdmbits ;
} at_data ;
} ;
net: Add a WWAN subsystem
This change introduces initial support for a WWAN framework. Given the
complexity and heterogeneity of existing WWAN hardwares and interfaces,
there is no strict definition of what a WWAN device is and how it should
be represented. It's often a collection of multiple devices that perform
the global WWAN feature (netdev, tty, chardev, etc).
One usual way to expose modem controls and configuration is via high
level protocols such as the well known AT command protocol, MBIM or
QMI. The USB modems started to expose them as character devices, and
user daemons such as ModemManager learnt to use them.
This initial version adds the concept of WWAN port, which is a logical
pipe to a modem control protocol. The protocols are rawly exposed to
user via character device, allowing straigthforward support in existing
tools (ModemManager, ofono...). The WWAN core takes care of the generic
part, including character device management, and relies on port driver
operations to receive/submit protocol data.
Since the different devices exposing protocols for a same WWAN hardware
do not necessarily know about each others (e.g. two different USB
interfaces, PCI/MHI channel devices...) and can be created/removed in
different orders, the WWAN core ensures that all WAN ports contributing
to the 'whole' WWAN feature are grouped under the same virtual WWAN
device, relying on the provided parent device (e.g. mhi controller,
USB device). It's a 'trick' I copied from Johannes's earlier WWAN
subsystem proposal.
This initial version is purposely minimalist, it's essentially moving
the generic part of the previously proposed mhi_wwan_ctrl driver inside
a common WWAN framework, but the implementation is open and flexible
enough to allow extension for further drivers.
Signed-off-by: Loic Poulain <loic.poulain@linaro.org>
Signed-off-by: David S. Miller <davem@davemloft.net>
2021-04-16 10:36:33 +02:00
} ;
2021-05-25 18:31:18 +02:00
static ssize_t index_show ( struct device * dev , struct device_attribute * attr , char * buf )
{
struct wwan_device * wwan = to_wwan_dev ( dev ) ;
return sprintf ( buf , " %d \n " , wwan - > id ) ;
}
static DEVICE_ATTR_RO ( index ) ;
static struct attribute * wwan_dev_attrs [ ] = {
& dev_attr_index . attr ,
NULL ,
} ;
ATTRIBUTE_GROUPS ( wwan_dev ) ;
net: Add a WWAN subsystem
This change introduces initial support for a WWAN framework. Given the
complexity and heterogeneity of existing WWAN hardwares and interfaces,
there is no strict definition of what a WWAN device is and how it should
be represented. It's often a collection of multiple devices that perform
the global WWAN feature (netdev, tty, chardev, etc).
One usual way to expose modem controls and configuration is via high
level protocols such as the well known AT command protocol, MBIM or
QMI. The USB modems started to expose them as character devices, and
user daemons such as ModemManager learnt to use them.
This initial version adds the concept of WWAN port, which is a logical
pipe to a modem control protocol. The protocols are rawly exposed to
user via character device, allowing straigthforward support in existing
tools (ModemManager, ofono...). The WWAN core takes care of the generic
part, including character device management, and relies on port driver
operations to receive/submit protocol data.
Since the different devices exposing protocols for a same WWAN hardware
do not necessarily know about each others (e.g. two different USB
interfaces, PCI/MHI channel devices...) and can be created/removed in
different orders, the WWAN core ensures that all WAN ports contributing
to the 'whole' WWAN feature are grouped under the same virtual WWAN
device, relying on the provided parent device (e.g. mhi controller,
USB device). It's a 'trick' I copied from Johannes's earlier WWAN
subsystem proposal.
This initial version is purposely minimalist, it's essentially moving
the generic part of the previously proposed mhi_wwan_ctrl driver inside
a common WWAN framework, but the implementation is open and flexible
enough to allow extension for further drivers.
Signed-off-by: Loic Poulain <loic.poulain@linaro.org>
Signed-off-by: David S. Miller <davem@davemloft.net>
2021-04-16 10:36:33 +02:00
static void wwan_dev_destroy ( struct device * dev )
{
struct wwan_device * wwandev = to_wwan_dev ( dev ) ;
ida_free ( & wwan_dev_ids , wwandev - > id ) ;
kfree ( wwandev ) ;
}
static const struct device_type wwan_dev_type = {
. name = " wwan_dev " ,
. release = wwan_dev_destroy ,
2021-05-25 18:31:18 +02:00
. groups = wwan_dev_groups ,
net: Add a WWAN subsystem
This change introduces initial support for a WWAN framework. Given the
complexity and heterogeneity of existing WWAN hardwares and interfaces,
there is no strict definition of what a WWAN device is and how it should
be represented. It's often a collection of multiple devices that perform
the global WWAN feature (netdev, tty, chardev, etc).
One usual way to expose modem controls and configuration is via high
level protocols such as the well known AT command protocol, MBIM or
QMI. The USB modems started to expose them as character devices, and
user daemons such as ModemManager learnt to use them.
This initial version adds the concept of WWAN port, which is a logical
pipe to a modem control protocol. The protocols are rawly exposed to
user via character device, allowing straigthforward support in existing
tools (ModemManager, ofono...). The WWAN core takes care of the generic
part, including character device management, and relies on port driver
operations to receive/submit protocol data.
Since the different devices exposing protocols for a same WWAN hardware
do not necessarily know about each others (e.g. two different USB
interfaces, PCI/MHI channel devices...) and can be created/removed in
different orders, the WWAN core ensures that all WAN ports contributing
to the 'whole' WWAN feature are grouped under the same virtual WWAN
device, relying on the provided parent device (e.g. mhi controller,
USB device). It's a 'trick' I copied from Johannes's earlier WWAN
subsystem proposal.
This initial version is purposely minimalist, it's essentially moving
the generic part of the previously proposed mhi_wwan_ctrl driver inside
a common WWAN framework, but the implementation is open and flexible
enough to allow extension for further drivers.
Signed-off-by: Loic Poulain <loic.poulain@linaro.org>
Signed-off-by: David S. Miller <davem@davemloft.net>
2021-04-16 10:36:33 +02:00
} ;
static int wwan_dev_parent_match ( struct device * dev , const void * parent )
{
2021-06-12 10:20:56 +02:00
return ( dev - > type = = & wwan_dev_type & &
( dev - > parent = = parent | | dev = = parent ) ) ;
net: Add a WWAN subsystem
This change introduces initial support for a WWAN framework. Given the
complexity and heterogeneity of existing WWAN hardwares and interfaces,
there is no strict definition of what a WWAN device is and how it should
be represented. It's often a collection of multiple devices that perform
the global WWAN feature (netdev, tty, chardev, etc).
One usual way to expose modem controls and configuration is via high
level protocols such as the well known AT command protocol, MBIM or
QMI. The USB modems started to expose them as character devices, and
user daemons such as ModemManager learnt to use them.
This initial version adds the concept of WWAN port, which is a logical
pipe to a modem control protocol. The protocols are rawly exposed to
user via character device, allowing straigthforward support in existing
tools (ModemManager, ofono...). The WWAN core takes care of the generic
part, including character device management, and relies on port driver
operations to receive/submit protocol data.
Since the different devices exposing protocols for a same WWAN hardware
do not necessarily know about each others (e.g. two different USB
interfaces, PCI/MHI channel devices...) and can be created/removed in
different orders, the WWAN core ensures that all WAN ports contributing
to the 'whole' WWAN feature are grouped under the same virtual WWAN
device, relying on the provided parent device (e.g. mhi controller,
USB device). It's a 'trick' I copied from Johannes's earlier WWAN
subsystem proposal.
This initial version is purposely minimalist, it's essentially moving
the generic part of the previously proposed mhi_wwan_ctrl driver inside
a common WWAN framework, but the implementation is open and flexible
enough to allow extension for further drivers.
Signed-off-by: Loic Poulain <loic.poulain@linaro.org>
Signed-off-by: David S. Miller <davem@davemloft.net>
2021-04-16 10:36:33 +02:00
}
static struct wwan_device * wwan_dev_get_by_parent ( struct device * parent )
{
struct device * dev ;
dev = class_find_device ( wwan_class , NULL , parent , wwan_dev_parent_match ) ;
if ( ! dev )
return ERR_PTR ( - ENODEV ) ;
return to_wwan_dev ( dev ) ;
}
2021-06-12 10:20:56 +02:00
static int wwan_dev_name_match ( struct device * dev , const void * name )
{
return dev - > type = = & wwan_dev_type & &
strcmp ( dev_name ( dev ) , name ) = = 0 ;
}
static struct wwan_device * wwan_dev_get_by_name ( const char * name )
{
struct device * dev ;
dev = class_find_device ( wwan_class , NULL , name , wwan_dev_name_match ) ;
if ( ! dev )
return ERR_PTR ( - ENODEV ) ;
return to_wwan_dev ( dev ) ;
}
net: Add a WWAN subsystem
This change introduces initial support for a WWAN framework. Given the
complexity and heterogeneity of existing WWAN hardwares and interfaces,
there is no strict definition of what a WWAN device is and how it should
be represented. It's often a collection of multiple devices that perform
the global WWAN feature (netdev, tty, chardev, etc).
One usual way to expose modem controls and configuration is via high
level protocols such as the well known AT command protocol, MBIM or
QMI. The USB modems started to expose them as character devices, and
user daemons such as ModemManager learnt to use them.
This initial version adds the concept of WWAN port, which is a logical
pipe to a modem control protocol. The protocols are rawly exposed to
user via character device, allowing straigthforward support in existing
tools (ModemManager, ofono...). The WWAN core takes care of the generic
part, including character device management, and relies on port driver
operations to receive/submit protocol data.
Since the different devices exposing protocols for a same WWAN hardware
do not necessarily know about each others (e.g. two different USB
interfaces, PCI/MHI channel devices...) and can be created/removed in
different orders, the WWAN core ensures that all WAN ports contributing
to the 'whole' WWAN feature are grouped under the same virtual WWAN
device, relying on the provided parent device (e.g. mhi controller,
USB device). It's a 'trick' I copied from Johannes's earlier WWAN
subsystem proposal.
This initial version is purposely minimalist, it's essentially moving
the generic part of the previously proposed mhi_wwan_ctrl driver inside
a common WWAN framework, but the implementation is open and flexible
enough to allow extension for further drivers.
Signed-off-by: Loic Poulain <loic.poulain@linaro.org>
Signed-off-by: David S. Miller <davem@davemloft.net>
2021-04-16 10:36:33 +02:00
/* This function allocates and registers a new WWAN device OR if a WWAN device
* already exist for the given parent , it gets a reference and return it .
* This function is not exported ( for now ) , it is called indirectly via
* wwan_create_port ( ) .
*/
static struct wwan_device * wwan_create_dev ( struct device * parent )
{
struct wwan_device * wwandev ;
int err , id ;
/* The 'find-alloc-register' operation must be protected against
* concurrent execution , a WWAN device is possibly shared between
* multiple callers or concurrently unregistered from wwan_remove_dev ( ) .
*/
mutex_lock ( & wwan_register_lock ) ;
/* If wwandev already exists, return it */
wwandev = wwan_dev_get_by_parent ( parent ) ;
if ( ! IS_ERR ( wwandev ) )
goto done_unlock ;
id = ida_alloc ( & wwan_dev_ids , GFP_KERNEL ) ;
if ( id < 0 )
goto done_unlock ;
wwandev = kzalloc ( sizeof ( * wwandev ) , GFP_KERNEL ) ;
if ( ! wwandev ) {
ida_free ( & wwan_dev_ids , id ) ;
goto done_unlock ;
}
wwandev - > dev . parent = parent ;
wwandev - > dev . class = wwan_class ;
wwandev - > dev . type = & wwan_dev_type ;
wwandev - > id = id ;
dev_set_name ( & wwandev - > dev , " wwan%d " , wwandev - > id ) ;
err = device_register ( & wwandev - > dev ) ;
if ( err ) {
put_device ( & wwandev - > dev ) ;
wwandev = NULL ;
}
done_unlock :
mutex_unlock ( & wwan_register_lock ) ;
return wwandev ;
}
static int is_wwan_child ( struct device * dev , void * data )
{
return dev - > class = = wwan_class ;
}
static void wwan_remove_dev ( struct wwan_device * wwandev )
{
int ret ;
/* Prevent concurrent picking from wwan_create_dev */
mutex_lock ( & wwan_register_lock ) ;
/* WWAN device is created and registered (get+add) along with its first
* child port , and subsequent port registrations only grab a reference
* ( get ) . The WWAN device must then be unregistered ( del + put ) along with
2021-06-12 10:20:56 +02:00
* its last port , and reference simply dropped ( put ) otherwise . In the
* same fashion , we must not unregister it when the ops are still there .
net: Add a WWAN subsystem
This change introduces initial support for a WWAN framework. Given the
complexity and heterogeneity of existing WWAN hardwares and interfaces,
there is no strict definition of what a WWAN device is and how it should
be represented. It's often a collection of multiple devices that perform
the global WWAN feature (netdev, tty, chardev, etc).
One usual way to expose modem controls and configuration is via high
level protocols such as the well known AT command protocol, MBIM or
QMI. The USB modems started to expose them as character devices, and
user daemons such as ModemManager learnt to use them.
This initial version adds the concept of WWAN port, which is a logical
pipe to a modem control protocol. The protocols are rawly exposed to
user via character device, allowing straigthforward support in existing
tools (ModemManager, ofono...). The WWAN core takes care of the generic
part, including character device management, and relies on port driver
operations to receive/submit protocol data.
Since the different devices exposing protocols for a same WWAN hardware
do not necessarily know about each others (e.g. two different USB
interfaces, PCI/MHI channel devices...) and can be created/removed in
different orders, the WWAN core ensures that all WAN ports contributing
to the 'whole' WWAN feature are grouped under the same virtual WWAN
device, relying on the provided parent device (e.g. mhi controller,
USB device). It's a 'trick' I copied from Johannes's earlier WWAN
subsystem proposal.
This initial version is purposely minimalist, it's essentially moving
the generic part of the previously proposed mhi_wwan_ctrl driver inside
a common WWAN framework, but the implementation is open and flexible
enough to allow extension for further drivers.
Signed-off-by: Loic Poulain <loic.poulain@linaro.org>
Signed-off-by: David S. Miller <davem@davemloft.net>
2021-04-16 10:36:33 +02:00
*/
2021-06-12 10:20:56 +02:00
if ( wwandev - > ops )
ret = 1 ;
else
ret = device_for_each_child ( & wwandev - > dev , NULL , is_wwan_child ) ;
net: Add a WWAN subsystem
This change introduces initial support for a WWAN framework. Given the
complexity and heterogeneity of existing WWAN hardwares and interfaces,
there is no strict definition of what a WWAN device is and how it should
be represented. It's often a collection of multiple devices that perform
the global WWAN feature (netdev, tty, chardev, etc).
One usual way to expose modem controls and configuration is via high
level protocols such as the well known AT command protocol, MBIM or
QMI. The USB modems started to expose them as character devices, and
user daemons such as ModemManager learnt to use them.
This initial version adds the concept of WWAN port, which is a logical
pipe to a modem control protocol. The protocols are rawly exposed to
user via character device, allowing straigthforward support in existing
tools (ModemManager, ofono...). The WWAN core takes care of the generic
part, including character device management, and relies on port driver
operations to receive/submit protocol data.
Since the different devices exposing protocols for a same WWAN hardware
do not necessarily know about each others (e.g. two different USB
interfaces, PCI/MHI channel devices...) and can be created/removed in
different orders, the WWAN core ensures that all WAN ports contributing
to the 'whole' WWAN feature are grouped under the same virtual WWAN
device, relying on the provided parent device (e.g. mhi controller,
USB device). It's a 'trick' I copied from Johannes's earlier WWAN
subsystem proposal.
This initial version is purposely minimalist, it's essentially moving
the generic part of the previously proposed mhi_wwan_ctrl driver inside
a common WWAN framework, but the implementation is open and flexible
enough to allow extension for further drivers.
Signed-off-by: Loic Poulain <loic.poulain@linaro.org>
Signed-off-by: David S. Miller <davem@davemloft.net>
2021-04-16 10:36:33 +02:00
if ( ! ret )
device_unregister ( & wwandev - > dev ) ;
else
put_device ( & wwandev - > dev ) ;
mutex_unlock ( & wwan_register_lock ) ;
}
/* ------- WWAN port management ------- */
2021-06-08 07:02:36 +03:00
static const struct {
const char * const name ; /* Port type name */
const char * const devsuf ; /* Port devce name suffix */
} wwan_port_types [ WWAN_PORT_MAX + 1 ] = {
[ WWAN_PORT_AT ] = {
. name = " AT " ,
. devsuf = " at " ,
} ,
[ WWAN_PORT_MBIM ] = {
. name = " MBIM " ,
. devsuf = " mbim " ,
} ,
[ WWAN_PORT_QMI ] = {
. name = " QMI " ,
. devsuf = " qmi " ,
} ,
[ WWAN_PORT_QCDM ] = {
. name = " QCDM " ,
. devsuf = " qcdm " ,
} ,
[ WWAN_PORT_FIREHOSE ] = {
. name = " FIREHOSE " ,
. devsuf = " firehose " ,
} ,
2021-05-17 11:53:34 +02:00
} ;
static ssize_t type_show ( struct device * dev , struct device_attribute * attr ,
char * buf )
{
struct wwan_port * port = to_wwan_port ( dev ) ;
2021-06-08 07:02:36 +03:00
return sprintf ( buf , " %s \n " , wwan_port_types [ port - > type ] . name ) ;
2021-05-17 11:53:34 +02:00
}
static DEVICE_ATTR_RO ( type ) ;
static struct attribute * wwan_port_attrs [ ] = {
& dev_attr_type . attr ,
NULL ,
} ;
ATTRIBUTE_GROUPS ( wwan_port ) ;
net: Add a WWAN subsystem
This change introduces initial support for a WWAN framework. Given the
complexity and heterogeneity of existing WWAN hardwares and interfaces,
there is no strict definition of what a WWAN device is and how it should
be represented. It's often a collection of multiple devices that perform
the global WWAN feature (netdev, tty, chardev, etc).
One usual way to expose modem controls and configuration is via high
level protocols such as the well known AT command protocol, MBIM or
QMI. The USB modems started to expose them as character devices, and
user daemons such as ModemManager learnt to use them.
This initial version adds the concept of WWAN port, which is a logical
pipe to a modem control protocol. The protocols are rawly exposed to
user via character device, allowing straigthforward support in existing
tools (ModemManager, ofono...). The WWAN core takes care of the generic
part, including character device management, and relies on port driver
operations to receive/submit protocol data.
Since the different devices exposing protocols for a same WWAN hardware
do not necessarily know about each others (e.g. two different USB
interfaces, PCI/MHI channel devices...) and can be created/removed in
different orders, the WWAN core ensures that all WAN ports contributing
to the 'whole' WWAN feature are grouped under the same virtual WWAN
device, relying on the provided parent device (e.g. mhi controller,
USB device). It's a 'trick' I copied from Johannes's earlier WWAN
subsystem proposal.
This initial version is purposely minimalist, it's essentially moving
the generic part of the previously proposed mhi_wwan_ctrl driver inside
a common WWAN framework, but the implementation is open and flexible
enough to allow extension for further drivers.
Signed-off-by: Loic Poulain <loic.poulain@linaro.org>
Signed-off-by: David S. Miller <davem@davemloft.net>
2021-04-16 10:36:33 +02:00
static void wwan_port_destroy ( struct device * dev )
{
struct wwan_port * port = to_wwan_port ( dev ) ;
ida_free ( & minors , MINOR ( port - > dev . devt ) ) ;
2021-06-08 07:02:40 +03:00
mutex_destroy ( & port - > data_lock ) ;
net: Add a WWAN subsystem
This change introduces initial support for a WWAN framework. Given the
complexity and heterogeneity of existing WWAN hardwares and interfaces,
there is no strict definition of what a WWAN device is and how it should
be represented. It's often a collection of multiple devices that perform
the global WWAN feature (netdev, tty, chardev, etc).
One usual way to expose modem controls and configuration is via high
level protocols such as the well known AT command protocol, MBIM or
QMI. The USB modems started to expose them as character devices, and
user daemons such as ModemManager learnt to use them.
This initial version adds the concept of WWAN port, which is a logical
pipe to a modem control protocol. The protocols are rawly exposed to
user via character device, allowing straigthforward support in existing
tools (ModemManager, ofono...). The WWAN core takes care of the generic
part, including character device management, and relies on port driver
operations to receive/submit protocol data.
Since the different devices exposing protocols for a same WWAN hardware
do not necessarily know about each others (e.g. two different USB
interfaces, PCI/MHI channel devices...) and can be created/removed in
different orders, the WWAN core ensures that all WAN ports contributing
to the 'whole' WWAN feature are grouped under the same virtual WWAN
device, relying on the provided parent device (e.g. mhi controller,
USB device). It's a 'trick' I copied from Johannes's earlier WWAN
subsystem proposal.
This initial version is purposely minimalist, it's essentially moving
the generic part of the previously proposed mhi_wwan_ctrl driver inside
a common WWAN framework, but the implementation is open and flexible
enough to allow extension for further drivers.
Signed-off-by: Loic Poulain <loic.poulain@linaro.org>
Signed-off-by: David S. Miller <davem@davemloft.net>
2021-04-16 10:36:33 +02:00
mutex_destroy ( & port - > ops_lock ) ;
kfree ( port ) ;
}
static const struct device_type wwan_port_dev_type = {
. name = " wwan_port " ,
. release = wwan_port_destroy ,
2021-05-17 11:53:34 +02:00
. groups = wwan_port_groups ,
net: Add a WWAN subsystem
This change introduces initial support for a WWAN framework. Given the
complexity and heterogeneity of existing WWAN hardwares and interfaces,
there is no strict definition of what a WWAN device is and how it should
be represented. It's often a collection of multiple devices that perform
the global WWAN feature (netdev, tty, chardev, etc).
One usual way to expose modem controls and configuration is via high
level protocols such as the well known AT command protocol, MBIM or
QMI. The USB modems started to expose them as character devices, and
user daemons such as ModemManager learnt to use them.
This initial version adds the concept of WWAN port, which is a logical
pipe to a modem control protocol. The protocols are rawly exposed to
user via character device, allowing straigthforward support in existing
tools (ModemManager, ofono...). The WWAN core takes care of the generic
part, including character device management, and relies on port driver
operations to receive/submit protocol data.
Since the different devices exposing protocols for a same WWAN hardware
do not necessarily know about each others (e.g. two different USB
interfaces, PCI/MHI channel devices...) and can be created/removed in
different orders, the WWAN core ensures that all WAN ports contributing
to the 'whole' WWAN feature are grouped under the same virtual WWAN
device, relying on the provided parent device (e.g. mhi controller,
USB device). It's a 'trick' I copied from Johannes's earlier WWAN
subsystem proposal.
This initial version is purposely minimalist, it's essentially moving
the generic part of the previously proposed mhi_wwan_ctrl driver inside
a common WWAN framework, but the implementation is open and flexible
enough to allow extension for further drivers.
Signed-off-by: Loic Poulain <loic.poulain@linaro.org>
Signed-off-by: David S. Miller <davem@davemloft.net>
2021-04-16 10:36:33 +02:00
} ;
static int wwan_port_minor_match ( struct device * dev , const void * minor )
{
return ( dev - > type = = & wwan_port_dev_type & &
MINOR ( dev - > devt ) = = * ( unsigned int * ) minor ) ;
}
static struct wwan_port * wwan_port_get_by_minor ( unsigned int minor )
{
struct device * dev ;
dev = class_find_device ( wwan_class , NULL , & minor , wwan_port_minor_match ) ;
if ( ! dev )
return ERR_PTR ( - ENODEV ) ;
return to_wwan_port ( dev ) ;
}
net: wwan: core: make port names more user-friendly
At the moment, the port name is allocated based on the parent device
name, port id and the port type. Where the port id specifies nothing but
the ports registration order and is only used to make the port name
unique.
Most likely, to configure a WWAN device, the user will look for a port
of a specific type (e.g. AT port or MBIM port, etc.). The current naming
scheme can make it difficult to find a port of a specific type.
Consider a WWAN device that has 3 ports: AT port, MBIM port, and another
one AT port. With the global port index, the port names will be:
* wwan0p1at
* wwan0p2mbim
* wwan0p3at
To find the MBIM port, user should know in advance the device ports
composition (i.e. the user should know that the MBIM port is the 2nd
one) or carefully examine the whole ports list. It is not unusual for
USB modems to have a different composition, even if they are build on a
same chipset. Moreover, some modems able to change the ports composition
based on the user's configuration. All this makes port names fully
unpredictable.
To make naming more user-friendly, remove the global port id and
enumerate ports by its type. E.g.:
* wwan0p1at -> wwan0at0
* wwan0p2mbim -> wwan0mbim0
* wwan0p3at -> wwan0at1
With this naming scheme, the first AT port name will always be wwanXat0,
the first MBIM port name will always be wwanXmbim0, etc.
Signed-off-by: Sergey Ryazanov <ryazanov.s.a@gmail.com>
Signed-off-by: David S. Miller <davem@davemloft.net>
2021-06-08 07:02:37 +03:00
/* Allocate and set unique name based on passed format
*
* Name allocation approach is highly inspired by the __dev_alloc_name ( )
* function .
*
* To avoid names collision , the caller must prevent the new port device
* registration as well as concurrent invocation of this function .
*/
static int __wwan_port_dev_assign_name ( struct wwan_port * port , const char * fmt )
{
struct wwan_device * wwandev = to_wwan_dev ( port - > dev . parent ) ;
const unsigned int max_ports = PAGE_SIZE * 8 ;
struct class_dev_iter iter ;
unsigned long * idmap ;
struct device * dev ;
char buf [ 0x20 ] ;
int id ;
idmap = ( unsigned long * ) get_zeroed_page ( GFP_KERNEL ) ;
if ( ! idmap )
return - ENOMEM ;
/* Collect ids of same name format ports */
class_dev_iter_init ( & iter , wwan_class , NULL , & wwan_port_dev_type ) ;
while ( ( dev = class_dev_iter_next ( & iter ) ) ) {
if ( dev - > parent ! = & wwandev - > dev )
continue ;
if ( sscanf ( dev_name ( dev ) , fmt , & id ) ! = 1 )
continue ;
if ( id < 0 | | id > = max_ports )
continue ;
set_bit ( id , idmap ) ;
}
class_dev_iter_exit ( & iter ) ;
/* Allocate unique id */
id = find_first_zero_bit ( idmap , max_ports ) ;
free_page ( ( unsigned long ) idmap ) ;
snprintf ( buf , sizeof ( buf ) , fmt , id ) ; /* Name generation */
dev = device_find_child_by_name ( & wwandev - > dev , buf ) ;
if ( dev ) {
put_device ( dev ) ;
return - ENFILE ;
}
return dev_set_name ( & port - > dev , buf ) ;
}
net: Add a WWAN subsystem
This change introduces initial support for a WWAN framework. Given the
complexity and heterogeneity of existing WWAN hardwares and interfaces,
there is no strict definition of what a WWAN device is and how it should
be represented. It's often a collection of multiple devices that perform
the global WWAN feature (netdev, tty, chardev, etc).
One usual way to expose modem controls and configuration is via high
level protocols such as the well known AT command protocol, MBIM or
QMI. The USB modems started to expose them as character devices, and
user daemons such as ModemManager learnt to use them.
This initial version adds the concept of WWAN port, which is a logical
pipe to a modem control protocol. The protocols are rawly exposed to
user via character device, allowing straigthforward support in existing
tools (ModemManager, ofono...). The WWAN core takes care of the generic
part, including character device management, and relies on port driver
operations to receive/submit protocol data.
Since the different devices exposing protocols for a same WWAN hardware
do not necessarily know about each others (e.g. two different USB
interfaces, PCI/MHI channel devices...) and can be created/removed in
different orders, the WWAN core ensures that all WAN ports contributing
to the 'whole' WWAN feature are grouped under the same virtual WWAN
device, relying on the provided parent device (e.g. mhi controller,
USB device). It's a 'trick' I copied from Johannes's earlier WWAN
subsystem proposal.
This initial version is purposely minimalist, it's essentially moving
the generic part of the previously proposed mhi_wwan_ctrl driver inside
a common WWAN framework, but the implementation is open and flexible
enough to allow extension for further drivers.
Signed-off-by: Loic Poulain <loic.poulain@linaro.org>
Signed-off-by: David S. Miller <davem@davemloft.net>
2021-04-16 10:36:33 +02:00
struct wwan_port * wwan_create_port ( struct device * parent ,
enum wwan_port_type type ,
const struct wwan_port_ops * ops ,
void * drvdata )
{
struct wwan_device * wwandev ;
struct wwan_port * port ;
int minor , err = - ENOMEM ;
net: wwan: core: make port names more user-friendly
At the moment, the port name is allocated based on the parent device
name, port id and the port type. Where the port id specifies nothing but
the ports registration order and is only used to make the port name
unique.
Most likely, to configure a WWAN device, the user will look for a port
of a specific type (e.g. AT port or MBIM port, etc.). The current naming
scheme can make it difficult to find a port of a specific type.
Consider a WWAN device that has 3 ports: AT port, MBIM port, and another
one AT port. With the global port index, the port names will be:
* wwan0p1at
* wwan0p2mbim
* wwan0p3at
To find the MBIM port, user should know in advance the device ports
composition (i.e. the user should know that the MBIM port is the 2nd
one) or carefully examine the whole ports list. It is not unusual for
USB modems to have a different composition, even if they are build on a
same chipset. Moreover, some modems able to change the ports composition
based on the user's configuration. All this makes port names fully
unpredictable.
To make naming more user-friendly, remove the global port id and
enumerate ports by its type. E.g.:
* wwan0p1at -> wwan0at0
* wwan0p2mbim -> wwan0mbim0
* wwan0p3at -> wwan0at1
With this naming scheme, the first AT port name will always be wwanXat0,
the first MBIM port name will always be wwanXmbim0, etc.
Signed-off-by: Sergey Ryazanov <ryazanov.s.a@gmail.com>
Signed-off-by: David S. Miller <davem@davemloft.net>
2021-06-08 07:02:37 +03:00
char namefmt [ 0x20 ] ;
net: Add a WWAN subsystem
This change introduces initial support for a WWAN framework. Given the
complexity and heterogeneity of existing WWAN hardwares and interfaces,
there is no strict definition of what a WWAN device is and how it should
be represented. It's often a collection of multiple devices that perform
the global WWAN feature (netdev, tty, chardev, etc).
One usual way to expose modem controls and configuration is via high
level protocols such as the well known AT command protocol, MBIM or
QMI. The USB modems started to expose them as character devices, and
user daemons such as ModemManager learnt to use them.
This initial version adds the concept of WWAN port, which is a logical
pipe to a modem control protocol. The protocols are rawly exposed to
user via character device, allowing straigthforward support in existing
tools (ModemManager, ofono...). The WWAN core takes care of the generic
part, including character device management, and relies on port driver
operations to receive/submit protocol data.
Since the different devices exposing protocols for a same WWAN hardware
do not necessarily know about each others (e.g. two different USB
interfaces, PCI/MHI channel devices...) and can be created/removed in
different orders, the WWAN core ensures that all WAN ports contributing
to the 'whole' WWAN feature are grouped under the same virtual WWAN
device, relying on the provided parent device (e.g. mhi controller,
USB device). It's a 'trick' I copied from Johannes's earlier WWAN
subsystem proposal.
This initial version is purposely minimalist, it's essentially moving
the generic part of the previously proposed mhi_wwan_ctrl driver inside
a common WWAN framework, but the implementation is open and flexible
enough to allow extension for further drivers.
Signed-off-by: Loic Poulain <loic.poulain@linaro.org>
Signed-off-by: David S. Miller <davem@davemloft.net>
2021-04-16 10:36:33 +02:00
2021-06-08 07:02:34 +03:00
if ( type > WWAN_PORT_MAX | | ! ops )
net: Add a WWAN subsystem
This change introduces initial support for a WWAN framework. Given the
complexity and heterogeneity of existing WWAN hardwares and interfaces,
there is no strict definition of what a WWAN device is and how it should
be represented. It's often a collection of multiple devices that perform
the global WWAN feature (netdev, tty, chardev, etc).
One usual way to expose modem controls and configuration is via high
level protocols such as the well known AT command protocol, MBIM or
QMI. The USB modems started to expose them as character devices, and
user daemons such as ModemManager learnt to use them.
This initial version adds the concept of WWAN port, which is a logical
pipe to a modem control protocol. The protocols are rawly exposed to
user via character device, allowing straigthforward support in existing
tools (ModemManager, ofono...). The WWAN core takes care of the generic
part, including character device management, and relies on port driver
operations to receive/submit protocol data.
Since the different devices exposing protocols for a same WWAN hardware
do not necessarily know about each others (e.g. two different USB
interfaces, PCI/MHI channel devices...) and can be created/removed in
different orders, the WWAN core ensures that all WAN ports contributing
to the 'whole' WWAN feature are grouped under the same virtual WWAN
device, relying on the provided parent device (e.g. mhi controller,
USB device). It's a 'trick' I copied from Johannes's earlier WWAN
subsystem proposal.
This initial version is purposely minimalist, it's essentially moving
the generic part of the previously proposed mhi_wwan_ctrl driver inside
a common WWAN framework, but the implementation is open and flexible
enough to allow extension for further drivers.
Signed-off-by: Loic Poulain <loic.poulain@linaro.org>
Signed-off-by: David S. Miller <davem@davemloft.net>
2021-04-16 10:36:33 +02:00
return ERR_PTR ( - EINVAL ) ;
/* A port is always a child of a WWAN device, retrieve (allocate or
* pick ) the WWAN device based on the provided parent device .
*/
wwandev = wwan_create_dev ( parent ) ;
if ( IS_ERR ( wwandev ) )
return ERR_CAST ( wwandev ) ;
/* A port is exposed as character device, get a minor */
minor = ida_alloc_range ( & minors , 0 , WWAN_MAX_MINORS - 1 , GFP_KERNEL ) ;
if ( minor < 0 )
goto error_wwandev_remove ;
port = kzalloc ( sizeof ( * port ) , GFP_KERNEL ) ;
if ( ! port ) {
ida_free ( & minors , minor ) ;
goto error_wwandev_remove ;
}
port - > type = type ;
port - > ops = ops ;
mutex_init ( & port - > ops_lock ) ;
skb_queue_head_init ( & port - > rxq ) ;
init_waitqueue_head ( & port - > waitqueue ) ;
2021-06-08 07:02:40 +03:00
mutex_init ( & port - > data_lock ) ;
net: Add a WWAN subsystem
This change introduces initial support for a WWAN framework. Given the
complexity and heterogeneity of existing WWAN hardwares and interfaces,
there is no strict definition of what a WWAN device is and how it should
be represented. It's often a collection of multiple devices that perform
the global WWAN feature (netdev, tty, chardev, etc).
One usual way to expose modem controls and configuration is via high
level protocols such as the well known AT command protocol, MBIM or
QMI. The USB modems started to expose them as character devices, and
user daemons such as ModemManager learnt to use them.
This initial version adds the concept of WWAN port, which is a logical
pipe to a modem control protocol. The protocols are rawly exposed to
user via character device, allowing straigthforward support in existing
tools (ModemManager, ofono...). The WWAN core takes care of the generic
part, including character device management, and relies on port driver
operations to receive/submit protocol data.
Since the different devices exposing protocols for a same WWAN hardware
do not necessarily know about each others (e.g. two different USB
interfaces, PCI/MHI channel devices...) and can be created/removed in
different orders, the WWAN core ensures that all WAN ports contributing
to the 'whole' WWAN feature are grouped under the same virtual WWAN
device, relying on the provided parent device (e.g. mhi controller,
USB device). It's a 'trick' I copied from Johannes's earlier WWAN
subsystem proposal.
This initial version is purposely minimalist, it's essentially moving
the generic part of the previously proposed mhi_wwan_ctrl driver inside
a common WWAN framework, but the implementation is open and flexible
enough to allow extension for further drivers.
Signed-off-by: Loic Poulain <loic.poulain@linaro.org>
Signed-off-by: David S. Miller <davem@davemloft.net>
2021-04-16 10:36:33 +02:00
port - > dev . parent = & wwandev - > dev ;
port - > dev . class = wwan_class ;
port - > dev . type = & wwan_port_dev_type ;
port - > dev . devt = MKDEV ( wwan_major , minor ) ;
dev_set_drvdata ( & port - > dev , drvdata ) ;
net: wwan: core: make port names more user-friendly
At the moment, the port name is allocated based on the parent device
name, port id and the port type. Where the port id specifies nothing but
the ports registration order and is only used to make the port name
unique.
Most likely, to configure a WWAN device, the user will look for a port
of a specific type (e.g. AT port or MBIM port, etc.). The current naming
scheme can make it difficult to find a port of a specific type.
Consider a WWAN device that has 3 ports: AT port, MBIM port, and another
one AT port. With the global port index, the port names will be:
* wwan0p1at
* wwan0p2mbim
* wwan0p3at
To find the MBIM port, user should know in advance the device ports
composition (i.e. the user should know that the MBIM port is the 2nd
one) or carefully examine the whole ports list. It is not unusual for
USB modems to have a different composition, even if they are build on a
same chipset. Moreover, some modems able to change the ports composition
based on the user's configuration. All this makes port names fully
unpredictable.
To make naming more user-friendly, remove the global port id and
enumerate ports by its type. E.g.:
* wwan0p1at -> wwan0at0
* wwan0p2mbim -> wwan0mbim0
* wwan0p3at -> wwan0at1
With this naming scheme, the first AT port name will always be wwanXat0,
the first MBIM port name will always be wwanXmbim0, etc.
Signed-off-by: Sergey Ryazanov <ryazanov.s.a@gmail.com>
Signed-off-by: David S. Miller <davem@davemloft.net>
2021-06-08 07:02:37 +03:00
/* allocate unique name based on wwan device id, port type and number */
snprintf ( namefmt , sizeof ( namefmt ) , " wwan%u%s%%d " , wwandev - > id ,
wwan_port_types [ port - > type ] . devsuf ) ;
net: Add a WWAN subsystem
This change introduces initial support for a WWAN framework. Given the
complexity and heterogeneity of existing WWAN hardwares and interfaces,
there is no strict definition of what a WWAN device is and how it should
be represented. It's often a collection of multiple devices that perform
the global WWAN feature (netdev, tty, chardev, etc).
One usual way to expose modem controls and configuration is via high
level protocols such as the well known AT command protocol, MBIM or
QMI. The USB modems started to expose them as character devices, and
user daemons such as ModemManager learnt to use them.
This initial version adds the concept of WWAN port, which is a logical
pipe to a modem control protocol. The protocols are rawly exposed to
user via character device, allowing straigthforward support in existing
tools (ModemManager, ofono...). The WWAN core takes care of the generic
part, including character device management, and relies on port driver
operations to receive/submit protocol data.
Since the different devices exposing protocols for a same WWAN hardware
do not necessarily know about each others (e.g. two different USB
interfaces, PCI/MHI channel devices...) and can be created/removed in
different orders, the WWAN core ensures that all WAN ports contributing
to the 'whole' WWAN feature are grouped under the same virtual WWAN
device, relying on the provided parent device (e.g. mhi controller,
USB device). It's a 'trick' I copied from Johannes's earlier WWAN
subsystem proposal.
This initial version is purposely minimalist, it's essentially moving
the generic part of the previously proposed mhi_wwan_ctrl driver inside
a common WWAN framework, but the implementation is open and flexible
enough to allow extension for further drivers.
Signed-off-by: Loic Poulain <loic.poulain@linaro.org>
Signed-off-by: David S. Miller <davem@davemloft.net>
2021-04-16 10:36:33 +02:00
net: wwan: core: make port names more user-friendly
At the moment, the port name is allocated based on the parent device
name, port id and the port type. Where the port id specifies nothing but
the ports registration order and is only used to make the port name
unique.
Most likely, to configure a WWAN device, the user will look for a port
of a specific type (e.g. AT port or MBIM port, etc.). The current naming
scheme can make it difficult to find a port of a specific type.
Consider a WWAN device that has 3 ports: AT port, MBIM port, and another
one AT port. With the global port index, the port names will be:
* wwan0p1at
* wwan0p2mbim
* wwan0p3at
To find the MBIM port, user should know in advance the device ports
composition (i.e. the user should know that the MBIM port is the 2nd
one) or carefully examine the whole ports list. It is not unusual for
USB modems to have a different composition, even if they are build on a
same chipset. Moreover, some modems able to change the ports composition
based on the user's configuration. All this makes port names fully
unpredictable.
To make naming more user-friendly, remove the global port id and
enumerate ports by its type. E.g.:
* wwan0p1at -> wwan0at0
* wwan0p2mbim -> wwan0mbim0
* wwan0p3at -> wwan0at1
With this naming scheme, the first AT port name will always be wwanXat0,
the first MBIM port name will always be wwanXmbim0, etc.
Signed-off-by: Sergey Ryazanov <ryazanov.s.a@gmail.com>
Signed-off-by: David S. Miller <davem@davemloft.net>
2021-06-08 07:02:37 +03:00
/* Serialize ports registration */
mutex_lock ( & wwan_register_lock ) ;
__wwan_port_dev_assign_name ( port , namefmt ) ;
net: Add a WWAN subsystem
This change introduces initial support for a WWAN framework. Given the
complexity and heterogeneity of existing WWAN hardwares and interfaces,
there is no strict definition of what a WWAN device is and how it should
be represented. It's often a collection of multiple devices that perform
the global WWAN feature (netdev, tty, chardev, etc).
One usual way to expose modem controls and configuration is via high
level protocols such as the well known AT command protocol, MBIM or
QMI. The USB modems started to expose them as character devices, and
user daemons such as ModemManager learnt to use them.
This initial version adds the concept of WWAN port, which is a logical
pipe to a modem control protocol. The protocols are rawly exposed to
user via character device, allowing straigthforward support in existing
tools (ModemManager, ofono...). The WWAN core takes care of the generic
part, including character device management, and relies on port driver
operations to receive/submit protocol data.
Since the different devices exposing protocols for a same WWAN hardware
do not necessarily know about each others (e.g. two different USB
interfaces, PCI/MHI channel devices...) and can be created/removed in
different orders, the WWAN core ensures that all WAN ports contributing
to the 'whole' WWAN feature are grouped under the same virtual WWAN
device, relying on the provided parent device (e.g. mhi controller,
USB device). It's a 'trick' I copied from Johannes's earlier WWAN
subsystem proposal.
This initial version is purposely minimalist, it's essentially moving
the generic part of the previously proposed mhi_wwan_ctrl driver inside
a common WWAN framework, but the implementation is open and flexible
enough to allow extension for further drivers.
Signed-off-by: Loic Poulain <loic.poulain@linaro.org>
Signed-off-by: David S. Miller <davem@davemloft.net>
2021-04-16 10:36:33 +02:00
err = device_register ( & port - > dev ) ;
net: wwan: core: make port names more user-friendly
At the moment, the port name is allocated based on the parent device
name, port id and the port type. Where the port id specifies nothing but
the ports registration order and is only used to make the port name
unique.
Most likely, to configure a WWAN device, the user will look for a port
of a specific type (e.g. AT port or MBIM port, etc.). The current naming
scheme can make it difficult to find a port of a specific type.
Consider a WWAN device that has 3 ports: AT port, MBIM port, and another
one AT port. With the global port index, the port names will be:
* wwan0p1at
* wwan0p2mbim
* wwan0p3at
To find the MBIM port, user should know in advance the device ports
composition (i.e. the user should know that the MBIM port is the 2nd
one) or carefully examine the whole ports list. It is not unusual for
USB modems to have a different composition, even if they are build on a
same chipset. Moreover, some modems able to change the ports composition
based on the user's configuration. All this makes port names fully
unpredictable.
To make naming more user-friendly, remove the global port id and
enumerate ports by its type. E.g.:
* wwan0p1at -> wwan0at0
* wwan0p2mbim -> wwan0mbim0
* wwan0p3at -> wwan0at1
With this naming scheme, the first AT port name will always be wwanXat0,
the first MBIM port name will always be wwanXmbim0, etc.
Signed-off-by: Sergey Ryazanov <ryazanov.s.a@gmail.com>
Signed-off-by: David S. Miller <davem@davemloft.net>
2021-06-08 07:02:37 +03:00
mutex_unlock ( & wwan_register_lock ) ;
net: Add a WWAN subsystem
This change introduces initial support for a WWAN framework. Given the
complexity and heterogeneity of existing WWAN hardwares and interfaces,
there is no strict definition of what a WWAN device is and how it should
be represented. It's often a collection of multiple devices that perform
the global WWAN feature (netdev, tty, chardev, etc).
One usual way to expose modem controls and configuration is via high
level protocols such as the well known AT command protocol, MBIM or
QMI. The USB modems started to expose them as character devices, and
user daemons such as ModemManager learnt to use them.
This initial version adds the concept of WWAN port, which is a logical
pipe to a modem control protocol. The protocols are rawly exposed to
user via character device, allowing straigthforward support in existing
tools (ModemManager, ofono...). The WWAN core takes care of the generic
part, including character device management, and relies on port driver
operations to receive/submit protocol data.
Since the different devices exposing protocols for a same WWAN hardware
do not necessarily know about each others (e.g. two different USB
interfaces, PCI/MHI channel devices...) and can be created/removed in
different orders, the WWAN core ensures that all WAN ports contributing
to the 'whole' WWAN feature are grouped under the same virtual WWAN
device, relying on the provided parent device (e.g. mhi controller,
USB device). It's a 'trick' I copied from Johannes's earlier WWAN
subsystem proposal.
This initial version is purposely minimalist, it's essentially moving
the generic part of the previously proposed mhi_wwan_ctrl driver inside
a common WWAN framework, but the implementation is open and flexible
enough to allow extension for further drivers.
Signed-off-by: Loic Poulain <loic.poulain@linaro.org>
Signed-off-by: David S. Miller <davem@davemloft.net>
2021-04-16 10:36:33 +02:00
if ( err )
goto error_put_device ;
return port ;
error_put_device :
put_device ( & port - > dev ) ;
error_wwandev_remove :
wwan_remove_dev ( wwandev ) ;
return ERR_PTR ( err ) ;
}
EXPORT_SYMBOL_GPL ( wwan_create_port ) ;
void wwan_remove_port ( struct wwan_port * port )
{
struct wwan_device * wwandev = to_wwan_dev ( port - > dev . parent ) ;
mutex_lock ( & port - > ops_lock ) ;
if ( port - > start_count )
port - > ops - > stop ( port ) ;
port - > ops = NULL ; /* Prevent any new port operations (e.g. from fops) */
mutex_unlock ( & port - > ops_lock ) ;
wake_up_interruptible ( & port - > waitqueue ) ;
skb_queue_purge ( & port - > rxq ) ;
dev_set_drvdata ( & port - > dev , NULL ) ;
device_unregister ( & port - > dev ) ;
/* Release related wwan device */
wwan_remove_dev ( wwandev ) ;
}
EXPORT_SYMBOL_GPL ( wwan_remove_port ) ;
void wwan_port_rx ( struct wwan_port * port , struct sk_buff * skb )
{
skb_queue_tail ( & port - > rxq , skb ) ;
wake_up_interruptible ( & port - > waitqueue ) ;
}
EXPORT_SYMBOL_GPL ( wwan_port_rx ) ;
void wwan_port_txon ( struct wwan_port * port )
{
clear_bit ( WWAN_PORT_TX_OFF , & port - > flags ) ;
wake_up_interruptible ( & port - > waitqueue ) ;
}
EXPORT_SYMBOL_GPL ( wwan_port_txon ) ;
void wwan_port_txoff ( struct wwan_port * port )
{
set_bit ( WWAN_PORT_TX_OFF , & port - > flags ) ;
}
EXPORT_SYMBOL_GPL ( wwan_port_txoff ) ;
void * wwan_port_get_drvdata ( struct wwan_port * port )
{
return dev_get_drvdata ( & port - > dev ) ;
}
EXPORT_SYMBOL_GPL ( wwan_port_get_drvdata ) ;
static int wwan_port_op_start ( struct wwan_port * port )
{
int ret = 0 ;
mutex_lock ( & port - > ops_lock ) ;
if ( ! port - > ops ) { /* Port got unplugged */
ret = - ENODEV ;
goto out_unlock ;
}
/* If port is already started, don't start again */
if ( ! port - > start_count )
ret = port - > ops - > start ( port ) ;
if ( ! ret )
port - > start_count + + ;
out_unlock :
mutex_unlock ( & port - > ops_lock ) ;
return ret ;
}
static void wwan_port_op_stop ( struct wwan_port * port )
{
mutex_lock ( & port - > ops_lock ) ;
port - > start_count - - ;
2021-06-08 07:02:41 +03:00
if ( ! port - > start_count ) {
if ( port - > ops )
port - > ops - > stop ( port ) ;
skb_queue_purge ( & port - > rxq ) ;
}
net: Add a WWAN subsystem
This change introduces initial support for a WWAN framework. Given the
complexity and heterogeneity of existing WWAN hardwares and interfaces,
there is no strict definition of what a WWAN device is and how it should
be represented. It's often a collection of multiple devices that perform
the global WWAN feature (netdev, tty, chardev, etc).
One usual way to expose modem controls and configuration is via high
level protocols such as the well known AT command protocol, MBIM or
QMI. The USB modems started to expose them as character devices, and
user daemons such as ModemManager learnt to use them.
This initial version adds the concept of WWAN port, which is a logical
pipe to a modem control protocol. The protocols are rawly exposed to
user via character device, allowing straigthforward support in existing
tools (ModemManager, ofono...). The WWAN core takes care of the generic
part, including character device management, and relies on port driver
operations to receive/submit protocol data.
Since the different devices exposing protocols for a same WWAN hardware
do not necessarily know about each others (e.g. two different USB
interfaces, PCI/MHI channel devices...) and can be created/removed in
different orders, the WWAN core ensures that all WAN ports contributing
to the 'whole' WWAN feature are grouped under the same virtual WWAN
device, relying on the provided parent device (e.g. mhi controller,
USB device). It's a 'trick' I copied from Johannes's earlier WWAN
subsystem proposal.
This initial version is purposely minimalist, it's essentially moving
the generic part of the previously proposed mhi_wwan_ctrl driver inside
a common WWAN framework, but the implementation is open and flexible
enough to allow extension for further drivers.
Signed-off-by: Loic Poulain <loic.poulain@linaro.org>
Signed-off-by: David S. Miller <davem@davemloft.net>
2021-04-16 10:36:33 +02:00
mutex_unlock ( & port - > ops_lock ) ;
}
net: wwan: Allow WWAN drivers to provide blocking tx and poll function
At the moment, the WWAN core provides wwan_port_txon/off() to implement
blocking writes. The tx() port operation should not block, instead
wwan_port_txon/off() should be called when the TX queue is full or has
free space again.
However, in some cases it is not straightforward to make use of that
functionality. For example, the RPMSG API used by rpmsg_wwan_ctrl.c
does not provide any way to be notified when the TX queue has space
again. Instead, it only provides the following operations:
- rpmsg_send(): blocking write (wait until there is space)
- rpmsg_trysend(): non-blocking write (return error if no space)
- rpmsg_poll(): set poll flags depending on TX queue state
Generally that's totally sufficient for implementing a char device,
but it does not fit well to the currently provided WWAN port ops.
Most of the time, using the non-blocking rpmsg_trysend() in the
WWAN tx() port operation works just fine. However, with high-frequent
writes to the char device it is possible to trigger a situation
where this causes issues. For example, consider the following
(somewhat unrealistic) example:
# dd if=/dev/zero bs=1000 of=/dev/wwan0qmi0
dd: error writing '/dev/wwan0qmi0': Resource temporarily unavailable
1+0 records out
This fails immediately after writing the first record. It's likely
only a matter of time until this triggers issues for some real application
(e.g. ModemManager sending a lot of large QMI packets).
The rpmsg_char device does not have this problem, because it uses
rpmsg_trysend() and rpmsg_poll() to support non-blocking operations.
Make it possible to use the same in the RPMSG WWAN driver by adding
two new optional wwan_port_ops:
- tx_blocking(): send data blocking if allowed
- tx_poll(): set additional TX poll flags
This integrates nicely with the RPMSG API and does not require
any change in existing WWAN drivers.
With these changes, the dd example above blocks instead of exiting
with an error.
Cc: Loic Poulain <loic.poulain@linaro.org>
Signed-off-by: Stephan Gerhold <stephan@gerhold.net>
Signed-off-by: David S. Miller <davem@davemloft.net>
2021-06-18 19:36:11 +02:00
static int wwan_port_op_tx ( struct wwan_port * port , struct sk_buff * skb ,
bool nonblock )
net: Add a WWAN subsystem
This change introduces initial support for a WWAN framework. Given the
complexity and heterogeneity of existing WWAN hardwares and interfaces,
there is no strict definition of what a WWAN device is and how it should
be represented. It's often a collection of multiple devices that perform
the global WWAN feature (netdev, tty, chardev, etc).
One usual way to expose modem controls and configuration is via high
level protocols such as the well known AT command protocol, MBIM or
QMI. The USB modems started to expose them as character devices, and
user daemons such as ModemManager learnt to use them.
This initial version adds the concept of WWAN port, which is a logical
pipe to a modem control protocol. The protocols are rawly exposed to
user via character device, allowing straigthforward support in existing
tools (ModemManager, ofono...). The WWAN core takes care of the generic
part, including character device management, and relies on port driver
operations to receive/submit protocol data.
Since the different devices exposing protocols for a same WWAN hardware
do not necessarily know about each others (e.g. two different USB
interfaces, PCI/MHI channel devices...) and can be created/removed in
different orders, the WWAN core ensures that all WAN ports contributing
to the 'whole' WWAN feature are grouped under the same virtual WWAN
device, relying on the provided parent device (e.g. mhi controller,
USB device). It's a 'trick' I copied from Johannes's earlier WWAN
subsystem proposal.
This initial version is purposely minimalist, it's essentially moving
the generic part of the previously proposed mhi_wwan_ctrl driver inside
a common WWAN framework, but the implementation is open and flexible
enough to allow extension for further drivers.
Signed-off-by: Loic Poulain <loic.poulain@linaro.org>
Signed-off-by: David S. Miller <davem@davemloft.net>
2021-04-16 10:36:33 +02:00
{
int ret ;
mutex_lock ( & port - > ops_lock ) ;
if ( ! port - > ops ) { /* Port got unplugged */
ret = - ENODEV ;
goto out_unlock ;
}
net: wwan: Allow WWAN drivers to provide blocking tx and poll function
At the moment, the WWAN core provides wwan_port_txon/off() to implement
blocking writes. The tx() port operation should not block, instead
wwan_port_txon/off() should be called when the TX queue is full or has
free space again.
However, in some cases it is not straightforward to make use of that
functionality. For example, the RPMSG API used by rpmsg_wwan_ctrl.c
does not provide any way to be notified when the TX queue has space
again. Instead, it only provides the following operations:
- rpmsg_send(): blocking write (wait until there is space)
- rpmsg_trysend(): non-blocking write (return error if no space)
- rpmsg_poll(): set poll flags depending on TX queue state
Generally that's totally sufficient for implementing a char device,
but it does not fit well to the currently provided WWAN port ops.
Most of the time, using the non-blocking rpmsg_trysend() in the
WWAN tx() port operation works just fine. However, with high-frequent
writes to the char device it is possible to trigger a situation
where this causes issues. For example, consider the following
(somewhat unrealistic) example:
# dd if=/dev/zero bs=1000 of=/dev/wwan0qmi0
dd: error writing '/dev/wwan0qmi0': Resource temporarily unavailable
1+0 records out
This fails immediately after writing the first record. It's likely
only a matter of time until this triggers issues for some real application
(e.g. ModemManager sending a lot of large QMI packets).
The rpmsg_char device does not have this problem, because it uses
rpmsg_trysend() and rpmsg_poll() to support non-blocking operations.
Make it possible to use the same in the RPMSG WWAN driver by adding
two new optional wwan_port_ops:
- tx_blocking(): send data blocking if allowed
- tx_poll(): set additional TX poll flags
This integrates nicely with the RPMSG API and does not require
any change in existing WWAN drivers.
With these changes, the dd example above blocks instead of exiting
with an error.
Cc: Loic Poulain <loic.poulain@linaro.org>
Signed-off-by: Stephan Gerhold <stephan@gerhold.net>
Signed-off-by: David S. Miller <davem@davemloft.net>
2021-06-18 19:36:11 +02:00
if ( nonblock | | ! port - > ops - > tx_blocking )
ret = port - > ops - > tx ( port , skb ) ;
else
ret = port - > ops - > tx_blocking ( port , skb ) ;
net: Add a WWAN subsystem
This change introduces initial support for a WWAN framework. Given the
complexity and heterogeneity of existing WWAN hardwares and interfaces,
there is no strict definition of what a WWAN device is and how it should
be represented. It's often a collection of multiple devices that perform
the global WWAN feature (netdev, tty, chardev, etc).
One usual way to expose modem controls and configuration is via high
level protocols such as the well known AT command protocol, MBIM or
QMI. The USB modems started to expose them as character devices, and
user daemons such as ModemManager learnt to use them.
This initial version adds the concept of WWAN port, which is a logical
pipe to a modem control protocol. The protocols are rawly exposed to
user via character device, allowing straigthforward support in existing
tools (ModemManager, ofono...). The WWAN core takes care of the generic
part, including character device management, and relies on port driver
operations to receive/submit protocol data.
Since the different devices exposing protocols for a same WWAN hardware
do not necessarily know about each others (e.g. two different USB
interfaces, PCI/MHI channel devices...) and can be created/removed in
different orders, the WWAN core ensures that all WAN ports contributing
to the 'whole' WWAN feature are grouped under the same virtual WWAN
device, relying on the provided parent device (e.g. mhi controller,
USB device). It's a 'trick' I copied from Johannes's earlier WWAN
subsystem proposal.
This initial version is purposely minimalist, it's essentially moving
the generic part of the previously proposed mhi_wwan_ctrl driver inside
a common WWAN framework, but the implementation is open and flexible
enough to allow extension for further drivers.
Signed-off-by: Loic Poulain <loic.poulain@linaro.org>
Signed-off-by: David S. Miller <davem@davemloft.net>
2021-04-16 10:36:33 +02:00
out_unlock :
mutex_unlock ( & port - > ops_lock ) ;
return ret ;
}
static bool is_read_blocked ( struct wwan_port * port )
{
return skb_queue_empty ( & port - > rxq ) & & port - > ops ;
}
static bool is_write_blocked ( struct wwan_port * port )
{
return test_bit ( WWAN_PORT_TX_OFF , & port - > flags ) & & port - > ops ;
}
static int wwan_wait_rx ( struct wwan_port * port , bool nonblock )
{
if ( ! is_read_blocked ( port ) )
return 0 ;
if ( nonblock )
return - EAGAIN ;
if ( wait_event_interruptible ( port - > waitqueue , ! is_read_blocked ( port ) ) )
return - ERESTARTSYS ;
return 0 ;
}
static int wwan_wait_tx ( struct wwan_port * port , bool nonblock )
{
if ( ! is_write_blocked ( port ) )
return 0 ;
if ( nonblock )
return - EAGAIN ;
if ( wait_event_interruptible ( port - > waitqueue , ! is_write_blocked ( port ) ) )
return - ERESTARTSYS ;
return 0 ;
}
static int wwan_port_fops_open ( struct inode * inode , struct file * file )
{
struct wwan_port * port ;
int err = 0 ;
port = wwan_port_get_by_minor ( iminor ( inode ) ) ;
if ( IS_ERR ( port ) )
return PTR_ERR ( port ) ;
file - > private_data = port ;
stream_open ( inode , file ) ;
err = wwan_port_op_start ( port ) ;
if ( err )
put_device ( & port - > dev ) ;
return err ;
}
static int wwan_port_fops_release ( struct inode * inode , struct file * filp )
{
struct wwan_port * port = filp - > private_data ;
wwan_port_op_stop ( port ) ;
put_device ( & port - > dev ) ;
return 0 ;
}
static ssize_t wwan_port_fops_read ( struct file * filp , char __user * buf ,
size_t count , loff_t * ppos )
{
struct wwan_port * port = filp - > private_data ;
struct sk_buff * skb ;
size_t copied ;
int ret ;
ret = wwan_wait_rx ( port , ! ! ( filp - > f_flags & O_NONBLOCK ) ) ;
if ( ret )
return ret ;
skb = skb_dequeue ( & port - > rxq ) ;
if ( ! skb )
return - EIO ;
copied = min_t ( size_t , count , skb - > len ) ;
if ( copy_to_user ( buf , skb - > data , copied ) ) {
kfree_skb ( skb ) ;
return - EFAULT ;
}
skb_pull ( skb , copied ) ;
/* skb is not fully consumed, keep it in the queue */
if ( skb - > len )
skb_queue_head ( & port - > rxq , skb ) ;
else
consume_skb ( skb ) ;
return copied ;
}
static ssize_t wwan_port_fops_write ( struct file * filp , const char __user * buf ,
size_t count , loff_t * offp )
{
struct wwan_port * port = filp - > private_data ;
struct sk_buff * skb ;
int ret ;
ret = wwan_wait_tx ( port , ! ! ( filp - > f_flags & O_NONBLOCK ) ) ;
if ( ret )
return ret ;
skb = alloc_skb ( count , GFP_KERNEL ) ;
if ( ! skb )
return - ENOMEM ;
if ( copy_from_user ( skb_put ( skb , count ) , buf , count ) ) {
kfree_skb ( skb ) ;
return - EFAULT ;
}
net: wwan: Allow WWAN drivers to provide blocking tx and poll function
At the moment, the WWAN core provides wwan_port_txon/off() to implement
blocking writes. The tx() port operation should not block, instead
wwan_port_txon/off() should be called when the TX queue is full or has
free space again.
However, in some cases it is not straightforward to make use of that
functionality. For example, the RPMSG API used by rpmsg_wwan_ctrl.c
does not provide any way to be notified when the TX queue has space
again. Instead, it only provides the following operations:
- rpmsg_send(): blocking write (wait until there is space)
- rpmsg_trysend(): non-blocking write (return error if no space)
- rpmsg_poll(): set poll flags depending on TX queue state
Generally that's totally sufficient for implementing a char device,
but it does not fit well to the currently provided WWAN port ops.
Most of the time, using the non-blocking rpmsg_trysend() in the
WWAN tx() port operation works just fine. However, with high-frequent
writes to the char device it is possible to trigger a situation
where this causes issues. For example, consider the following
(somewhat unrealistic) example:
# dd if=/dev/zero bs=1000 of=/dev/wwan0qmi0
dd: error writing '/dev/wwan0qmi0': Resource temporarily unavailable
1+0 records out
This fails immediately after writing the first record. It's likely
only a matter of time until this triggers issues for some real application
(e.g. ModemManager sending a lot of large QMI packets).
The rpmsg_char device does not have this problem, because it uses
rpmsg_trysend() and rpmsg_poll() to support non-blocking operations.
Make it possible to use the same in the RPMSG WWAN driver by adding
two new optional wwan_port_ops:
- tx_blocking(): send data blocking if allowed
- tx_poll(): set additional TX poll flags
This integrates nicely with the RPMSG API and does not require
any change in existing WWAN drivers.
With these changes, the dd example above blocks instead of exiting
with an error.
Cc: Loic Poulain <loic.poulain@linaro.org>
Signed-off-by: Stephan Gerhold <stephan@gerhold.net>
Signed-off-by: David S. Miller <davem@davemloft.net>
2021-06-18 19:36:11 +02:00
ret = wwan_port_op_tx ( port , skb , ! ! ( filp - > f_flags & O_NONBLOCK ) ) ;
net: Add a WWAN subsystem
This change introduces initial support for a WWAN framework. Given the
complexity and heterogeneity of existing WWAN hardwares and interfaces,
there is no strict definition of what a WWAN device is and how it should
be represented. It's often a collection of multiple devices that perform
the global WWAN feature (netdev, tty, chardev, etc).
One usual way to expose modem controls and configuration is via high
level protocols such as the well known AT command protocol, MBIM or
QMI. The USB modems started to expose them as character devices, and
user daemons such as ModemManager learnt to use them.
This initial version adds the concept of WWAN port, which is a logical
pipe to a modem control protocol. The protocols are rawly exposed to
user via character device, allowing straigthforward support in existing
tools (ModemManager, ofono...). The WWAN core takes care of the generic
part, including character device management, and relies on port driver
operations to receive/submit protocol data.
Since the different devices exposing protocols for a same WWAN hardware
do not necessarily know about each others (e.g. two different USB
interfaces, PCI/MHI channel devices...) and can be created/removed in
different orders, the WWAN core ensures that all WAN ports contributing
to the 'whole' WWAN feature are grouped under the same virtual WWAN
device, relying on the provided parent device (e.g. mhi controller,
USB device). It's a 'trick' I copied from Johannes's earlier WWAN
subsystem proposal.
This initial version is purposely minimalist, it's essentially moving
the generic part of the previously proposed mhi_wwan_ctrl driver inside
a common WWAN framework, but the implementation is open and flexible
enough to allow extension for further drivers.
Signed-off-by: Loic Poulain <loic.poulain@linaro.org>
Signed-off-by: David S. Miller <davem@davemloft.net>
2021-04-16 10:36:33 +02:00
if ( ret ) {
kfree_skb ( skb ) ;
return ret ;
}
return count ;
}
static __poll_t wwan_port_fops_poll ( struct file * filp , poll_table * wait )
{
struct wwan_port * port = filp - > private_data ;
__poll_t mask = 0 ;
poll_wait ( filp , & port - > waitqueue , wait ) ;
net: wwan: Allow WWAN drivers to provide blocking tx and poll function
At the moment, the WWAN core provides wwan_port_txon/off() to implement
blocking writes. The tx() port operation should not block, instead
wwan_port_txon/off() should be called when the TX queue is full or has
free space again.
However, in some cases it is not straightforward to make use of that
functionality. For example, the RPMSG API used by rpmsg_wwan_ctrl.c
does not provide any way to be notified when the TX queue has space
again. Instead, it only provides the following operations:
- rpmsg_send(): blocking write (wait until there is space)
- rpmsg_trysend(): non-blocking write (return error if no space)
- rpmsg_poll(): set poll flags depending on TX queue state
Generally that's totally sufficient for implementing a char device,
but it does not fit well to the currently provided WWAN port ops.
Most of the time, using the non-blocking rpmsg_trysend() in the
WWAN tx() port operation works just fine. However, with high-frequent
writes to the char device it is possible to trigger a situation
where this causes issues. For example, consider the following
(somewhat unrealistic) example:
# dd if=/dev/zero bs=1000 of=/dev/wwan0qmi0
dd: error writing '/dev/wwan0qmi0': Resource temporarily unavailable
1+0 records out
This fails immediately after writing the first record. It's likely
only a matter of time until this triggers issues for some real application
(e.g. ModemManager sending a lot of large QMI packets).
The rpmsg_char device does not have this problem, because it uses
rpmsg_trysend() and rpmsg_poll() to support non-blocking operations.
Make it possible to use the same in the RPMSG WWAN driver by adding
two new optional wwan_port_ops:
- tx_blocking(): send data blocking if allowed
- tx_poll(): set additional TX poll flags
This integrates nicely with the RPMSG API and does not require
any change in existing WWAN drivers.
With these changes, the dd example above blocks instead of exiting
with an error.
Cc: Loic Poulain <loic.poulain@linaro.org>
Signed-off-by: Stephan Gerhold <stephan@gerhold.net>
Signed-off-by: David S. Miller <davem@davemloft.net>
2021-06-18 19:36:11 +02:00
mutex_lock ( & port - > ops_lock ) ;
if ( port - > ops & & port - > ops - > tx_poll )
mask | = port - > ops - > tx_poll ( port , filp , wait ) ;
else if ( ! is_write_blocked ( port ) )
net: Add a WWAN subsystem
This change introduces initial support for a WWAN framework. Given the
complexity and heterogeneity of existing WWAN hardwares and interfaces,
there is no strict definition of what a WWAN device is and how it should
be represented. It's often a collection of multiple devices that perform
the global WWAN feature (netdev, tty, chardev, etc).
One usual way to expose modem controls and configuration is via high
level protocols such as the well known AT command protocol, MBIM or
QMI. The USB modems started to expose them as character devices, and
user daemons such as ModemManager learnt to use them.
This initial version adds the concept of WWAN port, which is a logical
pipe to a modem control protocol. The protocols are rawly exposed to
user via character device, allowing straigthforward support in existing
tools (ModemManager, ofono...). The WWAN core takes care of the generic
part, including character device management, and relies on port driver
operations to receive/submit protocol data.
Since the different devices exposing protocols for a same WWAN hardware
do not necessarily know about each others (e.g. two different USB
interfaces, PCI/MHI channel devices...) and can be created/removed in
different orders, the WWAN core ensures that all WAN ports contributing
to the 'whole' WWAN feature are grouped under the same virtual WWAN
device, relying on the provided parent device (e.g. mhi controller,
USB device). It's a 'trick' I copied from Johannes's earlier WWAN
subsystem proposal.
This initial version is purposely minimalist, it's essentially moving
the generic part of the previously proposed mhi_wwan_ctrl driver inside
a common WWAN framework, but the implementation is open and flexible
enough to allow extension for further drivers.
Signed-off-by: Loic Poulain <loic.poulain@linaro.org>
Signed-off-by: David S. Miller <davem@davemloft.net>
2021-04-16 10:36:33 +02:00
mask | = EPOLLOUT | EPOLLWRNORM ;
if ( ! is_read_blocked ( port ) )
mask | = EPOLLIN | EPOLLRDNORM ;
2021-04-22 16:06:01 +02:00
if ( ! port - > ops )
mask | = EPOLLHUP | EPOLLERR ;
net: wwan: Allow WWAN drivers to provide blocking tx and poll function
At the moment, the WWAN core provides wwan_port_txon/off() to implement
blocking writes. The tx() port operation should not block, instead
wwan_port_txon/off() should be called when the TX queue is full or has
free space again.
However, in some cases it is not straightforward to make use of that
functionality. For example, the RPMSG API used by rpmsg_wwan_ctrl.c
does not provide any way to be notified when the TX queue has space
again. Instead, it only provides the following operations:
- rpmsg_send(): blocking write (wait until there is space)
- rpmsg_trysend(): non-blocking write (return error if no space)
- rpmsg_poll(): set poll flags depending on TX queue state
Generally that's totally sufficient for implementing a char device,
but it does not fit well to the currently provided WWAN port ops.
Most of the time, using the non-blocking rpmsg_trysend() in the
WWAN tx() port operation works just fine. However, with high-frequent
writes to the char device it is possible to trigger a situation
where this causes issues. For example, consider the following
(somewhat unrealistic) example:
# dd if=/dev/zero bs=1000 of=/dev/wwan0qmi0
dd: error writing '/dev/wwan0qmi0': Resource temporarily unavailable
1+0 records out
This fails immediately after writing the first record. It's likely
only a matter of time until this triggers issues for some real application
(e.g. ModemManager sending a lot of large QMI packets).
The rpmsg_char device does not have this problem, because it uses
rpmsg_trysend() and rpmsg_poll() to support non-blocking operations.
Make it possible to use the same in the RPMSG WWAN driver by adding
two new optional wwan_port_ops:
- tx_blocking(): send data blocking if allowed
- tx_poll(): set additional TX poll flags
This integrates nicely with the RPMSG API and does not require
any change in existing WWAN drivers.
With these changes, the dd example above blocks instead of exiting
with an error.
Cc: Loic Poulain <loic.poulain@linaro.org>
Signed-off-by: Stephan Gerhold <stephan@gerhold.net>
Signed-off-by: David S. Miller <davem@davemloft.net>
2021-06-18 19:36:11 +02:00
mutex_unlock ( & port - > ops_lock ) ;
net: Add a WWAN subsystem
This change introduces initial support for a WWAN framework. Given the
complexity and heterogeneity of existing WWAN hardwares and interfaces,
there is no strict definition of what a WWAN device is and how it should
be represented. It's often a collection of multiple devices that perform
the global WWAN feature (netdev, tty, chardev, etc).
One usual way to expose modem controls and configuration is via high
level protocols such as the well known AT command protocol, MBIM or
QMI. The USB modems started to expose them as character devices, and
user daemons such as ModemManager learnt to use them.
This initial version adds the concept of WWAN port, which is a logical
pipe to a modem control protocol. The protocols are rawly exposed to
user via character device, allowing straigthforward support in existing
tools (ModemManager, ofono...). The WWAN core takes care of the generic
part, including character device management, and relies on port driver
operations to receive/submit protocol data.
Since the different devices exposing protocols for a same WWAN hardware
do not necessarily know about each others (e.g. two different USB
interfaces, PCI/MHI channel devices...) and can be created/removed in
different orders, the WWAN core ensures that all WAN ports contributing
to the 'whole' WWAN feature are grouped under the same virtual WWAN
device, relying on the provided parent device (e.g. mhi controller,
USB device). It's a 'trick' I copied from Johannes's earlier WWAN
subsystem proposal.
This initial version is purposely minimalist, it's essentially moving
the generic part of the previously proposed mhi_wwan_ctrl driver inside
a common WWAN framework, but the implementation is open and flexible
enough to allow extension for further drivers.
Signed-off-by: Loic Poulain <loic.poulain@linaro.org>
Signed-off-by: David S. Miller <davem@davemloft.net>
2021-04-16 10:36:33 +02:00
return mask ;
}
2021-06-08 07:02:40 +03:00
/* Implements minimalistic stub terminal IOCTLs support */
static long wwan_port_fops_at_ioctl ( struct wwan_port * port , unsigned int cmd ,
unsigned long arg )
{
int ret = 0 ;
mutex_lock ( & port - > data_lock ) ;
switch ( cmd ) {
case TCFLSH :
break ;
case TCGETS :
if ( copy_to_user ( ( void __user * ) arg , & port - > at_data . termios ,
sizeof ( struct termios ) ) )
ret = - EFAULT ;
break ;
case TCSETS :
case TCSETSW :
case TCSETSF :
if ( copy_from_user ( & port - > at_data . termios , ( void __user * ) arg ,
sizeof ( struct termios ) ) )
ret = - EFAULT ;
break ;
# ifdef TCGETS2
case TCGETS2 :
if ( copy_to_user ( ( void __user * ) arg , & port - > at_data . termios ,
sizeof ( struct termios2 ) ) )
ret = - EFAULT ;
break ;
case TCSETS2 :
case TCSETSW2 :
case TCSETSF2 :
if ( copy_from_user ( & port - > at_data . termios , ( void __user * ) arg ,
sizeof ( struct termios2 ) ) )
ret = - EFAULT ;
break ;
# endif
case TIOCMGET :
ret = put_user ( port - > at_data . mdmbits , ( int __user * ) arg ) ;
break ;
case TIOCMSET :
case TIOCMBIC :
case TIOCMBIS : {
int mdmbits ;
if ( copy_from_user ( & mdmbits , ( int __user * ) arg , sizeof ( int ) ) ) {
ret = - EFAULT ;
break ;
}
if ( cmd = = TIOCMBIC )
port - > at_data . mdmbits & = ~ mdmbits ;
else if ( cmd = = TIOCMBIS )
port - > at_data . mdmbits | = mdmbits ;
else
port - > at_data . mdmbits = mdmbits ;
break ;
}
default :
ret = - ENOIOCTLCMD ;
}
mutex_unlock ( & port - > data_lock ) ;
return ret ;
}
2021-06-08 07:02:39 +03:00
static long wwan_port_fops_ioctl ( struct file * filp , unsigned int cmd ,
unsigned long arg )
{
struct wwan_port * port = filp - > private_data ;
2021-06-08 07:02:40 +03:00
int res ;
if ( port - > type = = WWAN_PORT_AT ) { /* AT port specific IOCTLs */
res = wwan_port_fops_at_ioctl ( port , cmd , arg ) ;
if ( res ! = - ENOIOCTLCMD )
return res ;
}
2021-06-08 07:02:39 +03:00
switch ( cmd ) {
case TIOCINQ : { /* aka SIOCINQ aka FIONREAD */
unsigned long flags ;
struct sk_buff * skb ;
int amount = 0 ;
spin_lock_irqsave ( & port - > rxq . lock , flags ) ;
skb_queue_walk ( & port - > rxq , skb )
amount + = skb - > len ;
spin_unlock_irqrestore ( & port - > rxq . lock , flags ) ;
return put_user ( amount , ( int __user * ) arg ) ;
}
default :
return - ENOIOCTLCMD ;
}
}
net: Add a WWAN subsystem
This change introduces initial support for a WWAN framework. Given the
complexity and heterogeneity of existing WWAN hardwares and interfaces,
there is no strict definition of what a WWAN device is and how it should
be represented. It's often a collection of multiple devices that perform
the global WWAN feature (netdev, tty, chardev, etc).
One usual way to expose modem controls and configuration is via high
level protocols such as the well known AT command protocol, MBIM or
QMI. The USB modems started to expose them as character devices, and
user daemons such as ModemManager learnt to use them.
This initial version adds the concept of WWAN port, which is a logical
pipe to a modem control protocol. The protocols are rawly exposed to
user via character device, allowing straigthforward support in existing
tools (ModemManager, ofono...). The WWAN core takes care of the generic
part, including character device management, and relies on port driver
operations to receive/submit protocol data.
Since the different devices exposing protocols for a same WWAN hardware
do not necessarily know about each others (e.g. two different USB
interfaces, PCI/MHI channel devices...) and can be created/removed in
different orders, the WWAN core ensures that all WAN ports contributing
to the 'whole' WWAN feature are grouped under the same virtual WWAN
device, relying on the provided parent device (e.g. mhi controller,
USB device). It's a 'trick' I copied from Johannes's earlier WWAN
subsystem proposal.
This initial version is purposely minimalist, it's essentially moving
the generic part of the previously proposed mhi_wwan_ctrl driver inside
a common WWAN framework, but the implementation is open and flexible
enough to allow extension for further drivers.
Signed-off-by: Loic Poulain <loic.poulain@linaro.org>
Signed-off-by: David S. Miller <davem@davemloft.net>
2021-04-16 10:36:33 +02:00
static const struct file_operations wwan_port_fops = {
. owner = THIS_MODULE ,
. open = wwan_port_fops_open ,
. release = wwan_port_fops_release ,
. read = wwan_port_fops_read ,
. write = wwan_port_fops_write ,
. poll = wwan_port_fops_poll ,
2021-06-08 07:02:39 +03:00
. unlocked_ioctl = wwan_port_fops_ioctl ,
# ifdef CONFIG_COMPAT
. compat_ioctl = compat_ptr_ioctl ,
# endif
net: Add a WWAN subsystem
This change introduces initial support for a WWAN framework. Given the
complexity and heterogeneity of existing WWAN hardwares and interfaces,
there is no strict definition of what a WWAN device is and how it should
be represented. It's often a collection of multiple devices that perform
the global WWAN feature (netdev, tty, chardev, etc).
One usual way to expose modem controls and configuration is via high
level protocols such as the well known AT command protocol, MBIM or
QMI. The USB modems started to expose them as character devices, and
user daemons such as ModemManager learnt to use them.
This initial version adds the concept of WWAN port, which is a logical
pipe to a modem control protocol. The protocols are rawly exposed to
user via character device, allowing straigthforward support in existing
tools (ModemManager, ofono...). The WWAN core takes care of the generic
part, including character device management, and relies on port driver
operations to receive/submit protocol data.
Since the different devices exposing protocols for a same WWAN hardware
do not necessarily know about each others (e.g. two different USB
interfaces, PCI/MHI channel devices...) and can be created/removed in
different orders, the WWAN core ensures that all WAN ports contributing
to the 'whole' WWAN feature are grouped under the same virtual WWAN
device, relying on the provided parent device (e.g. mhi controller,
USB device). It's a 'trick' I copied from Johannes's earlier WWAN
subsystem proposal.
This initial version is purposely minimalist, it's essentially moving
the generic part of the previously proposed mhi_wwan_ctrl driver inside
a common WWAN framework, but the implementation is open and flexible
enough to allow extension for further drivers.
Signed-off-by: Loic Poulain <loic.poulain@linaro.org>
Signed-off-by: David S. Miller <davem@davemloft.net>
2021-04-16 10:36:33 +02:00
. llseek = noop_llseek ,
} ;
2021-06-12 10:20:56 +02:00
static int wwan_rtnl_validate ( struct nlattr * tb [ ] , struct nlattr * data [ ] ,
struct netlink_ext_ack * extack )
{
if ( ! data )
return - EINVAL ;
if ( ! tb [ IFLA_PARENT_DEV_NAME ] )
return - EINVAL ;
if ( ! data [ IFLA_WWAN_LINK_ID ] )
return - EINVAL ;
return 0 ;
}
static struct device_type wwan_type = { . name = " wwan " } ;
static struct net_device * wwan_rtnl_alloc ( struct nlattr * tb [ ] ,
const char * ifname ,
unsigned char name_assign_type ,
unsigned int num_tx_queues ,
unsigned int num_rx_queues )
{
const char * devname = nla_data ( tb [ IFLA_PARENT_DEV_NAME ] ) ;
struct wwan_device * wwandev = wwan_dev_get_by_name ( devname ) ;
struct net_device * dev ;
2021-06-22 01:51:00 +03:00
unsigned int priv_size ;
2021-06-12 10:20:56 +02:00
if ( IS_ERR ( wwandev ) )
return ERR_CAST ( wwandev ) ;
/* only supported if ops were registered (not just ports) */
if ( ! wwandev - > ops ) {
dev = ERR_PTR ( - EOPNOTSUPP ) ;
goto out ;
}
2021-06-22 01:51:00 +03:00
priv_size = sizeof ( struct wwan_netdev_priv ) + wwandev - > ops - > priv_size ;
dev = alloc_netdev_mqs ( priv_size , ifname , name_assign_type ,
2021-06-12 10:20:56 +02:00
wwandev - > ops - > setup , num_tx_queues , num_rx_queues ) ;
if ( dev ) {
SET_NETDEV_DEV ( dev , & wwandev - > dev ) ;
SET_NETDEV_DEVTYPE ( dev , & wwan_type ) ;
}
out :
/* release the reference */
put_device ( & wwandev - > dev ) ;
return dev ;
}
static int wwan_rtnl_newlink ( struct net * src_net , struct net_device * dev ,
struct nlattr * tb [ ] , struct nlattr * data [ ] ,
struct netlink_ext_ack * extack )
{
struct wwan_device * wwandev = wwan_dev_get_by_parent ( dev - > dev . parent ) ;
u32 link_id = nla_get_u32 ( data [ IFLA_WWAN_LINK_ID ] ) ;
2021-06-22 01:51:00 +03:00
struct wwan_netdev_priv * priv = netdev_priv ( dev ) ;
2021-06-12 10:20:56 +02:00
int ret ;
if ( IS_ERR ( wwandev ) )
return PTR_ERR ( wwandev ) ;
/* shouldn't have a netdev (left) with us as parent so WARN */
if ( WARN_ON ( ! wwandev - > ops ) ) {
ret = - EOPNOTSUPP ;
goto out ;
}
2021-06-22 01:51:00 +03:00
priv - > link_id = link_id ;
2021-06-12 10:20:56 +02:00
if ( wwandev - > ops - > newlink )
ret = wwandev - > ops - > newlink ( wwandev - > ops_ctxt , dev ,
link_id , extack ) ;
else
ret = register_netdevice ( dev ) ;
out :
/* release the reference */
put_device ( & wwandev - > dev ) ;
return ret ;
}
static void wwan_rtnl_dellink ( struct net_device * dev , struct list_head * head )
{
struct wwan_device * wwandev = wwan_dev_get_by_parent ( dev - > dev . parent ) ;
if ( IS_ERR ( wwandev ) )
return ;
/* shouldn't have a netdev (left) with us as parent so WARN */
if ( WARN_ON ( ! wwandev - > ops ) )
goto out ;
if ( wwandev - > ops - > dellink )
wwandev - > ops - > dellink ( wwandev - > ops_ctxt , dev , head ) ;
else
2021-06-22 01:50:54 +03:00
unregister_netdevice_queue ( dev , head ) ;
2021-06-12 10:20:56 +02:00
out :
/* release the reference */
put_device ( & wwandev - > dev ) ;
}
2021-06-22 01:51:00 +03:00
static size_t wwan_rtnl_get_size ( const struct net_device * dev )
{
return
nla_total_size ( 4 ) + /* IFLA_WWAN_LINK_ID */
0 ;
}
static int wwan_rtnl_fill_info ( struct sk_buff * skb ,
const struct net_device * dev )
{
struct wwan_netdev_priv * priv = netdev_priv ( dev ) ;
if ( nla_put_u32 ( skb , IFLA_WWAN_LINK_ID , priv - > link_id ) )
goto nla_put_failure ;
return 0 ;
nla_put_failure :
return - EMSGSIZE ;
}
2021-06-12 10:20:56 +02:00
static const struct nla_policy wwan_rtnl_policy [ IFLA_WWAN_MAX + 1 ] = {
[ IFLA_WWAN_LINK_ID ] = { . type = NLA_U32 } ,
} ;
static struct rtnl_link_ops wwan_rtnl_link_ops __read_mostly = {
. kind = " wwan " ,
. maxtype = __IFLA_WWAN_MAX ,
. alloc = wwan_rtnl_alloc ,
. validate = wwan_rtnl_validate ,
. newlink = wwan_rtnl_newlink ,
. dellink = wwan_rtnl_dellink ,
2021-06-22 01:51:00 +03:00
. get_size = wwan_rtnl_get_size ,
. fill_info = wwan_rtnl_fill_info ,
2021-06-12 10:20:56 +02:00
. policy = wwan_rtnl_policy ,
} ;
2021-06-22 01:50:58 +03:00
static void wwan_create_default_link ( struct wwan_device * wwandev ,
u32 def_link_id )
{
struct nlattr * tb [ IFLA_MAX + 1 ] , * linkinfo [ IFLA_INFO_MAX + 1 ] ;
struct nlattr * data [ IFLA_WWAN_MAX + 1 ] ;
struct net_device * dev ;
struct nlmsghdr * nlh ;
struct sk_buff * msg ;
/* Forge attributes required to create a WWAN netdev. We first
* build a netlink message and then parse it . This looks
* odd , but such approach is less error prone .
*/
msg = nlmsg_new ( NLMSG_DEFAULT_SIZE , GFP_KERNEL ) ;
if ( WARN_ON ( ! msg ) )
return ;
nlh = nlmsg_put ( msg , 0 , 0 , RTM_NEWLINK , 0 , 0 ) ;
if ( WARN_ON ( ! nlh ) )
goto free_attrs ;
if ( nla_put_string ( msg , IFLA_PARENT_DEV_NAME , dev_name ( & wwandev - > dev ) ) )
goto free_attrs ;
tb [ IFLA_LINKINFO ] = nla_nest_start ( msg , IFLA_LINKINFO ) ;
if ( ! tb [ IFLA_LINKINFO ] )
goto free_attrs ;
linkinfo [ IFLA_INFO_DATA ] = nla_nest_start ( msg , IFLA_INFO_DATA ) ;
if ( ! linkinfo [ IFLA_INFO_DATA ] )
goto free_attrs ;
if ( nla_put_u32 ( msg , IFLA_WWAN_LINK_ID , def_link_id ) )
goto free_attrs ;
nla_nest_end ( msg , linkinfo [ IFLA_INFO_DATA ] ) ;
nla_nest_end ( msg , tb [ IFLA_LINKINFO ] ) ;
nlmsg_end ( msg , nlh ) ;
/* The next three parsing calls can not fail */
nlmsg_parse_deprecated ( nlh , 0 , tb , IFLA_MAX , NULL , NULL ) ;
nla_parse_nested_deprecated ( linkinfo , IFLA_INFO_MAX , tb [ IFLA_LINKINFO ] ,
NULL , NULL ) ;
nla_parse_nested_deprecated ( data , IFLA_WWAN_MAX ,
linkinfo [ IFLA_INFO_DATA ] , NULL , NULL ) ;
rtnl_lock ( ) ;
dev = rtnl_create_link ( & init_net , " wwan%d " , NET_NAME_ENUM ,
& wwan_rtnl_link_ops , tb , NULL ) ;
if ( WARN_ON ( IS_ERR ( dev ) ) )
goto unlock ;
if ( WARN_ON ( wwan_rtnl_newlink ( & init_net , dev , tb , data , NULL ) ) ) {
free_netdev ( dev ) ;
goto unlock ;
}
2021-07-22 20:21:05 +02:00
rtnl_configure_link ( dev , NULL ) ; /* Link initialized, notify new link */
2021-06-22 01:50:58 +03:00
unlock :
rtnl_unlock ( ) ;
free_attrs :
nlmsg_free ( msg ) ;
}
2021-06-22 01:50:52 +03:00
/**
* wwan_register_ops - register WWAN device ops
* @ parent : Device to use as parent and shared by all WWAN ports and
* created netdevs
* @ ops : operations to register
* @ ctxt : context to pass to operations
2021-06-22 01:50:58 +03:00
* @ def_link_id : id of the default link that will be automatically created by
* the WWAN core for the WWAN device . The default link will not be created
* if the passed value is WWAN_NO_DEFAULT_LINK .
2021-06-22 01:50:52 +03:00
*
* Returns : 0 on success , a negative error code on failure
*/
int wwan_register_ops ( struct device * parent , const struct wwan_ops * ops ,
2021-06-22 01:50:58 +03:00
void * ctxt , u32 def_link_id )
2021-06-22 01:50:52 +03:00
{
struct wwan_device * wwandev ;
2021-06-22 01:50:53 +03:00
if ( WARN_ON ( ! parent | | ! ops | | ! ops - > setup ) )
2021-06-22 01:50:52 +03:00
return - EINVAL ;
wwandev = wwan_create_dev ( parent ) ;
if ( ! wwandev )
return - ENOMEM ;
if ( WARN_ON ( wwandev - > ops ) ) {
wwan_remove_dev ( wwandev ) ;
return - EBUSY ;
}
wwandev - > ops = ops ;
wwandev - > ops_ctxt = ctxt ;
2021-06-22 01:50:58 +03:00
/* NB: we do not abort ops registration in case of default link
* creation failure . Link ops is the management interface , while the
* default link creation is a service option . And we should not prevent
* a user from manually creating a link latter if service option failed
* now .
*/
if ( def_link_id ! = WWAN_NO_DEFAULT_LINK )
wwan_create_default_link ( wwandev , def_link_id ) ;
2021-06-22 01:50:52 +03:00
return 0 ;
}
EXPORT_SYMBOL_GPL ( wwan_register_ops ) ;
2021-06-22 01:50:55 +03:00
/* Enqueue child netdev deletion */
static int wwan_child_dellink ( struct device * dev , void * data )
{
struct list_head * kill_list = data ;
if ( dev - > type = = & wwan_type )
wwan_rtnl_dellink ( to_net_dev ( dev ) , kill_list ) ;
return 0 ;
}
2021-06-22 01:50:52 +03:00
/**
* wwan_unregister_ops - remove WWAN device ops
* @ parent : Device to use as parent and shared by all WWAN ports and
* created netdevs
*/
void wwan_unregister_ops ( struct device * parent )
{
struct wwan_device * wwandev = wwan_dev_get_by_parent ( parent ) ;
2021-06-22 01:50:55 +03:00
LIST_HEAD ( kill_list ) ;
2021-06-22 01:50:52 +03:00
if ( WARN_ON ( IS_ERR ( wwandev ) ) )
return ;
2021-06-22 01:50:55 +03:00
if ( WARN_ON ( ! wwandev - > ops ) ) {
put_device ( & wwandev - > dev ) ;
return ;
}
2021-06-22 01:50:52 +03:00
/* put the reference obtained by wwan_dev_get_by_parent(),
* we should still have one ( that the owner is giving back
2021-06-22 01:50:55 +03:00
* now ) due to the ops being assigned .
2021-06-22 01:50:52 +03:00
*/
put_device ( & wwandev - > dev ) ;
2021-06-22 01:50:55 +03:00
rtnl_lock ( ) ; /* Prevent concurent netdev(s) creation/destroying */
/* Remove all child netdev(s), using batch removing */
device_for_each_child ( & wwandev - > dev , & kill_list ,
wwan_child_dellink ) ;
unregister_netdevice_many ( & kill_list ) ;
wwandev - > ops = NULL ; /* Finally remove ops */
rtnl_unlock ( ) ;
2021-06-22 01:50:52 +03:00
wwandev - > ops_ctxt = NULL ;
wwan_remove_dev ( wwandev ) ;
}
EXPORT_SYMBOL_GPL ( wwan_unregister_ops ) ;
net: Add a WWAN subsystem
This change introduces initial support for a WWAN framework. Given the
complexity and heterogeneity of existing WWAN hardwares and interfaces,
there is no strict definition of what a WWAN device is and how it should
be represented. It's often a collection of multiple devices that perform
the global WWAN feature (netdev, tty, chardev, etc).
One usual way to expose modem controls and configuration is via high
level protocols such as the well known AT command protocol, MBIM or
QMI. The USB modems started to expose them as character devices, and
user daemons such as ModemManager learnt to use them.
This initial version adds the concept of WWAN port, which is a logical
pipe to a modem control protocol. The protocols are rawly exposed to
user via character device, allowing straigthforward support in existing
tools (ModemManager, ofono...). The WWAN core takes care of the generic
part, including character device management, and relies on port driver
operations to receive/submit protocol data.
Since the different devices exposing protocols for a same WWAN hardware
do not necessarily know about each others (e.g. two different USB
interfaces, PCI/MHI channel devices...) and can be created/removed in
different orders, the WWAN core ensures that all WAN ports contributing
to the 'whole' WWAN feature are grouped under the same virtual WWAN
device, relying on the provided parent device (e.g. mhi controller,
USB device). It's a 'trick' I copied from Johannes's earlier WWAN
subsystem proposal.
This initial version is purposely minimalist, it's essentially moving
the generic part of the previously proposed mhi_wwan_ctrl driver inside
a common WWAN framework, but the implementation is open and flexible
enough to allow extension for further drivers.
Signed-off-by: Loic Poulain <loic.poulain@linaro.org>
Signed-off-by: David S. Miller <davem@davemloft.net>
2021-04-16 10:36:33 +02:00
static int __init wwan_init ( void )
{
2021-06-12 10:20:56 +02:00
int err ;
err = rtnl_link_register ( & wwan_rtnl_link_ops ) ;
if ( err )
return err ;
net: Add a WWAN subsystem
This change introduces initial support for a WWAN framework. Given the
complexity and heterogeneity of existing WWAN hardwares and interfaces,
there is no strict definition of what a WWAN device is and how it should
be represented. It's often a collection of multiple devices that perform
the global WWAN feature (netdev, tty, chardev, etc).
One usual way to expose modem controls and configuration is via high
level protocols such as the well known AT command protocol, MBIM or
QMI. The USB modems started to expose them as character devices, and
user daemons such as ModemManager learnt to use them.
This initial version adds the concept of WWAN port, which is a logical
pipe to a modem control protocol. The protocols are rawly exposed to
user via character device, allowing straigthforward support in existing
tools (ModemManager, ofono...). The WWAN core takes care of the generic
part, including character device management, and relies on port driver
operations to receive/submit protocol data.
Since the different devices exposing protocols for a same WWAN hardware
do not necessarily know about each others (e.g. two different USB
interfaces, PCI/MHI channel devices...) and can be created/removed in
different orders, the WWAN core ensures that all WAN ports contributing
to the 'whole' WWAN feature are grouped under the same virtual WWAN
device, relying on the provided parent device (e.g. mhi controller,
USB device). It's a 'trick' I copied from Johannes's earlier WWAN
subsystem proposal.
This initial version is purposely minimalist, it's essentially moving
the generic part of the previously proposed mhi_wwan_ctrl driver inside
a common WWAN framework, but the implementation is open and flexible
enough to allow extension for further drivers.
Signed-off-by: Loic Poulain <loic.poulain@linaro.org>
Signed-off-by: David S. Miller <davem@davemloft.net>
2021-04-16 10:36:33 +02:00
wwan_class = class_create ( THIS_MODULE , " wwan " ) ;
2021-06-12 10:20:56 +02:00
if ( IS_ERR ( wwan_class ) ) {
err = PTR_ERR ( wwan_class ) ;
goto unregister ;
}
net: Add a WWAN subsystem
This change introduces initial support for a WWAN framework. Given the
complexity and heterogeneity of existing WWAN hardwares and interfaces,
there is no strict definition of what a WWAN device is and how it should
be represented. It's often a collection of multiple devices that perform
the global WWAN feature (netdev, tty, chardev, etc).
One usual way to expose modem controls and configuration is via high
level protocols such as the well known AT command protocol, MBIM or
QMI. The USB modems started to expose them as character devices, and
user daemons such as ModemManager learnt to use them.
This initial version adds the concept of WWAN port, which is a logical
pipe to a modem control protocol. The protocols are rawly exposed to
user via character device, allowing straigthforward support in existing
tools (ModemManager, ofono...). The WWAN core takes care of the generic
part, including character device management, and relies on port driver
operations to receive/submit protocol data.
Since the different devices exposing protocols for a same WWAN hardware
do not necessarily know about each others (e.g. two different USB
interfaces, PCI/MHI channel devices...) and can be created/removed in
different orders, the WWAN core ensures that all WAN ports contributing
to the 'whole' WWAN feature are grouped under the same virtual WWAN
device, relying on the provided parent device (e.g. mhi controller,
USB device). It's a 'trick' I copied from Johannes's earlier WWAN
subsystem proposal.
This initial version is purposely minimalist, it's essentially moving
the generic part of the previously proposed mhi_wwan_ctrl driver inside
a common WWAN framework, but the implementation is open and flexible
enough to allow extension for further drivers.
Signed-off-by: Loic Poulain <loic.poulain@linaro.org>
Signed-off-by: David S. Miller <davem@davemloft.net>
2021-04-16 10:36:33 +02:00
/* chrdev used for wwan ports */
2021-06-08 07:02:38 +03:00
wwan_major = __register_chrdev ( 0 , 0 , WWAN_MAX_MINORS , " wwan_port " ,
& wwan_port_fops ) ;
net: Add a WWAN subsystem
This change introduces initial support for a WWAN framework. Given the
complexity and heterogeneity of existing WWAN hardwares and interfaces,
there is no strict definition of what a WWAN device is and how it should
be represented. It's often a collection of multiple devices that perform
the global WWAN feature (netdev, tty, chardev, etc).
One usual way to expose modem controls and configuration is via high
level protocols such as the well known AT command protocol, MBIM or
QMI. The USB modems started to expose them as character devices, and
user daemons such as ModemManager learnt to use them.
This initial version adds the concept of WWAN port, which is a logical
pipe to a modem control protocol. The protocols are rawly exposed to
user via character device, allowing straigthforward support in existing
tools (ModemManager, ofono...). The WWAN core takes care of the generic
part, including character device management, and relies on port driver
operations to receive/submit protocol data.
Since the different devices exposing protocols for a same WWAN hardware
do not necessarily know about each others (e.g. two different USB
interfaces, PCI/MHI channel devices...) and can be created/removed in
different orders, the WWAN core ensures that all WAN ports contributing
to the 'whole' WWAN feature are grouped under the same virtual WWAN
device, relying on the provided parent device (e.g. mhi controller,
USB device). It's a 'trick' I copied from Johannes's earlier WWAN
subsystem proposal.
This initial version is purposely minimalist, it's essentially moving
the generic part of the previously proposed mhi_wwan_ctrl driver inside
a common WWAN framework, but the implementation is open and flexible
enough to allow extension for further drivers.
Signed-off-by: Loic Poulain <loic.poulain@linaro.org>
Signed-off-by: David S. Miller <davem@davemloft.net>
2021-04-16 10:36:33 +02:00
if ( wwan_major < 0 ) {
2021-06-12 10:20:56 +02:00
err = wwan_major ;
goto destroy ;
net: Add a WWAN subsystem
This change introduces initial support for a WWAN framework. Given the
complexity and heterogeneity of existing WWAN hardwares and interfaces,
there is no strict definition of what a WWAN device is and how it should
be represented. It's often a collection of multiple devices that perform
the global WWAN feature (netdev, tty, chardev, etc).
One usual way to expose modem controls and configuration is via high
level protocols such as the well known AT command protocol, MBIM or
QMI. The USB modems started to expose them as character devices, and
user daemons such as ModemManager learnt to use them.
This initial version adds the concept of WWAN port, which is a logical
pipe to a modem control protocol. The protocols are rawly exposed to
user via character device, allowing straigthforward support in existing
tools (ModemManager, ofono...). The WWAN core takes care of the generic
part, including character device management, and relies on port driver
operations to receive/submit protocol data.
Since the different devices exposing protocols for a same WWAN hardware
do not necessarily know about each others (e.g. two different USB
interfaces, PCI/MHI channel devices...) and can be created/removed in
different orders, the WWAN core ensures that all WAN ports contributing
to the 'whole' WWAN feature are grouped under the same virtual WWAN
device, relying on the provided parent device (e.g. mhi controller,
USB device). It's a 'trick' I copied from Johannes's earlier WWAN
subsystem proposal.
This initial version is purposely minimalist, it's essentially moving
the generic part of the previously proposed mhi_wwan_ctrl driver inside
a common WWAN framework, but the implementation is open and flexible
enough to allow extension for further drivers.
Signed-off-by: Loic Poulain <loic.poulain@linaro.org>
Signed-off-by: David S. Miller <davem@davemloft.net>
2021-04-16 10:36:33 +02:00
}
return 0 ;
2021-06-12 10:20:56 +02:00
destroy :
class_destroy ( wwan_class ) ;
unregister :
rtnl_link_unregister ( & wwan_rtnl_link_ops ) ;
return err ;
net: Add a WWAN subsystem
This change introduces initial support for a WWAN framework. Given the
complexity and heterogeneity of existing WWAN hardwares and interfaces,
there is no strict definition of what a WWAN device is and how it should
be represented. It's often a collection of multiple devices that perform
the global WWAN feature (netdev, tty, chardev, etc).
One usual way to expose modem controls and configuration is via high
level protocols such as the well known AT command protocol, MBIM or
QMI. The USB modems started to expose them as character devices, and
user daemons such as ModemManager learnt to use them.
This initial version adds the concept of WWAN port, which is a logical
pipe to a modem control protocol. The protocols are rawly exposed to
user via character device, allowing straigthforward support in existing
tools (ModemManager, ofono...). The WWAN core takes care of the generic
part, including character device management, and relies on port driver
operations to receive/submit protocol data.
Since the different devices exposing protocols for a same WWAN hardware
do not necessarily know about each others (e.g. two different USB
interfaces, PCI/MHI channel devices...) and can be created/removed in
different orders, the WWAN core ensures that all WAN ports contributing
to the 'whole' WWAN feature are grouped under the same virtual WWAN
device, relying on the provided parent device (e.g. mhi controller,
USB device). It's a 'trick' I copied from Johannes's earlier WWAN
subsystem proposal.
This initial version is purposely minimalist, it's essentially moving
the generic part of the previously proposed mhi_wwan_ctrl driver inside
a common WWAN framework, but the implementation is open and flexible
enough to allow extension for further drivers.
Signed-off-by: Loic Poulain <loic.poulain@linaro.org>
Signed-off-by: David S. Miller <davem@davemloft.net>
2021-04-16 10:36:33 +02:00
}
static void __exit wwan_exit ( void )
{
2021-06-08 07:02:38 +03:00
__unregister_chrdev ( wwan_major , 0 , WWAN_MAX_MINORS , " wwan_port " ) ;
2021-06-12 10:20:56 +02:00
rtnl_link_unregister ( & wwan_rtnl_link_ops ) ;
net: Add a WWAN subsystem
This change introduces initial support for a WWAN framework. Given the
complexity and heterogeneity of existing WWAN hardwares and interfaces,
there is no strict definition of what a WWAN device is and how it should
be represented. It's often a collection of multiple devices that perform
the global WWAN feature (netdev, tty, chardev, etc).
One usual way to expose modem controls and configuration is via high
level protocols such as the well known AT command protocol, MBIM or
QMI. The USB modems started to expose them as character devices, and
user daemons such as ModemManager learnt to use them.
This initial version adds the concept of WWAN port, which is a logical
pipe to a modem control protocol. The protocols are rawly exposed to
user via character device, allowing straigthforward support in existing
tools (ModemManager, ofono...). The WWAN core takes care of the generic
part, including character device management, and relies on port driver
operations to receive/submit protocol data.
Since the different devices exposing protocols for a same WWAN hardware
do not necessarily know about each others (e.g. two different USB
interfaces, PCI/MHI channel devices...) and can be created/removed in
different orders, the WWAN core ensures that all WAN ports contributing
to the 'whole' WWAN feature are grouped under the same virtual WWAN
device, relying on the provided parent device (e.g. mhi controller,
USB device). It's a 'trick' I copied from Johannes's earlier WWAN
subsystem proposal.
This initial version is purposely minimalist, it's essentially moving
the generic part of the previously proposed mhi_wwan_ctrl driver inside
a common WWAN framework, but the implementation is open and flexible
enough to allow extension for further drivers.
Signed-off-by: Loic Poulain <loic.poulain@linaro.org>
Signed-off-by: David S. Miller <davem@davemloft.net>
2021-04-16 10:36:33 +02:00
class_destroy ( wwan_class ) ;
}
module_init ( wwan_init ) ;
module_exit ( wwan_exit ) ;
MODULE_AUTHOR ( " Loic Poulain <loic.poulain@linaro.org> " ) ;
MODULE_DESCRIPTION ( " WWAN core " ) ;
MODULE_LICENSE ( " GPL v2 " ) ;