can: dev: add CAN interface API for fixed bitrates

Some CAN interfaces only support fixed fixed bitrates. This patch adds a
netlink interface to get the list of the CAN interface's fixed bitrates and
data bitrates.

Inside the driver arrays of supported data- bitrate values are defined.

const u32 drvname_bitrate[] = { 20000, 50000, 100000 };
const u32 drvname_data_bitrate[] = { 200000, 500000, 1000000 };

struct drvname_priv *priv;
priv = netdev_priv(dev);

priv->bitrate_const = drvname_bitrate;
priv->bitrate_const_cnt = ARRAY_SIZE(drvname_bitrate);
priv->data_bitrate_const = drvname_data_bitrate;
priv->data_bitrate_const_cnt = ARRAY_SIZE(drvname_data_bitrate);

Signed-off-by: Marc Kleine-Budde <mkl@pengutronix.de>
This commit is contained in:
Marc Kleine-Budde 2017-01-11 17:05:35 +01:00
parent c3606d438a
commit 431af77925
3 changed files with 71 additions and 16 deletions

View File

@ -279,8 +279,29 @@ static int can_fixup_bittiming(struct net_device *dev, struct can_bittiming *bt,
return 0; return 0;
} }
/* Checks the validity of predefined bitrate settings */
static int can_validate_bitrate(struct net_device *dev, struct can_bittiming *bt,
const u32 *bitrate_const,
const unsigned int bitrate_const_cnt)
{
struct can_priv *priv = netdev_priv(dev);
unsigned int i;
for (i = 0; i < bitrate_const_cnt; i++) {
if (bt->bitrate == bitrate_const[i])
break;
}
if (i >= priv->bitrate_const_cnt)
return -EINVAL;
return 0;
}
static int can_get_bittiming(struct net_device *dev, struct can_bittiming *bt, static int can_get_bittiming(struct net_device *dev, struct can_bittiming *bt,
const struct can_bittiming_const *btc) const struct can_bittiming_const *btc,
const u32 *bitrate_const,
const unsigned int bitrate_const_cnt)
{ {
int err; int err;
@ -290,10 +311,13 @@ static int can_get_bittiming(struct net_device *dev, struct can_bittiming *bt,
* alternatively the CAN timing parameters (tq, prop_seg, etc.) are * alternatively the CAN timing parameters (tq, prop_seg, etc.) are
* provided directly which are then checked and fixed up. * provided directly which are then checked and fixed up.
*/ */
if (!bt->tq && bt->bitrate) if (!bt->tq && bt->bitrate && btc)
err = can_calc_bittiming(dev, bt, btc); err = can_calc_bittiming(dev, bt, btc);
else if (bt->tq && !bt->bitrate) else if (bt->tq && !bt->bitrate && btc)
err = can_fixup_bittiming(dev, bt, btc); err = can_fixup_bittiming(dev, bt, btc);
else if (!bt->tq && bt->bitrate && bitrate_const)
err = can_validate_bitrate(dev, bt, bitrate_const,
bitrate_const_cnt);
else else
err = -EINVAL; err = -EINVAL;
@ -878,12 +902,12 @@ static int can_changelink(struct net_device *dev,
return -EOPNOTSUPP; return -EOPNOTSUPP;
memcpy(&bt, nla_data(data[IFLA_CAN_BITTIMING]), sizeof(bt)); memcpy(&bt, nla_data(data[IFLA_CAN_BITTIMING]), sizeof(bt));
if (priv->bittiming_const) { err = can_get_bittiming(dev, &bt,
err = can_get_bittiming(dev, &bt, priv->bittiming_const,
priv->bittiming_const); priv->bitrate_const,
if (err) priv->bitrate_const_cnt);
return err; if (err)
} return err;
memcpy(&priv->bittiming, &bt, sizeof(bt)); memcpy(&priv->bittiming, &bt, sizeof(bt));
if (priv->do_set_bittiming) { if (priv->do_set_bittiming) {
@ -962,12 +986,12 @@ static int can_changelink(struct net_device *dev,
memcpy(&dbt, nla_data(data[IFLA_CAN_DATA_BITTIMING]), memcpy(&dbt, nla_data(data[IFLA_CAN_DATA_BITTIMING]),
sizeof(dbt)); sizeof(dbt));
if (priv->data_bittiming_const) { err = can_get_bittiming(dev, &dbt,
err = can_get_bittiming(dev, &dbt, priv->data_bittiming_const,
priv->data_bittiming_const); priv->data_bitrate_const,
if (err) priv->data_bitrate_const_cnt);
return err; if (err)
} return err;
memcpy(&priv->data_bittiming, &dbt, sizeof(dbt)); memcpy(&priv->data_bittiming, &dbt, sizeof(dbt));
if (priv->do_set_data_bittiming) { if (priv->do_set_data_bittiming) {
@ -1029,6 +1053,12 @@ static size_t can_get_size(const struct net_device *dev)
size += nla_total_size(sizeof(*priv->termination_const) * /* IFLA_CAN_TERMINATION_CONST */ size += nla_total_size(sizeof(*priv->termination_const) * /* IFLA_CAN_TERMINATION_CONST */
priv->termination_const_cnt); priv->termination_const_cnt);
} }
if (priv->bitrate_const) /* IFLA_CAN_BITRATE_CONST */
size += nla_total_size(sizeof(*priv->bitrate_const) *
priv->bitrate_const_cnt);
if (priv->data_bitrate_const) /* IFLA_CAN_DATA_BITRATE_CONST */
size += nla_total_size(sizeof(*priv->data_bitrate_const) *
priv->data_bitrate_const_cnt);
return size; return size;
} }
@ -1074,7 +1104,20 @@ static int can_fill_info(struct sk_buff *skb, const struct net_device *dev)
nla_put(skb, IFLA_CAN_TERMINATION_CONST, nla_put(skb, IFLA_CAN_TERMINATION_CONST,
sizeof(*priv->termination_const) * sizeof(*priv->termination_const) *
priv->termination_const_cnt, priv->termination_const_cnt,
priv->termination_const)))) priv->termination_const))) ||
(priv->bitrate_const &&
nla_put(skb, IFLA_CAN_BITRATE_CONST,
sizeof(*priv->bitrate_const) *
priv->bitrate_const_cnt,
priv->bitrate_const)) ||
(priv->data_bitrate_const &&
nla_put(skb, IFLA_CAN_DATA_BITRATE_CONST,
sizeof(*priv->data_bitrate_const) *
priv->data_bitrate_const_cnt,
priv->data_bitrate_const))
)
return -EMSGSIZE; return -EMSGSIZE;
@ -1140,6 +1183,12 @@ int register_candev(struct net_device *dev)
(!priv->termination_const != !priv->do_set_termination)) (!priv->termination_const != !priv->do_set_termination))
return -EINVAL; return -EINVAL;
if (!priv->bitrate_const != !priv->bitrate_const_cnt)
return -EINVAL;
if (!priv->data_bitrate_const != !priv->data_bitrate_const_cnt)
return -EINVAL;
dev->rtnl_link_ops = &can_link_ops; dev->rtnl_link_ops = &can_link_ops;
return register_netdev(dev); return register_netdev(dev);
} }

View File

@ -41,6 +41,10 @@ struct can_priv {
const u16 *termination_const; const u16 *termination_const;
unsigned int termination_const_cnt; unsigned int termination_const_cnt;
u16 termination; u16 termination;
const u32 *bitrate_const;
unsigned int bitrate_const_cnt;
const u32 *data_bitrate_const;
unsigned int data_bitrate_const_cnt;
struct can_clock clock; struct can_clock clock;
enum can_state state; enum can_state state;

View File

@ -129,6 +129,8 @@ enum {
IFLA_CAN_DATA_BITTIMING_CONST, IFLA_CAN_DATA_BITTIMING_CONST,
IFLA_CAN_TERMINATION, IFLA_CAN_TERMINATION,
IFLA_CAN_TERMINATION_CONST, IFLA_CAN_TERMINATION_CONST,
IFLA_CAN_BITRATE_CONST,
IFLA_CAN_DATA_BITRATE_CONST,
__IFLA_CAN_MAX __IFLA_CAN_MAX
}; };