blob: c922fc672a0f6a96239de29ad74bb175ad94444b [file] [log] [blame]
/*
* Copyright (c) 2014-2018 The Linux Foundation. All rights reserved.
*
* Permission to use, copy, modify, and/or distribute this software for
* any purpose with or without fee is hereby granted, provided that the
* above copyright notice and this permission notice appear in all
* copies.
*
* THE SOFTWARE IS PROVIDED "AS IS" AND THE AUTHOR DISCLAIMS ALL
* WARRANTIES WITH REGARD TO THIS SOFTWARE INCLUDING ALL IMPLIED
* WARRANTIES OF MERCHANTABILITY AND FITNESS. IN NO EVENT SHALL THE
* AUTHOR BE LIABLE FOR ANY SPECIAL, DIRECT, INDIRECT, OR CONSEQUENTIAL
* DAMAGES OR ANY DAMAGES WHATSOEVER RESULTING FROM LOSS OF USE, DATA OR
* PROFITS, WHETHER IN AN ACTION OF CONTRACT, NEGLIGENCE OR OTHER
* TORTIOUS ACTION, ARISING OUT OF OR IN CONNECTION WITH THE USE OR
* PERFORMANCE OF THIS SOFTWARE.
*/
/**
* DOC: i_qdf_mem.h
* Linux-specific definitions for QDF memory API's
*/
#ifndef __I_QDF_MEM_H
#define __I_QDF_MEM_H
#ifdef __KERNEL__
#if LINUX_VERSION_CODE > KERNEL_VERSION(2, 6, 17)
#if LINUX_VERSION_CODE < KERNEL_VERSION(2, 6, 33)
#include <linux/autoconf.h>
#else
#include <generated/autoconf.h>
#endif
#endif
#include <linux/slab.h>
#include <linux/hardirq.h>
#include <linux/vmalloc.h>
#include <linux/pci.h> /* pci_alloc_consistent */
#include <linux/cache.h> /* L1_CACHE_BYTES */
#define __qdf_cache_line_sz L1_CACHE_BYTES
#if CONFIG_MCL
#include <cds_queue.h>
#else
#include <sys/queue.h>
#endif
#else
/*
* Provide dummy defs for kernel data types, functions, and enums
* used in this header file.
*/
#define GFP_KERNEL 0
#define GFP_ATOMIC 0
#define kzalloc(size, flags) NULL
#define vmalloc(size) NULL
#define kfree(buf)
#define vfree(buf)
#define pci_alloc_consistent(dev, size, paddr) NULL
#define __qdf_mempool_t void*
#define QDF_RET_IP NULL
#endif /* __KERNEL__ */
#include <qdf_status.h>
#ifdef CONFIG_ARM_SMMU
#include <pld_common.h>
#include <asm/dma-iommu.h>
#include <linux/iommu.h>
#endif
#ifdef __KERNEL__
typedef struct mempool_elem {
STAILQ_ENTRY(mempool_elem) mempool_entry;
} mempool_elem_t;
/**
* typedef __qdf_mempool_ctxt_t - Memory pool context
* @pool_id: pool identifier
* @flags: flags
* @elem_size: size of each pool element in bytes
* @pool_mem: pool_addr address of the pool created
* @mem_size: Total size of the pool in bytes
* @free_list: free pool list
* @lock: spinlock object
* @max_elem: Maximum number of elements in tha pool
* @free_cnt: Number of free elements available
*/
typedef struct __qdf_mempool_ctxt {
int pool_id;
u_int32_t flags;
size_t elem_size;
void *pool_mem;
u_int32_t mem_size;
STAILQ_HEAD(, mempool_elem) free_list;
spinlock_t lock;
u_int32_t max_elem;
u_int32_t free_cnt;
} __qdf_mempool_ctxt_t;
#endif /* __KERNEL__ */
#define __qdf_align(a, mask) ALIGN(a, mask)
/* typedef for dma_data_direction */
typedef enum dma_data_direction __dma_data_direction;
/**
* __qdf_dma_dir_to_os() - Convert DMA data direction to OS specific enum
* @dir: QDF DMA data direction
*
* Return:
* enum dma_data_direction
*/
static inline
enum dma_data_direction __qdf_dma_dir_to_os(qdf_dma_dir_t qdf_dir)
{
switch (qdf_dir) {
case QDF_DMA_BIDIRECTIONAL:
return DMA_BIDIRECTIONAL;
case QDF_DMA_TO_DEVICE:
return DMA_TO_DEVICE;
case QDF_DMA_FROM_DEVICE:
return DMA_FROM_DEVICE;
default:
return DMA_NONE;
}
}
/**
* __qdf_mem_map_nbytes_single - Map memory for DMA
* @osdev: pomter OS device context
* @buf: pointer to memory to be dma mapped
* @dir: DMA map direction
* @nbytes: number of bytes to be mapped.
* @phy_addr: ponter to recive physical address.
*
* Return: success/failure
*/
static inline uint32_t __qdf_mem_map_nbytes_single(qdf_device_t osdev,
void *buf, qdf_dma_dir_t dir,
int nbytes,
qdf_dma_addr_t *phy_addr)
{
/* assume that the OS only provides a single fragment */
*phy_addr = dma_map_single(osdev->dev, buf, nbytes,
__qdf_dma_dir_to_os(dir));
return dma_mapping_error(osdev->dev, *phy_addr) ?
QDF_STATUS_E_FAILURE : QDF_STATUS_SUCCESS;
}
/**
* __qdf_mem_unmap_nbytes_single() - un_map memory for DMA
*
* @osdev: pomter OS device context
* @phy_addr: physical address of memory to be dma unmapped
* @dir: DMA unmap direction
* @nbytes: number of bytes to be unmapped.
*
* Return - none
*/
static inline void __qdf_mem_unmap_nbytes_single(qdf_device_t osdev,
qdf_dma_addr_t phy_addr,
qdf_dma_dir_t dir, int nbytes)
{
dma_unmap_single(osdev->dev, phy_addr, nbytes,
__qdf_dma_dir_to_os(dir));
}
#ifdef __KERNEL__
typedef __qdf_mempool_ctxt_t *__qdf_mempool_t;
int __qdf_mempool_init(qdf_device_t osdev, __qdf_mempool_t *pool, int pool_cnt,
size_t pool_entry_size, u_int32_t flags);
void __qdf_mempool_destroy(qdf_device_t osdev, __qdf_mempool_t pool);
void *__qdf_mempool_alloc(qdf_device_t osdev, __qdf_mempool_t pool);
void __qdf_mempool_free(qdf_device_t osdev, __qdf_mempool_t pool, void *buf);
#define QDF_RET_IP ((void *)_RET_IP_)
#define __qdf_mempool_elem_size(_pool) ((_pool)->elem_size)
#endif
/**
* __qdf_mem_cmp() - memory compare
* @memory1: pointer to one location in memory to compare.
* @memory2: pointer to second location in memory to compare.
* @num_bytes: the number of bytes to compare.
*
* Function to compare two pieces of memory, similar to memcmp function
* in standard C.
* Return:
* int32_t - returns an int value that tells if the memory
* locations are equal or not equal.
* 0 -- equal
* < 0 -- *memory1 is less than *memory2
* > 0 -- *memory1 is bigger than *memory2
*/
static inline int32_t __qdf_mem_cmp(const void *memory1, const void *memory2,
uint32_t num_bytes)
{
return (int32_t) memcmp(memory1, memory2, num_bytes);
}
/**
* __qdf_mem_smmu_s1_enabled() - Return SMMU stage 1 translation enable status
* @osdev parent device instance
*
* Return: true if smmu s1 enabled, false if smmu s1 is bypassed
*/
static inline bool __qdf_mem_smmu_s1_enabled(qdf_device_t osdev)
{
return osdev->smmu_s1_enabled;
}
#ifdef CONFIG_ARM_SMMU
/**
* __qdf_mem_paddr_from_dmaaddr() - get actual physical address from dma_addr
* @osdev: parent device instance
* @dma_addr: dma_addr
*
* Get actual physical address from dma_addr based on SMMU enablement status.
* IF SMMU Stage 1 tranlation is enabled, DMA APIs return IO virtual address
* (IOVA) otherwise returns physical address. So get SMMU physical address
* mapping from IOVA.
*
* Return: dmaable physical address
*/
static inline unsigned long
__qdf_mem_paddr_from_dmaaddr(qdf_device_t osdev,
qdf_dma_addr_t dma_addr)
{
struct dma_iommu_mapping *mapping;
if (__qdf_mem_smmu_s1_enabled(osdev)) {
mapping = osdev->iommu_mapping;
if (mapping)
return iommu_iova_to_phys(mapping->domain, dma_addr);
}
return dma_addr;
}
#else
static inline unsigned long
__qdf_mem_paddr_from_dmaaddr(qdf_device_t osdev,
qdf_dma_addr_t dma_addr)
{
return dma_addr;
}
#endif
/**
* __qdf_os_mem_dma_get_sgtable() - Returns DMA memory scatter gather table
* @dev: device instace
* @sgt: scatter gather table pointer
* @cpu_addr: HLOS virtual address
* @dma_addr: dma/iova
* @size: allocated memory size
*
* Return: physical address
*/
static inline int
__qdf_os_mem_dma_get_sgtable(struct device *dev, void *sgt, void *cpu_addr,
qdf_dma_addr_t dma_addr, size_t size)
{
return dma_get_sgtable(dev, (struct sg_table *)sgt, cpu_addr, dma_addr,
size);
}
/**
* __qdf_mem_get_dma_addr() - Return dma addr based on SMMU translation status
* @osdev: parent device instance
* @mem_info: Pointer to allocated memory information
*
* Based on smmu stage 1 translation enablement status, return corresponding dma
* address from qdf_mem_info_t. If stage 1 translation enabled, return
* IO virtual address otherwise return physical address.
*
* Return: dma address
*/
static inline qdf_dma_addr_t __qdf_mem_get_dma_addr(qdf_device_t osdev,
qdf_mem_info_t *mem_info)
{
if (__qdf_mem_smmu_s1_enabled(osdev))
return (qdf_dma_addr_t)mem_info->iova;
else
return (qdf_dma_addr_t)mem_info->pa;
}
/**
* __qdf_mem_get_dma_addr_ptr() - Return DMA address storage pointer
* @osdev: parent device instance
* @mem_info: Pointer to allocated memory information
*
* Based on smmu stage 1 translation enablement status, return corresponding
* dma address pointer from qdf_mem_info_t structure. If stage 1 translation
* enabled, return pointer to IO virtual address otherwise return pointer to
* physical address
*
* Return: dma address storage pointer
*/
static inline qdf_dma_addr_t *
__qdf_mem_get_dma_addr_ptr(qdf_device_t osdev,
qdf_mem_info_t *mem_info)
{
if (__qdf_mem_smmu_s1_enabled(osdev))
return (qdf_dma_addr_t *)(&mem_info->iova);
else
return (qdf_dma_addr_t *)(&mem_info->pa);
}
/**
* __qdf_update_mem_map_table() - Update DMA memory map info
* @osdev: Parent device instance
* @mem_info: Pointer to shared memory information
* @dma_addr: dma address
* @mem_size: memory size allocated
*
* Store DMA shared memory information
*
* Return: none
*/
static inline void __qdf_update_mem_map_table(qdf_device_t osdev,
qdf_mem_info_t *mem_info,
qdf_dma_addr_t dma_addr,
uint32_t mem_size)
{
mem_info->pa = __qdf_mem_paddr_from_dmaaddr(osdev, dma_addr);
mem_info->iova = dma_addr;
mem_info->size = mem_size;
}
/**
* __qdf_mem_get_dma_size() - Return DMA memory size
* @osdev: parent device instance
* @mem_info: Pointer to allocated memory information
*
* Return: DMA memory size
*/
static inline uint32_t
__qdf_mem_get_dma_size(qdf_device_t osdev,
qdf_mem_info_t *mem_info)
{
return mem_info->size;
}
/**
* __qdf_mem_set_dma_size() - Set DMA memory size
* @osdev: parent device instance
* @mem_info: Pointer to allocated memory information
* @mem_size: memory size allocated
*
* Return: none
*/
static inline void
__qdf_mem_set_dma_size(qdf_device_t osdev,
qdf_mem_info_t *mem_info,
uint32_t mem_size)
{
mem_info->size = mem_size;
}
/**
* __qdf_mem_get_dma_size() - Return DMA physical address
* @osdev: parent device instance
* @mem_info: Pointer to allocated memory information
*
* Return: DMA physical address
*/
static inline qdf_dma_addr_t
__qdf_mem_get_dma_pa(qdf_device_t osdev,
qdf_mem_info_t *mem_info)
{
return mem_info->pa;
}
/**
* __qdf_mem_set_dma_size() - Set DMA physical address
* @osdev: parent device instance
* @mem_info: Pointer to allocated memory information
* @dma_pa: DMA phsical address
*
* Return: none
*/
static inline void
__qdf_mem_set_dma_pa(qdf_device_t osdev,
qdf_mem_info_t *mem_info,
qdf_dma_addr_t dma_pa)
{
mem_info->pa = dma_pa;
}
#endif /* __I_QDF_MEM_H */