2019-05-19 13:08:20 +01:00
// SPDX-License-Identifier: GPL-2.0-only
2008-08-27 01:13:12 -07:00
/* flash.c: Allow mmap access to the OBP Flash, for OBP updates.
2005-04-16 15:20:36 -07:00
*
* Copyright ( C ) 1997 Eddie C . Dost ( ecd @ skynet . be )
*/
# include <linux/module.h>
# include <linux/types.h>
# include <linux/errno.h>
# include <linux/miscdevice.h>
# include <linux/fcntl.h>
# include <linux/poll.h>
2010-07-20 23:36:39 -07:00
# include <linux/mutex.h>
2005-04-16 15:20:36 -07:00
# include <linux/spinlock.h>
2007-05-31 01:27:52 -07:00
# include <linux/mm.h>
2008-08-27 01:13:12 -07:00
# include <linux/of.h>
# include <linux/of_device.h>
2005-04-16 15:20:36 -07:00
2016-12-24 11:46:01 -08:00
# include <linux/uaccess.h>
2005-04-16 15:20:36 -07:00
# include <asm/io.h>
# include <asm/upa.h>
2010-07-20 23:36:39 -07:00
static DEFINE_MUTEX ( flash_mutex ) ;
2005-04-16 15:20:36 -07:00
static DEFINE_SPINLOCK ( flash_lock ) ;
static struct {
unsigned long read_base ; /* Physical read address */
unsigned long write_base ; /* Physical write address */
unsigned long read_size ; /* Size of read area */
unsigned long write_size ; /* Size of write area */
unsigned long busy ; /* In use? */
} flash ;
static int
flash_mmap ( struct file * file , struct vm_area_struct * vma )
{
unsigned long addr ;
unsigned long size ;
spin_lock ( & flash_lock ) ;
if ( flash . read_base = = flash . write_base ) {
addr = flash . read_base ;
size = flash . read_size ;
} else {
if ( ( vma - > vm_flags & VM_READ ) & &
( vma - > vm_flags & VM_WRITE ) ) {
spin_unlock ( & flash_lock ) ;
return - EINVAL ;
}
if ( vma - > vm_flags & VM_READ ) {
addr = flash . read_base ;
size = flash . read_size ;
} else if ( vma - > vm_flags & VM_WRITE ) {
addr = flash . write_base ;
size = flash . write_size ;
} else {
spin_unlock ( & flash_lock ) ;
return - ENXIO ;
}
}
spin_unlock ( & flash_lock ) ;
if ( ( vma - > vm_pgoff < < PAGE_SHIFT ) > size )
return - ENXIO ;
addr = vma - > vm_pgoff + ( addr > > PAGE_SHIFT ) ;
if ( vma - > vm_end - ( vma - > vm_start + ( vma - > vm_pgoff < < PAGE_SHIFT ) ) > size )
size = vma - > vm_end - ( vma - > vm_start + ( vma - > vm_pgoff < < PAGE_SHIFT ) ) ;
2006-03-21 02:29:39 -08:00
vma - > vm_page_prot = pgprot_noncached ( vma - > vm_page_prot ) ;
2005-04-16 15:20:36 -07:00
if ( io_remap_pfn_range ( vma , vma - > vm_start , addr , size , vma - > vm_page_prot ) )
return - EAGAIN ;
return 0 ;
}
static long long
flash_llseek ( struct file * file , long long offset , int origin )
{
2010-07-20 23:36:39 -07:00
mutex_lock ( & flash_mutex ) ;
2005-04-16 15:20:36 -07:00
switch ( origin ) {
case 0 :
file - > f_pos = offset ;
break ;
case 1 :
file - > f_pos + = offset ;
if ( file - > f_pos > flash . read_size )
file - > f_pos = flash . read_size ;
break ;
case 2 :
file - > f_pos = flash . read_size ;
break ;
default :
2010-07-20 23:36:39 -07:00
mutex_unlock ( & flash_mutex ) ;
2005-04-16 15:20:36 -07:00
return - EINVAL ;
}
2010-07-20 23:36:39 -07:00
mutex_unlock ( & flash_mutex ) ;
2005-04-16 15:20:36 -07:00
return file - > f_pos ;
}
static ssize_t
flash_read ( struct file * file , char __user * buf ,
size_t count , loff_t * ppos )
{
2010-04-27 18:48:52 -07:00
loff_t p = * ppos ;
2005-04-16 15:20:36 -07:00
int i ;
2010-04-27 18:48:52 -07:00
2005-04-16 15:20:36 -07:00
if ( count > flash . read_size - p )
count = flash . read_size - p ;
for ( i = 0 ; i < count ; i + + ) {
u8 data = upa_readb ( flash . read_base + p + i ) ;
if ( put_user ( data , buf ) )
return - EFAULT ;
buf + + ;
}
2010-04-27 18:48:52 -07:00
* ppos + = count ;
2005-04-16 15:20:36 -07:00
return count ;
}
static int
flash_open ( struct inode * inode , struct file * file )
{
2010-07-20 23:36:39 -07:00
mutex_lock ( & flash_mutex ) ;
2008-05-20 19:15:54 +02:00
if ( test_and_set_bit ( 0 , ( void * ) & flash . busy ) ! = 0 ) {
2010-07-20 23:36:39 -07:00
mutex_unlock ( & flash_mutex ) ;
2005-04-16 15:20:36 -07:00
return - EBUSY ;
2008-05-20 19:15:54 +02:00
}
2005-04-16 15:20:36 -07:00
2010-07-20 23:36:39 -07:00
mutex_unlock ( & flash_mutex ) ;
2005-04-16 15:20:36 -07:00
return 0 ;
}
static int
flash_release ( struct inode * inode , struct file * file )
{
spin_lock ( & flash_lock ) ;
flash . busy = 0 ;
spin_unlock ( & flash_lock ) ;
return 0 ;
}
2007-02-12 00:55:34 -08:00
static const struct file_operations flash_fops = {
2005-04-16 15:20:36 -07:00
/* no write to the Flash, use mmap
* and play flash dependent tricks .
*/
. owner = THIS_MODULE ,
. llseek = flash_llseek ,
. read = flash_read ,
. mmap = flash_mmap ,
. open = flash_open ,
. release = flash_release ,
} ;
2020-03-11 15:16:54 +08:00
static struct miscdevice flash_dev = { SBUS_FLASH_MINOR , " flash " , & flash_fops } ;
2005-04-16 15:20:36 -07:00
2012-12-21 13:24:23 -08:00
static int flash_probe ( struct platform_device * op )
2005-04-16 15:20:36 -07:00
{
2010-04-13 16:12:29 -07:00
struct device_node * dp = op - > dev . of_node ;
2008-08-27 01:13:12 -07:00
struct device_node * parent ;
2005-04-16 15:20:36 -07:00
2008-08-27 01:13:12 -07:00
parent = dp - > parent ;
2005-04-16 15:20:36 -07:00
2018-12-05 13:50:38 -06:00
if ( ! of_node_name_eq ( parent , " sbus " ) & &
! of_node_name_eq ( parent , " sbi " ) & &
! of_node_name_eq ( parent , " ebus " ) )
2005-04-16 15:20:36 -07:00
return - ENODEV ;
2008-08-27 01:13:12 -07:00
flash . read_base = op - > resource [ 0 ] . start ;
flash . read_size = resource_size ( & op - > resource [ 0 ] ) ;
if ( op - > resource [ 1 ] . flags ) {
flash . write_base = op - > resource [ 1 ] . start ;
flash . write_size = resource_size ( & op - > resource [ 1 ] ) ;
} else {
flash . write_base = op - > resource [ 0 ] . start ;
flash . write_size = resource_size ( & op - > resource [ 0 ] ) ;
2005-04-16 15:20:36 -07:00
}
2008-08-27 01:13:12 -07:00
flash . busy = 0 ;
2005-04-16 15:20:36 -07:00
2017-07-18 16:43:27 -05:00
printk ( KERN_INFO " %pOF: OBP Flash, RD %lx[%lx] WR %lx[%lx] \n " ,
op - > dev . of_node ,
2005-04-16 15:20:36 -07:00
flash . read_base , flash . read_size ,
flash . write_base , flash . write_size ) ;
2008-08-27 01:13:12 -07:00
return misc_register ( & flash_dev ) ;
}
2012-12-21 13:24:23 -08:00
static int flash_remove ( struct platform_device * op )
2008-08-27 01:13:12 -07:00
{
misc_deregister ( & flash_dev ) ;
2005-04-16 15:20:36 -07:00
return 0 ;
}
2008-08-31 01:23:17 -07:00
static const struct of_device_id flash_match [ ] = {
2008-08-27 01:13:12 -07:00
{
. name = " flashprom " ,
} ,
{ } ,
} ;
MODULE_DEVICE_TABLE ( of , flash_match ) ;
2011-02-22 20:01:33 -07:00
static struct platform_driver flash_driver = {
2010-04-13 16:13:02 -07:00
. driver = {
. name = " flash " ,
. of_match_table = flash_match ,
} ,
2008-08-27 01:13:12 -07:00
. probe = flash_probe ,
2012-12-21 13:24:23 -08:00
. remove = flash_remove ,
2008-08-27 01:13:12 -07:00
} ;
2011-11-26 19:04:25 +00:00
module_platform_driver ( flash_driver ) ;
2005-04-16 15:20:36 -07:00
MODULE_LICENSE ( " GPL " ) ;