/* * dev_iox.c - IOX (bridge to external I/O) generic device API implementation. * * Authors Alain Greiner (2017) * * Copyright (c) UPMC Sorbonne Universites * * This file is part of ALMOS-MKH. * * ALMOS-MKH is free software; you can redistribute it and/or modify it * under the terms of the GNU General Public License as published by * the Free Software Foundation; version 2.0 of the License. * * ALMOS-MKH 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 PARTIOXLAR PURPOSE. See the GNU * General Public License for more details. * * You should have received a copy of the GNU General Public License * along with ALMOS-MKH; if not, write to the Free Software Foundation, * Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA */ #include #include #include #include #include #include #include #include ////////////////////////////////// void dev_iox_init( xptr_t dev_xp ) { // get IOX device descriptor cluster and local pointer cxy_t dev_cxy = GET_CXY( dev_xp ); device_t * dev_ptr = (device_t *)GET_PTR( dev_xp ); // get implementation from device descriptor uint32_t impl = hal_remote_lw( XPTR( dev_cxy , &dev_ptr->impl ) ); // call driver init function if( impl == IMPL_IOX_IOB ) { soclib_iob_init( dev_xp ); } else { printk("\n[PANIC] in %s: undefined IOX device implementation\n", __FUNCTION__ ); hal_core_sleep(); } } ////////////////////////////////////////// void dev_iox_iommu_enable( xptr_t dev_xp ) { // get IOX device descriptor cluster and local pointer cxy_t dev_cxy = GET_CXY( dev_xp ); device_t * dev_ptr = (device_t *)GET_PTR( dev_xp ); // get implementation from device descriptor uint32_t impl = hal_remote_lw( XPTR( dev_cxy , &dev_ptr->impl ) ); // call driver function if( impl == IMPL_IOX_IOB ) { remote_spinlock_lock( XPTR( dev_cxy , &dev_ptr->wait_lock ) ); soclib_iob_set_active( dev_xp , 1 ); remote_spinlock_unlock( XPTR( dev_cxy , &dev_ptr->wait_lock ) ); } else { printk("\n[PANIC] in %s: undefined IOX device implementation\n", __FUNCTION__ ); hal_core_sleep(); } } /////////////////////////////////////////// void dev_iox_iommu_disable( xptr_t dev_xp ) { // get IOX device descriptor cluster and local pointer cxy_t dev_cxy = GET_CXY( dev_xp ); device_t * dev_ptr = (device_t *)GET_PTR( dev_xp ); // get implementation from device descriptor uint32_t impl = hal_remote_lw( XPTR( dev_cxy , &dev_ptr->impl ) ); // call driver function if( impl == IMPL_IOX_IOB ) { remote_spinlock_lock( XPTR( dev_cxy , &dev_ptr->wait_lock ) ); soclib_iob_set_active( dev_xp , 0 ); remote_spinlock_unlock( XPTR( dev_cxy , &dev_ptr->wait_lock ) ); } else { printk("\n[PANIC] in %s: undefined IOX device implementation\n", __FUNCTION__ ); hal_core_sleep(); } } //////////////////////////////////////// void dev_iox_set_ptpr( xptr_t dev_xp, uint32_t wdata ) { // get IOX device descriptor cluster and local pointer cxy_t dev_cxy = GET_CXY( dev_xp ); device_t * dev_ptr = (device_t *)GET_PTR( dev_xp ); // get implementation from device descriptor uint32_t impl = hal_remote_lw( XPTR( dev_cxy , &dev_ptr->impl ) ); // call driver function if( impl == IMPL_IOX_IOB ) { remote_spinlock_lock( XPTR( dev_cxy , &dev_ptr->wait_lock ) ); soclib_iob_set_ptpr( dev_xp , wdata ); remote_spinlock_unlock( XPTR( dev_cxy , &dev_ptr->wait_lock ) ); } else { printk("\n[PANIC] in %s: undefined IOX device implementation\n", __FUNCTION__ ); hal_core_sleep(); } } //////////////////////////////////////// void dev_iox_inval_page( xptr_t dev_xp, vpn_t vpn ) { // get IOX device descriptor cluster and local pointer cxy_t dev_cxy = GET_CXY( dev_xp ); device_t * dev_ptr = (device_t *)GET_PTR( dev_xp ); // get implementation from device descriptor uint32_t impl = hal_remote_lw( XPTR( dev_cxy , &dev_ptr->impl ) ); // call driver function if( impl == IMPL_IOX_IOB ) { remote_spinlock_lock( XPTR( dev_cxy , &dev_ptr->wait_lock ) ); soclib_iob_inval_page( dev_xp , vpn ); remote_spinlock_unlock( XPTR( dev_cxy , &dev_ptr->wait_lock ) ); } else { printk("\n[PANIC] in %s: undefined IOX device implementation\n", __FUNCTION__ ); hal_core_sleep(); } } /////////////////////////////////////////// void dev_iox_get_status( xptr_t dev_xp, uint32_t * error, uint32_t * bvar, uint32_t * srcid ) { // get IOX device descriptor cluster and local pointer cxy_t dev_cxy = GET_CXY( dev_xp ); device_t * dev_ptr = (device_t *)GET_PTR( dev_xp ); // get implementation from device descriptor uint32_t impl = hal_remote_lw( XPTR( dev_cxy , &dev_ptr->impl ) ); // call driver function if( impl == IMPL_IOX_IOB ) { remote_spinlock_lock( XPTR( dev_cxy , &dev_ptr->wait_lock ) ); *error = soclib_iob_get_error( dev_xp ); *srcid = soclib_iob_get_srcid( dev_xp ); *bvar = soclib_iob_get_bvar( dev_xp ); remote_spinlock_unlock( XPTR( dev_cxy , &dev_ptr->wait_lock ) ); } else { printk("\n[PANIC] in %s: undefined IOX device implementation\n", __FUNCTION__ ); hal_core_sleep(); } }