162306a36Sopenharmony_ci/* SPDX-License-Identifier: (GPL-2.0-only OR BSD-2-Clause) */
262306a36Sopenharmony_ci/* Copyright (C) 2016-2019 Netronome Systems, Inc. */
362306a36Sopenharmony_ci
462306a36Sopenharmony_ci#ifndef NFP_CCM_H
562306a36Sopenharmony_ci#define NFP_CCM_H 1
662306a36Sopenharmony_ci
762306a36Sopenharmony_ci#include <linux/bitmap.h>
862306a36Sopenharmony_ci#include <linux/skbuff.h>
962306a36Sopenharmony_ci#include <linux/wait.h>
1062306a36Sopenharmony_ci
1162306a36Sopenharmony_cistruct nfp_app;
1262306a36Sopenharmony_cistruct nfp_net;
1362306a36Sopenharmony_ci
1462306a36Sopenharmony_ci/* Firmware ABI */
1562306a36Sopenharmony_ci
1662306a36Sopenharmony_cienum nfp_ccm_type {
1762306a36Sopenharmony_ci	NFP_CCM_TYPE_BPF_MAP_ALLOC	= 1,
1862306a36Sopenharmony_ci	NFP_CCM_TYPE_BPF_MAP_FREE	= 2,
1962306a36Sopenharmony_ci	NFP_CCM_TYPE_BPF_MAP_LOOKUP	= 3,
2062306a36Sopenharmony_ci	NFP_CCM_TYPE_BPF_MAP_UPDATE	= 4,
2162306a36Sopenharmony_ci	NFP_CCM_TYPE_BPF_MAP_DELETE	= 5,
2262306a36Sopenharmony_ci	NFP_CCM_TYPE_BPF_MAP_GETNEXT	= 6,
2362306a36Sopenharmony_ci	NFP_CCM_TYPE_BPF_MAP_GETFIRST	= 7,
2462306a36Sopenharmony_ci	NFP_CCM_TYPE_BPF_BPF_EVENT	= 8,
2562306a36Sopenharmony_ci	NFP_CCM_TYPE_CRYPTO_RESET	= 9,
2662306a36Sopenharmony_ci	NFP_CCM_TYPE_CRYPTO_ADD		= 10,
2762306a36Sopenharmony_ci	NFP_CCM_TYPE_CRYPTO_DEL		= 11,
2862306a36Sopenharmony_ci	NFP_CCM_TYPE_CRYPTO_UPDATE	= 12,
2962306a36Sopenharmony_ci	NFP_CCM_TYPE_CRYPTO_RESYNC	= 13,
3062306a36Sopenharmony_ci	__NFP_CCM_TYPE_MAX,
3162306a36Sopenharmony_ci};
3262306a36Sopenharmony_ci
3362306a36Sopenharmony_ci#define NFP_CCM_ABI_VERSION		1
3462306a36Sopenharmony_ci
3562306a36Sopenharmony_ci#define NFP_CCM_TYPE_REPLY_BIT		7
3662306a36Sopenharmony_ci#define __NFP_CCM_REPLY(req)		(BIT(NFP_CCM_TYPE_REPLY_BIT) | (req))
3762306a36Sopenharmony_ci
3862306a36Sopenharmony_cistruct nfp_ccm_hdr {
3962306a36Sopenharmony_ci	union {
4062306a36Sopenharmony_ci		struct {
4162306a36Sopenharmony_ci			u8 type;
4262306a36Sopenharmony_ci			u8 ver;
4362306a36Sopenharmony_ci			__be16 tag;
4462306a36Sopenharmony_ci		};
4562306a36Sopenharmony_ci		__be32 raw;
4662306a36Sopenharmony_ci	};
4762306a36Sopenharmony_ci};
4862306a36Sopenharmony_ci
4962306a36Sopenharmony_cistatic inline u8 nfp_ccm_get_type(struct sk_buff *skb)
5062306a36Sopenharmony_ci{
5162306a36Sopenharmony_ci	struct nfp_ccm_hdr *hdr;
5262306a36Sopenharmony_ci
5362306a36Sopenharmony_ci	hdr = (struct nfp_ccm_hdr *)skb->data;
5462306a36Sopenharmony_ci
5562306a36Sopenharmony_ci	return hdr->type;
5662306a36Sopenharmony_ci}
5762306a36Sopenharmony_ci
5862306a36Sopenharmony_cistatic inline __be16 __nfp_ccm_get_tag(struct sk_buff *skb)
5962306a36Sopenharmony_ci{
6062306a36Sopenharmony_ci	struct nfp_ccm_hdr *hdr;
6162306a36Sopenharmony_ci
6262306a36Sopenharmony_ci	hdr = (struct nfp_ccm_hdr *)skb->data;
6362306a36Sopenharmony_ci
6462306a36Sopenharmony_ci	return hdr->tag;
6562306a36Sopenharmony_ci}
6662306a36Sopenharmony_ci
6762306a36Sopenharmony_cistatic inline unsigned int nfp_ccm_get_tag(struct sk_buff *skb)
6862306a36Sopenharmony_ci{
6962306a36Sopenharmony_ci	return be16_to_cpu(__nfp_ccm_get_tag(skb));
7062306a36Sopenharmony_ci}
7162306a36Sopenharmony_ci
7262306a36Sopenharmony_ci#define NFP_NET_MBOX_TLV_TYPE		GENMASK(31, 16)
7362306a36Sopenharmony_ci#define NFP_NET_MBOX_TLV_LEN		GENMASK(15, 0)
7462306a36Sopenharmony_ci
7562306a36Sopenharmony_cienum nfp_ccm_mbox_tlv_type {
7662306a36Sopenharmony_ci	NFP_NET_MBOX_TLV_TYPE_UNKNOWN	= 0,
7762306a36Sopenharmony_ci	NFP_NET_MBOX_TLV_TYPE_END	= 1,
7862306a36Sopenharmony_ci	NFP_NET_MBOX_TLV_TYPE_MSG	= 2,
7962306a36Sopenharmony_ci	NFP_NET_MBOX_TLV_TYPE_MSG_NOSUP	= 3,
8062306a36Sopenharmony_ci	NFP_NET_MBOX_TLV_TYPE_RESV	= 4,
8162306a36Sopenharmony_ci};
8262306a36Sopenharmony_ci
8362306a36Sopenharmony_ci/* Implementation */
8462306a36Sopenharmony_ci
8562306a36Sopenharmony_ci/**
8662306a36Sopenharmony_ci * struct nfp_ccm - common control message handling
8762306a36Sopenharmony_ci * @app:		APP handle
8862306a36Sopenharmony_ci *
8962306a36Sopenharmony_ci * @tag_allocator:	bitmap of control message tags in use
9062306a36Sopenharmony_ci * @tag_alloc_next:	next tag bit to allocate
9162306a36Sopenharmony_ci * @tag_alloc_last:	next tag bit to be freed
9262306a36Sopenharmony_ci *
9362306a36Sopenharmony_ci * @replies:		received cmsg replies waiting to be consumed
9462306a36Sopenharmony_ci * @wq:			work queue for waiting for cmsg replies
9562306a36Sopenharmony_ci */
9662306a36Sopenharmony_cistruct nfp_ccm {
9762306a36Sopenharmony_ci	struct nfp_app *app;
9862306a36Sopenharmony_ci
9962306a36Sopenharmony_ci	DECLARE_BITMAP(tag_allocator, U16_MAX + 1);
10062306a36Sopenharmony_ci	u16 tag_alloc_next;
10162306a36Sopenharmony_ci	u16 tag_alloc_last;
10262306a36Sopenharmony_ci
10362306a36Sopenharmony_ci	struct sk_buff_head replies;
10462306a36Sopenharmony_ci	wait_queue_head_t wq;
10562306a36Sopenharmony_ci};
10662306a36Sopenharmony_ci
10762306a36Sopenharmony_ciint nfp_ccm_init(struct nfp_ccm *ccm, struct nfp_app *app);
10862306a36Sopenharmony_civoid nfp_ccm_clean(struct nfp_ccm *ccm);
10962306a36Sopenharmony_civoid nfp_ccm_rx(struct nfp_ccm *ccm, struct sk_buff *skb);
11062306a36Sopenharmony_cistruct sk_buff *
11162306a36Sopenharmony_cinfp_ccm_communicate(struct nfp_ccm *ccm, struct sk_buff *skb,
11262306a36Sopenharmony_ci		    enum nfp_ccm_type type, unsigned int reply_size);
11362306a36Sopenharmony_ci
11462306a36Sopenharmony_ciint nfp_ccm_mbox_alloc(struct nfp_net *nn);
11562306a36Sopenharmony_civoid nfp_ccm_mbox_free(struct nfp_net *nn);
11662306a36Sopenharmony_ciint nfp_ccm_mbox_init(struct nfp_net *nn);
11762306a36Sopenharmony_civoid nfp_ccm_mbox_clean(struct nfp_net *nn);
11862306a36Sopenharmony_cibool nfp_ccm_mbox_fits(struct nfp_net *nn, unsigned int size);
11962306a36Sopenharmony_cistruct sk_buff *
12062306a36Sopenharmony_cinfp_ccm_mbox_msg_alloc(struct nfp_net *nn, unsigned int req_size,
12162306a36Sopenharmony_ci		       unsigned int reply_size, gfp_t flags);
12262306a36Sopenharmony_ciint __nfp_ccm_mbox_communicate(struct nfp_net *nn, struct sk_buff *skb,
12362306a36Sopenharmony_ci			       enum nfp_ccm_type type,
12462306a36Sopenharmony_ci			       unsigned int reply_size,
12562306a36Sopenharmony_ci			       unsigned int max_reply_size, bool critical);
12662306a36Sopenharmony_ciint nfp_ccm_mbox_communicate(struct nfp_net *nn, struct sk_buff *skb,
12762306a36Sopenharmony_ci			     enum nfp_ccm_type type,
12862306a36Sopenharmony_ci			     unsigned int reply_size,
12962306a36Sopenharmony_ci			     unsigned int max_reply_size);
13062306a36Sopenharmony_ciint nfp_ccm_mbox_post(struct nfp_net *nn, struct sk_buff *skb,
13162306a36Sopenharmony_ci		      enum nfp_ccm_type type, unsigned int max_reply_size);
13262306a36Sopenharmony_ci#endif
133