hfi1/user_exp_rcv/clear: implement TID_FREE ioctl
This commit is contained in:
committed by
Balazs Gerofi
parent
159c18b98b
commit
23f178d718
@ -499,39 +499,35 @@ long hfi1_file_ioctl(void *private_data, unsigned int cmd,
|
|||||||
break;
|
break;
|
||||||
|
|
||||||
case HFI1_IOCTL_TID_FREE:
|
case HFI1_IOCTL_TID_FREE:
|
||||||
#if 0
|
kprintf("%s: HFI1_IOCTL_TID_FREE \n", __FUNCTION__);
|
||||||
if (copy_from_user(&tinfo,
|
if (copy_from_user(&tinfo,
|
||||||
(struct hfi11_tid_info __user *)arg,
|
(struct hfi11_tid_info __user *)arg,
|
||||||
sizeof(tinfo)))
|
sizeof(tinfo)))
|
||||||
return -EFAULT;
|
return -EFAULT;
|
||||||
|
|
||||||
ret = hfi1_user_exp_rcv_clear(fp, &tinfo);
|
ret = hfi1_user_exp_rcv_clear(fd, &tinfo);
|
||||||
if (ret)
|
if (ret)
|
||||||
break;
|
break;
|
||||||
addr = arg + offsetof(struct hfi1_tid_info, tidcnt);
|
addr = arg + offsetof(struct hfi1_tid_info, tidcnt);
|
||||||
if (copy_to_user((void __user *)addr, &tinfo.tidcnt,
|
if (copy_to_user((void __user *)addr, &tinfo.tidcnt,
|
||||||
sizeof(tinfo.tidcnt)))
|
sizeof(tinfo.tidcnt)))
|
||||||
ret = -EFAULT;
|
ret = -EFAULT;
|
||||||
#endif
|
|
||||||
kprintf("%s: HFI1_IOCTL_TID_FREE \n", __FUNCTION__);
|
|
||||||
break;
|
break;
|
||||||
|
|
||||||
case HFI1_IOCTL_TID_INVAL_READ:
|
case HFI1_IOCTL_TID_INVAL_READ:
|
||||||
#if 0
|
kprintf("%s: HFI1_IOCTL_TID_INVAL_READ \n", __FUNCTION__);
|
||||||
if (copy_from_user(&tinfo,
|
if (copy_from_user(&tinfo,
|
||||||
(struct hfi11_tid_info __user *)arg,
|
(struct hfi11_tid_info __user *)arg,
|
||||||
sizeof(tinfo)))
|
sizeof(tinfo)))
|
||||||
return -EFAULT;
|
return -EFAULT;
|
||||||
|
|
||||||
ret = hfi1_user_exp_rcv_invalid(fp, &tinfo);
|
ret = hfi1_user_exp_rcv_invalid(fd, &tinfo);
|
||||||
if (ret)
|
if (ret)
|
||||||
break;
|
break;
|
||||||
addr = arg + offsetof(struct hfi1_tid_info, tidcnt);
|
addr = arg + offsetof(struct hfi1_tid_info, tidcnt);
|
||||||
if (copy_to_user((void __user *)addr, &tinfo.tidcnt,
|
if (copy_to_user((void __user *)addr, &tinfo.tidcnt,
|
||||||
sizeof(tinfo.tidcnt)))
|
sizeof(tinfo.tidcnt)))
|
||||||
ret = -EFAULT;
|
ret = -EFAULT;
|
||||||
#endif
|
|
||||||
kprintf("%s: HFI1_IOCTL_TID_INVAL_READ \n", __FUNCTION__);
|
|
||||||
break;
|
break;
|
||||||
|
|
||||||
case HFI1_IOCTL_RECV_CTRL:
|
case HFI1_IOCTL_RECV_CTRL:
|
||||||
|
|||||||
@ -1623,12 +1623,13 @@ struct hfi1_devdata {
|
|||||||
|
|
||||||
#ifdef __HFI1_ORIG__
|
#ifdef __HFI1_ORIG__
|
||||||
|
|
||||||
struct tid_rb_node;
|
|
||||||
struct mmu_rb_node;
|
struct mmu_rb_node;
|
||||||
struct mmu_rb_handler;
|
struct mmu_rb_handler;
|
||||||
|
|
||||||
#endif /* __HFI1_ORIG__ */
|
#endif /* __HFI1_ORIG__ */
|
||||||
|
|
||||||
|
struct tid_rb_node;
|
||||||
|
|
||||||
/* Original size on linux is 96 bytes */
|
/* Original size on linux is 96 bytes */
|
||||||
/* Private data for file operations */
|
/* Private data for file operations */
|
||||||
|
|
||||||
@ -1640,7 +1641,7 @@ struct hfi1_filedata {
|
|||||||
int rec_cpu_num;
|
int rec_cpu_num;
|
||||||
u32 tid_n_pinned;
|
u32 tid_n_pinned;
|
||||||
void *handler; //struct mmu_rb_handler *handler;
|
void *handler; //struct mmu_rb_handler *handler;
|
||||||
void **entry_to_rb; //struct tid_rb_node **entry_to_rb;
|
struct tid_rb_node **entry_to_rb;
|
||||||
spinlock_t tid_lock;
|
spinlock_t tid_lock;
|
||||||
u32 tid_limit;
|
u32 tid_limit;
|
||||||
u32 tid_used;
|
u32 tid_used;
|
||||||
|
|||||||
@ -57,6 +57,14 @@ static int set_rcvarray_entry(struct hfi1_filedata *, unsigned long,
|
|||||||
u32, struct tid_group *,
|
u32, struct tid_group *,
|
||||||
u32);
|
u32);
|
||||||
static int unprogram_rcvarray(struct hfi1_filedata *, u32, struct tid_group **);
|
static int unprogram_rcvarray(struct hfi1_filedata *, u32, struct tid_group **);
|
||||||
|
static void clear_tid_node(struct hfi1_filedata *, struct tid_rb_node *);
|
||||||
|
|
||||||
|
struct tid_rb_node {
|
||||||
|
uintptr_t phys;
|
||||||
|
u32 len;
|
||||||
|
u32 rcventry;
|
||||||
|
struct tid_group *grp;
|
||||||
|
};
|
||||||
|
|
||||||
|
|
||||||
/*
|
/*
|
||||||
@ -144,10 +152,6 @@ int hfi1_user_exp_rcv_setup(struct hfi1_filedata *fd, struct hfi1_tid_info *tinf
|
|||||||
grp = tid_group_pop(&uctxt->tid_used_list);
|
grp = tid_group_pop(&uctxt->tid_used_list);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
grp = list_first_entry(&uctxt->tid_used_list.list,
|
|
||||||
struct tid_group, list);
|
|
||||||
|
|
||||||
ret = program_rcvarray(fd, vaddr, grp, tinfo->length, &tid);
|
ret = program_rcvarray(fd, vaddr, grp, tinfo->length, &tid);
|
||||||
if (ret < 0) {
|
if (ret < 0) {
|
||||||
hfi1_cdbg(TID,
|
hfi1_cdbg(TID,
|
||||||
@ -158,7 +162,6 @@ int hfi1_user_exp_rcv_setup(struct hfi1_filedata *fd, struct hfi1_tid_info *tinf
|
|||||||
ret = -EFAULT;
|
ret = -EFAULT;
|
||||||
}
|
}
|
||||||
|
|
||||||
spin_lock(&fd->tid_lock);
|
|
||||||
if (grp->used == grp->size)
|
if (grp->used == grp->size)
|
||||||
tid_group_add_tail(grp, &uctxt->tid_full_list);
|
tid_group_add_tail(grp, &uctxt->tid_full_list);
|
||||||
else
|
else
|
||||||
@ -192,9 +195,7 @@ unlock:
|
|||||||
|
|
||||||
int hfi1_user_exp_rcv_clear(struct hfi1_filedata *fd, struct hfi1_tid_info *tinfo)
|
int hfi1_user_exp_rcv_clear(struct hfi1_filedata *fd, struct hfi1_tid_info *tinfo)
|
||||||
{
|
{
|
||||||
#if 0
|
|
||||||
int ret = 0;
|
int ret = 0;
|
||||||
struct hfi1_ctxtdata *uctxt = fd->uctxt;
|
|
||||||
u32 *tidinfo;
|
u32 *tidinfo;
|
||||||
unsigned tididx;
|
unsigned tididx;
|
||||||
|
|
||||||
@ -209,7 +210,7 @@ int hfi1_user_exp_rcv_clear(struct hfi1_filedata *fd, struct hfi1_tid_info *tinf
|
|||||||
goto done;
|
goto done;
|
||||||
}
|
}
|
||||||
|
|
||||||
mutex_lock(&uctxt->exp_mutex);
|
spin_lock(&fd->tid_lock);
|
||||||
for (tididx = 0; tididx < tinfo->tidcnt; tididx++) {
|
for (tididx = 0; tididx < tinfo->tidcnt; tididx++) {
|
||||||
ret = unprogram_rcvarray(fd, tidinfo[tididx], NULL);
|
ret = unprogram_rcvarray(fd, tidinfo[tididx], NULL);
|
||||||
if (ret) {
|
if (ret) {
|
||||||
@ -218,16 +219,12 @@ int hfi1_user_exp_rcv_clear(struct hfi1_filedata *fd, struct hfi1_tid_info *tinf
|
|||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
spin_lock(&fd->tid_lock);
|
|
||||||
fd->tid_used -= tididx;
|
fd->tid_used -= tididx;
|
||||||
spin_unlock(&fd->tid_lock);
|
spin_unlock(&fd->tid_lock);
|
||||||
tinfo->tidcnt = tididx;
|
tinfo->tidcnt = tididx;
|
||||||
mutex_unlock(&uctxt->exp_mutex);
|
|
||||||
done:
|
done:
|
||||||
kfree(tidinfo);
|
kfree(tidinfo);
|
||||||
return ret;
|
return ret;
|
||||||
#endif
|
|
||||||
return 0;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
int hfi1_user_exp_rcv_invalid(struct hfi1_filedata *fd, struct hfi1_tid_info *tinfo)
|
int hfi1_user_exp_rcv_invalid(struct hfi1_filedata *fd, struct hfi1_tid_info *tinfo)
|
||||||
@ -353,54 +350,19 @@ static int set_rcvarray_entry(struct hfi1_filedata *fd, unsigned long vaddr,
|
|||||||
struct hfi1_ctxtdata *uctxt = fd->uctxt;
|
struct hfi1_ctxtdata *uctxt = fd->uctxt;
|
||||||
struct hfi1_devdata *dd = uctxt->dd;
|
struct hfi1_devdata *dd = uctxt->dd;
|
||||||
u16 order = 1;
|
u16 order = 1;
|
||||||
#if 0
|
|
||||||
int ret;
|
|
||||||
struct tid_rb_node *node;
|
struct tid_rb_node *node;
|
||||||
dma_addr_t phys;
|
struct process_vm *vm = cpu_local_var(current)->vm;
|
||||||
|
size_t base_pgsize;
|
||||||
|
pte_t *ptep;
|
||||||
|
u64 phys;
|
||||||
|
|
||||||
/*
|
/*
|
||||||
* Allocate the node first so we can handle a potential
|
* Allocate the node first so we can handle a potential
|
||||||
* failure before we've programmed anything.
|
* failure before we've programmed anything.
|
||||||
*/
|
*/
|
||||||
node = kzalloc(sizeof(*node) + (sizeof(struct page *) * npages),
|
node = kcalloc(1, sizeof(*node), GFP_KERNEL);
|
||||||
GFP_KERNEL);
|
|
||||||
if (!node)
|
if (!node)
|
||||||
return -ENOMEM;
|
return -ENOMEM;
|
||||||
phys = pci_map_single(dd->pcidev,
|
|
||||||
__va(page_to_phys(pages[0])),
|
|
||||||
npages * PAGE_SIZE, PCI_DMA_FROMDEVICE);
|
|
||||||
if (dma_mapping_error(&dd->pcidev->dev, phys)) {
|
|
||||||
kprintf("Failed to DMA map Exp Rcv pages 0x%llx\n",
|
|
||||||
phys);
|
|
||||||
kfree(node);
|
|
||||||
return -EFAULT;
|
|
||||||
}
|
|
||||||
node->mmu.addr = vaddr;
|
|
||||||
node->mmu.len = npages * PAGE_SIZE;
|
|
||||||
node->phys = page_to_phys(pages[0]);
|
|
||||||
node->npages = npages;
|
|
||||||
node->rcventry = rcventry;
|
|
||||||
node->dma_addr = phys;
|
|
||||||
node->grp = grp;
|
|
||||||
node->freed = false;
|
|
||||||
memcpy(node->pages, pages, sizeof(struct page *) * npages);
|
|
||||||
if (!fd->handler)
|
|
||||||
ret = tid_rb_insert(fd, &node->mmu);
|
|
||||||
else
|
|
||||||
ret = hfi1_mmu_rb_insert(fd->handler, &node->mmu);
|
|
||||||
|
|
||||||
if (ret) {
|
|
||||||
hfi1_cdbg(TID, "Failed to insert RB node %u 0x%lx, 0x%lx %d",
|
|
||||||
node->rcventry, node->mmu.addr, node->phys, ret);
|
|
||||||
pci_unmap_single(dd->pcidev, phys, npages * PAGE_SIZE,
|
|
||||||
PCI_DMA_FROMDEVICE);
|
|
||||||
kfree(node);
|
|
||||||
return -EFAULT;
|
|
||||||
}
|
|
||||||
#endif
|
|
||||||
size_t base_pgsize;
|
|
||||||
pte_t *ptep;
|
|
||||||
struct process_vm *vm = cpu_local_var(current)->vm;
|
|
||||||
|
|
||||||
ptep = ihk_mc_pt_lookup_pte(vm->address_space->page_table,
|
ptep = ihk_mc_pt_lookup_pte(vm->address_space->page_table,
|
||||||
(void*)vaddr, 0, 0, &base_pgsize, 0);
|
(void*)vaddr, 0, 0, &base_pgsize, 0);
|
||||||
@ -409,7 +371,17 @@ static int set_rcvarray_entry(struct hfi1_filedata *fd, unsigned long vaddr,
|
|||||||
__FUNCTION__, vaddr);
|
__FUNCTION__, vaddr);
|
||||||
return -EFAULT;
|
return -EFAULT;
|
||||||
}
|
}
|
||||||
u64 phys = pte_get_phys(ptep);
|
phys = pte_get_phys(ptep);
|
||||||
|
|
||||||
|
kprintf("Registering rcventry %d, phys 0x%p, len %u\n", rcventry, phys, len);
|
||||||
|
|
||||||
|
node->phys = phys;
|
||||||
|
node->len = len;
|
||||||
|
node->rcventry = rcventry;
|
||||||
|
node->grp = grp;
|
||||||
|
// TODO: check node->rcventry - uctxt->expected_base is within
|
||||||
|
// [0; uctxt->expected_count[ ?
|
||||||
|
fd->entry_to_rb[node->rcventry - uctxt->expected_base] = node;
|
||||||
|
|
||||||
while (len > 0) {
|
while (len > 0) {
|
||||||
order++;
|
order++;
|
||||||
@ -420,6 +392,7 @@ static int set_rcvarray_entry(struct hfi1_filedata *fd, unsigned long vaddr,
|
|||||||
// so we need to make that mapping.
|
// so we need to make that mapping.
|
||||||
|
|
||||||
hfi1_put_tid(dd, rcventry, PT_EXPECTED, phys, order);
|
hfi1_put_tid(dd, rcventry, PT_EXPECTED, phys, order);
|
||||||
|
|
||||||
#if 0
|
#if 0
|
||||||
trace_hfi1_exp_tid_reg(uctxt->ctxt, fd->subctxt, rcventry, npages,
|
trace_hfi1_exp_tid_reg(uctxt->ctxt, fd->subctxt, rcventry, npages,
|
||||||
node->mmu.addr, node->phys, phys);
|
node->mmu.addr, node->phys, phys);
|
||||||
@ -430,9 +403,7 @@ static int set_rcvarray_entry(struct hfi1_filedata *fd, unsigned long vaddr,
|
|||||||
static int unprogram_rcvarray(struct hfi1_filedata *fd, u32 tidinfo,
|
static int unprogram_rcvarray(struct hfi1_filedata *fd, u32 tidinfo,
|
||||||
struct tid_group **grp)
|
struct tid_group **grp)
|
||||||
{
|
{
|
||||||
#if 0
|
|
||||||
struct hfi1_ctxtdata *uctxt = fd->uctxt;
|
struct hfi1_ctxtdata *uctxt = fd->uctxt;
|
||||||
struct hfi1_devdata *dd = uctxt->dd;
|
|
||||||
struct tid_rb_node *node;
|
struct tid_rb_node *node;
|
||||||
u8 tidctrl = EXP_TID_GET(tidinfo, CTRL);
|
u8 tidctrl = EXP_TID_GET(tidinfo, CTRL);
|
||||||
u32 tididx = EXP_TID_GET(tidinfo, IDX) << 1, rcventry;
|
u32 tididx = EXP_TID_GET(tidinfo, IDX) << 1, rcventry;
|
||||||
@ -455,11 +426,49 @@ static int unprogram_rcvarray(struct hfi1_filedata *fd, u32 tidinfo,
|
|||||||
if (grp)
|
if (grp)
|
||||||
*grp = node->grp;
|
*grp = node->grp;
|
||||||
|
|
||||||
if (!fd->handler)
|
kprintf("Clearing rcventry %d, phys 0x%p, len %u\n", node->rcventry,
|
||||||
cacheless_tid_rb_remove(fd, node);
|
node->phys, node->len);
|
||||||
else
|
|
||||||
hfi1_mmu_rb_remove(fd->handler, &node->mmu);
|
fd->entry_to_rb[rcventry] = NULL;
|
||||||
#endif
|
clear_tid_node(fd, node);
|
||||||
|
|
||||||
return 0;
|
return 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
static void clear_tid_node(struct hfi1_filedata *fd, struct tid_rb_node *node)
|
||||||
|
{
|
||||||
|
struct hfi1_ctxtdata *uctxt = fd->uctxt;
|
||||||
|
struct hfi1_devdata *dd = uctxt->dd;
|
||||||
|
|
||||||
|
#if 0
|
||||||
|
trace_hfi1_exp_tid_unreg(uctxt->ctxt, fd->subctxt, node->rcventry,
|
||||||
|
node->npages, node->mmu.addr, node->phys,
|
||||||
|
node->dma_addr);
|
||||||
|
#endif
|
||||||
|
|
||||||
|
hfi1_put_tid(dd, node->rcventry, PT_INVALID, 0, 0);
|
||||||
|
/*
|
||||||
|
* Make sure device has seen the write before we unpin the
|
||||||
|
* pages.
|
||||||
|
*/
|
||||||
|
flush_wc();
|
||||||
|
|
||||||
|
#if 0
|
||||||
|
pci_unmap_single(dd->pcidev, node->dma_addr, node->mmu.len,
|
||||||
|
PCI_DMA_FROMDEVICE);
|
||||||
|
hfi1_release_user_pages(fd->mm, node->pages, node->npages, true);
|
||||||
|
fd->tid_n_pinned -= node->npages;
|
||||||
|
#endif
|
||||||
|
|
||||||
|
|
||||||
|
node->grp->used--;
|
||||||
|
node->grp->map &= ~(1 << (node->rcventry - node->grp->base));
|
||||||
|
|
||||||
|
if (node->grp->used == node->grp->size - 1)
|
||||||
|
tid_group_move(node->grp, &uctxt->tid_full_list,
|
||||||
|
&uctxt->tid_used_list);
|
||||||
|
else if (!node->grp->used)
|
||||||
|
tid_group_move(node->grp, &uctxt->tid_used_list,
|
||||||
|
&uctxt->tid_group_list);
|
||||||
|
kfree(node);
|
||||||
|
}
|
||||||
|
|||||||
Reference in New Issue
Block a user