blob: 921b80c49a3b3feaadd4e36be16850c838c551ea [file] [log] [blame]
/*
* Copyright (c) 2012-2013, The Linux Foundation. All rights reserved.
*
* This program is free software; you can redistribute it and/or modify
* it under the terms of the GNU General Public License version 2 and
* only version 2 as published by the Free Software Foundation.
*
* This program is distributed in the hope that it will be useful,
* but WITHOUT ANY WARRANTY; without even the implied warranty of
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
* GNU General Public License for more details.
*
*/
#include "adsprpc_shared.h"
#include <linux/slab.h>
#include <linux/completion.h>
#include <linux/pagemap.h>
#include <linux/mm.h>
#include <linux/fs.h>
#include <linux/sched.h>
#include <linux/module.h>
#include <linux/cdev.h>
#include <linux/list.h>
#include <linux/hash.h>
#include <linux/msm_ion.h>
#include <mach/msm_smd.h>
#include <mach/ion.h>
#include <mach/iommu_domains.h>
#include <linux/scatterlist.h>
#include <linux/fs.h>
#include <linux/uaccess.h>
#include <linux/device.h>
#include <linux/of.h>
#include <linux/iommu.h>
#include <linux/kref.h>
#ifndef ION_ADSPRPC_HEAP_ID
#define ION_ADSPRPC_HEAP_ID ION_AUDIO_HEAP_ID
#endif /*ION_ADSPRPC_HEAP_ID*/
#define RPC_TIMEOUT (5 * HZ)
#define RPC_HASH_BITS 5
#define RPC_HASH_SZ (1 << RPC_HASH_BITS)
#define BALIGN 32
#define LOCK_MMAP(kernel)\
do {\
if (!kernel)\
down_read(&current->mm->mmap_sem);\
} while (0)
#define UNLOCK_MMAP(kernel)\
do {\
if (!kernel)\
up_read(&current->mm->mmap_sem);\
} while (0)
#define IS_CACHE_ALIGNED(x) (((x) & ((L1_CACHE_BYTES)-1)) == 0)
static inline uint32_t buf_page_start(void *buf)
{
uint32_t start = (uint32_t) buf & PAGE_MASK;
return start;
}
static inline uint32_t buf_page_offset(void *buf)
{
uint32_t offset = (uint32_t) buf & (PAGE_SIZE - 1);
return offset;
}
static inline int buf_num_pages(void *buf, int len)
{
uint32_t start = buf_page_start(buf) >> PAGE_SHIFT;
uint32_t end = (((uint32_t) buf + len - 1) & PAGE_MASK) >> PAGE_SHIFT;
int nPages = end - start + 1;
return nPages;
}
static inline uint32_t buf_page_size(uint32_t size)
{
uint32_t sz = (size + (PAGE_SIZE - 1)) & PAGE_MASK;
return sz > PAGE_SIZE ? sz : PAGE_SIZE;
}
static inline int buf_get_pages(void *addr, int sz, int nr_pages, int access,
struct smq_phy_page *pages, int nr_elems)
{
struct vm_area_struct *vma, *vmaend;
uint32_t start = buf_page_start(addr);
uint32_t end = buf_page_start((void *)((uint32_t)addr + sz - 1));
uint32_t len = nr_pages << PAGE_SHIFT;
unsigned long pfn, pfnend;
int n = -1, err = 0;
VERIFY(err, 0 != access_ok(access ? VERIFY_WRITE : VERIFY_READ,
(void __user *)start, len));
if (err)
goto bail;
VERIFY(err, 0 != (vma = find_vma(current->mm, start)));
if (err)
goto bail;
VERIFY(err, 0 != (vmaend = find_vma(current->mm, end)));
if (err)
goto bail;
n = 0;
VERIFY(err, 0 == follow_pfn(vma, start, &pfn));
if (err)
goto bail;
VERIFY(err, 0 == follow_pfn(vmaend, end, &pfnend));
if (err)
goto bail;
VERIFY(err, (pfn + nr_pages - 1) == pfnend);
if (err)
goto bail;
VERIFY(err, nr_elems > 0);
if (err)
goto bail;
pages->addr = __pfn_to_phys(pfn);
pages->size = len;
n++;
bail:
return n;
}
struct smq_invoke_ctx {
struct completion work;
int retval;
atomic_t free;
};
struct smq_context_list {
struct smq_invoke_ctx *ls;
int size;
int last;
};
struct fastrpc_smmu {
struct iommu_group *group;
struct iommu_domain *domain;
int domain_id;
bool enabled;
};
struct fastrpc_apps {
smd_channel_t *chan;
struct smq_context_list clst;
struct completion work;
struct ion_client *iclient;
struct cdev cdev;
struct class *class;
struct device *dev;
struct fastrpc_smmu smmu;
struct mutex smd_mutex;
dev_t dev_no;
spinlock_t wrlock;
spinlock_t hlock;
struct kref kref;
struct hlist_head htbl[RPC_HASH_SZ];
};
struct fastrpc_mmap {
struct hlist_node hn;
struct ion_handle *handle;
void *virt;
uint32_t vaddrin;
uint32_t vaddrout;
int size;
};
struct fastrpc_buf {
struct ion_handle *handle;
void *virt;
ion_phys_addr_t phys;
int size;
int used;
};
struct file_data {
spinlock_t hlock;
struct hlist_head hlst;
};
struct fastrpc_device {
uint32_t tgid;
struct hlist_node hn;
struct fastrpc_buf buf;
};
static struct fastrpc_apps gfa;
static void free_mem(struct fastrpc_buf *buf)
{
struct fastrpc_apps *me = &gfa;
if (!IS_ERR_OR_NULL(buf->handle)) {
if (me->smmu.enabled && buf->phys) {
ion_unmap_iommu(me->iclient, buf->handle,
me->smmu.domain_id, 0);
buf->phys = 0;
}
if (buf->virt) {
ion_unmap_kernel(me->iclient, buf->handle);
buf->virt = 0;
}
ion_free(me->iclient, buf->handle);
buf->handle = 0;
}
}
static void free_map(struct fastrpc_mmap *map)
{
struct fastrpc_apps *me = &gfa;
if (!IS_ERR_OR_NULL(map->handle)) {
if (map->virt) {
ion_unmap_kernel(me->iclient, map->handle);
map->virt = 0;
}
ion_free(me->iclient, map->handle);
}
map->handle = 0;
}
static int alloc_mem(struct fastrpc_buf *buf)
{
struct fastrpc_apps *me = &gfa;
struct ion_client *clnt = gfa.iclient;
struct sg_table *sg;
int err = 0;
unsigned int heap;
unsigned long len;
buf->handle = 0;
buf->virt = 0;
heap = me->smmu.enabled ? ION_HEAP(ION_IOMMU_HEAP_ID) :
ION_HEAP(ION_ADSP_HEAP_ID) | ION_HEAP(ION_AUDIO_HEAP_ID);
buf->handle = ion_alloc(clnt, buf->size, SZ_4K, heap, ION_FLAG_CACHED);
VERIFY(err, 0 == IS_ERR_OR_NULL(buf->handle));
if (err)
goto bail;
VERIFY(err, 0 != (buf->virt = ion_map_kernel(clnt, buf->handle)));
if (err)
goto bail;
if (me->smmu.enabled) {
len = buf->size;
VERIFY(err, 0 == ion_map_iommu(clnt, buf->handle,
me->smmu.domain_id, 0, SZ_4K, 0,
&buf->phys, &len, 0, 0));
if (err)
goto bail;
} else {
VERIFY(err, 0 != (sg = ion_sg_table(clnt, buf->handle)));
if (err)
goto bail;
VERIFY(err, 1 == sg->nents);
if (err)
goto bail;
buf->phys = sg_dma_address(sg->sgl);
}
bail:
if (err && !IS_ERR_OR_NULL(buf->handle))
free_mem(buf);
return err;
}
static int context_list_ctor(struct smq_context_list *me, int size)
{
int err = 0;
VERIFY(err, 0 != (me->ls = kzalloc(size, GFP_KERNEL)));
if (err)
goto bail;
me->size = size / sizeof(*me->ls);
me->last = 0;
bail:
return err;
}
static void context_list_dtor(struct smq_context_list *me)
{
kfree(me->ls);
}
static void context_list_alloc_ctx(struct smq_context_list *me,
struct smq_invoke_ctx **po)
{
int i = me->last;
struct smq_invoke_ctx *ctx;
for (;;) {
i = i % me->size;
ctx = &me->ls[i];
if (atomic_read(&ctx->free) == 0)
if (atomic_cmpxchg(&ctx->free, 0, 1) == 0)
break;
i++;
}
me->last = i;
ctx->retval = -1;
init_completion(&ctx->work);
*po = ctx;
}
static void context_free(struct smq_invoke_ctx *me)
{
if (me)
atomic_set(&me->free, 0);
}
static void context_notify_user(struct smq_invoke_ctx *me, int retval)
{
me->retval = retval;
complete(&me->work);
}
static void context_notify_all_users(struct smq_context_list *me)
{
int i;
if (!me->ls)
return;
for (i = 0; i < me->size; ++i) {
if (atomic_read(&me->ls[i].free) != 0)
complete(&me->ls[i].work);
}
}
static int get_page_list(uint32_t kernel, uint32_t sc, remote_arg_t *pra,
struct fastrpc_buf *ibuf, struct fastrpc_buf *obuf)
{
struct smq_phy_page *pgstart, *pages;
struct smq_invoke_buf *list;
int i, rlen, err = 0;
int inbufs = REMOTE_SCALARS_INBUFS(sc);
int outbufs = REMOTE_SCALARS_OUTBUFS(sc);
LOCK_MMAP(kernel);
*obuf = *ibuf;
retry:
list = smq_invoke_buf_start((remote_arg_t *)obuf->virt, sc);
pgstart = smq_phy_page_start(sc, list);
pages = pgstart + 1;
rlen = obuf->size - ((uint32_t)pages - (uint32_t)obuf->virt);
if (rlen < 0) {
rlen = ((uint32_t)pages - (uint32_t)obuf->virt) - obuf->size;
obuf->size += buf_page_size(rlen);
VERIFY(err, 0 == alloc_mem(obuf));
if (err)
goto bail;
goto retry;
}
pgstart->addr = obuf->phys;
pgstart->size = obuf->size;
for (i = 0; i < inbufs + outbufs; ++i) {
void *buf;
int len, num;
list[i].num = 0;
list[i].pgidx = 0;
len = pra[i].buf.len;
if (!len)
continue;
buf = pra[i].buf.pv;
num = buf_num_pages(buf, len);
if (!kernel)
list[i].num = buf_get_pages(buf, len, num,
i >= inbufs, pages, rlen / sizeof(*pages));
else
list[i].num = 0;
VERIFY(err, list[i].num >= 0);
if (err)
goto bail;
if (list[i].num) {
list[i].pgidx = pages - pgstart;
pages = pages + list[i].num;
} else if (rlen > sizeof(*pages)) {
list[i].pgidx = pages - pgstart;
pages = pages + 1;
} else {
if (obuf->handle != ibuf->handle)
free_mem(obuf);
obuf->size += buf_page_size(sizeof(*pages));
VERIFY(err, 0 == alloc_mem(obuf));
if (err)
goto bail;
goto retry;
}
rlen = obuf->size - ((uint32_t) pages - (uint32_t) obuf->virt);
}
obuf->used = obuf->size - rlen;
bail:
if (err && (obuf->handle != ibuf->handle))
free_mem(obuf);
UNLOCK_MMAP(kernel);
return err;
}
static int get_args(uint32_t kernel, uint32_t sc, remote_arg_t *pra,
remote_arg_t *rpra, remote_arg_t *upra,
struct fastrpc_buf *ibuf, struct fastrpc_buf **abufs,
int *nbufs, int *fds)
{
struct fastrpc_apps *me = &gfa;
struct smq_invoke_buf *list;
struct fastrpc_buf *pbuf = ibuf, *obufs = 0;
struct smq_phy_page *pages;
struct ion_handle **handles = NULL;
void *args;
int i, rlen, size, used, inh, bufs = 0, err = 0;
int inbufs = REMOTE_SCALARS_INBUFS(sc);
int outbufs = REMOTE_SCALARS_OUTBUFS(sc);
unsigned long iova, len;
list = smq_invoke_buf_start(rpra, sc);
pages = smq_phy_page_start(sc, list);
used = ALIGN(pbuf->used, BALIGN);
args = (void *)((char *)pbuf->virt + used);
rlen = pbuf->size - used;
if (fds)
handles = (struct ion_handle **)(fds + inbufs + outbufs);
for (i = 0; i < inbufs + outbufs; ++i) {
rpra[i].buf.len = pra[i].buf.len;
if (!rpra[i].buf.len)
continue;
if (list[i].num) {
rpra[i].buf.pv = pra[i].buf.pv;
continue;
} else if (me->smmu.enabled && fds && (fds[i] >= 0)) {
len = buf_page_size(pra[i].buf.len);
handles[i] = ion_import_dma_buf(me->iclient, fds[i]);
VERIFY(err, 0 == IS_ERR_OR_NULL(handles[i]));
if (err)
goto bail;
VERIFY(err, 0 == ion_map_iommu(me->iclient, handles[i],
me->smmu.domain_id, 0, SZ_4K, 0,
&iova, &len, 0, 0));
if (err)
goto bail;
rpra[i].buf.pv = pra[i].buf.pv;
list[i].num = 1;
pages[list[i].pgidx].addr = iova;
pages[list[i].pgidx].size = len;
continue;
}
if (rlen < pra[i].buf.len) {
struct fastrpc_buf *b;
pbuf->used = pbuf->size - rlen;
VERIFY(err, 0 != (b = krealloc(obufs,
(bufs + 1) * sizeof(*obufs), GFP_KERNEL)));
if (err)
goto bail;
obufs = b;
pbuf = obufs + bufs;
pbuf->size = buf_num_pages(0, pra[i].buf.len) *
PAGE_SIZE;
VERIFY(err, 0 == alloc_mem(pbuf));
if (err)
goto bail;
bufs++;
args = pbuf->virt;
rlen = pbuf->size;
}
list[i].num = 1;
pages[list[i].pgidx].addr =
buf_page_start((void *)(pbuf->phys +
(pbuf->size - rlen)));
pages[list[i].pgidx].size =
buf_page_size(pra[i].buf.len);
if (i < inbufs) {
if (!kernel) {
VERIFY(err, 0 == copy_from_user(args,
pra[i].buf.pv, pra[i].buf.len));
if (err)
goto bail;
} else {
memmove(args, pra[i].buf.pv, pra[i].buf.len);
}
}
rpra[i].buf.pv = args;
args = (void *)((char *)args + ALIGN(pra[i].buf.len, BALIGN));
rlen -= ALIGN(pra[i].buf.len, BALIGN);
}
for (i = 0; i < inbufs; ++i) {
if (rpra[i].buf.len)
dmac_flush_range(rpra[i].buf.pv,
(char *)rpra[i].buf.pv + rpra[i].buf.len);
}
pbuf->used = pbuf->size - rlen;
size = sizeof(*rpra) * REMOTE_SCALARS_INHANDLES(sc);
if (size) {
inh = inbufs + outbufs;
if (!kernel) {
VERIFY(err, 0 == copy_from_user(&rpra[inh], &upra[inh],
size));
if (err)
goto bail;
} else {
memmove(&rpra[inh], &upra[inh], size);
}
}
dmac_flush_range(rpra, (char *)rpra + used);
bail:
*abufs = obufs;
*nbufs = bufs;
return err;
}
static int put_args(uint32_t kernel, uint32_t sc, remote_arg_t *pra,
remote_arg_t *rpra, remote_arg_t *upra)
{
int i, inbufs, outbufs, outh, size;
int err = 0;
inbufs = REMOTE_SCALARS_INBUFS(sc);
outbufs = REMOTE_SCALARS_OUTBUFS(sc);
for (i = inbufs; i < inbufs + outbufs; ++i) {
if (rpra[i].buf.pv != pra[i].buf.pv) {
if (!kernel) {
VERIFY(err, 0 == copy_to_user(pra[i].buf.pv,
rpra[i].buf.pv, rpra[i].buf.len));
if (err)
goto bail;
} else {
memmove(pra[i].buf.pv, rpra[i].buf.pv,
rpra[i].buf.len);
}
}
}
size = sizeof(*rpra) * REMOTE_SCALARS_OUTHANDLES(sc);
if (size) {
outh = inbufs + outbufs + REMOTE_SCALARS_INHANDLES(sc);
if (!kernel) {
VERIFY(err, 0 == copy_to_user(&upra[outh], &rpra[outh],
size));
if (err)
goto bail;
} else {
memmove(&upra[outh], &rpra[outh], size);
}
}
bail:
return err;
}
static void inv_args_pre(uint32_t sc, remote_arg_t *rpra)
{
int i, inbufs, outbufs;
uint32_t end;
inbufs = REMOTE_SCALARS_INBUFS(sc);
outbufs = REMOTE_SCALARS_OUTBUFS(sc);
for (i = inbufs; i < inbufs + outbufs; ++i) {
if (!rpra[i].buf.len)
continue;
if (buf_page_start(rpra) == buf_page_start(rpra[i].buf.pv))
continue;
if (!IS_CACHE_ALIGNED((uint32_t)rpra[i].buf.pv))
dmac_flush_range(rpra[i].buf.pv,
(char *)rpra[i].buf.pv + 1);
end = (uint32_t)rpra[i].buf.pv + rpra[i].buf.len;
if (!IS_CACHE_ALIGNED(end))
dmac_flush_range((char *)end,
(char *)end + 1);
}
}
static void inv_args(uint32_t sc, remote_arg_t *rpra, int used)
{
int i, inbufs, outbufs;
int inv = 0;
inbufs = REMOTE_SCALARS_INBUFS(sc);
outbufs = REMOTE_SCALARS_OUTBUFS(sc);
for (i = inbufs; i < inbufs + outbufs; ++i) {
if (buf_page_start(rpra) == buf_page_start(rpra[i].buf.pv))
inv = 1;
else if (rpra[i].buf.len)
dmac_inv_range(rpra[i].buf.pv,
(char *)rpra[i].buf.pv + rpra[i].buf.len);
}
if (inv || REMOTE_SCALARS_OUTHANDLES(sc))
dmac_inv_range(rpra, (char *)rpra + used);
}
static int fastrpc_invoke_send(struct fastrpc_apps *me,
uint32_t kernel, uint32_t handle,
uint32_t sc, struct smq_invoke_ctx *ctx,
struct fastrpc_buf *buf)
{
struct smq_msg msg;
int err = 0, len;
msg.pid = current->tgid;
msg.tid = current->pid;
if (kernel)
msg.pid = 0;
msg.invoke.header.ctx = ctx;
msg.invoke.header.handle = handle;
msg.invoke.header.sc = sc;
msg.invoke.page.addr = buf->phys;
msg.invoke.page.size = buf_page_size(buf->used);
spin_lock(&me->wrlock);
len = smd_write(me->chan, &msg, sizeof(msg));
spin_unlock(&me->wrlock);
VERIFY(err, len == sizeof(msg));
return err;
}
static void fastrpc_deinit(void)
{
struct fastrpc_apps *me = &gfa;
smd_close(me->chan);
context_list_dtor(&me->clst);
ion_client_destroy(me->iclient);
me->iclient = 0;
me->chan = 0;
}
static void fastrpc_read_handler(void)
{
struct fastrpc_apps *me = &gfa;
struct smq_invoke_rsp rsp;
int err = 0;
do {
VERIFY(err, sizeof(rsp) ==
smd_read_from_cb(me->chan, &rsp, sizeof(rsp)));
if (err)
goto bail;
context_notify_user(rsp.ctx, rsp.retval);
} while (!err);
bail:
return;
}
static void smd_event_handler(void *priv, unsigned event)
{
struct fastrpc_apps *me = (struct fastrpc_apps *)priv;
switch (event) {
case SMD_EVENT_OPEN:
complete(&(me->work));
break;
case SMD_EVENT_CLOSE:
context_notify_all_users(&me->clst);
break;
case SMD_EVENT_DATA:
fastrpc_read_handler();
break;
}
}
static int fastrpc_init(void)
{
int err = 0;
struct fastrpc_apps *me = &gfa;
struct device_node *node;
bool enabled = 0;
if (me->chan == 0) {
int i;
spin_lock_init(&me->hlock);
spin_lock_init(&me->wrlock);
init_completion(&me->work);
mutex_init(&me->smd_mutex);
for (i = 0; i < RPC_HASH_SZ; ++i)
INIT_HLIST_HEAD(&me->htbl[i]);
VERIFY(err, 0 == context_list_ctor(&me->clst, SZ_4K));
if (err)
goto context_list_bail;
me->iclient = msm_ion_client_create(ION_HEAP_CARVEOUT_MASK,
DEVICE_NAME);
VERIFY(err, 0 == IS_ERR_OR_NULL(me->iclient));
if (err)
goto ion_bail;
node = of_find_compatible_node(NULL, NULL,
"qcom,msm-audio-ion");
if (node)
enabled = of_property_read_bool(node,
"qcom,smmu-enabled");
if (enabled)
me->smmu.group = iommu_group_find("lpass_audio");
if (me->smmu.group)
me->smmu.domain = iommu_group_get_iommudata(
me->smmu.group);
if (!IS_ERR_OR_NULL(me->smmu.domain)) {
me->smmu.domain_id = msm_find_domain_no(
me->smmu.domain);
if (me->smmu.domain_id >= 0)
me->smmu.enabled = enabled;
}
}
return 0;
ion_bail:
context_list_dtor(&me->clst);
context_list_bail:
return err;
}
static void free_dev(struct fastrpc_device *dev)
{
if (dev) {
free_mem(&dev->buf);
kfree(dev);
module_put(THIS_MODULE);
}
}
static int alloc_dev(struct fastrpc_device **dev)
{
int err = 0;
struct fastrpc_device *fd = 0;
VERIFY(err, 0 != try_module_get(THIS_MODULE));
if (err)
goto bail;
VERIFY(err, 0 != (fd = kzalloc(sizeof(*fd), GFP_KERNEL)));
if (err)
goto bail;
INIT_HLIST_NODE(&fd->hn);
fd->buf.size = PAGE_SIZE;
VERIFY(err, 0 == alloc_mem(&fd->buf));
if (err)
goto bail;
fd->tgid = current->tgid;
*dev = fd;
bail:
if (err)
free_dev(fd);
return err;
}
static int get_dev(struct fastrpc_apps *me, struct fastrpc_device **rdev)
{
struct hlist_head *head;
struct fastrpc_device *dev = 0, *devfree = 0;
struct hlist_node *pos, *n;
uint32_t h = hash_32(current->tgid, RPC_HASH_BITS);
int err = 0;
spin_lock(&me->hlock);
head = &me->htbl[h];
hlist_for_each_entry_safe(dev, pos, n, head, hn) {
if (dev->tgid == current->tgid) {
hlist_del(&dev->hn);
devfree = dev;
break;
}
}
spin_unlock(&me->hlock);
VERIFY(err, devfree != 0);
if (err)
goto bail;
*rdev = devfree;
bail:
if (err) {
free_dev(devfree);
err = alloc_dev(rdev);
}
return err;
}
static void add_dev(struct fastrpc_apps *me, struct fastrpc_device *dev)
{
struct hlist_head *head;
uint32_t h = hash_32(current->tgid, RPC_HASH_BITS);
spin_lock(&me->hlock);
head = &me->htbl[h];
hlist_add_head(&dev->hn, head);
spin_unlock(&me->hlock);
return;
}
static int fastrpc_release_current_dsp_process(void);
static int fastrpc_internal_invoke(struct fastrpc_apps *me, uint32_t kernel,
struct fastrpc_ioctl_invoke *invoke, remote_arg_t *pra,
int *fds)
{
remote_arg_t *rpra = 0;
struct fastrpc_device *dev = 0;
struct smq_invoke_ctx *ctx = 0;
struct fastrpc_buf obuf, *abufs = 0, *b;
struct ion_handle **handles = NULL;
int interrupted = 0;
uint32_t sc;
int i, bufs, nbufs = 0, err = 0;
sc = invoke->sc;
obuf.handle = 0;
if (me->smmu.enabled) {
VERIFY(err, 0 == iommu_attach_group(me->smmu.domain,
me->smmu.group));
if (err)
return err;
}
if (REMOTE_SCALARS_LENGTH(sc)) {
VERIFY(err, 0 == get_dev(me, &dev));
if (err)
goto bail;
VERIFY(err, 0 == get_page_list(kernel, sc, pra, &dev->buf,
&obuf));
if (err)
goto bail;
rpra = (remote_arg_t *)obuf.virt;
VERIFY(err, 0 == get_args(kernel, sc, pra, rpra, invoke->pra,
&obuf, &abufs, &nbufs, fds));
if (err)
goto bail;
}
context_list_alloc_ctx(&me->clst, &ctx);
inv_args_pre(sc, rpra);
VERIFY(err, 0 == fastrpc_invoke_send(me, kernel, invoke->handle, sc,
ctx, &obuf));
if (err)
goto bail;
inv_args(sc, rpra, obuf.used);
VERIFY(err, 0 == (interrupted =
wait_for_completion_interruptible(&ctx->work)));
if (err)
goto bail;
VERIFY(err, 0 == (err = ctx->retval));
if (err)
goto bail;
VERIFY(err, 0 == put_args(kernel, sc, pra, rpra, invoke->pra));
if (err)
goto bail;
bail:
if (interrupted) {
if (!kernel)
(void)fastrpc_release_current_dsp_process();
wait_for_completion(&ctx->work);
}
context_free(ctx);
if (me->smmu.enabled) {
bufs = REMOTE_SCALARS_LENGTH(sc);
if (fds) {
handles = (struct ion_handle **)(fds + bufs);
for (i = 0; i < bufs; i++)
if (!IS_ERR_OR_NULL(handles[i])) {
ion_unmap_iommu(me->iclient, handles[i],
me->smmu.domain_id, 0);
ion_free(me->iclient, handles[i]);
}
}
iommu_detach_group(me->smmu.domain, me->smmu.group);
}
for (i = 0, b = abufs; i < nbufs; ++i, ++b)
free_mem(b);
kfree(abufs);
if (dev) {
add_dev(me, dev);
if (obuf.handle != dev->buf.handle)
free_mem(&obuf);
}
return err;
}
static int fastrpc_create_current_dsp_process(void)
{
int err = 0;
struct fastrpc_ioctl_invoke ioctl;
struct fastrpc_apps *me = &gfa;
remote_arg_t ra[1];
int tgid = 0;
tgid = current->tgid;
ra[0].buf.pv = &tgid;
ra[0].buf.len = sizeof(tgid);
ioctl.handle = 1;
ioctl.sc = REMOTE_SCALARS_MAKE(0, 1, 0);
ioctl.pra = ra;
VERIFY(err, 0 == (err = fastrpc_internal_invoke(me, 1, &ioctl, ra, 0)));
return err;
}
static int fastrpc_release_current_dsp_process(void)
{
int err = 0;
struct fastrpc_apps *me = &gfa;
struct fastrpc_ioctl_invoke ioctl;
remote_arg_t ra[1];
int tgid = 0;
tgid = current->tgid;
ra[0].buf.pv = &tgid;
ra[0].buf.len = sizeof(tgid);
ioctl.handle = 1;
ioctl.sc = REMOTE_SCALARS_MAKE(1, 1, 0);
ioctl.pra = ra;
VERIFY(err, 0 == (err = fastrpc_internal_invoke(me, 1, &ioctl, ra, 0)));
return err;
}
static int fastrpc_mmap_on_dsp(struct fastrpc_apps *me,
struct fastrpc_ioctl_mmap *mmap,
struct smq_phy_page *pages,
int num)
{
struct fastrpc_ioctl_invoke ioctl;
remote_arg_t ra[3];
int err = 0;
struct {
int pid;
uint32_t flags;
uint32_t vaddrin;
int num;
} inargs;
struct {
uint32_t vaddrout;
} routargs;
inargs.pid = current->tgid;
inargs.vaddrin = mmap->vaddrin;
inargs.flags = mmap->flags;
inargs.num = num;
ra[0].buf.pv = &inargs;
ra[0].buf.len = sizeof(inargs);
ra[1].buf.pv = pages;
ra[1].buf.len = num * sizeof(*pages);
ra[2].buf.pv = &routargs;
ra[2].buf.len = sizeof(routargs);
ioctl.handle = 1;
ioctl.sc = REMOTE_SCALARS_MAKE(2, 2, 1);
ioctl.pra = ra;
VERIFY(err, 0 == (err = fastrpc_internal_invoke(me, 1, &ioctl, ra, 0)));
mmap->vaddrout = routargs.vaddrout;
if (err)
goto bail;
bail:
return err;
}
static int fastrpc_munmap_on_dsp(struct fastrpc_apps *me,
struct fastrpc_ioctl_munmap *munmap)
{
struct fastrpc_ioctl_invoke ioctl;
remote_arg_t ra[1];
int err = 0;
struct {
int pid;
uint32_t vaddrout;
int size;
} inargs;
inargs.pid = current->tgid;
inargs.size = munmap->size;
inargs.vaddrout = munmap->vaddrout;
ra[0].buf.pv = &inargs;
ra[0].buf.len = sizeof(inargs);
ioctl.handle = 1;
ioctl.sc = REMOTE_SCALARS_MAKE(3, 1, 0);
ioctl.pra = ra;
VERIFY(err, 0 == (err = fastrpc_internal_invoke(me, 1, &ioctl, ra, 0)));
return err;
}
static int fastrpc_internal_munmap(struct fastrpc_apps *me,
struct file_data *fdata,
struct fastrpc_ioctl_munmap *munmap)
{
int err = 0;
struct fastrpc_mmap *map = 0, *mapfree = 0;
struct hlist_node *pos, *n;
VERIFY(err, 0 == (err = fastrpc_munmap_on_dsp(me, munmap)));
if (err)
goto bail;
spin_lock(&fdata->hlock);
hlist_for_each_entry_safe(map, pos, n, &fdata->hlst, hn) {
if (map->vaddrout == munmap->vaddrout &&
map->size == munmap->size) {
hlist_del(&map->hn);
mapfree = map;
map = 0;
break;
}
}
spin_unlock(&fdata->hlock);
bail:
if (mapfree) {
free_map(mapfree);
kfree(mapfree);
}
return err;
}
static int fastrpc_internal_mmap(struct fastrpc_apps *me,
struct file_data *fdata,
struct fastrpc_ioctl_mmap *mmap)
{
struct ion_client *clnt = gfa.iclient;
struct fastrpc_mmap *map = 0;
struct smq_phy_page *pages = 0;
void *buf;
int len;
int num;
int err = 0;
VERIFY(err, 0 != (map = kzalloc(sizeof(*map), GFP_KERNEL)));
if (err)
goto bail;
map->handle = ion_import_dma_buf(clnt, mmap->fd);
VERIFY(err, 0 == IS_ERR_OR_NULL(map->handle));
if (err)
goto bail;
VERIFY(err, 0 != (map->virt = ion_map_kernel(clnt, map->handle)));
if (err)
goto bail;
buf = (void *)mmap->vaddrin;
len = mmap->size;
num = buf_num_pages(buf, len);
VERIFY(err, 0 != (pages = kzalloc(num * sizeof(*pages), GFP_KERNEL)));
if (err)
goto bail;
VERIFY(err, 0 < (num = buf_get_pages(buf, len, num, 1, pages, num)));
if (err)
goto bail;
VERIFY(err, 0 == fastrpc_mmap_on_dsp(me, mmap, pages, num));
if (err)
goto bail;
map->vaddrin = mmap->vaddrin;
map->vaddrout = mmap->vaddrout;
map->size = mmap->size;
INIT_HLIST_NODE(&map->hn);
spin_lock(&fdata->hlock);
hlist_add_head(&map->hn, &fdata->hlst);
spin_unlock(&fdata->hlock);
bail:
if (err && map) {
free_map(map);
kfree(map);
}
kfree(pages);
return err;
}
static void cleanup_current_dev(void)
{
struct fastrpc_apps *me = &gfa;
uint32_t h = hash_32(current->tgid, RPC_HASH_BITS);
struct hlist_head *head;
struct hlist_node *pos, *n;
struct fastrpc_device *dev, *devfree;
rnext:
devfree = dev = 0;
spin_lock(&me->hlock);
head = &me->htbl[h];
hlist_for_each_entry_safe(dev, pos, n, head, hn) {
if (dev->tgid == current->tgid) {
hlist_del(&dev->hn);
devfree = dev;
break;
}
}
spin_unlock(&me->hlock);
if (devfree) {
free_dev(devfree);
goto rnext;
}
return;
}
static void fastrpc_channel_close(struct kref *kref)
{
struct fastrpc_apps *me = &gfa;
smd_close(me->chan);
me->chan = 0;
mutex_unlock(&me->smd_mutex);
pr_info("'closed /dev/%s c %d 0'\n", DEVICE_NAME,
MAJOR(me->dev_no));
}
static int fastrpc_device_release(struct inode *inode, struct file *file)
{
struct file_data *fdata = (struct file_data *)file->private_data;
struct fastrpc_apps *me = &gfa;
(void)fastrpc_release_current_dsp_process();
cleanup_current_dev();
if (fdata) {
struct fastrpc_mmap *map;
struct hlist_node *n, *pos;
file->private_data = 0;
hlist_for_each_entry_safe(map, pos, n, &fdata->hlst, hn) {
hlist_del(&map->hn);
free_map(map);
kfree(map);
}
kfree(fdata);
kref_put_mutex(&me->kref, fastrpc_channel_close,
&me->smd_mutex);
}
return 0;
}
static int fastrpc_device_open(struct inode *inode, struct file *filp)
{
int err = 0;
struct fastrpc_apps *me = &gfa;
mutex_lock(&me->smd_mutex);
if (kref_get_unless_zero(&me->kref) == 0) {
VERIFY(err, 0 == smd_named_open_on_edge(FASTRPC_SMD_GUID,
SMD_APPS_QDSP, &me->chan,
me, smd_event_handler));
if (err)
goto smd_bail;
VERIFY(err, 0 != wait_for_completion_timeout(&me->work,
RPC_TIMEOUT));
if (err)
goto completion_bail;
kref_init(&me->kref);
pr_info("'opened /dev/%s c %d 0'\n", DEVICE_NAME,
MAJOR(me->dev_no));
}
mutex_unlock(&me->smd_mutex);
filp->private_data = 0;
if (0 != try_module_get(THIS_MODULE)) {
struct file_data *fdata = 0;
/* This call will cause a dev to be created
* which will addref this module
*/
VERIFY(err, 0 != (fdata = kzalloc(sizeof(*fdata), GFP_KERNEL)));
if (err)
goto bail;
spin_lock_init(&fdata->hlock);
INIT_HLIST_HEAD(&fdata->hlst);
VERIFY(err, 0 == fastrpc_create_current_dsp_process());
if (err)
goto bail;
filp->private_data = fdata;
bail:
if (err) {
cleanup_current_dev();
kfree(fdata);
kref_put_mutex(&me->kref, fastrpc_channel_close,
&me->smd_mutex);
}
module_put(THIS_MODULE);
}
return err;
completion_bail:
smd_close(me->chan);
me->chan = 0;
smd_bail:
mutex_unlock(&me->smd_mutex);
return err;
}
static long fastrpc_device_ioctl(struct file *file, unsigned int ioctl_num,
unsigned long ioctl_param)
{
struct fastrpc_apps *me = &gfa;
struct fastrpc_ioctl_invoke_fd invokefd;
struct fastrpc_ioctl_invoke *invoke = &invokefd.inv;
struct fastrpc_ioctl_mmap mmap;
struct fastrpc_ioctl_munmap munmap;
remote_arg_t *pra = 0;
void *param = (char *)ioctl_param;
struct file_data *fdata = (struct file_data *)file->private_data;
int *fds = 0;
int bufs, size = 0, err = 0;
switch (ioctl_num) {
case FASTRPC_IOCTL_INVOKE_FD:
case FASTRPC_IOCTL_INVOKE:
invokefd.fds = 0;
size = (ioctl_num == FASTRPC_IOCTL_INVOKE) ?
sizeof(*invoke) : sizeof(invokefd);
VERIFY(err, 0 == copy_from_user(&invokefd, param, size));
if (err)
goto bail;
bufs = REMOTE_SCALARS_INBUFS(invoke->sc) +
REMOTE_SCALARS_OUTBUFS(invoke->sc);
if (bufs) {
size = bufs * sizeof(*pra);
if (invokefd.fds)
size = size + bufs * sizeof(*fds) +
bufs * sizeof(struct ion_handle *);
VERIFY(err, 0 != (pra = kzalloc(size, GFP_KERNEL)));
if (err)
goto bail;
}
VERIFY(err, 0 == copy_from_user(pra, invoke->pra,
bufs * sizeof(*pra)));
if (err)
goto bail;
if (invokefd.fds) {
fds = (int *)(pra + bufs);
VERIFY(err, 0 == copy_from_user(fds, invokefd.fds,
bufs * sizeof(*fds)));
if (err)
goto bail;
}
VERIFY(err, 0 == (err = fastrpc_internal_invoke(me, 0, invoke,
pra, fds)));
if (err)
goto bail;
break;
case FASTRPC_IOCTL_MMAP:
VERIFY(err, 0 == copy_from_user(&mmap, param,
sizeof(mmap)));
if (err)
goto bail;
VERIFY(err, 0 == (err = fastrpc_internal_mmap(me, fdata,
&mmap)));
if (err)
goto bail;
VERIFY(err, 0 == copy_to_user(param, &mmap, sizeof(mmap)));
if (err)
goto bail;
break;
case FASTRPC_IOCTL_MUNMAP:
VERIFY(err, 0 == copy_from_user(&munmap, param,
sizeof(munmap)));
if (err)
goto bail;
VERIFY(err, 0 == (err = fastrpc_internal_munmap(me, fdata,
&munmap)));
if (err)
goto bail;
break;
default:
err = -ENOTTY;
break;
}
bail:
kfree(pra);
return err;
}
static const struct file_operations fops = {
.open = fastrpc_device_open,
.release = fastrpc_device_release,
.unlocked_ioctl = fastrpc_device_ioctl,
};
static int __init fastrpc_device_init(void)
{
struct fastrpc_apps *me = &gfa;
int err = 0;
memset(me, 0, sizeof(*me));
VERIFY(err, 0 == fastrpc_init());
if (err)
goto fastrpc_bail;
VERIFY(err, 0 == alloc_chrdev_region(&me->dev_no, 0, 1, DEVICE_NAME));
if (err)
goto alloc_chrdev_bail;
cdev_init(&me->cdev, &fops);
me->cdev.owner = THIS_MODULE;
VERIFY(err, 0 == cdev_add(&me->cdev, MKDEV(MAJOR(me->dev_no), 0), 1));
if (err)
goto cdev_init_bail;
me->class = class_create(THIS_MODULE, "fastrpc");
VERIFY(err, !IS_ERR(me->class));
if (err)
goto class_create_bail;
me->dev = device_create(me->class, NULL, MKDEV(MAJOR(me->dev_no), 0),
NULL, DEVICE_NAME);
VERIFY(err, !IS_ERR(me->dev));
if (err)
goto device_create_bail;
return 0;
device_create_bail:
class_destroy(me->class);
class_create_bail:
cdev_del(&me->cdev);
cdev_init_bail:
unregister_chrdev_region(me->dev_no, 1);
alloc_chrdev_bail:
fastrpc_deinit();
fastrpc_bail:
return err;
}
static void __exit fastrpc_device_exit(void)
{
struct fastrpc_apps *me = &gfa;
fastrpc_deinit();
device_destroy(me->class, MKDEV(MAJOR(me->dev_no), 0));
class_destroy(me->class);
cdev_del(&me->cdev);
unregister_chrdev_region(me->dev_no, 1);
}
module_init(fastrpc_device_init);
module_exit(fastrpc_device_exit);
MODULE_LICENSE("GPL v2");