Commit 9ce6bbbb authored by Jakub Kicinski's avatar Jakub Kicinski Committed by David S. Miller

nfp: add nfp_app cleanup callback and make flower use it

Add a cleanup callback for undoing what app init callback did.
Make flower allocate its private structure on init and free
it from the new callback.

While at it remember to set the app pointer to NULL on the
error path to avoid any races while probe path unwinds.
Signed-off-by: default avatarJakub Kicinski <jakub.kicinski@netronome.com>
Reviewed-by: default avatarSimon Horman <simon.horman@netronome.com>
Signed-off-by: default avatarDavid S. Miller <davem@davemloft.net>
parent 8a119cef
...@@ -296,26 +296,16 @@ static int nfp_flower_start(struct nfp_app *app) ...@@ -296,26 +296,16 @@ static int nfp_flower_start(struct nfp_app *app)
NFP_REPR_TYPE_PF, 1); NFP_REPR_TYPE_PF, 1);
} }
static void nfp_flower_vnic_clean(struct nfp_app *app, struct nfp_net *nn)
{
kfree(app->priv);
app->priv = NULL;
}
static int nfp_flower_vnic_init(struct nfp_app *app, struct nfp_net *nn, static int nfp_flower_vnic_init(struct nfp_app *app, struct nfp_net *nn,
unsigned int id) unsigned int id)
{ {
struct nfp_flower_priv *priv; struct nfp_flower_priv *priv = app->priv;
if (id > 0) { if (id > 0) {
nfp_warn(app->cpp, "FlowerNIC doesn't support more than one data vNIC\n"); nfp_warn(app->cpp, "FlowerNIC doesn't support more than one data vNIC\n");
goto err_invalid_port; goto err_invalid_port;
} }
priv = kzalloc(sizeof(*priv), GFP_KERNEL);
if (!priv)
return -ENOMEM;
app->priv = priv;
priv->nn = nn; priv->nn = nn;
eth_hw_addr_random(nn->dp.netdev); eth_hw_addr_random(nn->dp.netdev);
...@@ -347,9 +337,19 @@ static int nfp_flower_init(struct nfp_app *app) ...@@ -347,9 +337,19 @@ static int nfp_flower_init(struct nfp_app *app)
return -EINVAL; return -EINVAL;
} }
app->priv = kzalloc(sizeof(struct nfp_flower_priv), GFP_KERNEL);
if (!app->priv)
return -ENOMEM;
return 0; return 0;
} }
static void nfp_flower_clean(struct nfp_app *app)
{
kfree(app->priv);
app->priv = NULL;
}
const struct nfp_app_type app_flower = { const struct nfp_app_type app_flower = {
.id = NFP_APP_FLOWER_NIC, .id = NFP_APP_FLOWER_NIC,
.name = "flower", .name = "flower",
...@@ -358,9 +358,9 @@ const struct nfp_app_type app_flower = { ...@@ -358,9 +358,9 @@ const struct nfp_app_type app_flower = {
.extra_cap = nfp_flower_extra_cap, .extra_cap = nfp_flower_extra_cap,
.init = nfp_flower_init, .init = nfp_flower_init,
.clean = nfp_flower_clean,
.vnic_init = nfp_flower_vnic_init, .vnic_init = nfp_flower_vnic_init,
.vnic_clean = nfp_flower_vnic_clean,
.start = nfp_flower_start, .start = nfp_flower_start,
.stop = nfp_flower_stop, .stop = nfp_flower_stop,
......
...@@ -66,7 +66,8 @@ extern const struct nfp_app_type app_flower; ...@@ -66,7 +66,8 @@ extern const struct nfp_app_type app_flower;
* @ctrl_has_meta: control messages have prepend of type:5/port:CTRL * @ctrl_has_meta: control messages have prepend of type:5/port:CTRL
* *
* Callbacks * Callbacks
* @init: perform basic app checks * @init: perform basic app checks and init
* @clean: clean app state
* @extra_cap: extra capabilities string * @extra_cap: extra capabilities string
* @vnic_init: init vNICs (assign port types, etc.) * @vnic_init: init vNICs (assign port types, etc.)
* @vnic_clean: clean up app's vNIC state * @vnic_clean: clean up app's vNIC state
...@@ -88,6 +89,7 @@ struct nfp_app_type { ...@@ -88,6 +89,7 @@ struct nfp_app_type {
bool ctrl_has_meta; bool ctrl_has_meta;
int (*init)(struct nfp_app *app); int (*init)(struct nfp_app *app);
void (*clean)(struct nfp_app *app);
const char *(*extra_cap)(struct nfp_app *app, struct nfp_net *nn); const char *(*extra_cap)(struct nfp_app *app, struct nfp_net *nn);
...@@ -144,6 +146,12 @@ static inline int nfp_app_init(struct nfp_app *app) ...@@ -144,6 +146,12 @@ static inline int nfp_app_init(struct nfp_app *app)
return app->type->init(app); return app->type->init(app);
} }
static inline void nfp_app_clean(struct nfp_app *app)
{
if (app->type->clean)
app->type->clean(app);
}
static inline int nfp_app_vnic_init(struct nfp_app *app, struct nfp_net *nn, static inline int nfp_app_vnic_init(struct nfp_app *app, struct nfp_net *nn,
unsigned int id) unsigned int id)
{ {
......
...@@ -414,7 +414,7 @@ nfp_net_pf_app_init(struct nfp_pf *pf, u8 __iomem *qc_bar, unsigned int stride) ...@@ -414,7 +414,7 @@ nfp_net_pf_app_init(struct nfp_pf *pf, u8 __iomem *qc_bar, unsigned int stride)
if (IS_ERR(ctrl_bar)) { if (IS_ERR(ctrl_bar)) {
nfp_err(pf->cpp, "Failed to find data vNIC memory symbol\n"); nfp_err(pf->cpp, "Failed to find data vNIC memory symbol\n");
err = PTR_ERR(ctrl_bar); err = PTR_ERR(ctrl_bar);
goto err_free; goto err_app_clean;
} }
pf->ctrl_vnic = nfp_net_pf_alloc_vnic(pf, false, ctrl_bar, qc_bar, pf->ctrl_vnic = nfp_net_pf_alloc_vnic(pf, false, ctrl_bar, qc_bar,
...@@ -428,8 +428,11 @@ nfp_net_pf_app_init(struct nfp_pf *pf, u8 __iomem *qc_bar, unsigned int stride) ...@@ -428,8 +428,11 @@ nfp_net_pf_app_init(struct nfp_pf *pf, u8 __iomem *qc_bar, unsigned int stride)
err_unmap: err_unmap:
nfp_cpp_area_release_free(pf->ctrl_vnic_bar); nfp_cpp_area_release_free(pf->ctrl_vnic_bar);
err_app_clean:
nfp_app_clean(pf->app);
err_free: err_free:
nfp_app_free(pf->app); nfp_app_free(pf->app);
pf->app = NULL;
return err; return err;
} }
...@@ -439,6 +442,7 @@ static void nfp_net_pf_app_clean(struct nfp_pf *pf) ...@@ -439,6 +442,7 @@ static void nfp_net_pf_app_clean(struct nfp_pf *pf)
nfp_net_pf_free_vnic(pf, pf->ctrl_vnic); nfp_net_pf_free_vnic(pf, pf->ctrl_vnic);
nfp_cpp_area_release_free(pf->ctrl_vnic_bar); nfp_cpp_area_release_free(pf->ctrl_vnic_bar);
} }
nfp_app_clean(pf->app);
nfp_app_free(pf->app); nfp_app_free(pf->app);
pf->app = NULL; pf->app = NULL;
} }
......
Markdown is supported
0%
or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment