xref: /openbmc/linux/net/devlink/param.c (revision c900529f3d9161bfde5cca0754f83b4d3c3e0220)
1*830c41e1SJiri Pirko // SPDX-License-Identifier: GPL-2.0-or-later
2*830c41e1SJiri Pirko /*
3*830c41e1SJiri Pirko  * Copyright (c) 2016 Mellanox Technologies. All rights reserved.
4*830c41e1SJiri Pirko  * Copyright (c) 2016 Jiri Pirko <jiri@mellanox.com>
5*830c41e1SJiri Pirko  */
6*830c41e1SJiri Pirko 
7*830c41e1SJiri Pirko #include "devl_internal.h"
8*830c41e1SJiri Pirko 
9*830c41e1SJiri Pirko static const struct devlink_param devlink_param_generic[] = {
10*830c41e1SJiri Pirko 	{
11*830c41e1SJiri Pirko 		.id = DEVLINK_PARAM_GENERIC_ID_INT_ERR_RESET,
12*830c41e1SJiri Pirko 		.name = DEVLINK_PARAM_GENERIC_INT_ERR_RESET_NAME,
13*830c41e1SJiri Pirko 		.type = DEVLINK_PARAM_GENERIC_INT_ERR_RESET_TYPE,
14*830c41e1SJiri Pirko 	},
15*830c41e1SJiri Pirko 	{
16*830c41e1SJiri Pirko 		.id = DEVLINK_PARAM_GENERIC_ID_MAX_MACS,
17*830c41e1SJiri Pirko 		.name = DEVLINK_PARAM_GENERIC_MAX_MACS_NAME,
18*830c41e1SJiri Pirko 		.type = DEVLINK_PARAM_GENERIC_MAX_MACS_TYPE,
19*830c41e1SJiri Pirko 	},
20*830c41e1SJiri Pirko 	{
21*830c41e1SJiri Pirko 		.id = DEVLINK_PARAM_GENERIC_ID_ENABLE_SRIOV,
22*830c41e1SJiri Pirko 		.name = DEVLINK_PARAM_GENERIC_ENABLE_SRIOV_NAME,
23*830c41e1SJiri Pirko 		.type = DEVLINK_PARAM_GENERIC_ENABLE_SRIOV_TYPE,
24*830c41e1SJiri Pirko 	},
25*830c41e1SJiri Pirko 	{
26*830c41e1SJiri Pirko 		.id = DEVLINK_PARAM_GENERIC_ID_REGION_SNAPSHOT,
27*830c41e1SJiri Pirko 		.name = DEVLINK_PARAM_GENERIC_REGION_SNAPSHOT_NAME,
28*830c41e1SJiri Pirko 		.type = DEVLINK_PARAM_GENERIC_REGION_SNAPSHOT_TYPE,
29*830c41e1SJiri Pirko 	},
30*830c41e1SJiri Pirko 	{
31*830c41e1SJiri Pirko 		.id = DEVLINK_PARAM_GENERIC_ID_IGNORE_ARI,
32*830c41e1SJiri Pirko 		.name = DEVLINK_PARAM_GENERIC_IGNORE_ARI_NAME,
33*830c41e1SJiri Pirko 		.type = DEVLINK_PARAM_GENERIC_IGNORE_ARI_TYPE,
34*830c41e1SJiri Pirko 	},
35*830c41e1SJiri Pirko 	{
36*830c41e1SJiri Pirko 		.id = DEVLINK_PARAM_GENERIC_ID_MSIX_VEC_PER_PF_MAX,
37*830c41e1SJiri Pirko 		.name = DEVLINK_PARAM_GENERIC_MSIX_VEC_PER_PF_MAX_NAME,
38*830c41e1SJiri Pirko 		.type = DEVLINK_PARAM_GENERIC_MSIX_VEC_PER_PF_MAX_TYPE,
39*830c41e1SJiri Pirko 	},
40*830c41e1SJiri Pirko 	{
41*830c41e1SJiri Pirko 		.id = DEVLINK_PARAM_GENERIC_ID_MSIX_VEC_PER_PF_MIN,
42*830c41e1SJiri Pirko 		.name = DEVLINK_PARAM_GENERIC_MSIX_VEC_PER_PF_MIN_NAME,
43*830c41e1SJiri Pirko 		.type = DEVLINK_PARAM_GENERIC_MSIX_VEC_PER_PF_MIN_TYPE,
44*830c41e1SJiri Pirko 	},
45*830c41e1SJiri Pirko 	{
46*830c41e1SJiri Pirko 		.id = DEVLINK_PARAM_GENERIC_ID_FW_LOAD_POLICY,
47*830c41e1SJiri Pirko 		.name = DEVLINK_PARAM_GENERIC_FW_LOAD_POLICY_NAME,
48*830c41e1SJiri Pirko 		.type = DEVLINK_PARAM_GENERIC_FW_LOAD_POLICY_TYPE,
49*830c41e1SJiri Pirko 	},
50*830c41e1SJiri Pirko 	{
51*830c41e1SJiri Pirko 		.id = DEVLINK_PARAM_GENERIC_ID_RESET_DEV_ON_DRV_PROBE,
52*830c41e1SJiri Pirko 		.name = DEVLINK_PARAM_GENERIC_RESET_DEV_ON_DRV_PROBE_NAME,
53*830c41e1SJiri Pirko 		.type = DEVLINK_PARAM_GENERIC_RESET_DEV_ON_DRV_PROBE_TYPE,
54*830c41e1SJiri Pirko 	},
55*830c41e1SJiri Pirko 	{
56*830c41e1SJiri Pirko 		.id = DEVLINK_PARAM_GENERIC_ID_ENABLE_ROCE,
57*830c41e1SJiri Pirko 		.name = DEVLINK_PARAM_GENERIC_ENABLE_ROCE_NAME,
58*830c41e1SJiri Pirko 		.type = DEVLINK_PARAM_GENERIC_ENABLE_ROCE_TYPE,
59*830c41e1SJiri Pirko 	},
60*830c41e1SJiri Pirko 	{
61*830c41e1SJiri Pirko 		.id = DEVLINK_PARAM_GENERIC_ID_ENABLE_REMOTE_DEV_RESET,
62*830c41e1SJiri Pirko 		.name = DEVLINK_PARAM_GENERIC_ENABLE_REMOTE_DEV_RESET_NAME,
63*830c41e1SJiri Pirko 		.type = DEVLINK_PARAM_GENERIC_ENABLE_REMOTE_DEV_RESET_TYPE,
64*830c41e1SJiri Pirko 	},
65*830c41e1SJiri Pirko 	{
66*830c41e1SJiri Pirko 		.id = DEVLINK_PARAM_GENERIC_ID_ENABLE_ETH,
67*830c41e1SJiri Pirko 		.name = DEVLINK_PARAM_GENERIC_ENABLE_ETH_NAME,
68*830c41e1SJiri Pirko 		.type = DEVLINK_PARAM_GENERIC_ENABLE_ETH_TYPE,
69*830c41e1SJiri Pirko 	},
70*830c41e1SJiri Pirko 	{
71*830c41e1SJiri Pirko 		.id = DEVLINK_PARAM_GENERIC_ID_ENABLE_RDMA,
72*830c41e1SJiri Pirko 		.name = DEVLINK_PARAM_GENERIC_ENABLE_RDMA_NAME,
73*830c41e1SJiri Pirko 		.type = DEVLINK_PARAM_GENERIC_ENABLE_RDMA_TYPE,
74*830c41e1SJiri Pirko 	},
75*830c41e1SJiri Pirko 	{
76*830c41e1SJiri Pirko 		.id = DEVLINK_PARAM_GENERIC_ID_ENABLE_VNET,
77*830c41e1SJiri Pirko 		.name = DEVLINK_PARAM_GENERIC_ENABLE_VNET_NAME,
78*830c41e1SJiri Pirko 		.type = DEVLINK_PARAM_GENERIC_ENABLE_VNET_TYPE,
79*830c41e1SJiri Pirko 	},
80*830c41e1SJiri Pirko 	{
81*830c41e1SJiri Pirko 		.id = DEVLINK_PARAM_GENERIC_ID_ENABLE_IWARP,
82*830c41e1SJiri Pirko 		.name = DEVLINK_PARAM_GENERIC_ENABLE_IWARP_NAME,
83*830c41e1SJiri Pirko 		.type = DEVLINK_PARAM_GENERIC_ENABLE_IWARP_TYPE,
84*830c41e1SJiri Pirko 	},
85*830c41e1SJiri Pirko 	{
86*830c41e1SJiri Pirko 		.id = DEVLINK_PARAM_GENERIC_ID_IO_EQ_SIZE,
87*830c41e1SJiri Pirko 		.name = DEVLINK_PARAM_GENERIC_IO_EQ_SIZE_NAME,
88*830c41e1SJiri Pirko 		.type = DEVLINK_PARAM_GENERIC_IO_EQ_SIZE_TYPE,
89*830c41e1SJiri Pirko 	},
90*830c41e1SJiri Pirko 	{
91*830c41e1SJiri Pirko 		.id = DEVLINK_PARAM_GENERIC_ID_EVENT_EQ_SIZE,
92*830c41e1SJiri Pirko 		.name = DEVLINK_PARAM_GENERIC_EVENT_EQ_SIZE_NAME,
93*830c41e1SJiri Pirko 		.type = DEVLINK_PARAM_GENERIC_EVENT_EQ_SIZE_TYPE,
94*830c41e1SJiri Pirko 	},
95*830c41e1SJiri Pirko };
96*830c41e1SJiri Pirko 
devlink_param_generic_verify(const struct devlink_param * param)97*830c41e1SJiri Pirko static int devlink_param_generic_verify(const struct devlink_param *param)
98*830c41e1SJiri Pirko {
99*830c41e1SJiri Pirko 	/* verify it match generic parameter by id and name */
100*830c41e1SJiri Pirko 	if (param->id > DEVLINK_PARAM_GENERIC_ID_MAX)
101*830c41e1SJiri Pirko 		return -EINVAL;
102*830c41e1SJiri Pirko 	if (strcmp(param->name, devlink_param_generic[param->id].name))
103*830c41e1SJiri Pirko 		return -ENOENT;
104*830c41e1SJiri Pirko 
105*830c41e1SJiri Pirko 	WARN_ON(param->type != devlink_param_generic[param->id].type);
106*830c41e1SJiri Pirko 
107*830c41e1SJiri Pirko 	return 0;
108*830c41e1SJiri Pirko }
109*830c41e1SJiri Pirko 
devlink_param_driver_verify(const struct devlink_param * param)110*830c41e1SJiri Pirko static int devlink_param_driver_verify(const struct devlink_param *param)
111*830c41e1SJiri Pirko {
112*830c41e1SJiri Pirko 	int i;
113*830c41e1SJiri Pirko 
114*830c41e1SJiri Pirko 	if (param->id <= DEVLINK_PARAM_GENERIC_ID_MAX)
115*830c41e1SJiri Pirko 		return -EINVAL;
116*830c41e1SJiri Pirko 	/* verify no such name in generic params */
117*830c41e1SJiri Pirko 	for (i = 0; i <= DEVLINK_PARAM_GENERIC_ID_MAX; i++)
118*830c41e1SJiri Pirko 		if (!strcmp(param->name, devlink_param_generic[i].name))
119*830c41e1SJiri Pirko 			return -EEXIST;
120*830c41e1SJiri Pirko 
121*830c41e1SJiri Pirko 	return 0;
122*830c41e1SJiri Pirko }
123*830c41e1SJiri Pirko 
124*830c41e1SJiri Pirko static struct devlink_param_item *
devlink_param_find_by_name(struct xarray * params,const char * param_name)125*830c41e1SJiri Pirko devlink_param_find_by_name(struct xarray *params, const char *param_name)
126*830c41e1SJiri Pirko {
127*830c41e1SJiri Pirko 	struct devlink_param_item *param_item;
128*830c41e1SJiri Pirko 	unsigned long param_id;
129*830c41e1SJiri Pirko 
130*830c41e1SJiri Pirko 	xa_for_each(params, param_id, param_item) {
131*830c41e1SJiri Pirko 		if (!strcmp(param_item->param->name, param_name))
132*830c41e1SJiri Pirko 			return param_item;
133*830c41e1SJiri Pirko 	}
134*830c41e1SJiri Pirko 	return NULL;
135*830c41e1SJiri Pirko }
136*830c41e1SJiri Pirko 
137*830c41e1SJiri Pirko static struct devlink_param_item *
devlink_param_find_by_id(struct xarray * params,u32 param_id)138*830c41e1SJiri Pirko devlink_param_find_by_id(struct xarray *params, u32 param_id)
139*830c41e1SJiri Pirko {
140*830c41e1SJiri Pirko 	return xa_load(params, param_id);
141*830c41e1SJiri Pirko }
142*830c41e1SJiri Pirko 
143*830c41e1SJiri Pirko static bool
devlink_param_cmode_is_supported(const struct devlink_param * param,enum devlink_param_cmode cmode)144*830c41e1SJiri Pirko devlink_param_cmode_is_supported(const struct devlink_param *param,
145*830c41e1SJiri Pirko 				 enum devlink_param_cmode cmode)
146*830c41e1SJiri Pirko {
147*830c41e1SJiri Pirko 	return test_bit(cmode, &param->supported_cmodes);
148*830c41e1SJiri Pirko }
149*830c41e1SJiri Pirko 
devlink_param_get(struct devlink * devlink,const struct devlink_param * param,struct devlink_param_gset_ctx * ctx)150*830c41e1SJiri Pirko static int devlink_param_get(struct devlink *devlink,
151*830c41e1SJiri Pirko 			     const struct devlink_param *param,
152*830c41e1SJiri Pirko 			     struct devlink_param_gset_ctx *ctx)
153*830c41e1SJiri Pirko {
154*830c41e1SJiri Pirko 	if (!param->get)
155*830c41e1SJiri Pirko 		return -EOPNOTSUPP;
156*830c41e1SJiri Pirko 	return param->get(devlink, param->id, ctx);
157*830c41e1SJiri Pirko }
158*830c41e1SJiri Pirko 
devlink_param_set(struct devlink * devlink,const struct devlink_param * param,struct devlink_param_gset_ctx * ctx)159*830c41e1SJiri Pirko static int devlink_param_set(struct devlink *devlink,
160*830c41e1SJiri Pirko 			     const struct devlink_param *param,
161*830c41e1SJiri Pirko 			     struct devlink_param_gset_ctx *ctx)
162*830c41e1SJiri Pirko {
163*830c41e1SJiri Pirko 	if (!param->set)
164*830c41e1SJiri Pirko 		return -EOPNOTSUPP;
165*830c41e1SJiri Pirko 	return param->set(devlink, param->id, ctx);
166*830c41e1SJiri Pirko }
167*830c41e1SJiri Pirko 
168*830c41e1SJiri Pirko static int
devlink_param_type_to_nla_type(enum devlink_param_type param_type)169*830c41e1SJiri Pirko devlink_param_type_to_nla_type(enum devlink_param_type param_type)
170*830c41e1SJiri Pirko {
171*830c41e1SJiri Pirko 	switch (param_type) {
172*830c41e1SJiri Pirko 	case DEVLINK_PARAM_TYPE_U8:
173*830c41e1SJiri Pirko 		return NLA_U8;
174*830c41e1SJiri Pirko 	case DEVLINK_PARAM_TYPE_U16:
175*830c41e1SJiri Pirko 		return NLA_U16;
176*830c41e1SJiri Pirko 	case DEVLINK_PARAM_TYPE_U32:
177*830c41e1SJiri Pirko 		return NLA_U32;
178*830c41e1SJiri Pirko 	case DEVLINK_PARAM_TYPE_STRING:
179*830c41e1SJiri Pirko 		return NLA_STRING;
180*830c41e1SJiri Pirko 	case DEVLINK_PARAM_TYPE_BOOL:
181*830c41e1SJiri Pirko 		return NLA_FLAG;
182*830c41e1SJiri Pirko 	default:
183*830c41e1SJiri Pirko 		return -EINVAL;
184*830c41e1SJiri Pirko 	}
185*830c41e1SJiri Pirko }
186*830c41e1SJiri Pirko 
187*830c41e1SJiri Pirko static int
devlink_nl_param_value_fill_one(struct sk_buff * msg,enum devlink_param_type type,enum devlink_param_cmode cmode,union devlink_param_value val)188*830c41e1SJiri Pirko devlink_nl_param_value_fill_one(struct sk_buff *msg,
189*830c41e1SJiri Pirko 				enum devlink_param_type type,
190*830c41e1SJiri Pirko 				enum devlink_param_cmode cmode,
191*830c41e1SJiri Pirko 				union devlink_param_value val)
192*830c41e1SJiri Pirko {
193*830c41e1SJiri Pirko 	struct nlattr *param_value_attr;
194*830c41e1SJiri Pirko 
195*830c41e1SJiri Pirko 	param_value_attr = nla_nest_start_noflag(msg,
196*830c41e1SJiri Pirko 						 DEVLINK_ATTR_PARAM_VALUE);
197*830c41e1SJiri Pirko 	if (!param_value_attr)
198*830c41e1SJiri Pirko 		goto nla_put_failure;
199*830c41e1SJiri Pirko 
200*830c41e1SJiri Pirko 	if (nla_put_u8(msg, DEVLINK_ATTR_PARAM_VALUE_CMODE, cmode))
201*830c41e1SJiri Pirko 		goto value_nest_cancel;
202*830c41e1SJiri Pirko 
203*830c41e1SJiri Pirko 	switch (type) {
204*830c41e1SJiri Pirko 	case DEVLINK_PARAM_TYPE_U8:
205*830c41e1SJiri Pirko 		if (nla_put_u8(msg, DEVLINK_ATTR_PARAM_VALUE_DATA, val.vu8))
206*830c41e1SJiri Pirko 			goto value_nest_cancel;
207*830c41e1SJiri Pirko 		break;
208*830c41e1SJiri Pirko 	case DEVLINK_PARAM_TYPE_U16:
209*830c41e1SJiri Pirko 		if (nla_put_u16(msg, DEVLINK_ATTR_PARAM_VALUE_DATA, val.vu16))
210*830c41e1SJiri Pirko 			goto value_nest_cancel;
211*830c41e1SJiri Pirko 		break;
212*830c41e1SJiri Pirko 	case DEVLINK_PARAM_TYPE_U32:
213*830c41e1SJiri Pirko 		if (nla_put_u32(msg, DEVLINK_ATTR_PARAM_VALUE_DATA, val.vu32))
214*830c41e1SJiri Pirko 			goto value_nest_cancel;
215*830c41e1SJiri Pirko 		break;
216*830c41e1SJiri Pirko 	case DEVLINK_PARAM_TYPE_STRING:
217*830c41e1SJiri Pirko 		if (nla_put_string(msg, DEVLINK_ATTR_PARAM_VALUE_DATA,
218*830c41e1SJiri Pirko 				   val.vstr))
219*830c41e1SJiri Pirko 			goto value_nest_cancel;
220*830c41e1SJiri Pirko 		break;
221*830c41e1SJiri Pirko 	case DEVLINK_PARAM_TYPE_BOOL:
222*830c41e1SJiri Pirko 		if (val.vbool &&
223*830c41e1SJiri Pirko 		    nla_put_flag(msg, DEVLINK_ATTR_PARAM_VALUE_DATA))
224*830c41e1SJiri Pirko 			goto value_nest_cancel;
225*830c41e1SJiri Pirko 		break;
226*830c41e1SJiri Pirko 	}
227*830c41e1SJiri Pirko 
228*830c41e1SJiri Pirko 	nla_nest_end(msg, param_value_attr);
229*830c41e1SJiri Pirko 	return 0;
230*830c41e1SJiri Pirko 
231*830c41e1SJiri Pirko value_nest_cancel:
232*830c41e1SJiri Pirko 	nla_nest_cancel(msg, param_value_attr);
233*830c41e1SJiri Pirko nla_put_failure:
234*830c41e1SJiri Pirko 	return -EMSGSIZE;
235*830c41e1SJiri Pirko }
236*830c41e1SJiri Pirko 
devlink_nl_param_fill(struct sk_buff * msg,struct devlink * devlink,unsigned int port_index,struct devlink_param_item * param_item,enum devlink_command cmd,u32 portid,u32 seq,int flags)237*830c41e1SJiri Pirko static int devlink_nl_param_fill(struct sk_buff *msg, struct devlink *devlink,
238*830c41e1SJiri Pirko 				 unsigned int port_index,
239*830c41e1SJiri Pirko 				 struct devlink_param_item *param_item,
240*830c41e1SJiri Pirko 				 enum devlink_command cmd,
241*830c41e1SJiri Pirko 				 u32 portid, u32 seq, int flags)
242*830c41e1SJiri Pirko {
243*830c41e1SJiri Pirko 	union devlink_param_value param_value[DEVLINK_PARAM_CMODE_MAX + 1];
244*830c41e1SJiri Pirko 	bool param_value_set[DEVLINK_PARAM_CMODE_MAX + 1] = {};
245*830c41e1SJiri Pirko 	const struct devlink_param *param = param_item->param;
246*830c41e1SJiri Pirko 	struct devlink_param_gset_ctx ctx;
247*830c41e1SJiri Pirko 	struct nlattr *param_values_list;
248*830c41e1SJiri Pirko 	struct nlattr *param_attr;
249*830c41e1SJiri Pirko 	int nla_type;
250*830c41e1SJiri Pirko 	void *hdr;
251*830c41e1SJiri Pirko 	int err;
252*830c41e1SJiri Pirko 	int i;
253*830c41e1SJiri Pirko 
254*830c41e1SJiri Pirko 	/* Get value from driver part to driverinit configuration mode */
255*830c41e1SJiri Pirko 	for (i = 0; i <= DEVLINK_PARAM_CMODE_MAX; i++) {
256*830c41e1SJiri Pirko 		if (!devlink_param_cmode_is_supported(param, i))
257*830c41e1SJiri Pirko 			continue;
258*830c41e1SJiri Pirko 		if (i == DEVLINK_PARAM_CMODE_DRIVERINIT) {
259*830c41e1SJiri Pirko 			if (param_item->driverinit_value_new_valid)
260*830c41e1SJiri Pirko 				param_value[i] = param_item->driverinit_value_new;
261*830c41e1SJiri Pirko 			else if (param_item->driverinit_value_valid)
262*830c41e1SJiri Pirko 				param_value[i] = param_item->driverinit_value;
263*830c41e1SJiri Pirko 			else
264*830c41e1SJiri Pirko 				return -EOPNOTSUPP;
265*830c41e1SJiri Pirko 		} else {
266*830c41e1SJiri Pirko 			ctx.cmode = i;
267*830c41e1SJiri Pirko 			err = devlink_param_get(devlink, param, &ctx);
268*830c41e1SJiri Pirko 			if (err)
269*830c41e1SJiri Pirko 				return err;
270*830c41e1SJiri Pirko 			param_value[i] = ctx.val;
271*830c41e1SJiri Pirko 		}
272*830c41e1SJiri Pirko 		param_value_set[i] = true;
273*830c41e1SJiri Pirko 	}
274*830c41e1SJiri Pirko 
275*830c41e1SJiri Pirko 	hdr = genlmsg_put(msg, portid, seq, &devlink_nl_family, flags, cmd);
276*830c41e1SJiri Pirko 	if (!hdr)
277*830c41e1SJiri Pirko 		return -EMSGSIZE;
278*830c41e1SJiri Pirko 
279*830c41e1SJiri Pirko 	if (devlink_nl_put_handle(msg, devlink))
280*830c41e1SJiri Pirko 		goto genlmsg_cancel;
281*830c41e1SJiri Pirko 
282*830c41e1SJiri Pirko 	if (cmd == DEVLINK_CMD_PORT_PARAM_GET ||
283*830c41e1SJiri Pirko 	    cmd == DEVLINK_CMD_PORT_PARAM_NEW ||
284*830c41e1SJiri Pirko 	    cmd == DEVLINK_CMD_PORT_PARAM_DEL)
285*830c41e1SJiri Pirko 		if (nla_put_u32(msg, DEVLINK_ATTR_PORT_INDEX, port_index))
286*830c41e1SJiri Pirko 			goto genlmsg_cancel;
287*830c41e1SJiri Pirko 
288*830c41e1SJiri Pirko 	param_attr = nla_nest_start_noflag(msg, DEVLINK_ATTR_PARAM);
289*830c41e1SJiri Pirko 	if (!param_attr)
290*830c41e1SJiri Pirko 		goto genlmsg_cancel;
291*830c41e1SJiri Pirko 	if (nla_put_string(msg, DEVLINK_ATTR_PARAM_NAME, param->name))
292*830c41e1SJiri Pirko 		goto param_nest_cancel;
293*830c41e1SJiri Pirko 	if (param->generic && nla_put_flag(msg, DEVLINK_ATTR_PARAM_GENERIC))
294*830c41e1SJiri Pirko 		goto param_nest_cancel;
295*830c41e1SJiri Pirko 
296*830c41e1SJiri Pirko 	nla_type = devlink_param_type_to_nla_type(param->type);
297*830c41e1SJiri Pirko 	if (nla_type < 0)
298*830c41e1SJiri Pirko 		goto param_nest_cancel;
299*830c41e1SJiri Pirko 	if (nla_put_u8(msg, DEVLINK_ATTR_PARAM_TYPE, nla_type))
300*830c41e1SJiri Pirko 		goto param_nest_cancel;
301*830c41e1SJiri Pirko 
302*830c41e1SJiri Pirko 	param_values_list = nla_nest_start_noflag(msg,
303*830c41e1SJiri Pirko 						  DEVLINK_ATTR_PARAM_VALUES_LIST);
304*830c41e1SJiri Pirko 	if (!param_values_list)
305*830c41e1SJiri Pirko 		goto param_nest_cancel;
306*830c41e1SJiri Pirko 
307*830c41e1SJiri Pirko 	for (i = 0; i <= DEVLINK_PARAM_CMODE_MAX; i++) {
308*830c41e1SJiri Pirko 		if (!param_value_set[i])
309*830c41e1SJiri Pirko 			continue;
310*830c41e1SJiri Pirko 		err = devlink_nl_param_value_fill_one(msg, param->type,
311*830c41e1SJiri Pirko 						      i, param_value[i]);
312*830c41e1SJiri Pirko 		if (err)
313*830c41e1SJiri Pirko 			goto values_list_nest_cancel;
314*830c41e1SJiri Pirko 	}
315*830c41e1SJiri Pirko 
316*830c41e1SJiri Pirko 	nla_nest_end(msg, param_values_list);
317*830c41e1SJiri Pirko 	nla_nest_end(msg, param_attr);
318*830c41e1SJiri Pirko 	genlmsg_end(msg, hdr);
319*830c41e1SJiri Pirko 	return 0;
320*830c41e1SJiri Pirko 
321*830c41e1SJiri Pirko values_list_nest_cancel:
322*830c41e1SJiri Pirko 	nla_nest_end(msg, param_values_list);
323*830c41e1SJiri Pirko param_nest_cancel:
324*830c41e1SJiri Pirko 	nla_nest_cancel(msg, param_attr);
325*830c41e1SJiri Pirko genlmsg_cancel:
326*830c41e1SJiri Pirko 	genlmsg_cancel(msg, hdr);
327*830c41e1SJiri Pirko 	return -EMSGSIZE;
328*830c41e1SJiri Pirko }
329*830c41e1SJiri Pirko 
devlink_param_notify(struct devlink * devlink,unsigned int port_index,struct devlink_param_item * param_item,enum devlink_command cmd)330*830c41e1SJiri Pirko static void devlink_param_notify(struct devlink *devlink,
331*830c41e1SJiri Pirko 				 unsigned int port_index,
332*830c41e1SJiri Pirko 				 struct devlink_param_item *param_item,
333*830c41e1SJiri Pirko 				 enum devlink_command cmd)
334*830c41e1SJiri Pirko {
335*830c41e1SJiri Pirko 	struct sk_buff *msg;
336*830c41e1SJiri Pirko 	int err;
337*830c41e1SJiri Pirko 
338*830c41e1SJiri Pirko 	WARN_ON(cmd != DEVLINK_CMD_PARAM_NEW && cmd != DEVLINK_CMD_PARAM_DEL &&
339*830c41e1SJiri Pirko 		cmd != DEVLINK_CMD_PORT_PARAM_NEW &&
340*830c41e1SJiri Pirko 		cmd != DEVLINK_CMD_PORT_PARAM_DEL);
341*830c41e1SJiri Pirko 
342*830c41e1SJiri Pirko 	/* devlink_notify_register() / devlink_notify_unregister()
343*830c41e1SJiri Pirko 	 * will replay the notifications if the params are added/removed
344*830c41e1SJiri Pirko 	 * outside of the lifetime of the instance.
345*830c41e1SJiri Pirko 	 */
346*830c41e1SJiri Pirko 	if (!devl_is_registered(devlink))
347*830c41e1SJiri Pirko 		return;
348*830c41e1SJiri Pirko 
349*830c41e1SJiri Pirko 	msg = nlmsg_new(NLMSG_DEFAULT_SIZE, GFP_KERNEL);
350*830c41e1SJiri Pirko 	if (!msg)
351*830c41e1SJiri Pirko 		return;
352*830c41e1SJiri Pirko 	err = devlink_nl_param_fill(msg, devlink, port_index, param_item, cmd,
353*830c41e1SJiri Pirko 				    0, 0, 0);
354*830c41e1SJiri Pirko 	if (err) {
355*830c41e1SJiri Pirko 		nlmsg_free(msg);
356*830c41e1SJiri Pirko 		return;
357*830c41e1SJiri Pirko 	}
358*830c41e1SJiri Pirko 
359*830c41e1SJiri Pirko 	genlmsg_multicast_netns(&devlink_nl_family, devlink_net(devlink),
360*830c41e1SJiri Pirko 				msg, 0, DEVLINK_MCGRP_CONFIG, GFP_KERNEL);
361*830c41e1SJiri Pirko }
362*830c41e1SJiri Pirko 
devlink_params_notify(struct devlink * devlink,enum devlink_command cmd)363*830c41e1SJiri Pirko static void devlink_params_notify(struct devlink *devlink,
364*830c41e1SJiri Pirko 				  enum devlink_command cmd)
365*830c41e1SJiri Pirko {
366*830c41e1SJiri Pirko 	struct devlink_param_item *param_item;
367*830c41e1SJiri Pirko 	unsigned long param_id;
368*830c41e1SJiri Pirko 
369*830c41e1SJiri Pirko 	xa_for_each(&devlink->params, param_id, param_item)
370*830c41e1SJiri Pirko 		devlink_param_notify(devlink, 0, param_item, cmd);
371*830c41e1SJiri Pirko }
372*830c41e1SJiri Pirko 
devlink_params_notify_register(struct devlink * devlink)373*830c41e1SJiri Pirko void devlink_params_notify_register(struct devlink *devlink)
374*830c41e1SJiri Pirko {
375*830c41e1SJiri Pirko 	devlink_params_notify(devlink, DEVLINK_CMD_PARAM_NEW);
376*830c41e1SJiri Pirko }
377*830c41e1SJiri Pirko 
devlink_params_notify_unregister(struct devlink * devlink)378*830c41e1SJiri Pirko void devlink_params_notify_unregister(struct devlink *devlink)
379*830c41e1SJiri Pirko {
380*830c41e1SJiri Pirko 	devlink_params_notify(devlink, DEVLINK_CMD_PARAM_DEL);
381*830c41e1SJiri Pirko }
382*830c41e1SJiri Pirko 
devlink_nl_param_get_dump_one(struct sk_buff * msg,struct devlink * devlink,struct netlink_callback * cb,int flags)383*830c41e1SJiri Pirko static int devlink_nl_param_get_dump_one(struct sk_buff *msg,
384*830c41e1SJiri Pirko 					 struct devlink *devlink,
385*830c41e1SJiri Pirko 					 struct netlink_callback *cb,
386*830c41e1SJiri Pirko 					 int flags)
387*830c41e1SJiri Pirko {
388*830c41e1SJiri Pirko 	struct devlink_nl_dump_state *state = devlink_dump_state(cb);
389*830c41e1SJiri Pirko 	struct devlink_param_item *param_item;
390*830c41e1SJiri Pirko 	unsigned long param_id;
391*830c41e1SJiri Pirko 	int err = 0;
392*830c41e1SJiri Pirko 
393*830c41e1SJiri Pirko 	xa_for_each_start(&devlink->params, param_id, param_item, state->idx) {
394*830c41e1SJiri Pirko 		err = devlink_nl_param_fill(msg, devlink, 0, param_item,
395*830c41e1SJiri Pirko 					    DEVLINK_CMD_PARAM_GET,
396*830c41e1SJiri Pirko 					    NETLINK_CB(cb->skb).portid,
397*830c41e1SJiri Pirko 					    cb->nlh->nlmsg_seq, flags);
398*830c41e1SJiri Pirko 		if (err == -EOPNOTSUPP) {
399*830c41e1SJiri Pirko 			err = 0;
400*830c41e1SJiri Pirko 		} else if (err) {
401*830c41e1SJiri Pirko 			state->idx = param_id;
402*830c41e1SJiri Pirko 			break;
403*830c41e1SJiri Pirko 		}
404*830c41e1SJiri Pirko 	}
405*830c41e1SJiri Pirko 
406*830c41e1SJiri Pirko 	return err;
407*830c41e1SJiri Pirko }
408*830c41e1SJiri Pirko 
devlink_nl_param_get_dumpit(struct sk_buff * skb,struct netlink_callback * cb)409*830c41e1SJiri Pirko int devlink_nl_param_get_dumpit(struct sk_buff *skb,
410*830c41e1SJiri Pirko 				struct netlink_callback *cb)
411*830c41e1SJiri Pirko {
412*830c41e1SJiri Pirko 	return devlink_nl_dumpit(skb, cb, devlink_nl_param_get_dump_one);
413*830c41e1SJiri Pirko }
414*830c41e1SJiri Pirko 
415*830c41e1SJiri Pirko static int
devlink_param_type_get_from_info(struct genl_info * info,enum devlink_param_type * param_type)416*830c41e1SJiri Pirko devlink_param_type_get_from_info(struct genl_info *info,
417*830c41e1SJiri Pirko 				 enum devlink_param_type *param_type)
418*830c41e1SJiri Pirko {
419*830c41e1SJiri Pirko 	if (GENL_REQ_ATTR_CHECK(info, DEVLINK_ATTR_PARAM_TYPE))
420*830c41e1SJiri Pirko 		return -EINVAL;
421*830c41e1SJiri Pirko 
422*830c41e1SJiri Pirko 	switch (nla_get_u8(info->attrs[DEVLINK_ATTR_PARAM_TYPE])) {
423*830c41e1SJiri Pirko 	case NLA_U8:
424*830c41e1SJiri Pirko 		*param_type = DEVLINK_PARAM_TYPE_U8;
425*830c41e1SJiri Pirko 		break;
426*830c41e1SJiri Pirko 	case NLA_U16:
427*830c41e1SJiri Pirko 		*param_type = DEVLINK_PARAM_TYPE_U16;
428*830c41e1SJiri Pirko 		break;
429*830c41e1SJiri Pirko 	case NLA_U32:
430*830c41e1SJiri Pirko 		*param_type = DEVLINK_PARAM_TYPE_U32;
431*830c41e1SJiri Pirko 		break;
432*830c41e1SJiri Pirko 	case NLA_STRING:
433*830c41e1SJiri Pirko 		*param_type = DEVLINK_PARAM_TYPE_STRING;
434*830c41e1SJiri Pirko 		break;
435*830c41e1SJiri Pirko 	case NLA_FLAG:
436*830c41e1SJiri Pirko 		*param_type = DEVLINK_PARAM_TYPE_BOOL;
437*830c41e1SJiri Pirko 		break;
438*830c41e1SJiri Pirko 	default:
439*830c41e1SJiri Pirko 		return -EINVAL;
440*830c41e1SJiri Pirko 	}
441*830c41e1SJiri Pirko 
442*830c41e1SJiri Pirko 	return 0;
443*830c41e1SJiri Pirko }
444*830c41e1SJiri Pirko 
445*830c41e1SJiri Pirko static int
devlink_param_value_get_from_info(const struct devlink_param * param,struct genl_info * info,union devlink_param_value * value)446*830c41e1SJiri Pirko devlink_param_value_get_from_info(const struct devlink_param *param,
447*830c41e1SJiri Pirko 				  struct genl_info *info,
448*830c41e1SJiri Pirko 				  union devlink_param_value *value)
449*830c41e1SJiri Pirko {
450*830c41e1SJiri Pirko 	struct nlattr *param_data;
451*830c41e1SJiri Pirko 	int len;
452*830c41e1SJiri Pirko 
453*830c41e1SJiri Pirko 	param_data = info->attrs[DEVLINK_ATTR_PARAM_VALUE_DATA];
454*830c41e1SJiri Pirko 
455*830c41e1SJiri Pirko 	if (param->type != DEVLINK_PARAM_TYPE_BOOL && !param_data)
456*830c41e1SJiri Pirko 		return -EINVAL;
457*830c41e1SJiri Pirko 
458*830c41e1SJiri Pirko 	switch (param->type) {
459*830c41e1SJiri Pirko 	case DEVLINK_PARAM_TYPE_U8:
460*830c41e1SJiri Pirko 		if (nla_len(param_data) != sizeof(u8))
461*830c41e1SJiri Pirko 			return -EINVAL;
462*830c41e1SJiri Pirko 		value->vu8 = nla_get_u8(param_data);
463*830c41e1SJiri Pirko 		break;
464*830c41e1SJiri Pirko 	case DEVLINK_PARAM_TYPE_U16:
465*830c41e1SJiri Pirko 		if (nla_len(param_data) != sizeof(u16))
466*830c41e1SJiri Pirko 			return -EINVAL;
467*830c41e1SJiri Pirko 		value->vu16 = nla_get_u16(param_data);
468*830c41e1SJiri Pirko 		break;
469*830c41e1SJiri Pirko 	case DEVLINK_PARAM_TYPE_U32:
470*830c41e1SJiri Pirko 		if (nla_len(param_data) != sizeof(u32))
471*830c41e1SJiri Pirko 			return -EINVAL;
472*830c41e1SJiri Pirko 		value->vu32 = nla_get_u32(param_data);
473*830c41e1SJiri Pirko 		break;
474*830c41e1SJiri Pirko 	case DEVLINK_PARAM_TYPE_STRING:
475*830c41e1SJiri Pirko 		len = strnlen(nla_data(param_data), nla_len(param_data));
476*830c41e1SJiri Pirko 		if (len == nla_len(param_data) ||
477*830c41e1SJiri Pirko 		    len >= __DEVLINK_PARAM_MAX_STRING_VALUE)
478*830c41e1SJiri Pirko 			return -EINVAL;
479*830c41e1SJiri Pirko 		strcpy(value->vstr, nla_data(param_data));
480*830c41e1SJiri Pirko 		break;
481*830c41e1SJiri Pirko 	case DEVLINK_PARAM_TYPE_BOOL:
482*830c41e1SJiri Pirko 		if (param_data && nla_len(param_data))
483*830c41e1SJiri Pirko 			return -EINVAL;
484*830c41e1SJiri Pirko 		value->vbool = nla_get_flag(param_data);
485*830c41e1SJiri Pirko 		break;
486*830c41e1SJiri Pirko 	}
487*830c41e1SJiri Pirko 	return 0;
488*830c41e1SJiri Pirko }
489*830c41e1SJiri Pirko 
490*830c41e1SJiri Pirko static struct devlink_param_item *
devlink_param_get_from_info(struct xarray * params,struct genl_info * info)491*830c41e1SJiri Pirko devlink_param_get_from_info(struct xarray *params, struct genl_info *info)
492*830c41e1SJiri Pirko {
493*830c41e1SJiri Pirko 	char *param_name;
494*830c41e1SJiri Pirko 
495*830c41e1SJiri Pirko 	if (GENL_REQ_ATTR_CHECK(info, DEVLINK_ATTR_PARAM_NAME))
496*830c41e1SJiri Pirko 		return NULL;
497*830c41e1SJiri Pirko 
498*830c41e1SJiri Pirko 	param_name = nla_data(info->attrs[DEVLINK_ATTR_PARAM_NAME]);
499*830c41e1SJiri Pirko 	return devlink_param_find_by_name(params, param_name);
500*830c41e1SJiri Pirko }
501*830c41e1SJiri Pirko 
devlink_nl_param_get_doit(struct sk_buff * skb,struct genl_info * info)502*830c41e1SJiri Pirko int devlink_nl_param_get_doit(struct sk_buff *skb,
503*830c41e1SJiri Pirko 			      struct genl_info *info)
504*830c41e1SJiri Pirko {
505*830c41e1SJiri Pirko 	struct devlink *devlink = info->user_ptr[0];
506*830c41e1SJiri Pirko 	struct devlink_param_item *param_item;
507*830c41e1SJiri Pirko 	struct sk_buff *msg;
508*830c41e1SJiri Pirko 	int err;
509*830c41e1SJiri Pirko 
510*830c41e1SJiri Pirko 	param_item = devlink_param_get_from_info(&devlink->params, info);
511*830c41e1SJiri Pirko 	if (!param_item)
512*830c41e1SJiri Pirko 		return -EINVAL;
513*830c41e1SJiri Pirko 
514*830c41e1SJiri Pirko 	msg = nlmsg_new(NLMSG_DEFAULT_SIZE, GFP_KERNEL);
515*830c41e1SJiri Pirko 	if (!msg)
516*830c41e1SJiri Pirko 		return -ENOMEM;
517*830c41e1SJiri Pirko 
518*830c41e1SJiri Pirko 	err = devlink_nl_param_fill(msg, devlink, 0, param_item,
519*830c41e1SJiri Pirko 				    DEVLINK_CMD_PARAM_GET,
520*830c41e1SJiri Pirko 				    info->snd_portid, info->snd_seq, 0);
521*830c41e1SJiri Pirko 	if (err) {
522*830c41e1SJiri Pirko 		nlmsg_free(msg);
523*830c41e1SJiri Pirko 		return err;
524*830c41e1SJiri Pirko 	}
525*830c41e1SJiri Pirko 
526*830c41e1SJiri Pirko 	return genlmsg_reply(msg, info);
527*830c41e1SJiri Pirko }
528*830c41e1SJiri Pirko 
__devlink_nl_cmd_param_set_doit(struct devlink * devlink,unsigned int port_index,struct xarray * params,struct genl_info * info,enum devlink_command cmd)529*830c41e1SJiri Pirko static int __devlink_nl_cmd_param_set_doit(struct devlink *devlink,
530*830c41e1SJiri Pirko 					   unsigned int port_index,
531*830c41e1SJiri Pirko 					   struct xarray *params,
532*830c41e1SJiri Pirko 					   struct genl_info *info,
533*830c41e1SJiri Pirko 					   enum devlink_command cmd)
534*830c41e1SJiri Pirko {
535*830c41e1SJiri Pirko 	enum devlink_param_type param_type;
536*830c41e1SJiri Pirko 	struct devlink_param_gset_ctx ctx;
537*830c41e1SJiri Pirko 	enum devlink_param_cmode cmode;
538*830c41e1SJiri Pirko 	struct devlink_param_item *param_item;
539*830c41e1SJiri Pirko 	const struct devlink_param *param;
540*830c41e1SJiri Pirko 	union devlink_param_value value;
541*830c41e1SJiri Pirko 	int err = 0;
542*830c41e1SJiri Pirko 
543*830c41e1SJiri Pirko 	param_item = devlink_param_get_from_info(params, info);
544*830c41e1SJiri Pirko 	if (!param_item)
545*830c41e1SJiri Pirko 		return -EINVAL;
546*830c41e1SJiri Pirko 	param = param_item->param;
547*830c41e1SJiri Pirko 	err = devlink_param_type_get_from_info(info, &param_type);
548*830c41e1SJiri Pirko 	if (err)
549*830c41e1SJiri Pirko 		return err;
550*830c41e1SJiri Pirko 	if (param_type != param->type)
551*830c41e1SJiri Pirko 		return -EINVAL;
552*830c41e1SJiri Pirko 	err = devlink_param_value_get_from_info(param, info, &value);
553*830c41e1SJiri Pirko 	if (err)
554*830c41e1SJiri Pirko 		return err;
555*830c41e1SJiri Pirko 	if (param->validate) {
556*830c41e1SJiri Pirko 		err = param->validate(devlink, param->id, value, info->extack);
557*830c41e1SJiri Pirko 		if (err)
558*830c41e1SJiri Pirko 			return err;
559*830c41e1SJiri Pirko 	}
560*830c41e1SJiri Pirko 
561*830c41e1SJiri Pirko 	if (GENL_REQ_ATTR_CHECK(info, DEVLINK_ATTR_PARAM_VALUE_CMODE))
562*830c41e1SJiri Pirko 		return -EINVAL;
563*830c41e1SJiri Pirko 	cmode = nla_get_u8(info->attrs[DEVLINK_ATTR_PARAM_VALUE_CMODE]);
564*830c41e1SJiri Pirko 	if (!devlink_param_cmode_is_supported(param, cmode))
565*830c41e1SJiri Pirko 		return -EOPNOTSUPP;
566*830c41e1SJiri Pirko 
567*830c41e1SJiri Pirko 	if (cmode == DEVLINK_PARAM_CMODE_DRIVERINIT) {
568*830c41e1SJiri Pirko 		param_item->driverinit_value_new = value;
569*830c41e1SJiri Pirko 		param_item->driverinit_value_new_valid = true;
570*830c41e1SJiri Pirko 	} else {
571*830c41e1SJiri Pirko 		if (!param->set)
572*830c41e1SJiri Pirko 			return -EOPNOTSUPP;
573*830c41e1SJiri Pirko 		ctx.val = value;
574*830c41e1SJiri Pirko 		ctx.cmode = cmode;
575*830c41e1SJiri Pirko 		err = devlink_param_set(devlink, param, &ctx);
576*830c41e1SJiri Pirko 		if (err)
577*830c41e1SJiri Pirko 			return err;
578*830c41e1SJiri Pirko 	}
579*830c41e1SJiri Pirko 
580*830c41e1SJiri Pirko 	devlink_param_notify(devlink, port_index, param_item, cmd);
581*830c41e1SJiri Pirko 	return 0;
582*830c41e1SJiri Pirko }
583*830c41e1SJiri Pirko 
devlink_nl_cmd_param_set_doit(struct sk_buff * skb,struct genl_info * info)584*830c41e1SJiri Pirko int devlink_nl_cmd_param_set_doit(struct sk_buff *skb, struct genl_info *info)
585*830c41e1SJiri Pirko {
586*830c41e1SJiri Pirko 	struct devlink *devlink = info->user_ptr[0];
587*830c41e1SJiri Pirko 
588*830c41e1SJiri Pirko 	return __devlink_nl_cmd_param_set_doit(devlink, 0, &devlink->params,
589*830c41e1SJiri Pirko 					       info, DEVLINK_CMD_PARAM_NEW);
590*830c41e1SJiri Pirko }
591*830c41e1SJiri Pirko 
devlink_nl_cmd_port_param_get_dumpit(struct sk_buff * msg,struct netlink_callback * cb)592*830c41e1SJiri Pirko int devlink_nl_cmd_port_param_get_dumpit(struct sk_buff *msg,
593*830c41e1SJiri Pirko 					 struct netlink_callback *cb)
594*830c41e1SJiri Pirko {
595*830c41e1SJiri Pirko 	NL_SET_ERR_MSG(cb->extack, "Port params are not supported");
596*830c41e1SJiri Pirko 	return msg->len;
597*830c41e1SJiri Pirko }
598*830c41e1SJiri Pirko 
devlink_nl_cmd_port_param_get_doit(struct sk_buff * skb,struct genl_info * info)599*830c41e1SJiri Pirko int devlink_nl_cmd_port_param_get_doit(struct sk_buff *skb,
600*830c41e1SJiri Pirko 				       struct genl_info *info)
601*830c41e1SJiri Pirko {
602*830c41e1SJiri Pirko 	NL_SET_ERR_MSG(info->extack, "Port params are not supported");
603*830c41e1SJiri Pirko 	return -EINVAL;
604*830c41e1SJiri Pirko }
605*830c41e1SJiri Pirko 
devlink_nl_cmd_port_param_set_doit(struct sk_buff * skb,struct genl_info * info)606*830c41e1SJiri Pirko int devlink_nl_cmd_port_param_set_doit(struct sk_buff *skb,
607*830c41e1SJiri Pirko 				       struct genl_info *info)
608*830c41e1SJiri Pirko {
609*830c41e1SJiri Pirko 	NL_SET_ERR_MSG(info->extack, "Port params are not supported");
610*830c41e1SJiri Pirko 	return -EINVAL;
611*830c41e1SJiri Pirko }
612*830c41e1SJiri Pirko 
devlink_param_verify(const struct devlink_param * param)613*830c41e1SJiri Pirko static int devlink_param_verify(const struct devlink_param *param)
614*830c41e1SJiri Pirko {
615*830c41e1SJiri Pirko 	if (!param || !param->name || !param->supported_cmodes)
616*830c41e1SJiri Pirko 		return -EINVAL;
617*830c41e1SJiri Pirko 	if (param->generic)
618*830c41e1SJiri Pirko 		return devlink_param_generic_verify(param);
619*830c41e1SJiri Pirko 	else
620*830c41e1SJiri Pirko 		return devlink_param_driver_verify(param);
621*830c41e1SJiri Pirko }
622*830c41e1SJiri Pirko 
devlink_param_register(struct devlink * devlink,const struct devlink_param * param)623*830c41e1SJiri Pirko static int devlink_param_register(struct devlink *devlink,
624*830c41e1SJiri Pirko 				  const struct devlink_param *param)
625*830c41e1SJiri Pirko {
626*830c41e1SJiri Pirko 	struct devlink_param_item *param_item;
627*830c41e1SJiri Pirko 	int err;
628*830c41e1SJiri Pirko 
629*830c41e1SJiri Pirko 	WARN_ON(devlink_param_verify(param));
630*830c41e1SJiri Pirko 	WARN_ON(devlink_param_find_by_name(&devlink->params, param->name));
631*830c41e1SJiri Pirko 
632*830c41e1SJiri Pirko 	if (param->supported_cmodes == BIT(DEVLINK_PARAM_CMODE_DRIVERINIT))
633*830c41e1SJiri Pirko 		WARN_ON(param->get || param->set);
634*830c41e1SJiri Pirko 	else
635*830c41e1SJiri Pirko 		WARN_ON(!param->get || !param->set);
636*830c41e1SJiri Pirko 
637*830c41e1SJiri Pirko 	param_item = kzalloc(sizeof(*param_item), GFP_KERNEL);
638*830c41e1SJiri Pirko 	if (!param_item)
639*830c41e1SJiri Pirko 		return -ENOMEM;
640*830c41e1SJiri Pirko 
641*830c41e1SJiri Pirko 	param_item->param = param;
642*830c41e1SJiri Pirko 
643*830c41e1SJiri Pirko 	err = xa_insert(&devlink->params, param->id, param_item, GFP_KERNEL);
644*830c41e1SJiri Pirko 	if (err)
645*830c41e1SJiri Pirko 		goto err_xa_insert;
646*830c41e1SJiri Pirko 
647*830c41e1SJiri Pirko 	devlink_param_notify(devlink, 0, param_item, DEVLINK_CMD_PARAM_NEW);
648*830c41e1SJiri Pirko 	return 0;
649*830c41e1SJiri Pirko 
650*830c41e1SJiri Pirko err_xa_insert:
651*830c41e1SJiri Pirko 	kfree(param_item);
652*830c41e1SJiri Pirko 	return err;
653*830c41e1SJiri Pirko }
654*830c41e1SJiri Pirko 
devlink_param_unregister(struct devlink * devlink,const struct devlink_param * param)655*830c41e1SJiri Pirko static void devlink_param_unregister(struct devlink *devlink,
656*830c41e1SJiri Pirko 				     const struct devlink_param *param)
657*830c41e1SJiri Pirko {
658*830c41e1SJiri Pirko 	struct devlink_param_item *param_item;
659*830c41e1SJiri Pirko 
660*830c41e1SJiri Pirko 	param_item = devlink_param_find_by_id(&devlink->params, param->id);
661*830c41e1SJiri Pirko 	if (WARN_ON(!param_item))
662*830c41e1SJiri Pirko 		return;
663*830c41e1SJiri Pirko 	devlink_param_notify(devlink, 0, param_item, DEVLINK_CMD_PARAM_DEL);
664*830c41e1SJiri Pirko 	xa_erase(&devlink->params, param->id);
665*830c41e1SJiri Pirko 	kfree(param_item);
666*830c41e1SJiri Pirko }
667*830c41e1SJiri Pirko 
668*830c41e1SJiri Pirko /**
669*830c41e1SJiri Pirko  *	devl_params_register - register configuration parameters
670*830c41e1SJiri Pirko  *
671*830c41e1SJiri Pirko  *	@devlink: devlink
672*830c41e1SJiri Pirko  *	@params: configuration parameters array
673*830c41e1SJiri Pirko  *	@params_count: number of parameters provided
674*830c41e1SJiri Pirko  *
675*830c41e1SJiri Pirko  *	Register the configuration parameters supported by the driver.
676*830c41e1SJiri Pirko  */
devl_params_register(struct devlink * devlink,const struct devlink_param * params,size_t params_count)677*830c41e1SJiri Pirko int devl_params_register(struct devlink *devlink,
678*830c41e1SJiri Pirko 			 const struct devlink_param *params,
679*830c41e1SJiri Pirko 			 size_t params_count)
680*830c41e1SJiri Pirko {
681*830c41e1SJiri Pirko 	const struct devlink_param *param = params;
682*830c41e1SJiri Pirko 	int i, err;
683*830c41e1SJiri Pirko 
684*830c41e1SJiri Pirko 	lockdep_assert_held(&devlink->lock);
685*830c41e1SJiri Pirko 
686*830c41e1SJiri Pirko 	for (i = 0; i < params_count; i++, param++) {
687*830c41e1SJiri Pirko 		err = devlink_param_register(devlink, param);
688*830c41e1SJiri Pirko 		if (err)
689*830c41e1SJiri Pirko 			goto rollback;
690*830c41e1SJiri Pirko 	}
691*830c41e1SJiri Pirko 	return 0;
692*830c41e1SJiri Pirko 
693*830c41e1SJiri Pirko rollback:
694*830c41e1SJiri Pirko 	if (!i)
695*830c41e1SJiri Pirko 		return err;
696*830c41e1SJiri Pirko 
697*830c41e1SJiri Pirko 	for (param--; i > 0; i--, param--)
698*830c41e1SJiri Pirko 		devlink_param_unregister(devlink, param);
699*830c41e1SJiri Pirko 	return err;
700*830c41e1SJiri Pirko }
701*830c41e1SJiri Pirko EXPORT_SYMBOL_GPL(devl_params_register);
702*830c41e1SJiri Pirko 
devlink_params_register(struct devlink * devlink,const struct devlink_param * params,size_t params_count)703*830c41e1SJiri Pirko int devlink_params_register(struct devlink *devlink,
704*830c41e1SJiri Pirko 			    const struct devlink_param *params,
705*830c41e1SJiri Pirko 			    size_t params_count)
706*830c41e1SJiri Pirko {
707*830c41e1SJiri Pirko 	int err;
708*830c41e1SJiri Pirko 
709*830c41e1SJiri Pirko 	devl_lock(devlink);
710*830c41e1SJiri Pirko 	err = devl_params_register(devlink, params, params_count);
711*830c41e1SJiri Pirko 	devl_unlock(devlink);
712*830c41e1SJiri Pirko 	return err;
713*830c41e1SJiri Pirko }
714*830c41e1SJiri Pirko EXPORT_SYMBOL_GPL(devlink_params_register);
715*830c41e1SJiri Pirko 
716*830c41e1SJiri Pirko /**
717*830c41e1SJiri Pirko  *	devl_params_unregister - unregister configuration parameters
718*830c41e1SJiri Pirko  *	@devlink: devlink
719*830c41e1SJiri Pirko  *	@params: configuration parameters to unregister
720*830c41e1SJiri Pirko  *	@params_count: number of parameters provided
721*830c41e1SJiri Pirko  */
devl_params_unregister(struct devlink * devlink,const struct devlink_param * params,size_t params_count)722*830c41e1SJiri Pirko void devl_params_unregister(struct devlink *devlink,
723*830c41e1SJiri Pirko 			    const struct devlink_param *params,
724*830c41e1SJiri Pirko 			    size_t params_count)
725*830c41e1SJiri Pirko {
726*830c41e1SJiri Pirko 	const struct devlink_param *param = params;
727*830c41e1SJiri Pirko 	int i;
728*830c41e1SJiri Pirko 
729*830c41e1SJiri Pirko 	lockdep_assert_held(&devlink->lock);
730*830c41e1SJiri Pirko 
731*830c41e1SJiri Pirko 	for (i = 0; i < params_count; i++, param++)
732*830c41e1SJiri Pirko 		devlink_param_unregister(devlink, param);
733*830c41e1SJiri Pirko }
734*830c41e1SJiri Pirko EXPORT_SYMBOL_GPL(devl_params_unregister);
735*830c41e1SJiri Pirko 
devlink_params_unregister(struct devlink * devlink,const struct devlink_param * params,size_t params_count)736*830c41e1SJiri Pirko void devlink_params_unregister(struct devlink *devlink,
737*830c41e1SJiri Pirko 			       const struct devlink_param *params,
738*830c41e1SJiri Pirko 			       size_t params_count)
739*830c41e1SJiri Pirko {
740*830c41e1SJiri Pirko 	devl_lock(devlink);
741*830c41e1SJiri Pirko 	devl_params_unregister(devlink, params, params_count);
742*830c41e1SJiri Pirko 	devl_unlock(devlink);
743*830c41e1SJiri Pirko }
744*830c41e1SJiri Pirko EXPORT_SYMBOL_GPL(devlink_params_unregister);
745*830c41e1SJiri Pirko 
746*830c41e1SJiri Pirko /**
747*830c41e1SJiri Pirko  *	devl_param_driverinit_value_get - get configuration parameter
748*830c41e1SJiri Pirko  *					  value for driver initializing
749*830c41e1SJiri Pirko  *
750*830c41e1SJiri Pirko  *	@devlink: devlink
751*830c41e1SJiri Pirko  *	@param_id: parameter ID
752*830c41e1SJiri Pirko  *	@val: pointer to store the value of parameter in driverinit
753*830c41e1SJiri Pirko  *	      configuration mode
754*830c41e1SJiri Pirko  *
755*830c41e1SJiri Pirko  *	This function should be used by the driver to get driverinit
756*830c41e1SJiri Pirko  *	configuration for initialization after reload command.
757*830c41e1SJiri Pirko  *
758*830c41e1SJiri Pirko  *	Note that lockless call of this function relies on the
759*830c41e1SJiri Pirko  *	driver to maintain following basic sane behavior:
760*830c41e1SJiri Pirko  *	1) Driver ensures a call to this function cannot race with
761*830c41e1SJiri Pirko  *	   registering/unregistering the parameter with the same parameter ID.
762*830c41e1SJiri Pirko  *	2) Driver ensures a call to this function cannot race with
763*830c41e1SJiri Pirko  *	   devl_param_driverinit_value_set() call with the same parameter ID.
764*830c41e1SJiri Pirko  *	3) Driver ensures a call to this function cannot race with
765*830c41e1SJiri Pirko  *	   reload operation.
766*830c41e1SJiri Pirko  *	If the driver is not able to comply, it has to take the devlink->lock
767*830c41e1SJiri Pirko  *	while calling this.
768*830c41e1SJiri Pirko  */
devl_param_driverinit_value_get(struct devlink * devlink,u32 param_id,union devlink_param_value * val)769*830c41e1SJiri Pirko int devl_param_driverinit_value_get(struct devlink *devlink, u32 param_id,
770*830c41e1SJiri Pirko 				    union devlink_param_value *val)
771*830c41e1SJiri Pirko {
772*830c41e1SJiri Pirko 	struct devlink_param_item *param_item;
773*830c41e1SJiri Pirko 
774*830c41e1SJiri Pirko 	if (WARN_ON(!devlink_reload_supported(devlink->ops)))
775*830c41e1SJiri Pirko 		return -EOPNOTSUPP;
776*830c41e1SJiri Pirko 
777*830c41e1SJiri Pirko 	param_item = devlink_param_find_by_id(&devlink->params, param_id);
778*830c41e1SJiri Pirko 	if (!param_item)
779*830c41e1SJiri Pirko 		return -EINVAL;
780*830c41e1SJiri Pirko 
781*830c41e1SJiri Pirko 	if (!param_item->driverinit_value_valid)
782*830c41e1SJiri Pirko 		return -EOPNOTSUPP;
783*830c41e1SJiri Pirko 
784*830c41e1SJiri Pirko 	if (WARN_ON(!devlink_param_cmode_is_supported(param_item->param,
785*830c41e1SJiri Pirko 						      DEVLINK_PARAM_CMODE_DRIVERINIT)))
786*830c41e1SJiri Pirko 		return -EOPNOTSUPP;
787*830c41e1SJiri Pirko 
788*830c41e1SJiri Pirko 	*val = param_item->driverinit_value;
789*830c41e1SJiri Pirko 
790*830c41e1SJiri Pirko 	return 0;
791*830c41e1SJiri Pirko }
792*830c41e1SJiri Pirko EXPORT_SYMBOL_GPL(devl_param_driverinit_value_get);
793*830c41e1SJiri Pirko 
794*830c41e1SJiri Pirko /**
795*830c41e1SJiri Pirko  *	devl_param_driverinit_value_set - set value of configuration
796*830c41e1SJiri Pirko  *					  parameter for driverinit
797*830c41e1SJiri Pirko  *					  configuration mode
798*830c41e1SJiri Pirko  *
799*830c41e1SJiri Pirko  *	@devlink: devlink
800*830c41e1SJiri Pirko  *	@param_id: parameter ID
801*830c41e1SJiri Pirko  *	@init_val: value of parameter to set for driverinit configuration mode
802*830c41e1SJiri Pirko  *
803*830c41e1SJiri Pirko  *	This function should be used by the driver to set driverinit
804*830c41e1SJiri Pirko  *	configuration mode default value.
805*830c41e1SJiri Pirko  */
devl_param_driverinit_value_set(struct devlink * devlink,u32 param_id,union devlink_param_value init_val)806*830c41e1SJiri Pirko void devl_param_driverinit_value_set(struct devlink *devlink, u32 param_id,
807*830c41e1SJiri Pirko 				     union devlink_param_value init_val)
808*830c41e1SJiri Pirko {
809*830c41e1SJiri Pirko 	struct devlink_param_item *param_item;
810*830c41e1SJiri Pirko 
811*830c41e1SJiri Pirko 	devl_assert_locked(devlink);
812*830c41e1SJiri Pirko 
813*830c41e1SJiri Pirko 	param_item = devlink_param_find_by_id(&devlink->params, param_id);
814*830c41e1SJiri Pirko 	if (WARN_ON(!param_item))
815*830c41e1SJiri Pirko 		return;
816*830c41e1SJiri Pirko 
817*830c41e1SJiri Pirko 	if (WARN_ON(!devlink_param_cmode_is_supported(param_item->param,
818*830c41e1SJiri Pirko 						      DEVLINK_PARAM_CMODE_DRIVERINIT)))
819*830c41e1SJiri Pirko 		return;
820*830c41e1SJiri Pirko 
821*830c41e1SJiri Pirko 	param_item->driverinit_value = init_val;
822*830c41e1SJiri Pirko 	param_item->driverinit_value_valid = true;
823*830c41e1SJiri Pirko 
824*830c41e1SJiri Pirko 	devlink_param_notify(devlink, 0, param_item, DEVLINK_CMD_PARAM_NEW);
825*830c41e1SJiri Pirko }
826*830c41e1SJiri Pirko EXPORT_SYMBOL_GPL(devl_param_driverinit_value_set);
827*830c41e1SJiri Pirko 
devlink_params_driverinit_load_new(struct devlink * devlink)828*830c41e1SJiri Pirko void devlink_params_driverinit_load_new(struct devlink *devlink)
829*830c41e1SJiri Pirko {
830*830c41e1SJiri Pirko 	struct devlink_param_item *param_item;
831*830c41e1SJiri Pirko 	unsigned long param_id;
832*830c41e1SJiri Pirko 
833*830c41e1SJiri Pirko 	xa_for_each(&devlink->params, param_id, param_item) {
834*830c41e1SJiri Pirko 		if (!devlink_param_cmode_is_supported(param_item->param,
835*830c41e1SJiri Pirko 						      DEVLINK_PARAM_CMODE_DRIVERINIT) ||
836*830c41e1SJiri Pirko 		    !param_item->driverinit_value_new_valid)
837*830c41e1SJiri Pirko 			continue;
838*830c41e1SJiri Pirko 		param_item->driverinit_value = param_item->driverinit_value_new;
839*830c41e1SJiri Pirko 		param_item->driverinit_value_valid = true;
840*830c41e1SJiri Pirko 		param_item->driverinit_value_new_valid = false;
841*830c41e1SJiri Pirko 	}
842*830c41e1SJiri Pirko }
843*830c41e1SJiri Pirko 
844*830c41e1SJiri Pirko /**
845*830c41e1SJiri Pirko  *	devl_param_value_changed - notify devlink on a parameter's value
846*830c41e1SJiri Pirko  *				   change. Should be called by the driver
847*830c41e1SJiri Pirko  *				   right after the change.
848*830c41e1SJiri Pirko  *
849*830c41e1SJiri Pirko  *	@devlink: devlink
850*830c41e1SJiri Pirko  *	@param_id: parameter ID
851*830c41e1SJiri Pirko  *
852*830c41e1SJiri Pirko  *	This function should be used by the driver to notify devlink on value
853*830c41e1SJiri Pirko  *	change, excluding driverinit configuration mode.
854*830c41e1SJiri Pirko  *	For driverinit configuration mode driver should use the function
855*830c41e1SJiri Pirko  */
devl_param_value_changed(struct devlink * devlink,u32 param_id)856*830c41e1SJiri Pirko void devl_param_value_changed(struct devlink *devlink, u32 param_id)
857*830c41e1SJiri Pirko {
858*830c41e1SJiri Pirko 	struct devlink_param_item *param_item;
859*830c41e1SJiri Pirko 
860*830c41e1SJiri Pirko 	param_item = devlink_param_find_by_id(&devlink->params, param_id);
861*830c41e1SJiri Pirko 	WARN_ON(!param_item);
862*830c41e1SJiri Pirko 
863*830c41e1SJiri Pirko 	devlink_param_notify(devlink, 0, param_item, DEVLINK_CMD_PARAM_NEW);
864*830c41e1SJiri Pirko }
865*830c41e1SJiri Pirko EXPORT_SYMBOL_GPL(devl_param_value_changed);
866