162306a36Sopenharmony_ci/*
262306a36Sopenharmony_ci * Copyright (c) 2013, Cisco Systems, Inc. All rights reserved.
362306a36Sopenharmony_ci *
462306a36Sopenharmony_ci * This software is available to you under a choice of one of two
562306a36Sopenharmony_ci * licenses.  You may choose to be licensed under the terms of the GNU
662306a36Sopenharmony_ci * General Public License (GPL) Version 2, available from the file
762306a36Sopenharmony_ci * COPYING in the main directory of this source tree, or the
862306a36Sopenharmony_ci * BSD license below:
962306a36Sopenharmony_ci *
1062306a36Sopenharmony_ci *     Redistribution and use in source and binary forms, with or
1162306a36Sopenharmony_ci *     without modification, are permitted provided that the following
1262306a36Sopenharmony_ci *     conditions are met:
1362306a36Sopenharmony_ci *
1462306a36Sopenharmony_ci *      - Redistributions of source code must retain the above
1562306a36Sopenharmony_ci *        copyright notice, this list of conditions and the following
1662306a36Sopenharmony_ci *        disclaimer.
1762306a36Sopenharmony_ci *
1862306a36Sopenharmony_ci *      - Redistributions in binary form must reproduce the above
1962306a36Sopenharmony_ci *        copyright notice, this list of conditions and the following
2062306a36Sopenharmony_ci *        disclaimer in the documentation and/or other materials
2162306a36Sopenharmony_ci *        provided with the distribution.
2262306a36Sopenharmony_ci *
2362306a36Sopenharmony_ci * THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND,
2462306a36Sopenharmony_ci * EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF
2562306a36Sopenharmony_ci * MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND
2662306a36Sopenharmony_ci * NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS
2762306a36Sopenharmony_ci * BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN
2862306a36Sopenharmony_ci * ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN
2962306a36Sopenharmony_ci * CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE
3062306a36Sopenharmony_ci * SOFTWARE.
3162306a36Sopenharmony_ci *
3262306a36Sopenharmony_ci */
3362306a36Sopenharmony_ci
3462306a36Sopenharmony_ci#ifndef USNIC_UIOM_H_
3562306a36Sopenharmony_ci#define USNIC_UIOM_H_
3662306a36Sopenharmony_ci
3762306a36Sopenharmony_ci#include <linux/list.h>
3862306a36Sopenharmony_ci#include <linux/scatterlist.h>
3962306a36Sopenharmony_ci
4062306a36Sopenharmony_ci#include "usnic_uiom_interval_tree.h"
4162306a36Sopenharmony_ci
4262306a36Sopenharmony_cistruct ib_ucontext;
4362306a36Sopenharmony_ci
4462306a36Sopenharmony_ci#define USNIC_UIOM_READ			(1)
4562306a36Sopenharmony_ci#define USNIC_UIOM_WRITE		(2)
4662306a36Sopenharmony_ci
4762306a36Sopenharmony_ci#define USNIC_UIOM_MAX_PD_CNT		(1000)
4862306a36Sopenharmony_ci#define USNIC_UIOM_MAX_MR_CNT		(1000000)
4962306a36Sopenharmony_ci#define USNIC_UIOM_MAX_MR_SIZE		(~0UL)
5062306a36Sopenharmony_ci#define USNIC_UIOM_PAGE_SIZE		(PAGE_SIZE)
5162306a36Sopenharmony_ci
5262306a36Sopenharmony_cistruct usnic_uiom_dev {
5362306a36Sopenharmony_ci	struct device			*dev;
5462306a36Sopenharmony_ci	struct list_head		link;
5562306a36Sopenharmony_ci};
5662306a36Sopenharmony_ci
5762306a36Sopenharmony_cistruct usnic_uiom_pd {
5862306a36Sopenharmony_ci	struct iommu_domain		*domain;
5962306a36Sopenharmony_ci	spinlock_t			lock;
6062306a36Sopenharmony_ci	struct rb_root_cached		root;
6162306a36Sopenharmony_ci	struct list_head		devs;
6262306a36Sopenharmony_ci	int				dev_cnt;
6362306a36Sopenharmony_ci};
6462306a36Sopenharmony_ci
6562306a36Sopenharmony_cistruct usnic_uiom_reg {
6662306a36Sopenharmony_ci	struct usnic_uiom_pd		*pd;
6762306a36Sopenharmony_ci	unsigned long			va;
6862306a36Sopenharmony_ci	size_t				length;
6962306a36Sopenharmony_ci	int				offset;
7062306a36Sopenharmony_ci	int				page_size;
7162306a36Sopenharmony_ci	int				writable;
7262306a36Sopenharmony_ci	struct list_head		chunk_list;
7362306a36Sopenharmony_ci	struct work_struct		work;
7462306a36Sopenharmony_ci	struct mm_struct		*owning_mm;
7562306a36Sopenharmony_ci};
7662306a36Sopenharmony_ci
7762306a36Sopenharmony_cistruct usnic_uiom_chunk {
7862306a36Sopenharmony_ci	struct list_head		list;
7962306a36Sopenharmony_ci	int				nents;
8062306a36Sopenharmony_ci	struct scatterlist		page_list[];
8162306a36Sopenharmony_ci};
8262306a36Sopenharmony_ci
8362306a36Sopenharmony_cistruct usnic_uiom_pd *usnic_uiom_alloc_pd(struct device *dev);
8462306a36Sopenharmony_civoid usnic_uiom_dealloc_pd(struct usnic_uiom_pd *pd);
8562306a36Sopenharmony_ciint usnic_uiom_attach_dev_to_pd(struct usnic_uiom_pd *pd, struct device *dev);
8662306a36Sopenharmony_civoid usnic_uiom_detach_dev_from_pd(struct usnic_uiom_pd *pd,
8762306a36Sopenharmony_ci					struct device *dev);
8862306a36Sopenharmony_cistruct device **usnic_uiom_get_dev_list(struct usnic_uiom_pd *pd);
8962306a36Sopenharmony_civoid usnic_uiom_free_dev_list(struct device **devs);
9062306a36Sopenharmony_cistruct usnic_uiom_reg *usnic_uiom_reg_get(struct usnic_uiom_pd *pd,
9162306a36Sopenharmony_ci						unsigned long addr, size_t size,
9262306a36Sopenharmony_ci						int access, int dmasync);
9362306a36Sopenharmony_civoid usnic_uiom_reg_release(struct usnic_uiom_reg *uiomr);
9462306a36Sopenharmony_ci#endif /* USNIC_UIOM_H_ */
95