2019-05-27 08:55:21 +02:00
// SPDX-License-Identifier: GPL-2.0-only
2011-10-20 16:52:46 +02:00
/*
* Remote Processor Framework
*
* Copyright ( C ) 2011 Texas Instruments , Inc .
* Copyright ( C ) 2011 Google , Inc .
*
* Ohad Ben - Cohen < ohad @ wizery . com >
* Brian Swetland < swetland @ google . com >
* Mark Grosen < mgrosen @ ti . com >
* Fernando Guzman Lugo < fernando . lugo @ ti . com >
* Suman Anna < s - anna @ ti . com >
* Robert Tivy < rtivy @ ti . com >
* Armando Uribe De Leon < x0095078 @ ti . com >
*/
# define pr_fmt(fmt) "%s: " fmt, __func__
2020-03-23 22:29:02 -07:00
# include <linux/delay.h>
2011-10-20 16:52:46 +02:00
# include <linux/kernel.h>
# include <linux/module.h>
# include <linux/device.h>
# include <linux/slab.h>
# include <linux/mutex.h>
2020-09-22 15:31:03 +02:00
# include <linux/dma-map-ops.h>
2011-10-20 16:52:46 +02:00
# include <linux/dma-mapping.h>
2020-09-17 18:43:40 +02:00
# include <linux/dma-direct.h> /* XXX: pokes into bus_dma_range */
2011-10-20 16:52:46 +02:00
# include <linux/firmware.h>
# include <linux/string.h>
# include <linux/debugfs.h>
2020-03-23 22:29:01 -07:00
# include <linux/rculist.h>
2011-10-20 16:52:46 +02:00
# include <linux/remoteproc.h>
# include <linux/iommu.h>
remoteproc: maintain a generic child device for each rproc
For each registered rproc, maintain a generic remoteproc device whose
parent is the low level platform-specific device (commonly a pdev, but
it may certainly be any other type of device too).
With this in hand, the resulting device hierarchy might then look like:
omap-rproc.0
|
- remoteproc0 <---- new !
|
- virtio0
|
- virtio1
|
- rpmsg0
|
- rpmsg1
|
- rpmsg2
Where:
- omap-rproc.0 is the low level device that's bound to the
driver which invokes rproc_register()
- remoteproc0 is the result of this patch, and will be added by the
remoteproc framework when rproc_register() is invoked
- virtio0 and virtio1 are vdevs that are registered by remoteproc
when it realizes that they are supported by the firmware
of the physical remote processor represented by omap-rproc.0
- rpmsg0, rpmsg1 and rpmsg2 are rpmsg devices that represent rpmsg
channels, and are registerd by the rpmsg bus when it gets notified
about their existence
Technically, this patch:
- changes 'struct rproc' to contain this generic remoteproc.x device
- creates a new "remoteproc" type, to which this new generic remoteproc.x
device belong to.
- adds a super simple enumeration method for the indices of the
remoteproc.x devices
- updates all dev_* messaging to use the generic remoteproc.x device
instead of the low level platform-specific device
- updates all dma_* allocations to use the parent of remoteproc.x (where
the platform-specific memory pools, most commonly CMA, are to be found)
Adding this generic device has several merits:
- we can now add remoteproc runtime PM support simply by hooking onto the
new "remoteproc" type
- all remoteproc log messages will now carry a common name prefix
instead of having a platform-specific one
- having a device as part of the rproc struct makes it possible to simplify
refcounting (see subsequent patch)
Thanks to Stephen Boyd <sboyd@codeaurora.org> for suggesting and
discussing these ideas in one of the remoteproc review threads and
to Fernando Guzman Lugo <fernando.lugo@ti.com> for trying them out
with the (upcoming) runtime PM support for remoteproc.
Cc: Fernando Guzman Lugo <fernando.lugo@ti.com>
Reviewed-by: Stephen Boyd <sboyd@codeaurora.org>
Signed-off-by: Ohad Ben-Cohen <ohad@wizery.com>
2012-05-30 22:01:25 +03:00
# include <linux/idr.h>
2011-10-20 16:52:46 +02:00
# include <linux/elf.h>
2013-04-07 14:06:07 +03:00
# include <linux/crc32.h>
2019-01-10 14:50:49 +01:00
# include <linux/of_reserved_mem.h>
2011-10-20 16:52:46 +02:00
# include <linux/virtio_ids.h>
# include <linux/virtio_ring.h>
2012-01-31 15:23:41 +02:00
# include <asm/byteorder.h>
2019-01-10 14:50:49 +01:00
# include <linux/platform_device.h>
2011-10-20 16:52:46 +02:00
# include "remoteproc_internal.h"
2019-01-10 14:49:08 +01:00
# define HIGH_BITS_MASK 0xFFFFFFFF00000000ULL
2015-05-22 15:45:27 -05:00
static DEFINE_MUTEX ( rproc_list_mutex ) ;
static LIST_HEAD ( rproc_list ) ;
2020-03-23 22:29:02 -07:00
static struct notifier_block rproc_panic_nb ;
2015-05-22 15:45:27 -05:00
2013-04-07 14:06:07 +03:00
typedef int ( * rproc_handle_resource_t ) ( struct rproc * rproc ,
void * , int offset , int avail ) ;
2011-10-20 16:52:46 +02:00
2018-07-27 15:14:47 +02:00
static int rproc_alloc_carveout ( struct rproc * rproc ,
struct rproc_mem_entry * mem ) ;
static int rproc_release_carveout ( struct rproc * rproc ,
struct rproc_mem_entry * mem ) ;
remoteproc: maintain a generic child device for each rproc
For each registered rproc, maintain a generic remoteproc device whose
parent is the low level platform-specific device (commonly a pdev, but
it may certainly be any other type of device too).
With this in hand, the resulting device hierarchy might then look like:
omap-rproc.0
|
- remoteproc0 <---- new !
|
- virtio0
|
- virtio1
|
- rpmsg0
|
- rpmsg1
|
- rpmsg2
Where:
- omap-rproc.0 is the low level device that's bound to the
driver which invokes rproc_register()
- remoteproc0 is the result of this patch, and will be added by the
remoteproc framework when rproc_register() is invoked
- virtio0 and virtio1 are vdevs that are registered by remoteproc
when it realizes that they are supported by the firmware
of the physical remote processor represented by omap-rproc.0
- rpmsg0, rpmsg1 and rpmsg2 are rpmsg devices that represent rpmsg
channels, and are registerd by the rpmsg bus when it gets notified
about their existence
Technically, this patch:
- changes 'struct rproc' to contain this generic remoteproc.x device
- creates a new "remoteproc" type, to which this new generic remoteproc.x
device belong to.
- adds a super simple enumeration method for the indices of the
remoteproc.x devices
- updates all dev_* messaging to use the generic remoteproc.x device
instead of the low level platform-specific device
- updates all dma_* allocations to use the parent of remoteproc.x (where
the platform-specific memory pools, most commonly CMA, are to be found)
Adding this generic device has several merits:
- we can now add remoteproc runtime PM support simply by hooking onto the
new "remoteproc" type
- all remoteproc log messages will now carry a common name prefix
instead of having a platform-specific one
- having a device as part of the rproc struct makes it possible to simplify
refcounting (see subsequent patch)
Thanks to Stephen Boyd <sboyd@codeaurora.org> for suggesting and
discussing these ideas in one of the remoteproc review threads and
to Fernando Guzman Lugo <fernando.lugo@ti.com> for trying them out
with the (upcoming) runtime PM support for remoteproc.
Cc: Fernando Guzman Lugo <fernando.lugo@ti.com>
Reviewed-by: Stephen Boyd <sboyd@codeaurora.org>
Signed-off-by: Ohad Ben-Cohen <ohad@wizery.com>
2012-05-30 22:01:25 +03:00
/* Unique indices for remoteproc devices */
static DEFINE_IDA ( rproc_dev_index ) ;
2012-08-30 13:26:12 -05:00
static const char * const rproc_crash_names [ ] = {
[ RPROC_MMUFAULT ] = " mmufault " ,
2016-03-28 20:36:59 -07:00
[ RPROC_WATCHDOG ] = " watchdog " ,
[ RPROC_FATAL_ERROR ] = " fatal error " ,
2012-08-30 13:26:12 -05:00
} ;
/* translate rproc_crash_type to string */
static const char * rproc_crash_to_string ( enum rproc_crash_type type )
{
if ( type < ARRAY_SIZE ( rproc_crash_names ) )
return rproc_crash_names [ type ] ;
2013-04-18 00:12:55 +09:00
return " unknown " ;
2012-08-30 13:26:12 -05:00
}
2011-10-20 16:52:46 +02:00
/*
* This is the IOMMU fault handler we register with the IOMMU API
* ( when relevant ; not all remote processors access memory through
* an IOMMU ) .
*
* IOMMU core will invoke this handler whenever the remote processor
* will try to access an unmapped device address .
*/
static int rproc_iommu_fault ( struct iommu_domain * domain , struct device * dev ,
2016-08-12 18:42:20 -05:00
unsigned long iova , int flags , void * token )
2011-10-20 16:52:46 +02:00
{
2012-08-30 13:26:12 -05:00
struct rproc * rproc = token ;
2011-10-20 16:52:46 +02:00
dev_err ( dev , " iommu fault: da 0x%lx flags 0x%x \n " , iova , flags ) ;
2012-08-30 13:26:12 -05:00
rproc_report_crash ( rproc , RPROC_MMUFAULT ) ;
2011-10-20 16:52:46 +02:00
/*
* Let the iommu core know we ' re not really handling this fault ;
2012-08-30 13:26:12 -05:00
* we just used it as a recovery trigger .
2011-10-20 16:52:46 +02:00
*/
return - ENOSYS ;
}
static int rproc_enable_iommu ( struct rproc * rproc )
{
struct iommu_domain * domain ;
remoteproc: maintain a generic child device for each rproc
For each registered rproc, maintain a generic remoteproc device whose
parent is the low level platform-specific device (commonly a pdev, but
it may certainly be any other type of device too).
With this in hand, the resulting device hierarchy might then look like:
omap-rproc.0
|
- remoteproc0 <---- new !
|
- virtio0
|
- virtio1
|
- rpmsg0
|
- rpmsg1
|
- rpmsg2
Where:
- omap-rproc.0 is the low level device that's bound to the
driver which invokes rproc_register()
- remoteproc0 is the result of this patch, and will be added by the
remoteproc framework when rproc_register() is invoked
- virtio0 and virtio1 are vdevs that are registered by remoteproc
when it realizes that they are supported by the firmware
of the physical remote processor represented by omap-rproc.0
- rpmsg0, rpmsg1 and rpmsg2 are rpmsg devices that represent rpmsg
channels, and are registerd by the rpmsg bus when it gets notified
about their existence
Technically, this patch:
- changes 'struct rproc' to contain this generic remoteproc.x device
- creates a new "remoteproc" type, to which this new generic remoteproc.x
device belong to.
- adds a super simple enumeration method for the indices of the
remoteproc.x devices
- updates all dev_* messaging to use the generic remoteproc.x device
instead of the low level platform-specific device
- updates all dma_* allocations to use the parent of remoteproc.x (where
the platform-specific memory pools, most commonly CMA, are to be found)
Adding this generic device has several merits:
- we can now add remoteproc runtime PM support simply by hooking onto the
new "remoteproc" type
- all remoteproc log messages will now carry a common name prefix
instead of having a platform-specific one
- having a device as part of the rproc struct makes it possible to simplify
refcounting (see subsequent patch)
Thanks to Stephen Boyd <sboyd@codeaurora.org> for suggesting and
discussing these ideas in one of the remoteproc review threads and
to Fernando Guzman Lugo <fernando.lugo@ti.com> for trying them out
with the (upcoming) runtime PM support for remoteproc.
Cc: Fernando Guzman Lugo <fernando.lugo@ti.com>
Reviewed-by: Stephen Boyd <sboyd@codeaurora.org>
Signed-off-by: Ohad Ben-Cohen <ohad@wizery.com>
2012-05-30 22:01:25 +03:00
struct device * dev = rproc - > dev . parent ;
2011-10-20 16:52:46 +02:00
int ret ;
2015-01-09 15:21:58 -06:00
if ( ! rproc - > has_iommu ) {
dev_dbg ( dev , " iommu not present \n " ) ;
2011-12-13 08:41:47 +02:00
return 0 ;
2011-10-20 16:52:46 +02:00
}
domain = iommu_domain_alloc ( dev - > bus ) ;
if ( ! domain ) {
dev_err ( dev , " can't alloc iommu domain \n " ) ;
return - ENOMEM ;
}
2012-05-21 20:20:05 +03:00
iommu_set_fault_handler ( domain , rproc_iommu_fault , rproc ) ;
2011-10-20 16:52:46 +02:00
ret = iommu_attach_device ( domain , dev ) ;
if ( ret ) {
dev_err ( dev , " can't attach iommu device: %d \n " , ret ) ;
goto free_domain ;
}
rproc - > domain = domain ;
return 0 ;
free_domain :
iommu_domain_free ( domain ) ;
return ret ;
}
static void rproc_disable_iommu ( struct rproc * rproc )
{
struct iommu_domain * domain = rproc - > domain ;
remoteproc: maintain a generic child device for each rproc
For each registered rproc, maintain a generic remoteproc device whose
parent is the low level platform-specific device (commonly a pdev, but
it may certainly be any other type of device too).
With this in hand, the resulting device hierarchy might then look like:
omap-rproc.0
|
- remoteproc0 <---- new !
|
- virtio0
|
- virtio1
|
- rpmsg0
|
- rpmsg1
|
- rpmsg2
Where:
- omap-rproc.0 is the low level device that's bound to the
driver which invokes rproc_register()
- remoteproc0 is the result of this patch, and will be added by the
remoteproc framework when rproc_register() is invoked
- virtio0 and virtio1 are vdevs that are registered by remoteproc
when it realizes that they are supported by the firmware
of the physical remote processor represented by omap-rproc.0
- rpmsg0, rpmsg1 and rpmsg2 are rpmsg devices that represent rpmsg
channels, and are registerd by the rpmsg bus when it gets notified
about their existence
Technically, this patch:
- changes 'struct rproc' to contain this generic remoteproc.x device
- creates a new "remoteproc" type, to which this new generic remoteproc.x
device belong to.
- adds a super simple enumeration method for the indices of the
remoteproc.x devices
- updates all dev_* messaging to use the generic remoteproc.x device
instead of the low level platform-specific device
- updates all dma_* allocations to use the parent of remoteproc.x (where
the platform-specific memory pools, most commonly CMA, are to be found)
Adding this generic device has several merits:
- we can now add remoteproc runtime PM support simply by hooking onto the
new "remoteproc" type
- all remoteproc log messages will now carry a common name prefix
instead of having a platform-specific one
- having a device as part of the rproc struct makes it possible to simplify
refcounting (see subsequent patch)
Thanks to Stephen Boyd <sboyd@codeaurora.org> for suggesting and
discussing these ideas in one of the remoteproc review threads and
to Fernando Guzman Lugo <fernando.lugo@ti.com> for trying them out
with the (upcoming) runtime PM support for remoteproc.
Cc: Fernando Guzman Lugo <fernando.lugo@ti.com>
Reviewed-by: Stephen Boyd <sboyd@codeaurora.org>
Signed-off-by: Ohad Ben-Cohen <ohad@wizery.com>
2012-05-30 22:01:25 +03:00
struct device * dev = rproc - > dev . parent ;
2011-10-20 16:52:46 +02:00
if ( ! domain )
return ;
iommu_detach_device ( domain , dev ) ;
iommu_domain_free ( domain ) ;
}
2019-01-10 14:50:49 +01:00
phys_addr_t rproc_va_to_pa ( void * cpu_addr )
2018-07-27 15:14:37 +02:00
{
/*
* Return physical address according to virtual address location
* - in vmalloc : if region ioremapped or defined as dma_alloc_coherent
* - in kernel : if region allocated in generic dma memory pool
*/
if ( is_vmalloc_addr ( cpu_addr ) ) {
return page_to_phys ( vmalloc_to_page ( cpu_addr ) ) +
offset_in_page ( cpu_addr ) ;
}
WARN_ON ( ! virt_addr_valid ( cpu_addr ) ) ;
return virt_to_phys ( cpu_addr ) ;
}
2019-01-10 14:50:49 +01:00
EXPORT_SYMBOL ( rproc_va_to_pa ) ;
2018-07-27 15:14:37 +02:00
2015-05-22 15:45:28 -05:00
/**
* rproc_da_to_va ( ) - lookup the kernel virtual address for a remoteproc address
* @ rproc : handle of a remote processor
* @ da : remoteproc device address to translate
* @ len : length of the memory region @ da is pointing to
*
2011-10-20 16:52:46 +02:00
* Some remote processors will ask us to allocate them physically contiguous
* memory regions ( which we call " carveouts " ) , and map them to specific
2015-05-22 15:45:28 -05:00
* device addresses ( which are hardcoded in the firmware ) . They may also have
* dedicated memory regions internal to the processors , and use them either
* exclusively or alongside carveouts .
2011-10-20 16:52:46 +02:00
*
* They may then ask us to copy objects into specific device addresses ( e . g .
* code / data sections ) or expose us certain symbols in other device address
* ( e . g . their trace buffer ) .
*
2015-05-22 15:45:28 -05:00
* This function is a helper function with which we can go over the allocated
* carveouts and translate specific device addresses to kernel virtual addresses
* so we can access the referenced memory . This function also allows to perform
* translations on the internal remoteproc memory regions through a platform
* implementation specific da_to_va ops , if present .
*
* The function returns a valid kernel address on success or NULL on failure .
2011-10-20 16:52:46 +02:00
*
* Note : phys_to_virt ( iommu_iova_to_phys ( rproc - > domain , da ) ) will work too ,
* but only on kernel direct mapped RAM memory . Instead , we ' re just using
2015-05-22 15:45:28 -05:00
* here the output of the DMA API for the carveouts , which should be more
* correct .
2011-10-20 16:52:46 +02:00
*/
2021-03-06 19:24:19 +08:00
void * rproc_da_to_va ( struct rproc * rproc , u64 da , size_t len , bool * is_iomem )
2011-10-20 16:52:46 +02:00
{
struct rproc_mem_entry * carveout ;
void * ptr = NULL ;
2015-05-22 15:45:28 -05:00
if ( rproc - > ops - > da_to_va ) {
2021-03-06 19:24:19 +08:00
ptr = rproc - > ops - > da_to_va ( rproc , da , len , is_iomem ) ;
2015-05-22 15:45:28 -05:00
if ( ptr )
goto out ;
}
2011-10-20 16:52:46 +02:00
list_for_each_entry ( carveout , & rproc - > carveouts , node ) {
int offset = da - carveout - > da ;
2019-01-10 14:49:06 +01:00
/* Verify that carveout is allocated */
if ( ! carveout - > va )
continue ;
2011-10-20 16:52:46 +02:00
/* try next carveout if da is too small */
if ( offset < 0 )
continue ;
/* try next carveout if da is too large */
if ( offset + len > carveout - > len )
continue ;
ptr = carveout - > va + offset ;
2021-03-06 19:24:19 +08:00
if ( is_iomem )
* is_iomem = carveout - > is_iomem ;
2011-10-20 16:52:46 +02:00
break ;
}
2015-05-22 15:45:28 -05:00
out :
2011-10-20 16:52:46 +02:00
return ptr ;
}
2012-06-19 10:08:18 +03:00
EXPORT_SYMBOL ( rproc_da_to_va ) ;
2011-10-20 16:52:46 +02:00
2018-07-27 15:14:42 +02:00
/**
* rproc_find_carveout_by_name ( ) - lookup the carveout region by a name
* @ rproc : handle of a remote processor
2020-02-12 17:19:56 +01:00
* @ name : carveout name to find ( format string )
* @ . . . : optional parameters matching @ name string
2018-07-27 15:14:42 +02:00
*
* Platform driver has the capability to register some pre - allacoted carveout
* ( physically contiguous memory regions ) before rproc firmware loading and
* associated resource table analysis . These regions may be dedicated memory
* regions internal to the coprocessor or specified DDR region with specific
* attributes
*
* This function is a helper function with which we can go over the
* allocated carveouts and return associated region characteristics like
* coprocessor address , length or processor virtual address .
*
* Return : a valid pointer on carveout entry on success or NULL on failure .
*/
2020-07-15 13:35:50 +01:00
__printf ( 2 , 3 )
2018-07-27 15:14:42 +02:00
struct rproc_mem_entry *
rproc_find_carveout_by_name ( struct rproc * rproc , const char * name , . . . )
{
va_list args ;
char _name [ 32 ] ;
struct rproc_mem_entry * carveout , * mem = NULL ;
if ( ! name )
return NULL ;
va_start ( args , name ) ;
vsnprintf ( _name , sizeof ( _name ) , name , args ) ;
va_end ( args ) ;
list_for_each_entry ( carveout , & rproc - > carveouts , node ) {
/* Compare carveout and requested names */
if ( ! strcmp ( carveout - > name , _name ) ) {
mem = carveout ;
break ;
}
}
return mem ;
}
2018-07-27 15:14:45 +02:00
/**
* rproc_check_carveout_da ( ) - Check specified carveout da configuration
* @ rproc : handle of a remote processor
* @ mem : pointer on carveout to check
* @ da : area device address
* @ len : associated area size
*
* This function is a helper function to verify requested device area ( couple
2019-01-10 14:49:11 +01:00
* da , len ) is part of specified carveout .
* If da is not set ( defined as FW_RSC_ADDR_ANY ) , only requested length is
* checked .
2018-07-27 15:14:45 +02:00
*
2019-01-10 14:49:11 +01:00
* Return : 0 if carveout matches request else error
2018-07-27 15:14:45 +02:00
*/
2019-01-10 14:49:11 +01:00
static int rproc_check_carveout_da ( struct rproc * rproc ,
struct rproc_mem_entry * mem , u32 da , u32 len )
2018-07-27 15:14:45 +02:00
{
struct device * dev = & rproc - > dev ;
2019-01-10 14:49:11 +01:00
int delta ;
2018-07-27 15:14:45 +02:00
/* Check requested resource length */
if ( len > mem - > len ) {
dev_err ( dev , " Registered carveout doesn't fit len request \n " ) ;
2019-01-10 14:49:11 +01:00
return - EINVAL ;
2018-07-27 15:14:45 +02:00
}
if ( da ! = FW_RSC_ADDR_ANY & & mem - > da = = FW_RSC_ADDR_ANY ) {
2019-01-10 14:49:11 +01:00
/* Address doesn't match registered carveout configuration */
return - EINVAL ;
2018-07-27 15:14:45 +02:00
} else if ( da ! = FW_RSC_ADDR_ANY & & mem - > da ! = FW_RSC_ADDR_ANY ) {
delta = da - mem - > da ;
/* Check requested resource belongs to registered carveout */
if ( delta < 0 ) {
dev_err ( dev ,
" Registered carveout doesn't fit da request \n " ) ;
2019-01-10 14:49:11 +01:00
return - EINVAL ;
2018-07-27 15:14:45 +02:00
}
if ( delta + len > mem - > len ) {
dev_err ( dev ,
" Registered carveout doesn't fit len request \n " ) ;
2019-01-10 14:49:11 +01:00
return - EINVAL ;
2018-07-27 15:14:45 +02:00
}
}
return 0 ;
}
2012-05-17 14:23:59 +03:00
int rproc_alloc_vring ( struct rproc_vdev * rvdev , int i )
2011-10-20 16:52:46 +02:00
{
2012-02-13 22:30:39 +01:00
struct rproc * rproc = rvdev - > rproc ;
remoteproc: maintain a generic child device for each rproc
For each registered rproc, maintain a generic remoteproc device whose
parent is the low level platform-specific device (commonly a pdev, but
it may certainly be any other type of device too).
With this in hand, the resulting device hierarchy might then look like:
omap-rproc.0
|
- remoteproc0 <---- new !
|
- virtio0
|
- virtio1
|
- rpmsg0
|
- rpmsg1
|
- rpmsg2
Where:
- omap-rproc.0 is the low level device that's bound to the
driver which invokes rproc_register()
- remoteproc0 is the result of this patch, and will be added by the
remoteproc framework when rproc_register() is invoked
- virtio0 and virtio1 are vdevs that are registered by remoteproc
when it realizes that they are supported by the firmware
of the physical remote processor represented by omap-rproc.0
- rpmsg0, rpmsg1 and rpmsg2 are rpmsg devices that represent rpmsg
channels, and are registerd by the rpmsg bus when it gets notified
about their existence
Technically, this patch:
- changes 'struct rproc' to contain this generic remoteproc.x device
- creates a new "remoteproc" type, to which this new generic remoteproc.x
device belong to.
- adds a super simple enumeration method for the indices of the
remoteproc.x devices
- updates all dev_* messaging to use the generic remoteproc.x device
instead of the low level platform-specific device
- updates all dma_* allocations to use the parent of remoteproc.x (where
the platform-specific memory pools, most commonly CMA, are to be found)
Adding this generic device has several merits:
- we can now add remoteproc runtime PM support simply by hooking onto the
new "remoteproc" type
- all remoteproc log messages will now carry a common name prefix
instead of having a platform-specific one
- having a device as part of the rproc struct makes it possible to simplify
refcounting (see subsequent patch)
Thanks to Stephen Boyd <sboyd@codeaurora.org> for suggesting and
discussing these ideas in one of the remoteproc review threads and
to Fernando Guzman Lugo <fernando.lugo@ti.com> for trying them out
with the (upcoming) runtime PM support for remoteproc.
Cc: Fernando Guzman Lugo <fernando.lugo@ti.com>
Reviewed-by: Stephen Boyd <sboyd@codeaurora.org>
Signed-off-by: Ohad Ben-Cohen <ohad@wizery.com>
2012-05-30 22:01:25 +03:00
struct device * dev = & rproc - > dev ;
2012-05-17 14:23:59 +03:00
struct rproc_vring * rvring = & rvdev - > vring [ i ] ;
2013-02-21 18:15:40 +01:00
struct fw_rsc_vdev * rsc ;
2020-03-02 10:38:56 +01:00
int ret , notifyid ;
2018-07-27 15:14:47 +02:00
struct rproc_mem_entry * mem ;
2020-03-02 10:38:56 +01:00
size_t size ;
2011-10-20 16:52:46 +02:00
2012-02-13 22:30:39 +01:00
/* actual size of vring (in bytes) */
2012-05-17 14:23:59 +03:00
size = PAGE_ALIGN ( vring_size ( rvring - > len , rvring - > align ) ) ;
2012-02-13 22:30:39 +01:00
2018-07-27 15:14:47 +02:00
rsc = ( void * ) rproc - > table_ptr + rvdev - > rsc_offset ;
/* Search for pre-registered carveout */
mem = rproc_find_carveout_by_name ( rproc , " vdev%dvring%d " , rvdev - > index ,
i ) ;
if ( mem ) {
if ( rproc_check_carveout_da ( rproc , mem , rsc - > vring [ i ] . da , size ) )
return - ENOMEM ;
} else {
/* Register carveout in in list */
2019-10-17 12:59:52 +01:00
mem = rproc_mem_entry_init ( dev , NULL , 0 ,
size , rsc - > vring [ i ] . da ,
2018-07-27 15:14:47 +02:00
rproc_alloc_carveout ,
rproc_release_carveout ,
" vdev%dvring%d " ,
rvdev - > index , i ) ;
if ( ! mem ) {
dev_err ( dev , " Can't allocate memory entry structure \n " ) ;
return - ENOMEM ;
}
rproc_add_carveout ( rproc , mem ) ;
2011-10-20 16:52:46 +02:00
}
2012-05-17 14:23:59 +03:00
/*
* Assign an rproc - wide unique index for this vring
* TODO : assign a notifyid for rvdev updates as well
* TODO : support predefined notifyids ( via resource table )
*/
2013-02-27 17:04:39 -08:00
ret = idr_alloc ( & rproc - > notifyids , rvring , 0 , 0 , GFP_KERNEL ) ;
2013-03-06 16:56:48 -06:00
if ( ret < 0 ) {
2013-02-27 17:04:39 -08:00
dev_err ( dev , " idr_alloc failed: %d \n " , ret ) ;
2012-02-13 22:30:39 +01:00
return ret ;
}
2013-02-27 17:04:39 -08:00
notifyid = ret ;
2011-10-20 16:52:46 +02:00
2016-10-19 19:40:10 -07:00
/* Potentially bump max_notifyid */
if ( notifyid > rproc - > max_notifyid )
rproc - > max_notifyid = notifyid ;
2012-05-17 14:23:59 +03:00
rvring - > notifyid = notifyid ;
2011-10-20 16:52:46 +02:00
2018-07-27 15:14:47 +02:00
/* Let the rproc know the notifyid of this vring.*/
2013-02-21 18:15:40 +01:00
rsc - > vring [ i ] . notifyid = notifyid ;
2011-10-20 16:52:46 +02:00
return 0 ;
}
2012-05-17 14:23:59 +03:00
static int
rproc_parse_vring ( struct rproc_vdev * rvdev , struct fw_rsc_vdev * rsc , int i )
2012-02-13 22:30:39 +01:00
{
struct rproc * rproc = rvdev - > rproc ;
remoteproc: maintain a generic child device for each rproc
For each registered rproc, maintain a generic remoteproc device whose
parent is the low level platform-specific device (commonly a pdev, but
it may certainly be any other type of device too).
With this in hand, the resulting device hierarchy might then look like:
omap-rproc.0
|
- remoteproc0 <---- new !
|
- virtio0
|
- virtio1
|
- rpmsg0
|
- rpmsg1
|
- rpmsg2
Where:
- omap-rproc.0 is the low level device that's bound to the
driver which invokes rproc_register()
- remoteproc0 is the result of this patch, and will be added by the
remoteproc framework when rproc_register() is invoked
- virtio0 and virtio1 are vdevs that are registered by remoteproc
when it realizes that they are supported by the firmware
of the physical remote processor represented by omap-rproc.0
- rpmsg0, rpmsg1 and rpmsg2 are rpmsg devices that represent rpmsg
channels, and are registerd by the rpmsg bus when it gets notified
about their existence
Technically, this patch:
- changes 'struct rproc' to contain this generic remoteproc.x device
- creates a new "remoteproc" type, to which this new generic remoteproc.x
device belong to.
- adds a super simple enumeration method for the indices of the
remoteproc.x devices
- updates all dev_* messaging to use the generic remoteproc.x device
instead of the low level platform-specific device
- updates all dma_* allocations to use the parent of remoteproc.x (where
the platform-specific memory pools, most commonly CMA, are to be found)
Adding this generic device has several merits:
- we can now add remoteproc runtime PM support simply by hooking onto the
new "remoteproc" type
- all remoteproc log messages will now carry a common name prefix
instead of having a platform-specific one
- having a device as part of the rproc struct makes it possible to simplify
refcounting (see subsequent patch)
Thanks to Stephen Boyd <sboyd@codeaurora.org> for suggesting and
discussing these ideas in one of the remoteproc review threads and
to Fernando Guzman Lugo <fernando.lugo@ti.com> for trying them out
with the (upcoming) runtime PM support for remoteproc.
Cc: Fernando Guzman Lugo <fernando.lugo@ti.com>
Reviewed-by: Stephen Boyd <sboyd@codeaurora.org>
Signed-off-by: Ohad Ben-Cohen <ohad@wizery.com>
2012-05-30 22:01:25 +03:00
struct device * dev = & rproc - > dev ;
2012-05-17 14:23:59 +03:00
struct fw_rsc_vdev_vring * vring = & rsc - > vring [ i ] ;
struct rproc_vring * rvring = & rvdev - > vring [ i ] ;
2012-02-13 22:30:39 +01:00
2016-08-12 18:42:21 -05:00
dev_dbg ( dev , " vdev rsc: vring%d: da 0x%x, qsz %d, align %d \n " ,
2016-08-12 18:42:20 -05:00
i , vring - > da , vring - > num , vring - > align ) ;
2012-02-13 22:30:39 +01:00
2012-05-17 14:23:59 +03:00
/* verify queue size and vring alignment are sane */
if ( ! vring - > num | | ! vring - > align ) {
dev_err ( dev , " invalid qsz (%d) or alignment (%d) \n " ,
2016-08-12 18:42:20 -05:00
vring - > num , vring - > align ) ;
2012-05-17 14:23:59 +03:00
return - EINVAL ;
2012-02-13 22:30:39 +01:00
}
2012-05-17 14:23:59 +03:00
rvring - > len = vring - > num ;
rvring - > align = vring - > align ;
rvring - > rvdev = rvdev ;
return 0 ;
}
void rproc_free_vring ( struct rproc_vring * rvring )
{
struct rproc * rproc = rvring - > rvdev - > rproc ;
2019-10-04 09:37:36 +02:00
int idx = rvring - rvring - > rvdev - > vring ;
2013-02-21 18:15:40 +01:00
struct fw_rsc_vdev * rsc ;
2012-05-17 14:23:59 +03:00
idr_remove ( & rproc - > notifyids , rvring - > notifyid ) ;
2012-09-18 20:32:45 +02:00
2020-07-14 13:50:34 -06:00
/*
* At this point rproc_stop ( ) has been called and the installed resource
* table in the remote processor memory may no longer be accessible . As
* such and as per rproc_stop ( ) , rproc - > table_ptr points to the cached
* resource table ( rproc - > cached_table ) . The cached resource table is
* only available when a remote processor has been booted by the
* remoteproc core , otherwise it is NULL .
*
* Based on the above , reset the virtio device section in the cached
* resource table only if there is one to work with .
*/
if ( rproc - > table_ptr ) {
rsc = ( void * ) rproc - > table_ptr + rvring - > rvdev - > rsc_offset ;
rsc - > vring [ idx ] . da = 0 ;
rsc - > vring [ idx ] . notifyid = - 1 ;
}
2012-02-13 22:30:39 +01:00
}
2018-06-26 07:11:58 -05:00
static int rproc_vdev_do_start ( struct rproc_subdev * subdev )
2016-10-19 19:40:09 -07:00
{
struct rproc_vdev * rvdev = container_of ( subdev , struct rproc_vdev , subdev ) ;
return rproc_add_virtio_dev ( rvdev , rvdev - > id ) ;
}
2018-06-26 07:11:58 -05:00
static void rproc_vdev_do_stop ( struct rproc_subdev * subdev , bool crashed )
2016-10-19 19:40:09 -07:00
{
struct rproc_vdev * rvdev = container_of ( subdev , struct rproc_vdev , subdev ) ;
2019-01-21 14:55:15 +01:00
int ret ;
2016-10-19 19:40:09 -07:00
2019-01-21 14:55:15 +01:00
ret = device_for_each_child ( & rvdev - > dev , NULL , rproc_remove_virtio_dev ) ;
if ( ret )
dev_warn ( & rvdev - > dev , " can't remove vdev child device: %d \n " , ret ) ;
2016-10-19 19:40:09 -07:00
}
2019-01-10 14:50:49 +01:00
/**
* rproc_rvdev_release ( ) - release the existence of a rvdev
*
* @ dev : the subdevice ' s dev
*/
static void rproc_rvdev_release ( struct device * dev )
{
struct rproc_vdev * rvdev = container_of ( dev , struct rproc_vdev , dev ) ;
of_reserved_mem_device_release ( dev ) ;
kfree ( rvdev ) ;
}
2020-09-17 18:43:40 +02:00
static int copy_dma_range_map ( struct device * to , struct device * from )
{
const struct bus_dma_region * map = from - > dma_range_map , * new_map , * r ;
int num_ranges = 0 ;
if ( ! map )
return 0 ;
for ( r = map ; r - > size ; r + + )
num_ranges + + ;
new_map = kmemdup ( map , array_size ( num_ranges + 1 , sizeof ( * map ) ) ,
GFP_KERNEL ) ;
if ( ! new_map )
return - ENOMEM ;
to - > dma_range_map = new_map ;
return 0 ;
}
2011-10-20 16:52:46 +02:00
/**
2012-02-01 21:56:16 +02:00
* rproc_handle_vdev ( ) - handle a vdev fw resource
2011-10-20 16:52:46 +02:00
* @ rproc : the remote processor
2021-02-24 13:58:25 +08:00
* @ ptr : the vring resource descriptor
2020-02-12 17:19:56 +01:00
* @ offset : offset of the resource entry
2012-02-01 21:56:16 +02:00
* @ avail : size of available data ( for sanity checking the image )
2011-10-20 16:52:46 +02:00
*
2012-02-13 22:30:39 +01:00
* This resource entry requests the host to statically register a virtio
* device ( vdev ) , and setup everything needed to support it . It contains
* everything needed to make it possible : the virtio device id , virtio
* device features , vrings information , virtio config space , etc . . .
*
* Before registering the vdev , the vrings are allocated from non - cacheable
* physically contiguous memory . Currently we only support two vrings per
* remote processor ( temporary limitation ) . We might also want to consider
* doing the vring allocation only later when - > find_vqs ( ) is invoked , and
* then release them upon - > del_vqs ( ) .
*
* Note : @ da is currently not really handled correctly : we dynamically
* allocate it using the DMA API , ignoring requested hard coded addresses ,
* and we don ' t take care of any required IOMMU programming . This is all
* going to be taken care of when the generic iommu - based DMA API will be
* merged . Meanwhile , statically - addressed iommu - based firmware images should
* use RSC_DEVMEM resource entries to map their required @ da to the physical
* address of their base CMA region ( ouch , hacky ! ) .
2011-10-20 16:52:46 +02:00
*
* Returns 0 on success , or an appropriate error code otherwise
*/
2021-02-24 13:58:25 +08:00
static int rproc_handle_vdev ( struct rproc * rproc , void * ptr ,
2016-08-12 18:42:20 -05:00
int offset , int avail )
2011-10-20 16:52:46 +02:00
{
2021-02-24 13:58:25 +08:00
struct fw_rsc_vdev * rsc = ptr ;
remoteproc: maintain a generic child device for each rproc
For each registered rproc, maintain a generic remoteproc device whose
parent is the low level platform-specific device (commonly a pdev, but
it may certainly be any other type of device too).
With this in hand, the resulting device hierarchy might then look like:
omap-rproc.0
|
- remoteproc0 <---- new !
|
- virtio0
|
- virtio1
|
- rpmsg0
|
- rpmsg1
|
- rpmsg2
Where:
- omap-rproc.0 is the low level device that's bound to the
driver which invokes rproc_register()
- remoteproc0 is the result of this patch, and will be added by the
remoteproc framework when rproc_register() is invoked
- virtio0 and virtio1 are vdevs that are registered by remoteproc
when it realizes that they are supported by the firmware
of the physical remote processor represented by omap-rproc.0
- rpmsg0, rpmsg1 and rpmsg2 are rpmsg devices that represent rpmsg
channels, and are registerd by the rpmsg bus when it gets notified
about their existence
Technically, this patch:
- changes 'struct rproc' to contain this generic remoteproc.x device
- creates a new "remoteproc" type, to which this new generic remoteproc.x
device belong to.
- adds a super simple enumeration method for the indices of the
remoteproc.x devices
- updates all dev_* messaging to use the generic remoteproc.x device
instead of the low level platform-specific device
- updates all dma_* allocations to use the parent of remoteproc.x (where
the platform-specific memory pools, most commonly CMA, are to be found)
Adding this generic device has several merits:
- we can now add remoteproc runtime PM support simply by hooking onto the
new "remoteproc" type
- all remoteproc log messages will now carry a common name prefix
instead of having a platform-specific one
- having a device as part of the rproc struct makes it possible to simplify
refcounting (see subsequent patch)
Thanks to Stephen Boyd <sboyd@codeaurora.org> for suggesting and
discussing these ideas in one of the remoteproc review threads and
to Fernando Guzman Lugo <fernando.lugo@ti.com> for trying them out
with the (upcoming) runtime PM support for remoteproc.
Cc: Fernando Guzman Lugo <fernando.lugo@ti.com>
Reviewed-by: Stephen Boyd <sboyd@codeaurora.org>
Signed-off-by: Ohad Ben-Cohen <ohad@wizery.com>
2012-05-30 22:01:25 +03:00
struct device * dev = & rproc - > dev ;
2012-02-13 22:30:39 +01:00
struct rproc_vdev * rvdev ;
int i , ret ;
2019-01-10 14:50:49 +01:00
char name [ 16 ] ;
2011-10-20 16:52:46 +02:00
2012-02-01 21:56:16 +02:00
/* make sure resource isn't truncated */
2019-08-30 10:14:06 -05:00
if ( struct_size ( rsc , vring , rsc - > num_of_vrings ) + rsc - > config_len >
avail ) {
remoteproc: maintain a generic child device for each rproc
For each registered rproc, maintain a generic remoteproc device whose
parent is the low level platform-specific device (commonly a pdev, but
it may certainly be any other type of device too).
With this in hand, the resulting device hierarchy might then look like:
omap-rproc.0
|
- remoteproc0 <---- new !
|
- virtio0
|
- virtio1
|
- rpmsg0
|
- rpmsg1
|
- rpmsg2
Where:
- omap-rproc.0 is the low level device that's bound to the
driver which invokes rproc_register()
- remoteproc0 is the result of this patch, and will be added by the
remoteproc framework when rproc_register() is invoked
- virtio0 and virtio1 are vdevs that are registered by remoteproc
when it realizes that they are supported by the firmware
of the physical remote processor represented by omap-rproc.0
- rpmsg0, rpmsg1 and rpmsg2 are rpmsg devices that represent rpmsg
channels, and are registerd by the rpmsg bus when it gets notified
about their existence
Technically, this patch:
- changes 'struct rproc' to contain this generic remoteproc.x device
- creates a new "remoteproc" type, to which this new generic remoteproc.x
device belong to.
- adds a super simple enumeration method for the indices of the
remoteproc.x devices
- updates all dev_* messaging to use the generic remoteproc.x device
instead of the low level platform-specific device
- updates all dma_* allocations to use the parent of remoteproc.x (where
the platform-specific memory pools, most commonly CMA, are to be found)
Adding this generic device has several merits:
- we can now add remoteproc runtime PM support simply by hooking onto the
new "remoteproc" type
- all remoteproc log messages will now carry a common name prefix
instead of having a platform-specific one
- having a device as part of the rproc struct makes it possible to simplify
refcounting (see subsequent patch)
Thanks to Stephen Boyd <sboyd@codeaurora.org> for suggesting and
discussing these ideas in one of the remoteproc review threads and
to Fernando Guzman Lugo <fernando.lugo@ti.com> for trying them out
with the (upcoming) runtime PM support for remoteproc.
Cc: Fernando Guzman Lugo <fernando.lugo@ti.com>
Reviewed-by: Stephen Boyd <sboyd@codeaurora.org>
Signed-off-by: Ohad Ben-Cohen <ohad@wizery.com>
2012-05-30 22:01:25 +03:00
dev_err ( dev , " vdev rsc is truncated \n " ) ;
2011-10-20 16:52:46 +02:00
return - EINVAL ;
}
2012-02-01 21:56:16 +02:00
/* make sure reserved bytes are zeroes */
if ( rsc - > reserved [ 0 ] | | rsc - > reserved [ 1 ] ) {
dev_err ( dev , " vdev rsc has non zero reserved bytes \n " ) ;
2011-10-20 16:52:46 +02:00
return - EINVAL ;
}
2016-08-12 18:42:21 -05:00
dev_dbg ( dev , " vdev rsc: id %d, dfeatures 0x%x, cfg len %d, %d vrings \n " ,
2012-02-01 21:56:16 +02:00
rsc - > id , rsc - > dfeatures , rsc - > config_len , rsc - > num_of_vrings ) ;
2012-02-13 22:30:39 +01:00
/* we currently support only two vrings per rvdev */
if ( rsc - > num_of_vrings > ARRAY_SIZE ( rvdev - > vring ) ) {
2012-02-01 21:56:16 +02:00
dev_err ( dev , " too many vrings: %d \n " , rsc - > num_of_vrings ) ;
2011-10-20 16:52:46 +02:00
return - EINVAL ;
}
2016-08-12 18:42:18 -05:00
rvdev = kzalloc ( sizeof ( * rvdev ) , GFP_KERNEL ) ;
2012-02-13 22:30:39 +01:00
if ( ! rvdev )
return - ENOMEM ;
2011-10-20 16:52:46 +02:00
2016-10-19 19:40:06 -07:00
kref_init ( & rvdev - > refcount ) ;
2016-10-19 19:40:09 -07:00
rvdev - > id = rsc - > id ;
2012-02-13 22:30:39 +01:00
rvdev - > rproc = rproc ;
2018-07-27 15:14:47 +02:00
rvdev - > index = rproc - > nb_vdev + + ;
2011-10-20 16:52:46 +02:00
2019-01-10 14:50:49 +01:00
/* Initialise vdev subdevice */
snprintf ( name , sizeof ( name ) , " vdev%dbuffer " , rvdev - > index ) ;
2020-04-20 11:06:00 -05:00
rvdev - > dev . parent = & rproc - > dev ;
2020-09-17 18:43:40 +02:00
ret = copy_dma_range_map ( & rvdev - > dev , rproc - > dev . parent ) ;
if ( ret )
return ret ;
2019-01-10 14:50:49 +01:00
rvdev - > dev . release = rproc_rvdev_release ;
dev_set_name ( & rvdev - > dev , " %s#%s " , dev_name ( rvdev - > dev . parent ) , name ) ;
dev_set_drvdata ( & rvdev - > dev , rvdev ) ;
ret = device_register ( & rvdev - > dev ) ;
if ( ret ) {
put_device ( & rvdev - > dev ) ;
return ret ;
}
/* Make device dma capable by inheriting from parent's capabilities */
set_dma_ops ( & rvdev - > dev , get_dma_ops ( rproc - > dev . parent ) ) ;
ret = dma_coerce_mask_and_coherent ( & rvdev - > dev ,
dma_get_mask ( rproc - > dev . parent ) ) ;
if ( ret ) {
dev_warn ( dev ,
" Failed to set DMA mask %llx. Trying to continue... %x \n " ,
dma_get_mask ( rproc - > dev . parent ) , ret ) ;
}
2012-05-17 14:23:59 +03:00
/* parse the vrings */
2012-02-13 22:30:39 +01:00
for ( i = 0 ; i < rsc - > num_of_vrings ; i + + ) {
2012-05-17 14:23:59 +03:00
ret = rproc_parse_vring ( rvdev , rsc , i ) ;
2012-02-13 22:30:39 +01:00
if ( ret )
2012-05-17 14:23:59 +03:00
goto free_rvdev ;
2012-02-13 22:30:39 +01:00
}
2011-10-20 16:52:46 +02:00
2013-04-07 14:06:07 +03:00
/* remember the resource offset*/
rvdev - > rsc_offset = offset ;
2012-02-01 21:56:16 +02:00
2016-10-19 19:40:07 -07:00
/* allocate the vring resources */
for ( i = 0 ; i < rsc - > num_of_vrings ; i + + ) {
ret = rproc_alloc_vring ( rvdev , i ) ;
if ( ret )
goto unwind_vring_allocations ;
}
2012-02-13 22:30:39 +01:00
list_add_tail ( & rvdev - > node , & rproc - > rvdevs ) ;
2012-02-01 21:56:16 +02:00
2018-06-26 07:11:58 -05:00
rvdev - > subdev . start = rproc_vdev_do_start ;
rvdev - > subdev . stop = rproc_vdev_do_stop ;
2018-06-26 07:11:57 -05:00
rproc_add_subdev ( rproc , & rvdev - > subdev ) ;
2011-10-20 16:52:46 +02:00
return 0 ;
2012-02-13 22:30:39 +01:00
2016-10-19 19:40:07 -07:00
unwind_vring_allocations :
for ( i - - ; i > = 0 ; i - - )
rproc_free_vring ( & rvdev - > vring [ i ] ) ;
2012-05-17 14:23:59 +03:00
free_rvdev :
2019-01-10 14:50:49 +01:00
device_unregister ( & rvdev - > dev ) ;
2012-02-13 22:30:39 +01:00
return ret ;
2011-10-20 16:52:46 +02:00
}
2016-10-19 19:40:06 -07:00
void rproc_vdev_release ( struct kref * ref )
{
struct rproc_vdev * rvdev = container_of ( ref , struct rproc_vdev , refcount ) ;
2016-10-19 19:40:07 -07:00
struct rproc_vring * rvring ;
2016-10-19 19:40:09 -07:00
struct rproc * rproc = rvdev - > rproc ;
2016-10-19 19:40:07 -07:00
int id ;
for ( id = 0 ; id < ARRAY_SIZE ( rvdev - > vring ) ; id + + ) {
rvring = & rvdev - > vring [ id ] ;
rproc_free_vring ( rvring ) ;
}
2016-10-19 19:40:06 -07:00
2016-10-19 19:40:09 -07:00
rproc_remove_subdev ( rproc , & rvdev - > subdev ) ;
2016-10-19 19:40:06 -07:00
list_del ( & rvdev - > node ) ;
2019-01-10 14:50:49 +01:00
device_unregister ( & rvdev - > dev ) ;
2016-10-19 19:40:06 -07:00
}
2011-10-20 16:52:46 +02:00
/**
* rproc_handle_trace ( ) - handle a shared trace buffer resource
* @ rproc : the remote processor
2021-02-24 13:58:25 +08:00
* @ ptr : the trace resource descriptor
2020-02-12 17:19:56 +01:00
* @ offset : offset of the resource entry
2012-02-01 21:56:16 +02:00
* @ avail : size of available data ( for sanity checking the image )
2011-10-20 16:52:46 +02:00
*
* In case the remote processor dumps trace logs into memory ,
* export it via debugfs .
*
* Currently , the ' da ' member of @ rsc should contain the device address
* where the remote processor is dumping the traces . Later we could also
* support dynamically allocating this address using the generic
* DMA API ( but currently there isn ' t a use case for that ) .
*
* Returns 0 on success , or an appropriate error code otherwise
*/
2021-02-24 13:58:25 +08:00
static int rproc_handle_trace ( struct rproc * rproc , void * ptr ,
2016-08-12 18:42:20 -05:00
int offset , int avail )
2011-10-20 16:52:46 +02:00
{
2021-02-24 13:58:25 +08:00
struct fw_rsc_trace * rsc = ptr ;
2019-01-10 14:49:10 +01:00
struct rproc_debug_trace * trace ;
remoteproc: maintain a generic child device for each rproc
For each registered rproc, maintain a generic remoteproc device whose
parent is the low level platform-specific device (commonly a pdev, but
it may certainly be any other type of device too).
With this in hand, the resulting device hierarchy might then look like:
omap-rproc.0
|
- remoteproc0 <---- new !
|
- virtio0
|
- virtio1
|
- rpmsg0
|
- rpmsg1
|
- rpmsg2
Where:
- omap-rproc.0 is the low level device that's bound to the
driver which invokes rproc_register()
- remoteproc0 is the result of this patch, and will be added by the
remoteproc framework when rproc_register() is invoked
- virtio0 and virtio1 are vdevs that are registered by remoteproc
when it realizes that they are supported by the firmware
of the physical remote processor represented by omap-rproc.0
- rpmsg0, rpmsg1 and rpmsg2 are rpmsg devices that represent rpmsg
channels, and are registerd by the rpmsg bus when it gets notified
about their existence
Technically, this patch:
- changes 'struct rproc' to contain this generic remoteproc.x device
- creates a new "remoteproc" type, to which this new generic remoteproc.x
device belong to.
- adds a super simple enumeration method for the indices of the
remoteproc.x devices
- updates all dev_* messaging to use the generic remoteproc.x device
instead of the low level platform-specific device
- updates all dma_* allocations to use the parent of remoteproc.x (where
the platform-specific memory pools, most commonly CMA, are to be found)
Adding this generic device has several merits:
- we can now add remoteproc runtime PM support simply by hooking onto the
new "remoteproc" type
- all remoteproc log messages will now carry a common name prefix
instead of having a platform-specific one
- having a device as part of the rproc struct makes it possible to simplify
refcounting (see subsequent patch)
Thanks to Stephen Boyd <sboyd@codeaurora.org> for suggesting and
discussing these ideas in one of the remoteproc review threads and
to Fernando Guzman Lugo <fernando.lugo@ti.com> for trying them out
with the (upcoming) runtime PM support for remoteproc.
Cc: Fernando Guzman Lugo <fernando.lugo@ti.com>
Reviewed-by: Stephen Boyd <sboyd@codeaurora.org>
Signed-off-by: Ohad Ben-Cohen <ohad@wizery.com>
2012-05-30 22:01:25 +03:00
struct device * dev = & rproc - > dev ;
2011-10-20 16:52:46 +02:00
char name [ 15 ] ;
2012-02-01 21:56:16 +02:00
if ( sizeof ( * rsc ) > avail ) {
remoteproc: maintain a generic child device for each rproc
For each registered rproc, maintain a generic remoteproc device whose
parent is the low level platform-specific device (commonly a pdev, but
it may certainly be any other type of device too).
With this in hand, the resulting device hierarchy might then look like:
omap-rproc.0
|
- remoteproc0 <---- new !
|
- virtio0
|
- virtio1
|
- rpmsg0
|
- rpmsg1
|
- rpmsg2
Where:
- omap-rproc.0 is the low level device that's bound to the
driver which invokes rproc_register()
- remoteproc0 is the result of this patch, and will be added by the
remoteproc framework when rproc_register() is invoked
- virtio0 and virtio1 are vdevs that are registered by remoteproc
when it realizes that they are supported by the firmware
of the physical remote processor represented by omap-rproc.0
- rpmsg0, rpmsg1 and rpmsg2 are rpmsg devices that represent rpmsg
channels, and are registerd by the rpmsg bus when it gets notified
about their existence
Technically, this patch:
- changes 'struct rproc' to contain this generic remoteproc.x device
- creates a new "remoteproc" type, to which this new generic remoteproc.x
device belong to.
- adds a super simple enumeration method for the indices of the
remoteproc.x devices
- updates all dev_* messaging to use the generic remoteproc.x device
instead of the low level platform-specific device
- updates all dma_* allocations to use the parent of remoteproc.x (where
the platform-specific memory pools, most commonly CMA, are to be found)
Adding this generic device has several merits:
- we can now add remoteproc runtime PM support simply by hooking onto the
new "remoteproc" type
- all remoteproc log messages will now carry a common name prefix
instead of having a platform-specific one
- having a device as part of the rproc struct makes it possible to simplify
refcounting (see subsequent patch)
Thanks to Stephen Boyd <sboyd@codeaurora.org> for suggesting and
discussing these ideas in one of the remoteproc review threads and
to Fernando Guzman Lugo <fernando.lugo@ti.com> for trying them out
with the (upcoming) runtime PM support for remoteproc.
Cc: Fernando Guzman Lugo <fernando.lugo@ti.com>
Reviewed-by: Stephen Boyd <sboyd@codeaurora.org>
Signed-off-by: Ohad Ben-Cohen <ohad@wizery.com>
2012-05-30 22:01:25 +03:00
dev_err ( dev , " trace rsc is truncated \n " ) ;
2012-02-01 21:56:16 +02:00
return - EINVAL ;
}
/* make sure reserved bytes are zeroes */
if ( rsc - > reserved ) {
dev_err ( dev , " trace rsc has non zero reserved bytes \n " ) ;
return - EINVAL ;
}
2011-10-20 16:52:46 +02:00
trace = kzalloc ( sizeof ( * trace ) , GFP_KERNEL ) ;
2015-02-27 17:18:23 -06:00
if ( ! trace )
2011-10-20 16:52:46 +02:00
return - ENOMEM ;
/* set the trace buffer dma properties */
2019-01-10 14:49:10 +01:00
trace - > trace_mem . len = rsc - > len ;
trace - > trace_mem . da = rsc - > da ;
/* set pointer on rproc device */
trace - > rproc = rproc ;
2011-10-20 16:52:46 +02:00
/* make sure snprintf always null terminates, even if truncating */
snprintf ( name , sizeof ( name ) , " trace%d " , rproc - > num_traces ) ;
/* create the debugfs entry */
2019-01-10 14:49:10 +01:00
trace - > tfile = rproc_create_trace_file ( name , rproc , trace ) ;
if ( ! trace - > tfile ) {
2011-10-20 16:52:46 +02:00
kfree ( trace ) ;
return - EINVAL ;
}
list_add_tail ( & trace - > node , & rproc - > traces ) ;
rproc - > num_traces + + ;
2019-01-10 14:49:10 +01:00
dev_dbg ( dev , " %s added: da 0x%x, len 0x%x \n " ,
name , rsc - > da , rsc - > len ) ;
2011-10-20 16:52:46 +02:00
return 0 ;
}
/**
* rproc_handle_devmem ( ) - handle devmem resource entry
* @ rproc : remote processor handle
2021-02-24 13:58:25 +08:00
* @ ptr : the devmem resource entry
2020-02-12 17:19:56 +01:00
* @ offset : offset of the resource entry
2012-02-01 21:56:16 +02:00
* @ avail : size of available data ( for sanity checking the image )
2011-10-20 16:52:46 +02:00
*
* Remote processors commonly need to access certain on - chip peripherals .
*
* Some of these remote processors access memory via an iommu device ,
* and might require us to configure their iommu before they can access
* the on - chip peripherals they need .
*
* This resource entry is a request to map such a peripheral device .
*
* These devmem entries will contain the physical address of the device in
* the ' pa ' member . If a specific device address is expected , then ' da ' will
* contain it ( currently this is the only use case supported ) . ' len ' will
* contain the size of the physical region we need to map .
*
* Currently we just " trust " those devmem entries to contain valid physical
* addresses , but this is going to change : we want the implementations to
* tell us ranges of physical addresses the firmware is allowed to request ,
* and not allow firmwares to request access to physical addresses that
* are outside those ranges .
*/
2021-02-24 13:58:25 +08:00
static int rproc_handle_devmem ( struct rproc * rproc , void * ptr ,
2016-08-12 18:42:20 -05:00
int offset , int avail )
2011-10-20 16:52:46 +02:00
{
2021-02-24 13:58:25 +08:00
struct fw_rsc_devmem * rsc = ptr ;
2011-10-20 16:52:46 +02:00
struct rproc_mem_entry * mapping ;
remoteproc: maintain a generic child device for each rproc
For each registered rproc, maintain a generic remoteproc device whose
parent is the low level platform-specific device (commonly a pdev, but
it may certainly be any other type of device too).
With this in hand, the resulting device hierarchy might then look like:
omap-rproc.0
|
- remoteproc0 <---- new !
|
- virtio0
|
- virtio1
|
- rpmsg0
|
- rpmsg1
|
- rpmsg2
Where:
- omap-rproc.0 is the low level device that's bound to the
driver which invokes rproc_register()
- remoteproc0 is the result of this patch, and will be added by the
remoteproc framework when rproc_register() is invoked
- virtio0 and virtio1 are vdevs that are registered by remoteproc
when it realizes that they are supported by the firmware
of the physical remote processor represented by omap-rproc.0
- rpmsg0, rpmsg1 and rpmsg2 are rpmsg devices that represent rpmsg
channels, and are registerd by the rpmsg bus when it gets notified
about their existence
Technically, this patch:
- changes 'struct rproc' to contain this generic remoteproc.x device
- creates a new "remoteproc" type, to which this new generic remoteproc.x
device belong to.
- adds a super simple enumeration method for the indices of the
remoteproc.x devices
- updates all dev_* messaging to use the generic remoteproc.x device
instead of the low level platform-specific device
- updates all dma_* allocations to use the parent of remoteproc.x (where
the platform-specific memory pools, most commonly CMA, are to be found)
Adding this generic device has several merits:
- we can now add remoteproc runtime PM support simply by hooking onto the
new "remoteproc" type
- all remoteproc log messages will now carry a common name prefix
instead of having a platform-specific one
- having a device as part of the rproc struct makes it possible to simplify
refcounting (see subsequent patch)
Thanks to Stephen Boyd <sboyd@codeaurora.org> for suggesting and
discussing these ideas in one of the remoteproc review threads and
to Fernando Guzman Lugo <fernando.lugo@ti.com> for trying them out
with the (upcoming) runtime PM support for remoteproc.
Cc: Fernando Guzman Lugo <fernando.lugo@ti.com>
Reviewed-by: Stephen Boyd <sboyd@codeaurora.org>
Signed-off-by: Ohad Ben-Cohen <ohad@wizery.com>
2012-05-30 22:01:25 +03:00
struct device * dev = & rproc - > dev ;
2011-10-20 16:52:46 +02:00
int ret ;
/* no point in handling this resource without a valid iommu domain */
if ( ! rproc - > domain )
return - EINVAL ;
2012-02-01 21:56:16 +02:00
if ( sizeof ( * rsc ) > avail ) {
remoteproc: maintain a generic child device for each rproc
For each registered rproc, maintain a generic remoteproc device whose
parent is the low level platform-specific device (commonly a pdev, but
it may certainly be any other type of device too).
With this in hand, the resulting device hierarchy might then look like:
omap-rproc.0
|
- remoteproc0 <---- new !
|
- virtio0
|
- virtio1
|
- rpmsg0
|
- rpmsg1
|
- rpmsg2
Where:
- omap-rproc.0 is the low level device that's bound to the
driver which invokes rproc_register()
- remoteproc0 is the result of this patch, and will be added by the
remoteproc framework when rproc_register() is invoked
- virtio0 and virtio1 are vdevs that are registered by remoteproc
when it realizes that they are supported by the firmware
of the physical remote processor represented by omap-rproc.0
- rpmsg0, rpmsg1 and rpmsg2 are rpmsg devices that represent rpmsg
channels, and are registerd by the rpmsg bus when it gets notified
about their existence
Technically, this patch:
- changes 'struct rproc' to contain this generic remoteproc.x device
- creates a new "remoteproc" type, to which this new generic remoteproc.x
device belong to.
- adds a super simple enumeration method for the indices of the
remoteproc.x devices
- updates all dev_* messaging to use the generic remoteproc.x device
instead of the low level platform-specific device
- updates all dma_* allocations to use the parent of remoteproc.x (where
the platform-specific memory pools, most commonly CMA, are to be found)
Adding this generic device has several merits:
- we can now add remoteproc runtime PM support simply by hooking onto the
new "remoteproc" type
- all remoteproc log messages will now carry a common name prefix
instead of having a platform-specific one
- having a device as part of the rproc struct makes it possible to simplify
refcounting (see subsequent patch)
Thanks to Stephen Boyd <sboyd@codeaurora.org> for suggesting and
discussing these ideas in one of the remoteproc review threads and
to Fernando Guzman Lugo <fernando.lugo@ti.com> for trying them out
with the (upcoming) runtime PM support for remoteproc.
Cc: Fernando Guzman Lugo <fernando.lugo@ti.com>
Reviewed-by: Stephen Boyd <sboyd@codeaurora.org>
Signed-off-by: Ohad Ben-Cohen <ohad@wizery.com>
2012-05-30 22:01:25 +03:00
dev_err ( dev , " devmem rsc is truncated \n " ) ;
2012-02-01 21:56:16 +02:00
return - EINVAL ;
}
/* make sure reserved bytes are zeroes */
if ( rsc - > reserved ) {
remoteproc: maintain a generic child device for each rproc
For each registered rproc, maintain a generic remoteproc device whose
parent is the low level platform-specific device (commonly a pdev, but
it may certainly be any other type of device too).
With this in hand, the resulting device hierarchy might then look like:
omap-rproc.0
|
- remoteproc0 <---- new !
|
- virtio0
|
- virtio1
|
- rpmsg0
|
- rpmsg1
|
- rpmsg2
Where:
- omap-rproc.0 is the low level device that's bound to the
driver which invokes rproc_register()
- remoteproc0 is the result of this patch, and will be added by the
remoteproc framework when rproc_register() is invoked
- virtio0 and virtio1 are vdevs that are registered by remoteproc
when it realizes that they are supported by the firmware
of the physical remote processor represented by omap-rproc.0
- rpmsg0, rpmsg1 and rpmsg2 are rpmsg devices that represent rpmsg
channels, and are registerd by the rpmsg bus when it gets notified
about their existence
Technically, this patch:
- changes 'struct rproc' to contain this generic remoteproc.x device
- creates a new "remoteproc" type, to which this new generic remoteproc.x
device belong to.
- adds a super simple enumeration method for the indices of the
remoteproc.x devices
- updates all dev_* messaging to use the generic remoteproc.x device
instead of the low level platform-specific device
- updates all dma_* allocations to use the parent of remoteproc.x (where
the platform-specific memory pools, most commonly CMA, are to be found)
Adding this generic device has several merits:
- we can now add remoteproc runtime PM support simply by hooking onto the
new "remoteproc" type
- all remoteproc log messages will now carry a common name prefix
instead of having a platform-specific one
- having a device as part of the rproc struct makes it possible to simplify
refcounting (see subsequent patch)
Thanks to Stephen Boyd <sboyd@codeaurora.org> for suggesting and
discussing these ideas in one of the remoteproc review threads and
to Fernando Guzman Lugo <fernando.lugo@ti.com> for trying them out
with the (upcoming) runtime PM support for remoteproc.
Cc: Fernando Guzman Lugo <fernando.lugo@ti.com>
Reviewed-by: Stephen Boyd <sboyd@codeaurora.org>
Signed-off-by: Ohad Ben-Cohen <ohad@wizery.com>
2012-05-30 22:01:25 +03:00
dev_err ( dev , " devmem rsc has non zero reserved bytes \n " ) ;
2012-02-01 21:56:16 +02:00
return - EINVAL ;
}
2011-10-20 16:52:46 +02:00
mapping = kzalloc ( sizeof ( * mapping ) , GFP_KERNEL ) ;
2015-02-27 17:18:23 -06:00
if ( ! mapping )
2011-10-20 16:52:46 +02:00
return - ENOMEM ;
ret = iommu_map ( rproc - > domain , rsc - > da , rsc - > pa , rsc - > len , rsc - > flags ) ;
if ( ret ) {
remoteproc: maintain a generic child device for each rproc
For each registered rproc, maintain a generic remoteproc device whose
parent is the low level platform-specific device (commonly a pdev, but
it may certainly be any other type of device too).
With this in hand, the resulting device hierarchy might then look like:
omap-rproc.0
|
- remoteproc0 <---- new !
|
- virtio0
|
- virtio1
|
- rpmsg0
|
- rpmsg1
|
- rpmsg2
Where:
- omap-rproc.0 is the low level device that's bound to the
driver which invokes rproc_register()
- remoteproc0 is the result of this patch, and will be added by the
remoteproc framework when rproc_register() is invoked
- virtio0 and virtio1 are vdevs that are registered by remoteproc
when it realizes that they are supported by the firmware
of the physical remote processor represented by omap-rproc.0
- rpmsg0, rpmsg1 and rpmsg2 are rpmsg devices that represent rpmsg
channels, and are registerd by the rpmsg bus when it gets notified
about their existence
Technically, this patch:
- changes 'struct rproc' to contain this generic remoteproc.x device
- creates a new "remoteproc" type, to which this new generic remoteproc.x
device belong to.
- adds a super simple enumeration method for the indices of the
remoteproc.x devices
- updates all dev_* messaging to use the generic remoteproc.x device
instead of the low level platform-specific device
- updates all dma_* allocations to use the parent of remoteproc.x (where
the platform-specific memory pools, most commonly CMA, are to be found)
Adding this generic device has several merits:
- we can now add remoteproc runtime PM support simply by hooking onto the
new "remoteproc" type
- all remoteproc log messages will now carry a common name prefix
instead of having a platform-specific one
- having a device as part of the rproc struct makes it possible to simplify
refcounting (see subsequent patch)
Thanks to Stephen Boyd <sboyd@codeaurora.org> for suggesting and
discussing these ideas in one of the remoteproc review threads and
to Fernando Guzman Lugo <fernando.lugo@ti.com> for trying them out
with the (upcoming) runtime PM support for remoteproc.
Cc: Fernando Guzman Lugo <fernando.lugo@ti.com>
Reviewed-by: Stephen Boyd <sboyd@codeaurora.org>
Signed-off-by: Ohad Ben-Cohen <ohad@wizery.com>
2012-05-30 22:01:25 +03:00
dev_err ( dev , " failed to map devmem: %d \n " , ret ) ;
2011-10-20 16:52:46 +02:00
goto out ;
}
/*
* We ' ll need this info later when we ' ll want to unmap everything
* ( e . g . on shutdown ) .
*
* We can ' t trust the remote processor not to change the resource
* table , so we must maintain this info independently .
*/
mapping - > da = rsc - > da ;
mapping - > len = rsc - > len ;
list_add_tail ( & mapping - > node , & rproc - > mappings ) ;
remoteproc: maintain a generic child device for each rproc
For each registered rproc, maintain a generic remoteproc device whose
parent is the low level platform-specific device (commonly a pdev, but
it may certainly be any other type of device too).
With this in hand, the resulting device hierarchy might then look like:
omap-rproc.0
|
- remoteproc0 <---- new !
|
- virtio0
|
- virtio1
|
- rpmsg0
|
- rpmsg1
|
- rpmsg2
Where:
- omap-rproc.0 is the low level device that's bound to the
driver which invokes rproc_register()
- remoteproc0 is the result of this patch, and will be added by the
remoteproc framework when rproc_register() is invoked
- virtio0 and virtio1 are vdevs that are registered by remoteproc
when it realizes that they are supported by the firmware
of the physical remote processor represented by omap-rproc.0
- rpmsg0, rpmsg1 and rpmsg2 are rpmsg devices that represent rpmsg
channels, and are registerd by the rpmsg bus when it gets notified
about their existence
Technically, this patch:
- changes 'struct rproc' to contain this generic remoteproc.x device
- creates a new "remoteproc" type, to which this new generic remoteproc.x
device belong to.
- adds a super simple enumeration method for the indices of the
remoteproc.x devices
- updates all dev_* messaging to use the generic remoteproc.x device
instead of the low level platform-specific device
- updates all dma_* allocations to use the parent of remoteproc.x (where
the platform-specific memory pools, most commonly CMA, are to be found)
Adding this generic device has several merits:
- we can now add remoteproc runtime PM support simply by hooking onto the
new "remoteproc" type
- all remoteproc log messages will now carry a common name prefix
instead of having a platform-specific one
- having a device as part of the rproc struct makes it possible to simplify
refcounting (see subsequent patch)
Thanks to Stephen Boyd <sboyd@codeaurora.org> for suggesting and
discussing these ideas in one of the remoteproc review threads and
to Fernando Guzman Lugo <fernando.lugo@ti.com> for trying them out
with the (upcoming) runtime PM support for remoteproc.
Cc: Fernando Guzman Lugo <fernando.lugo@ti.com>
Reviewed-by: Stephen Boyd <sboyd@codeaurora.org>
Signed-off-by: Ohad Ben-Cohen <ohad@wizery.com>
2012-05-30 22:01:25 +03:00
dev_dbg ( dev , " mapped devmem pa 0x%x, da 0x%x, len 0x%x \n " ,
2016-08-12 18:42:20 -05:00
rsc - > pa , rsc - > da , rsc - > len ) ;
2011-10-20 16:52:46 +02:00
return 0 ;
out :
kfree ( mapping ) ;
return ret ;
}
2018-07-27 15:14:38 +02:00
/**
2018-07-27 15:14:43 +02:00
* rproc_alloc_carveout ( ) - allocated specified carveout
2018-07-27 15:14:38 +02:00
* @ rproc : rproc handle
2018-07-27 15:14:43 +02:00
* @ mem : the memory entry to allocate
2011-10-20 16:52:46 +02:00
*
2018-07-27 15:14:43 +02:00
* This function allocate specified memory entry @ mem using
* dma_alloc_coherent ( ) as default allocator
2011-10-20 16:52:46 +02:00
*/
2018-07-27 15:14:43 +02:00
static int rproc_alloc_carveout ( struct rproc * rproc ,
struct rproc_mem_entry * mem )
2011-10-20 16:52:46 +02:00
{
2018-07-27 15:14:43 +02:00
struct rproc_mem_entry * mapping = NULL ;
remoteproc: maintain a generic child device for each rproc
For each registered rproc, maintain a generic remoteproc device whose
parent is the low level platform-specific device (commonly a pdev, but
it may certainly be any other type of device too).
With this in hand, the resulting device hierarchy might then look like:
omap-rproc.0
|
- remoteproc0 <---- new !
|
- virtio0
|
- virtio1
|
- rpmsg0
|
- rpmsg1
|
- rpmsg2
Where:
- omap-rproc.0 is the low level device that's bound to the
driver which invokes rproc_register()
- remoteproc0 is the result of this patch, and will be added by the
remoteproc framework when rproc_register() is invoked
- virtio0 and virtio1 are vdevs that are registered by remoteproc
when it realizes that they are supported by the firmware
of the physical remote processor represented by omap-rproc.0
- rpmsg0, rpmsg1 and rpmsg2 are rpmsg devices that represent rpmsg
channels, and are registerd by the rpmsg bus when it gets notified
about their existence
Technically, this patch:
- changes 'struct rproc' to contain this generic remoteproc.x device
- creates a new "remoteproc" type, to which this new generic remoteproc.x
device belong to.
- adds a super simple enumeration method for the indices of the
remoteproc.x devices
- updates all dev_* messaging to use the generic remoteproc.x device
instead of the low level platform-specific device
- updates all dma_* allocations to use the parent of remoteproc.x (where
the platform-specific memory pools, most commonly CMA, are to be found)
Adding this generic device has several merits:
- we can now add remoteproc runtime PM support simply by hooking onto the
new "remoteproc" type
- all remoteproc log messages will now carry a common name prefix
instead of having a platform-specific one
- having a device as part of the rproc struct makes it possible to simplify
refcounting (see subsequent patch)
Thanks to Stephen Boyd <sboyd@codeaurora.org> for suggesting and
discussing these ideas in one of the remoteproc review threads and
to Fernando Guzman Lugo <fernando.lugo@ti.com> for trying them out
with the (upcoming) runtime PM support for remoteproc.
Cc: Fernando Guzman Lugo <fernando.lugo@ti.com>
Reviewed-by: Stephen Boyd <sboyd@codeaurora.org>
Signed-off-by: Ohad Ben-Cohen <ohad@wizery.com>
2012-05-30 22:01:25 +03:00
struct device * dev = & rproc - > dev ;
2011-10-20 16:52:46 +02:00
dma_addr_t dma ;
void * va ;
int ret ;
2018-07-27 15:14:43 +02:00
va = dma_alloc_coherent ( dev - > parent , mem - > len , & dma , GFP_KERNEL ) ;
2011-10-20 16:52:46 +02:00
if ( ! va ) {
2016-08-04 10:21:45 +01:00
dev_err ( dev - > parent ,
2020-03-02 10:38:56 +01:00
" failed to allocate dma memory: len 0x%zx \n " ,
mem - > len ) ;
2018-07-27 15:14:40 +02:00
return - ENOMEM ;
2011-10-20 16:52:46 +02:00
}
2020-03-02 10:38:56 +01:00
dev_dbg ( dev , " carveout va %pK, dma %pad, len 0x%zx \n " ,
2018-07-27 15:14:43 +02:00
va , & dma , mem - > len ) ;
2011-10-20 16:52:46 +02:00
2019-01-10 14:49:09 +01:00
if ( mem - > da ! = FW_RSC_ADDR_ANY & & ! rproc - > domain ) {
/*
* Check requested da is equal to dma address
* and print a warn message in case of missalignment .
* Don ' t stop rproc_start sequence as coprocessor may
* build pa to da translation on its side .
*/
if ( mem - > da ! = ( u32 ) dma )
dev_warn ( dev - > parent ,
" Allocated carveout doesn't fit device address request \n " ) ;
}
2011-10-20 16:52:46 +02:00
/*
* Ok , this is non - standard .
*
* Sometimes we can ' t rely on the generic iommu - based DMA API
* to dynamically allocate the device address and then set the IOMMU
* tables accordingly , because some remote processors might
* _require_ us to use hard coded device addresses that their
* firmware was compiled with .
*
* In this case , we must use the IOMMU API directly and map
* the memory to the device address as expected by the remote
* processor .
*
* Obviously such remote processor devices should not be configured
* to use the iommu - based DMA API : we expect ' dma ' to contain the
* physical address in this case .
*/
2019-01-10 14:49:09 +01:00
if ( mem - > da ! = FW_RSC_ADDR_ANY & & rproc - > domain ) {
2012-09-25 10:01:56 +03:00
mapping = kzalloc ( sizeof ( * mapping ) , GFP_KERNEL ) ;
if ( ! mapping ) {
ret = - ENOMEM ;
goto dma_free ;
}
2018-07-27 15:14:43 +02:00
ret = iommu_map ( rproc - > domain , mem - > da , dma , mem - > len ,
mem - > flags ) ;
2011-10-20 16:52:46 +02:00
if ( ret ) {
dev_err ( dev , " iommu_map failed: %d \n " , ret ) ;
2012-09-25 10:01:56 +03:00
goto free_mapping ;
2011-10-20 16:52:46 +02:00
}
/*
* We ' ll need this info later when we ' ll want to unmap
* everything ( e . g . on shutdown ) .
*
* We can ' t trust the remote processor not to change the
* resource table , so we must maintain this info independently .
*/
2018-07-27 15:14:43 +02:00
mapping - > da = mem - > da ;
mapping - > len = mem - > len ;
2011-10-20 16:52:46 +02:00
list_add_tail ( & mapping - > node , & rproc - > mappings ) ;
2016-08-12 18:42:16 -05:00
dev_dbg ( dev , " carveout mapped 0x%x to %pad \n " ,
2018-07-27 15:14:43 +02:00
mem - > da , & dma ) ;
2019-01-10 14:49:09 +01:00
}
if ( mem - > da = = FW_RSC_ADDR_ANY ) {
2019-01-10 14:49:08 +01:00
/* Update device address as undefined by requester */
if ( ( u64 ) dma & HIGH_BITS_MASK )
dev_warn ( dev , " DMA address cast in 32bit to fit resource table format \n " ) ;
2018-07-27 15:14:43 +02:00
mem - > da = ( u32 ) dma ;
2011-10-20 16:52:46 +02:00
}
2019-01-10 14:49:07 +01:00
mem - > dma = dma ;
2018-07-27 15:14:43 +02:00
mem - > va = va ;
2011-10-20 16:52:46 +02:00
return 0 ;
2012-09-25 10:01:56 +03:00
free_mapping :
kfree ( mapping ) ;
2011-10-20 16:52:46 +02:00
dma_free :
2018-07-27 15:14:43 +02:00
dma_free_coherent ( dev - > parent , mem - > len , va , dma ) ;
2011-10-20 16:52:46 +02:00
return ret ;
}
2018-07-27 15:14:43 +02:00
/**
* rproc_release_carveout ( ) - release acquired carveout
* @ rproc : rproc handle
* @ mem : the memory entry to release
*
* This function releases specified memory entry @ mem allocated via
* rproc_alloc_carveout ( ) function by @ rproc .
*/
static int rproc_release_carveout ( struct rproc * rproc ,
struct rproc_mem_entry * mem )
{
struct device * dev = & rproc - > dev ;
/* clean up carveout allocations */
dma_free_coherent ( dev - > parent , mem - > len , mem - > va , mem - > dma ) ;
return 0 ;
}
/**
* rproc_handle_carveout ( ) - handle phys contig memory allocation requests
* @ rproc : rproc handle
2021-02-24 13:58:25 +08:00
* @ ptr : the resource entry
2020-02-12 17:19:56 +01:00
* @ offset : offset of the resource entry
2018-07-27 15:14:43 +02:00
* @ avail : size of available data ( for image validation )
*
* This function will handle firmware requests for allocation of physically
* contiguous memory regions .
*
* These request entries should come first in the firmware ' s resource table ,
* as other firmware entries might request placing other data objects inside
* these memory regions ( e . g . data / code segments , trace resource entries , . . . ) .
*
* Allocating memory this way helps utilizing the reserved physical memory
* ( e . g . CMA ) more efficiently , and also minimizes the number of TLB entries
* needed to map it ( in case @ rproc is using an IOMMU ) . Reducing the TLB
* pressure is important ; it may have a substantial impact on performance .
*/
static int rproc_handle_carveout ( struct rproc * rproc ,
2021-02-24 13:58:25 +08:00
void * ptr , int offset , int avail )
2018-07-27 15:14:43 +02:00
{
2021-02-24 13:58:25 +08:00
struct fw_rsc_carveout * rsc = ptr ;
2018-07-27 15:14:43 +02:00
struct rproc_mem_entry * carveout ;
struct device * dev = & rproc - > dev ;
if ( sizeof ( * rsc ) > avail ) {
dev_err ( dev , " carveout rsc is truncated \n " ) ;
return - EINVAL ;
}
/* make sure reserved bytes are zeroes */
if ( rsc - > reserved ) {
dev_err ( dev , " carveout rsc has non zero reserved bytes \n " ) ;
return - EINVAL ;
}
dev_dbg ( dev , " carveout rsc: name: %s, da 0x%x, pa 0x%x, len 0x%x, flags 0x%x \n " ,
rsc - > name , rsc - > da , rsc - > pa , rsc - > len , rsc - > flags ) ;
2018-07-27 15:14:46 +02:00
/*
* Check carveout rsc already part of a registered carveout ,
* Search by name , then check the da and length
*/
carveout = rproc_find_carveout_by_name ( rproc , rsc - > name ) ;
if ( carveout ) {
if ( carveout - > rsc_offset ! = FW_RSC_ADDR_ANY ) {
dev_err ( dev ,
" Carveout already associated to resource table \n " ) ;
return - ENOMEM ;
}
if ( rproc_check_carveout_da ( rproc , carveout , rsc - > da , rsc - > len ) )
return - ENOMEM ;
/* Update memory carveout with resource table info */
carveout - > rsc_offset = offset ;
carveout - > flags = rsc - > flags ;
return 0 ;
}
2018-07-27 15:14:43 +02:00
/* Register carveout in in list */
2019-10-17 12:59:52 +01:00
carveout = rproc_mem_entry_init ( dev , NULL , 0 , rsc - > len , rsc - > da ,
2018-07-27 15:14:43 +02:00
rproc_alloc_carveout ,
rproc_release_carveout , rsc - > name ) ;
if ( ! carveout ) {
dev_err ( dev , " Can't allocate memory entry structure \n " ) ;
return - ENOMEM ;
}
carveout - > flags = rsc - > flags ;
carveout - > rsc_offset = offset ;
rproc_add_carveout ( rproc , carveout ) ;
return 0 ;
}
2018-07-27 15:14:41 +02:00
/**
* rproc_add_carveout ( ) - register an allocated carveout region
* @ rproc : rproc handle
* @ mem : memory entry to register
*
* This function registers specified memory entry in @ rproc carveouts list .
* Specified carveout should have been allocated before registering .
*/
void rproc_add_carveout ( struct rproc * rproc , struct rproc_mem_entry * mem )
{
list_add_tail ( & mem - > node , & rproc - > carveouts ) ;
}
EXPORT_SYMBOL ( rproc_add_carveout ) ;
2018-07-27 15:14:40 +02:00
/**
* rproc_mem_entry_init ( ) - allocate and initialize rproc_mem_entry struct
* @ dev : pointer on device struct
* @ va : virtual address
* @ dma : dma address
* @ len : memory carveout length
* @ da : device address
2019-01-10 14:49:05 +01:00
* @ alloc : memory carveout allocation function
* @ release : memory carveout release function
2018-07-27 15:14:40 +02:00
* @ name : carveout name
*
* This function allocates a rproc_mem_entry struct and fill it with parameters
* provided by client .
*/
2020-07-15 13:35:50 +01:00
__printf ( 8 , 9 )
2018-07-27 15:14:40 +02:00
struct rproc_mem_entry *
rproc_mem_entry_init ( struct device * dev ,
2020-03-02 10:38:56 +01:00
void * va , dma_addr_t dma , size_t len , u32 da ,
2018-07-27 15:14:43 +02:00
int ( * alloc ) ( struct rproc * , struct rproc_mem_entry * ) ,
2018-07-27 15:14:40 +02:00
int ( * release ) ( struct rproc * , struct rproc_mem_entry * ) ,
const char * name , . . . )
{
struct rproc_mem_entry * mem ;
va_list args ;
mem = kzalloc ( sizeof ( * mem ) , GFP_KERNEL ) ;
if ( ! mem )
return mem ;
mem - > va = va ;
mem - > dma = dma ;
mem - > da = da ;
mem - > len = len ;
2018-07-27 15:14:43 +02:00
mem - > alloc = alloc ;
2018-07-27 15:14:40 +02:00
mem - > release = release ;
2018-07-27 15:14:43 +02:00
mem - > rsc_offset = FW_RSC_ADDR_ANY ;
2018-07-27 15:14:44 +02:00
mem - > of_resm_idx = - 1 ;
2018-07-27 15:14:40 +02:00
va_start ( args , name ) ;
vsnprintf ( mem - > name , sizeof ( mem - > name ) , name , args ) ;
va_end ( args ) ;
return mem ;
}
EXPORT_SYMBOL ( rproc_mem_entry_init ) ;
2018-07-27 15:14:44 +02:00
/**
* rproc_of_resm_mem_entry_init ( ) - allocate and initialize rproc_mem_entry struct
* from a reserved memory phandle
* @ dev : pointer on device struct
* @ of_resm_idx : reserved memory phandle index in " memory-region "
* @ len : memory carveout length
* @ da : device address
* @ name : carveout name
*
* This function allocates a rproc_mem_entry struct and fill it with parameters
* provided by client .
*/
2020-07-15 13:35:50 +01:00
__printf ( 5 , 6 )
2018-07-27 15:14:44 +02:00
struct rproc_mem_entry *
2020-03-02 10:38:56 +01:00
rproc_of_resm_mem_entry_init ( struct device * dev , u32 of_resm_idx , size_t len ,
2018-07-27 15:14:44 +02:00
u32 da , const char * name , . . . )
{
struct rproc_mem_entry * mem ;
va_list args ;
mem = kzalloc ( sizeof ( * mem ) , GFP_KERNEL ) ;
if ( ! mem )
return mem ;
mem - > da = da ;
mem - > len = len ;
mem - > rsc_offset = FW_RSC_ADDR_ANY ;
mem - > of_resm_idx = of_resm_idx ;
va_start ( args , name ) ;
vsnprintf ( mem - > name , sizeof ( mem - > name ) , name , args ) ;
va_end ( args ) ;
return mem ;
}
EXPORT_SYMBOL ( rproc_of_resm_mem_entry_init ) ;
2020-07-21 17:36:13 -05:00
/**
* rproc_of_parse_firmware ( ) - parse and return the firmware - name
* @ dev : pointer on device struct representing a rproc
* @ index : index to use for the firmware - name retrieval
* @ fw_name : pointer to a character string , in which the firmware
* name is returned on success and unmodified otherwise .
*
* This is an OF helper function that parses a device ' s DT node for
* the " firmware-name " property and returns the firmware name pointer
* in @ fw_name on success .
*
* Return : 0 on success , or an appropriate failure .
*/
int rproc_of_parse_firmware ( struct device * dev , int index , const char * * fw_name )
{
int ret ;
ret = of_property_read_string_index ( dev - > of_node , " firmware-name " ,
index , fw_name ) ;
return ret ? ret : 0 ;
}
EXPORT_SYMBOL ( rproc_of_parse_firmware ) ;
2020-02-12 17:19:56 +01:00
/*
2012-01-31 16:07:27 +02:00
* A lookup table for resource handlers . The indices are defined in
* enum fw_resource_type .
*/
2013-02-21 18:15:36 +01:00
static rproc_handle_resource_t rproc_loading_handlers [ RSC_LAST ] = {
2021-02-24 13:58:25 +08:00
[ RSC_CARVEOUT ] = rproc_handle_carveout ,
[ RSC_DEVMEM ] = rproc_handle_devmem ,
[ RSC_TRACE ] = rproc_handle_trace ,
[ RSC_VDEV ] = rproc_handle_vdev ,
2013-02-21 18:15:36 +01:00
} ;
2011-10-20 16:52:46 +02:00
/* handle firmware resource entries before booting the remote processor */
2018-01-05 15:57:59 -08:00
static int rproc_handle_resources ( struct rproc * rproc ,
2013-02-21 18:15:36 +01:00
rproc_handle_resource_t handlers [ RSC_LAST ] )
2011-10-20 16:52:46 +02:00
{
remoteproc: maintain a generic child device for each rproc
For each registered rproc, maintain a generic remoteproc device whose
parent is the low level platform-specific device (commonly a pdev, but
it may certainly be any other type of device too).
With this in hand, the resulting device hierarchy might then look like:
omap-rproc.0
|
- remoteproc0 <---- new !
|
- virtio0
|
- virtio1
|
- rpmsg0
|
- rpmsg1
|
- rpmsg2
Where:
- omap-rproc.0 is the low level device that's bound to the
driver which invokes rproc_register()
- remoteproc0 is the result of this patch, and will be added by the
remoteproc framework when rproc_register() is invoked
- virtio0 and virtio1 are vdevs that are registered by remoteproc
when it realizes that they are supported by the firmware
of the physical remote processor represented by omap-rproc.0
- rpmsg0, rpmsg1 and rpmsg2 are rpmsg devices that represent rpmsg
channels, and are registerd by the rpmsg bus when it gets notified
about their existence
Technically, this patch:
- changes 'struct rproc' to contain this generic remoteproc.x device
- creates a new "remoteproc" type, to which this new generic remoteproc.x
device belong to.
- adds a super simple enumeration method for the indices of the
remoteproc.x devices
- updates all dev_* messaging to use the generic remoteproc.x device
instead of the low level platform-specific device
- updates all dma_* allocations to use the parent of remoteproc.x (where
the platform-specific memory pools, most commonly CMA, are to be found)
Adding this generic device has several merits:
- we can now add remoteproc runtime PM support simply by hooking onto the
new "remoteproc" type
- all remoteproc log messages will now carry a common name prefix
instead of having a platform-specific one
- having a device as part of the rproc struct makes it possible to simplify
refcounting (see subsequent patch)
Thanks to Stephen Boyd <sboyd@codeaurora.org> for suggesting and
discussing these ideas in one of the remoteproc review threads and
to Fernando Guzman Lugo <fernando.lugo@ti.com> for trying them out
with the (upcoming) runtime PM support for remoteproc.
Cc: Fernando Guzman Lugo <fernando.lugo@ti.com>
Reviewed-by: Stephen Boyd <sboyd@codeaurora.org>
Signed-off-by: Ohad Ben-Cohen <ohad@wizery.com>
2012-05-30 22:01:25 +03:00
struct device * dev = & rproc - > dev ;
2012-01-31 16:07:27 +02:00
rproc_handle_resource_t handler ;
2012-02-01 21:56:16 +02:00
int ret = 0 , i ;
2018-01-05 15:58:02 -08:00
if ( ! rproc - > table_ptr )
return 0 ;
2013-04-07 14:06:07 +03:00
for ( i = 0 ; i < rproc - > table_ptr - > num ; i + + ) {
int offset = rproc - > table_ptr - > offset [ i ] ;
struct fw_rsc_hdr * hdr = ( void * ) rproc - > table_ptr + offset ;
2018-01-05 15:57:59 -08:00
int avail = rproc - > table_sz - offset - sizeof ( * hdr ) ;
2012-02-01 21:56:16 +02:00
void * rsc = ( void * ) hdr + sizeof ( * hdr ) ;
/* make sure table isn't truncated */
if ( avail < 0 ) {
dev_err ( dev , " rsc table is truncated \n " ) ;
return - EINVAL ;
}
2011-10-20 16:52:46 +02:00
2012-02-01 21:56:16 +02:00
dev_dbg ( dev , " rsc: type %d \n " , hdr - > type ) ;
2011-10-20 16:52:46 +02:00
2019-06-17 14:57:30 +02:00
if ( hdr - > type > = RSC_VENDOR_START & &
hdr - > type < = RSC_VENDOR_END ) {
ret = rproc_handle_rsc ( rproc , hdr - > type , rsc ,
offset + sizeof ( * hdr ) , avail ) ;
if ( ret = = RSC_HANDLED )
continue ;
else if ( ret < 0 )
break ;
dev_warn ( dev , " unsupported vendor resource %d \n " ,
hdr - > type ) ;
continue ;
}
2012-02-01 21:56:16 +02:00
if ( hdr - > type > = RSC_LAST ) {
dev_warn ( dev , " unsupported resource %d \n " , hdr - > type ) ;
2012-01-31 16:07:27 +02:00
continue ;
2011-10-20 16:52:46 +02:00
}
2013-02-21 18:15:36 +01:00
handler = handlers [ hdr - > type ] ;
2012-01-31 16:07:27 +02:00
if ( ! handler )
continue ;
2013-04-07 14:06:07 +03:00
ret = handler ( rproc , rsc , offset + sizeof ( * hdr ) , avail ) ;
2012-02-13 22:30:39 +01:00
if ( ret )
2011-10-20 16:52:46 +02:00
break ;
2012-02-01 21:56:16 +02:00
}
2011-10-20 16:52:46 +02:00
return ret ;
}
2018-06-26 07:11:59 -05:00
static int rproc_prepare_subdevices ( struct rproc * rproc )
{
struct rproc_subdev * subdev ;
int ret ;
list_for_each_entry ( subdev , & rproc - > subdevs , node ) {
if ( subdev - > prepare ) {
ret = subdev - > prepare ( subdev ) ;
if ( ret )
goto unroll_preparation ;
}
}
return 0 ;
unroll_preparation :
list_for_each_entry_continue_reverse ( subdev , & rproc - > subdevs , node ) {
if ( subdev - > unprepare )
subdev - > unprepare ( subdev ) ;
}
return ret ;
}
2018-06-26 07:11:55 -05:00
static int rproc_start_subdevices ( struct rproc * rproc )
2016-10-19 19:40:02 -07:00
{
struct rproc_subdev * subdev ;
int ret ;
list_for_each_entry ( subdev , & rproc - > subdevs , node ) {
2018-06-26 07:11:56 -05:00
if ( subdev - > start ) {
ret = subdev - > start ( subdev ) ;
if ( ret )
goto unroll_registration ;
}
2016-10-19 19:40:02 -07:00
}
return 0 ;
unroll_registration :
2018-06-26 07:11:56 -05:00
list_for_each_entry_continue_reverse ( subdev , & rproc - > subdevs , node ) {
if ( subdev - > stop )
subdev - > stop ( subdev , true ) ;
}
2016-10-19 19:40:02 -07:00
return ret ;
}
2018-06-26 07:11:55 -05:00
static void rproc_stop_subdevices ( struct rproc * rproc , bool crashed )
2016-10-19 19:40:02 -07:00
{
struct rproc_subdev * subdev ;
2018-06-26 07:11:56 -05:00
list_for_each_entry_reverse ( subdev , & rproc - > subdevs , node ) {
if ( subdev - > stop )
subdev - > stop ( subdev , crashed ) ;
}
2016-10-19 19:40:02 -07:00
}
2018-06-26 07:11:59 -05:00
static void rproc_unprepare_subdevices ( struct rproc * rproc )
{
struct rproc_subdev * subdev ;
list_for_each_entry_reverse ( subdev , & rproc - > subdevs , node ) {
if ( subdev - > unprepare )
subdev - > unprepare ( subdev ) ;
}
}
2018-07-27 15:14:43 +02:00
/**
* rproc_alloc_registered_carveouts ( ) - allocate all carveouts registered
* in the list
* @ rproc : the remote processor handle
*
* This function parses registered carveout list , performs allocation
* if alloc ( ) ops registered and updates resource table information
* if rsc_offset set .
*
* Return : 0 on success
*/
static int rproc_alloc_registered_carveouts ( struct rproc * rproc )
{
struct rproc_mem_entry * entry , * tmp ;
struct fw_rsc_carveout * rsc ;
struct device * dev = & rproc - > dev ;
2019-01-10 14:49:08 +01:00
u64 pa ;
2018-07-27 15:14:43 +02:00
int ret ;
list_for_each_entry_safe ( entry , tmp , & rproc - > carveouts , node ) {
if ( entry - > alloc ) {
ret = entry - > alloc ( rproc , entry ) ;
if ( ret ) {
dev_err ( dev , " Unable to allocate carveout %s: %d \n " ,
entry - > name , ret ) ;
return - ENOMEM ;
}
}
if ( entry - > rsc_offset ! = FW_RSC_ADDR_ANY ) {
/* update resource table */
rsc = ( void * ) rproc - > table_ptr + entry - > rsc_offset ;
/*
* Some remote processors might need to know the pa
* even though they are behind an IOMMU . E . g . , OMAP4 ' s
* remote M3 processor needs this so it can control
* on - chip hardware accelerators that are not behind
* the IOMMU , and therefor must know the pa .
*
* Generally we don ' t want to expose physical addresses
* if we don ' t have to ( remote processors are generally
* _not_ trusted ) , so we might want to do this only for
* remote processor that _must_ have this ( e . g . OMAP4 ' s
* dual M3 subsystem ) .
*
* Non - IOMMU processors might also want to have this info .
* In this case , the device address and the physical address
* are the same .
*/
2018-07-27 15:14:46 +02:00
/* Use va if defined else dma to generate pa */
2018-07-27 15:14:43 +02:00
if ( entry - > va )
2019-01-10 14:49:08 +01:00
pa = ( u64 ) rproc_va_to_pa ( entry - > va ) ;
2018-07-27 15:14:46 +02:00
else
2019-01-10 14:49:08 +01:00
pa = ( u64 ) entry - > dma ;
if ( ( ( u64 ) pa ) & HIGH_BITS_MASK )
dev_warn ( dev ,
" Physical address cast in 32bit to fit resource table format \n " ) ;
2018-07-27 15:14:46 +02:00
2019-01-10 14:49:08 +01:00
rsc - > pa = ( u32 ) pa ;
2018-07-27 15:14:46 +02:00
rsc - > da = entry - > da ;
rsc - > len = entry - > len ;
2018-07-27 15:14:43 +02:00
}
}
return 0 ;
}
2018-01-05 16:04:17 -08:00
2011-10-20 16:52:46 +02:00
/**
* rproc_resource_cleanup ( ) - clean up and free all acquired resources
* @ rproc : rproc handle
*
* This function will free all resources acquired for @ rproc , and it
2012-02-13 22:30:39 +01:00
* is called whenever @ rproc either shuts down or fails to boot .
2011-10-20 16:52:46 +02:00
*/
2020-07-14 14:04:41 -06:00
void rproc_resource_cleanup ( struct rproc * rproc )
2011-10-20 16:52:46 +02:00
{
struct rproc_mem_entry * entry , * tmp ;
2019-01-10 14:49:10 +01:00
struct rproc_debug_trace * trace , * ttmp ;
2016-08-11 14:52:52 -07:00
struct rproc_vdev * rvdev , * rvtmp ;
remoteproc: maintain a generic child device for each rproc
For each registered rproc, maintain a generic remoteproc device whose
parent is the low level platform-specific device (commonly a pdev, but
it may certainly be any other type of device too).
With this in hand, the resulting device hierarchy might then look like:
omap-rproc.0
|
- remoteproc0 <---- new !
|
- virtio0
|
- virtio1
|
- rpmsg0
|
- rpmsg1
|
- rpmsg2
Where:
- omap-rproc.0 is the low level device that's bound to the
driver which invokes rproc_register()
- remoteproc0 is the result of this patch, and will be added by the
remoteproc framework when rproc_register() is invoked
- virtio0 and virtio1 are vdevs that are registered by remoteproc
when it realizes that they are supported by the firmware
of the physical remote processor represented by omap-rproc.0
- rpmsg0, rpmsg1 and rpmsg2 are rpmsg devices that represent rpmsg
channels, and are registerd by the rpmsg bus when it gets notified
about their existence
Technically, this patch:
- changes 'struct rproc' to contain this generic remoteproc.x device
- creates a new "remoteproc" type, to which this new generic remoteproc.x
device belong to.
- adds a super simple enumeration method for the indices of the
remoteproc.x devices
- updates all dev_* messaging to use the generic remoteproc.x device
instead of the low level platform-specific device
- updates all dma_* allocations to use the parent of remoteproc.x (where
the platform-specific memory pools, most commonly CMA, are to be found)
Adding this generic device has several merits:
- we can now add remoteproc runtime PM support simply by hooking onto the
new "remoteproc" type
- all remoteproc log messages will now carry a common name prefix
instead of having a platform-specific one
- having a device as part of the rproc struct makes it possible to simplify
refcounting (see subsequent patch)
Thanks to Stephen Boyd <sboyd@codeaurora.org> for suggesting and
discussing these ideas in one of the remoteproc review threads and
to Fernando Guzman Lugo <fernando.lugo@ti.com> for trying them out
with the (upcoming) runtime PM support for remoteproc.
Cc: Fernando Guzman Lugo <fernando.lugo@ti.com>
Reviewed-by: Stephen Boyd <sboyd@codeaurora.org>
Signed-off-by: Ohad Ben-Cohen <ohad@wizery.com>
2012-05-30 22:01:25 +03:00
struct device * dev = & rproc - > dev ;
2011-10-20 16:52:46 +02:00
/* clean up debugfs trace entries */
2019-01-10 14:49:10 +01:00
list_for_each_entry_safe ( trace , ttmp , & rproc - > traces , node ) {
rproc_remove_trace_file ( trace - > tfile ) ;
2011-10-20 16:52:46 +02:00
rproc - > num_traces - - ;
2019-01-10 14:49:10 +01:00
list_del ( & trace - > node ) ;
kfree ( trace ) ;
2011-10-20 16:52:46 +02:00
}
/* clean up iommu mapping entries */
list_for_each_entry_safe ( entry , tmp , & rproc - > mappings , node ) {
size_t unmapped ;
unmapped = iommu_unmap ( rproc - > domain , entry - > da , entry - > len ) ;
if ( unmapped ! = entry - > len ) {
/* nothing much to do besides complaining */
2020-03-02 10:38:56 +01:00
dev_err ( dev , " failed to unmap %zx/%zu \n " , entry - > len ,
2016-08-12 18:42:20 -05:00
unmapped ) ;
2011-10-20 16:52:46 +02:00
}
list_del ( & entry - > node ) ;
kfree ( entry ) ;
}
2013-07-01 17:01:56 +03:00
/* clean up carveout allocations */
list_for_each_entry_safe ( entry , tmp , & rproc - > carveouts , node ) {
2018-07-27 15:14:38 +02:00
if ( entry - > release )
entry - > release ( rproc , entry ) ;
2013-07-01 17:01:56 +03:00
list_del ( & entry - > node ) ;
kfree ( entry ) ;
}
2016-08-11 14:52:52 -07:00
/* clean up remote vdev entries */
2016-10-19 19:40:09 -07:00
list_for_each_entry_safe ( rvdev , rvtmp , & rproc - > rvdevs , node )
2016-10-19 19:40:08 -07:00
kref_put ( & rvdev - > refcount , rproc_vdev_release ) ;
2018-01-05 16:04:17 -08:00
rproc_coredump_cleanup ( rproc ) ;
2011-10-20 16:52:46 +02:00
}
2020-07-14 14:04:41 -06:00
EXPORT_SYMBOL ( rproc_resource_cleanup ) ;
2011-10-20 16:52:46 +02:00
2017-05-26 16:51:00 -07:00
static int rproc_start ( struct rproc * rproc , const struct firmware * fw )
{
2018-01-05 15:57:59 -08:00
struct resource_table * loaded_table ;
2017-05-26 16:51:00 -07:00
struct device * dev = & rproc - > dev ;
2018-01-05 15:57:59 -08:00
int ret ;
2017-05-26 16:51:00 -07:00
/* load the ELF segments to memory */
ret = rproc_load_segments ( rproc , fw ) ;
if ( ret ) {
dev_err ( dev , " Failed to load program segments: %d \n " , ret ) ;
return ret ;
}
/*
* The starting device has been given the rproc - > cached_table as the
* resource table . The address of the vring along with the other
* allocated resources ( carveouts etc ) is stored in cached_table .
* In order to pass this information to the remote device we must copy
* this information to device memory . We also update the table_ptr so
* that any subsequent changes will be applied to the loaded version .
*/
loaded_table = rproc_find_loaded_rsc_table ( rproc , fw ) ;
if ( loaded_table ) {
2018-01-05 15:57:59 -08:00
memcpy ( loaded_table , rproc - > cached_table , rproc - > table_sz ) ;
2017-05-26 16:51:00 -07:00
rproc - > table_ptr = loaded_table ;
}
2018-06-26 07:11:59 -05:00
ret = rproc_prepare_subdevices ( rproc ) ;
if ( ret ) {
dev_err ( dev , " failed to prepare subdevices for %s: %d \n " ,
rproc - > name , ret ) ;
2018-07-26 20:15:35 -05:00
goto reset_table_ptr ;
2018-06-26 07:11:59 -05:00
}
2017-05-26 16:51:00 -07:00
/* power up the remote processor */
ret = rproc - > ops - > start ( rproc ) ;
if ( ret ) {
dev_err ( dev , " can't start rproc %s: %d \n " , rproc - > name , ret ) ;
2018-06-26 07:11:59 -05:00
goto unprepare_subdevices ;
2017-05-26 16:51:00 -07:00
}
2018-06-26 07:11:55 -05:00
/* Start any subdevices for the remote processor */
ret = rproc_start_subdevices ( rproc ) ;
2017-05-26 16:51:00 -07:00
if ( ret ) {
dev_err ( dev , " failed to probe subdevices for %s: %d \n " ,
rproc - > name , ret ) ;
2018-06-26 07:11:59 -05:00
goto stop_rproc ;
2017-05-26 16:51:00 -07:00
}
rproc - > state = RPROC_RUNNING ;
dev_info ( dev , " remote processor %s is now up \n " , rproc - > name ) ;
return 0 ;
2018-06-26 07:11:59 -05:00
stop_rproc :
rproc - > ops - > stop ( rproc ) ;
unprepare_subdevices :
rproc_unprepare_subdevices ( rproc ) ;
2018-07-26 20:15:35 -05:00
reset_table_ptr :
rproc - > table_ptr = rproc - > cached_table ;
2018-06-26 07:11:59 -05:00
return ret ;
2017-05-26 16:51:00 -07:00
}
2021-03-12 09:24:38 -07:00
static int __rproc_attach ( struct rproc * rproc )
2020-07-14 13:50:29 -06:00
{
struct device * dev = & rproc - > dev ;
int ret ;
ret = rproc_prepare_subdevices ( rproc ) ;
if ( ret ) {
dev_err ( dev , " failed to prepare subdevices for %s: %d \n " ,
rproc - > name , ret ) ;
goto out ;
}
/* Attach to the remote processor */
ret = rproc_attach_device ( rproc ) ;
if ( ret ) {
dev_err ( dev , " can't attach to rproc %s: %d \n " ,
rproc - > name , ret ) ;
goto unprepare_subdevices ;
}
/* Start any subdevices for the remote processor */
ret = rproc_start_subdevices ( rproc ) ;
if ( ret ) {
dev_err ( dev , " failed to probe subdevices for %s: %d \n " ,
rproc - > name , ret ) ;
goto stop_rproc ;
}
rproc - > state = RPROC_RUNNING ;
dev_info ( dev , " remote processor %s is now attached \n " , rproc - > name ) ;
return 0 ;
stop_rproc :
rproc - > ops - > stop ( rproc ) ;
unprepare_subdevices :
rproc_unprepare_subdevices ( rproc ) ;
out :
return ret ;
}
2011-10-20 16:52:46 +02:00
/*
* take a firmware and boot a remote processor with it .
*/
static int rproc_fw_boot ( struct rproc * rproc , const struct firmware * fw )
{
remoteproc: maintain a generic child device for each rproc
For each registered rproc, maintain a generic remoteproc device whose
parent is the low level platform-specific device (commonly a pdev, but
it may certainly be any other type of device too).
With this in hand, the resulting device hierarchy might then look like:
omap-rproc.0
|
- remoteproc0 <---- new !
|
- virtio0
|
- virtio1
|
- rpmsg0
|
- rpmsg1
|
- rpmsg2
Where:
- omap-rproc.0 is the low level device that's bound to the
driver which invokes rproc_register()
- remoteproc0 is the result of this patch, and will be added by the
remoteproc framework when rproc_register() is invoked
- virtio0 and virtio1 are vdevs that are registered by remoteproc
when it realizes that they are supported by the firmware
of the physical remote processor represented by omap-rproc.0
- rpmsg0, rpmsg1 and rpmsg2 are rpmsg devices that represent rpmsg
channels, and are registerd by the rpmsg bus when it gets notified
about their existence
Technically, this patch:
- changes 'struct rproc' to contain this generic remoteproc.x device
- creates a new "remoteproc" type, to which this new generic remoteproc.x
device belong to.
- adds a super simple enumeration method for the indices of the
remoteproc.x devices
- updates all dev_* messaging to use the generic remoteproc.x device
instead of the low level platform-specific device
- updates all dma_* allocations to use the parent of remoteproc.x (where
the platform-specific memory pools, most commonly CMA, are to be found)
Adding this generic device has several merits:
- we can now add remoteproc runtime PM support simply by hooking onto the
new "remoteproc" type
- all remoteproc log messages will now carry a common name prefix
instead of having a platform-specific one
- having a device as part of the rproc struct makes it possible to simplify
refcounting (see subsequent patch)
Thanks to Stephen Boyd <sboyd@codeaurora.org> for suggesting and
discussing these ideas in one of the remoteproc review threads and
to Fernando Guzman Lugo <fernando.lugo@ti.com> for trying them out
with the (upcoming) runtime PM support for remoteproc.
Cc: Fernando Guzman Lugo <fernando.lugo@ti.com>
Reviewed-by: Stephen Boyd <sboyd@codeaurora.org>
Signed-off-by: Ohad Ben-Cohen <ohad@wizery.com>
2012-05-30 22:01:25 +03:00
struct device * dev = & rproc - > dev ;
2011-10-20 16:52:46 +02:00
const char * name = rproc - > firmware ;
2018-01-05 15:58:03 -08:00
int ret ;
2011-10-20 16:52:46 +02:00
ret = rproc_fw_sanity_check ( rproc , fw ) ;
if ( ret )
return ret ;
2012-06-10 14:37:07 +03:00
dev_info ( dev , " Booting fw image %s, size %zd \n " , name , fw - > size ) ;
2011-10-20 16:52:46 +02:00
/*
* if enabling an IOMMU isn ' t relevant for this rproc , this is
* just a nop
*/
ret = rproc_enable_iommu ( rproc ) ;
if ( ret ) {
dev_err ( dev , " can't enable iommu: %d \n " , ret ) ;
2020-06-30 10:31:18 -06:00
return ret ;
2011-10-20 16:52:46 +02:00
}
2020-04-16 19:20:35 -05:00
/* Prepare rproc for firmware loading if needed */
ret = rproc_prepare_device ( rproc ) ;
if ( ret ) {
dev_err ( dev , " can't prepare rproc %s: %d \n " , rproc - > name , ret ) ;
goto disable_iommu ;
}
2012-06-19 09:56:44 +03:00
rproc - > bootaddr = rproc_get_boot_addr ( rproc , fw ) ;
2016-08-11 14:52:53 -07:00
2018-01-05 16:04:18 -08:00
/* Load resource table, core dump segment list etc from the firmware */
ret = rproc_parse_fw ( rproc , fw ) ;
2018-01-05 15:58:03 -08:00
if ( ret )
2020-04-16 19:20:35 -05:00
goto unprepare_rproc ;
2016-12-30 03:21:38 -08:00
2016-08-11 14:52:51 -07:00
/* reset max_notifyid */
rproc - > max_notifyid = - 1 ;
2018-07-27 15:14:47 +02:00
/* reset handled vdev */
rproc - > nb_vdev = 0 ;
2011-10-20 16:52:46 +02:00
/* handle fw resources which are required to boot rproc */
2018-01-05 15:57:59 -08:00
ret = rproc_handle_resources ( rproc , rproc_loading_handlers ) ;
2011-10-20 16:52:46 +02:00
if ( ret ) {
dev_err ( dev , " Failed to process resources: %d \n " , ret ) ;
2016-10-02 17:41:29 -07:00
goto clean_up_resources ;
2011-10-20 16:52:46 +02:00
}
2018-07-27 15:14:43 +02:00
/* Allocate carveout resources associated to rproc */
ret = rproc_alloc_registered_carveouts ( rproc ) ;
if ( ret ) {
dev_err ( dev , " Failed to allocate associated carveouts: %d \n " ,
ret ) ;
goto clean_up_resources ;
}
2017-05-26 16:51:00 -07:00
ret = rproc_start ( rproc , fw ) ;
if ( ret )
2016-10-02 17:41:29 -07:00
goto clean_up_resources ;
2011-10-20 16:52:46 +02:00
return 0 ;
2016-10-02 17:41:29 -07:00
clean_up_resources :
rproc_resource_cleanup ( rproc ) ;
2016-12-30 03:21:38 -08:00
kfree ( rproc - > cached_table ) ;
rproc - > cached_table = NULL ;
2016-08-11 14:52:53 -07:00
rproc - > table_ptr = NULL ;
2020-04-16 19:20:35 -05:00
unprepare_rproc :
/* release HW resources if needed */
rproc_unprepare_device ( rproc ) ;
2018-01-05 15:58:03 -08:00
disable_iommu :
2011-10-20 16:52:46 +02:00
rproc_disable_iommu ( rproc ) ;
return ret ;
}
2020-07-14 13:50:30 -06:00
/*
* Attach to remote processor - similar to rproc_fw_boot ( ) but without
* the steps that deal with the firmware image .
*/
2021-03-12 09:24:38 -07:00
static int rproc_attach ( struct rproc * rproc )
2020-07-14 13:50:30 -06:00
{
struct device * dev = & rproc - > dev ;
int ret ;
/*
* if enabling an IOMMU isn ' t relevant for this rproc , this is
* just a nop
*/
ret = rproc_enable_iommu ( rproc ) ;
if ( ret ) {
dev_err ( dev , " can't enable iommu: %d \n " , ret ) ;
return ret ;
}
/* reset max_notifyid */
rproc - > max_notifyid = - 1 ;
/* reset handled vdev */
rproc - > nb_vdev = 0 ;
/*
* Handle firmware resources required to attach to a remote processor .
* Because we are attaching rather than booting the remote processor ,
* we expect the platform driver to properly set rproc - > table_ptr .
*/
ret = rproc_handle_resources ( rproc , rproc_loading_handlers ) ;
if ( ret ) {
dev_err ( dev , " Failed to process resources: %d \n " , ret ) ;
goto disable_iommu ;
}
/* Allocate carveout resources associated to rproc */
ret = rproc_alloc_registered_carveouts ( rproc ) ;
if ( ret ) {
dev_err ( dev , " Failed to allocate associated carveouts: %d \n " ,
ret ) ;
goto clean_up_resources ;
}
2021-03-12 09:24:38 -07:00
ret = __rproc_attach ( rproc ) ;
2020-07-14 13:50:30 -06:00
if ( ret )
goto clean_up_resources ;
return 0 ;
clean_up_resources :
rproc_resource_cleanup ( rproc ) ;
disable_iommu :
rproc_disable_iommu ( rproc ) ;
return ret ;
}
2011-10-20 16:52:46 +02:00
/*
2017-01-24 15:13:01 -08:00
* take a firmware and boot it up .
2011-10-20 16:52:46 +02:00
*
* Note : this function is called asynchronously upon registration of the
* remote processor ( so we must wait until it completes before we try
* to unregister the device . one other option is just to use kref here ,
* that might be cleaner ) .
*/
2017-01-24 15:13:01 -08:00
static void rproc_auto_boot_callback ( const struct firmware * fw , void * context )
2011-10-20 16:52:46 +02:00
{
struct rproc * rproc = context ;
2013-04-07 14:06:07 +03:00
2017-01-24 15:13:00 -08:00
rproc_boot ( rproc ) ;
2016-08-11 14:52:50 -07:00
2012-04-09 22:51:25 +02:00
release_firmware ( fw ) ;
2011-10-20 16:52:46 +02:00
}
2017-01-24 15:13:01 -08:00
static int rproc_trigger_auto_boot ( struct rproc * rproc )
2012-08-30 13:26:13 -05:00
{
int ret ;
2020-07-14 13:50:33 -06:00
/*
* Since the remote processor is in a detached state , it has already
* been booted by another entity . As such there is no point in waiting
* for a firmware image to be loaded , we can simply initiate the process
* of attaching to it immediately .
*/
if ( rproc - > state = = RPROC_DETACHED )
return rproc_boot ( rproc ) ;
2012-08-30 13:26:13 -05:00
/*
* We ' re initiating an asynchronous firmware loading , so we can
* be built - in kernel code , without hanging the boot process .
*/
ret = request_firmware_nowait ( THIS_MODULE , FW_ACTION_HOTPLUG ,
rproc - > firmware , & rproc - > dev , GFP_KERNEL ,
2017-01-24 15:13:01 -08:00
rproc , rproc_auto_boot_callback ) ;
2017-01-23 17:53:19 -08:00
if ( ret < 0 )
2012-08-30 13:26:13 -05:00
dev_err ( & rproc - > dev , " request_firmware_nowait err: %d \n " , ret ) ;
return ret ;
}
2017-10-30 23:11:14 -07:00
static int rproc_stop ( struct rproc * rproc , bool crashed )
2017-05-26 16:51:00 -07:00
{
struct device * dev = & rproc - > dev ;
int ret ;
2018-06-26 07:11:55 -05:00
/* Stop any subdevices for the remote processor */
rproc_stop_subdevices ( rproc , crashed ) ;
2017-05-26 16:51:00 -07:00
2018-01-05 15:58:05 -08:00
/* the installed resource table is no longer accessible */
rproc - > table_ptr = rproc - > cached_table ;
2017-05-26 16:51:00 -07:00
/* power off the remote processor */
ret = rproc - > ops - > stop ( rproc ) ;
if ( ret ) {
dev_err ( dev , " can't stop rproc: %d \n " , ret ) ;
return ret ;
}
2018-06-26 07:11:59 -05:00
rproc_unprepare_subdevices ( rproc ) ;
2017-05-26 16:51:00 -07:00
rproc - > state = RPROC_OFFLINE ;
2020-07-14 13:50:35 -06:00
/*
* The remote processor has been stopped and is now offline , which means
* that the next time it is brought back online the remoteproc core will
* be responsible to load its firmware . As such it is no longer
* autonomous .
*/
rproc - > autonomous = false ;
2017-05-26 16:51:00 -07:00
dev_info ( dev , " stopped remote processor %s \n " , rproc - > name ) ;
return 0 ;
}
2018-01-05 16:04:17 -08:00
2012-08-30 13:26:13 -05:00
/**
* rproc_trigger_recovery ( ) - recover a remoteproc
* @ rproc : the remote processor
*
2016-08-12 18:42:17 -05:00
* The recovery is done by resetting all the virtio devices , that way all the
2012-08-30 13:26:13 -05:00
* rpmsg drivers will be reseted along with the remote processor making the
* remoteproc functional again .
*
* This function can sleep , so it cannot be called from atomic context .
*/
int rproc_trigger_recovery ( struct rproc * rproc )
{
2017-05-26 16:51:01 -07:00
const struct firmware * firmware_p ;
struct device * dev = & rproc - > dev ;
int ret ;
ret = mutex_lock_interruptible ( & rproc - > lock ) ;
if ( ret )
return ret ;
2020-02-28 12:33:56 -06:00
/* State could have changed before we got the mutex */
if ( rproc - > state ! = RPROC_CRASHED )
goto unlock_mutex ;
dev_err ( dev , " recovering %s \n " , rproc - > name ) ;
2018-04-10 10:57:25 +02:00
ret = rproc_stop ( rproc , true ) ;
2017-05-26 16:51:01 -07:00
if ( ret )
goto unlock_mutex ;
2016-08-11 14:52:50 -07:00
2018-01-05 16:04:17 -08:00
/* generate coredump */
2020-11-19 13:05:32 -08:00
rproc - > ops - > coredump ( rproc ) ;
2018-01-05 16:04:17 -08:00
2017-05-26 16:51:01 -07:00
/* load firmware */
ret = request_firmware ( & firmware_p , rproc - > firmware , dev ) ;
if ( ret < 0 ) {
dev_err ( dev , " request_firmware failed: %d \n " , ret ) ;
goto unlock_mutex ;
}
2016-08-11 14:52:50 -07:00
2017-05-26 16:51:01 -07:00
/* boot the remote processor up again */
ret = rproc_start ( rproc , firmware_p ) ;
release_firmware ( firmware_p ) ;
unlock_mutex :
mutex_unlock ( & rproc - > lock ) ;
return ret ;
2012-08-30 13:26:13 -05:00
}
2012-08-30 13:26:12 -05:00
/**
* rproc_crash_handler_work ( ) - handle a crash
2020-02-12 17:19:56 +01:00
* @ work : work treating the crash
2012-08-30 13:26:12 -05:00
*
* This function needs to handle everything related to a crash , like cpu
* registers and stack dump , information to help to debug the fatal error , etc .
*/
static void rproc_crash_handler_work ( struct work_struct * work )
{
struct rproc * rproc = container_of ( work , struct rproc , crash_handler ) ;
struct device * dev = & rproc - > dev ;
dev_dbg ( dev , " enter %s \n " , __func__ ) ;
mutex_lock ( & rproc - > lock ) ;
if ( rproc - > state = = RPROC_CRASHED | | rproc - > state = = RPROC_OFFLINE ) {
/* handle only the first crash detected */
mutex_unlock ( & rproc - > lock ) ;
return ;
}
rproc - > state = RPROC_CRASHED ;
dev_err ( dev , " handling crash #%u in %s \n " , + + rproc - > crash_cnt ,
rproc - > name ) ;
mutex_unlock ( & rproc - > lock ) ;
2012-09-18 12:26:35 +03:00
if ( ! rproc - > recovery_disabled )
rproc_trigger_recovery ( rproc ) ;
2020-04-29 11:04:42 -07:00
pm_relax ( rproc - > dev . parent ) ;
2012-08-30 13:26:12 -05:00
}
2011-10-20 16:52:46 +02:00
/**
2017-07-20 15:19:22 -05:00
* rproc_boot ( ) - boot a remote processor
2011-10-20 16:52:46 +02:00
* @ rproc : handle of a remote processor
*
* Boot a remote processor ( i . e . load its firmware , power it on , . . . ) .
*
* If the remote processor is already powered on , this function immediately
* returns ( successfully ) .
*
* Returns 0 on success , and an appropriate error value otherwise .
*/
2017-07-20 15:19:22 -05:00
int rproc_boot ( struct rproc * rproc )
2011-10-20 16:52:46 +02:00
{
const struct firmware * firmware_p ;
struct device * dev ;
int ret ;
if ( ! rproc ) {
pr_err ( " invalid rproc handle \n " ) ;
return - EINVAL ;
}
remoteproc: maintain a generic child device for each rproc
For each registered rproc, maintain a generic remoteproc device whose
parent is the low level platform-specific device (commonly a pdev, but
it may certainly be any other type of device too).
With this in hand, the resulting device hierarchy might then look like:
omap-rproc.0
|
- remoteproc0 <---- new !
|
- virtio0
|
- virtio1
|
- rpmsg0
|
- rpmsg1
|
- rpmsg2
Where:
- omap-rproc.0 is the low level device that's bound to the
driver which invokes rproc_register()
- remoteproc0 is the result of this patch, and will be added by the
remoteproc framework when rproc_register() is invoked
- virtio0 and virtio1 are vdevs that are registered by remoteproc
when it realizes that they are supported by the firmware
of the physical remote processor represented by omap-rproc.0
- rpmsg0, rpmsg1 and rpmsg2 are rpmsg devices that represent rpmsg
channels, and are registerd by the rpmsg bus when it gets notified
about their existence
Technically, this patch:
- changes 'struct rproc' to contain this generic remoteproc.x device
- creates a new "remoteproc" type, to which this new generic remoteproc.x
device belong to.
- adds a super simple enumeration method for the indices of the
remoteproc.x devices
- updates all dev_* messaging to use the generic remoteproc.x device
instead of the low level platform-specific device
- updates all dma_* allocations to use the parent of remoteproc.x (where
the platform-specific memory pools, most commonly CMA, are to be found)
Adding this generic device has several merits:
- we can now add remoteproc runtime PM support simply by hooking onto the
new "remoteproc" type
- all remoteproc log messages will now carry a common name prefix
instead of having a platform-specific one
- having a device as part of the rproc struct makes it possible to simplify
refcounting (see subsequent patch)
Thanks to Stephen Boyd <sboyd@codeaurora.org> for suggesting and
discussing these ideas in one of the remoteproc review threads and
to Fernando Guzman Lugo <fernando.lugo@ti.com> for trying them out
with the (upcoming) runtime PM support for remoteproc.
Cc: Fernando Guzman Lugo <fernando.lugo@ti.com>
Reviewed-by: Stephen Boyd <sboyd@codeaurora.org>
Signed-off-by: Ohad Ben-Cohen <ohad@wizery.com>
2012-05-30 22:01:25 +03:00
dev = & rproc - > dev ;
2011-10-20 16:52:46 +02:00
ret = mutex_lock_interruptible ( & rproc - > lock ) ;
if ( ret ) {
dev_err ( dev , " can't lock rproc %s: %d \n " , rproc - > name , ret ) ;
return ret ;
}
2017-01-23 17:53:19 -08:00
if ( rproc - > state = = RPROC_DELETED ) {
ret = - ENODEV ;
dev_err ( dev , " can't boot deleted rproc %s \n " , rproc - > name ) ;
goto unlock_mutex ;
}
2020-07-14 13:50:32 -06:00
/* skip the boot or attach process if rproc is already powered up */
2011-10-20 16:52:46 +02:00
if ( atomic_inc_return ( & rproc - > power ) > 1 ) {
ret = 0 ;
goto unlock_mutex ;
}
2020-07-14 13:50:32 -06:00
if ( rproc - > state = = RPROC_DETACHED ) {
dev_info ( dev , " attaching to %s \n " , rproc - > name ) ;
2011-10-20 16:52:46 +02:00
2021-03-12 09:24:38 -07:00
ret = rproc_attach ( rproc ) ;
2020-07-14 13:50:32 -06:00
} else {
dev_info ( dev , " powering up %s \n " , rproc - > name ) ;
2011-10-20 16:52:46 +02:00
2020-07-14 13:50:32 -06:00
/* load firmware */
ret = request_firmware ( & firmware_p , rproc - > firmware , dev ) ;
if ( ret < 0 ) {
dev_err ( dev , " request_firmware failed: %d \n " , ret ) ;
goto downref_rproc ;
}
2011-10-20 16:52:46 +02:00
2020-07-14 13:50:32 -06:00
ret = rproc_fw_boot ( rproc , firmware_p ) ;
release_firmware ( firmware_p ) ;
}
2011-10-20 16:52:46 +02:00
downref_rproc :
2016-10-02 17:46:39 -07:00
if ( ret )
2011-10-20 16:52:46 +02:00
atomic_dec ( & rproc - > power ) ;
unlock_mutex :
mutex_unlock ( & rproc - > lock ) ;
return ret ;
}
EXPORT_SYMBOL ( rproc_boot ) ;
/**
* rproc_shutdown ( ) - power off the remote processor
* @ rproc : the remote processor
*
* Power off a remote processor ( previously booted with rproc_boot ( ) ) .
*
* In case @ rproc is still being used by an additional user ( s ) , then
* this function will just decrement the power refcount and exit ,
* without really powering off the device .
*
* Every call to rproc_boot ( ) must ( eventually ) be accompanied by a call
* to rproc_shutdown ( ) . Calling rproc_shutdown ( ) redundantly is a bug .
*
* Notes :
* - we ' re not decrementing the rproc ' s refcount , only the power refcount .
* which means that the @ rproc handle stays valid even after rproc_shutdown ( )
* returns , and users can still use it with a subsequent rproc_boot ( ) , if
* needed .
*/
void rproc_shutdown ( struct rproc * rproc )
{
remoteproc: maintain a generic child device for each rproc
For each registered rproc, maintain a generic remoteproc device whose
parent is the low level platform-specific device (commonly a pdev, but
it may certainly be any other type of device too).
With this in hand, the resulting device hierarchy might then look like:
omap-rproc.0
|
- remoteproc0 <---- new !
|
- virtio0
|
- virtio1
|
- rpmsg0
|
- rpmsg1
|
- rpmsg2
Where:
- omap-rproc.0 is the low level device that's bound to the
driver which invokes rproc_register()
- remoteproc0 is the result of this patch, and will be added by the
remoteproc framework when rproc_register() is invoked
- virtio0 and virtio1 are vdevs that are registered by remoteproc
when it realizes that they are supported by the firmware
of the physical remote processor represented by omap-rproc.0
- rpmsg0, rpmsg1 and rpmsg2 are rpmsg devices that represent rpmsg
channels, and are registerd by the rpmsg bus when it gets notified
about their existence
Technically, this patch:
- changes 'struct rproc' to contain this generic remoteproc.x device
- creates a new "remoteproc" type, to which this new generic remoteproc.x
device belong to.
- adds a super simple enumeration method for the indices of the
remoteproc.x devices
- updates all dev_* messaging to use the generic remoteproc.x device
instead of the low level platform-specific device
- updates all dma_* allocations to use the parent of remoteproc.x (where
the platform-specific memory pools, most commonly CMA, are to be found)
Adding this generic device has several merits:
- we can now add remoteproc runtime PM support simply by hooking onto the
new "remoteproc" type
- all remoteproc log messages will now carry a common name prefix
instead of having a platform-specific one
- having a device as part of the rproc struct makes it possible to simplify
refcounting (see subsequent patch)
Thanks to Stephen Boyd <sboyd@codeaurora.org> for suggesting and
discussing these ideas in one of the remoteproc review threads and
to Fernando Guzman Lugo <fernando.lugo@ti.com> for trying them out
with the (upcoming) runtime PM support for remoteproc.
Cc: Fernando Guzman Lugo <fernando.lugo@ti.com>
Reviewed-by: Stephen Boyd <sboyd@codeaurora.org>
Signed-off-by: Ohad Ben-Cohen <ohad@wizery.com>
2012-05-30 22:01:25 +03:00
struct device * dev = & rproc - > dev ;
2011-10-20 16:52:46 +02:00
int ret ;
ret = mutex_lock_interruptible ( & rproc - > lock ) ;
if ( ret ) {
dev_err ( dev , " can't lock rproc %s: %d \n " , rproc - > name , ret ) ;
return ;
}
/* if the remote proc is still needed, bail out */
if ( ! atomic_dec_and_test ( & rproc - > power ) )
goto out ;
2018-04-10 10:57:25 +02:00
ret = rproc_stop ( rproc , false ) ;
2011-10-20 16:52:46 +02:00
if ( ret ) {
atomic_inc ( & rproc - > power ) ;
goto out ;
}
/* clean up all acquired resources */
rproc_resource_cleanup ( rproc ) ;
2020-04-16 19:20:35 -05:00
/* release HW resources if needed */
rproc_unprepare_device ( rproc ) ;
2011-10-20 16:52:46 +02:00
rproc_disable_iommu ( rproc ) ;
2016-08-11 14:52:53 -07:00
/* Free the copy of the resource table */
2016-12-30 03:21:38 -08:00
kfree ( rproc - > cached_table ) ;
rproc - > cached_table = NULL ;
2016-08-11 14:52:53 -07:00
rproc - > table_ptr = NULL ;
2011-10-20 16:52:46 +02:00
out :
mutex_unlock ( & rproc - > lock ) ;
}
EXPORT_SYMBOL ( rproc_shutdown ) ;
2015-05-22 15:45:27 -05:00
/**
* rproc_get_by_phandle ( ) - find a remote processor by phandle
* @ phandle : phandle to the rproc
*
* Finds an rproc handle using the remote processor ' s phandle , and then
* return a handle to the rproc .
*
* This function increments the remote processor ' s refcount , so always
* use rproc_put ( ) to decrement it back once rproc isn ' t needed anymore .
*
* Returns the rproc handle on success , and NULL on failure .
*/
2015-06-18 11:44:41 +03:00
# ifdef CONFIG_OF
2015-05-22 15:45:27 -05:00
struct rproc * rproc_get_by_phandle ( phandle phandle )
{
struct rproc * rproc = NULL , * r ;
struct device_node * np ;
np = of_find_node_by_phandle ( phandle ) ;
if ( ! np )
return NULL ;
2020-03-23 22:29:01 -07:00
rcu_read_lock ( ) ;
list_for_each_entry_rcu ( r , & rproc_list , node ) {
2015-05-22 15:45:27 -05:00
if ( r - > dev . parent & & r - > dev . parent - > of_node = = np ) {
2016-10-02 17:46:39 -07:00
/* prevent underlying implementation from being removed */
if ( ! try_module_get ( r - > dev . parent - > driver - > owner ) ) {
dev_err ( & r - > dev , " can't get owner \n " ) ;
break ;
}
2015-05-22 15:45:27 -05:00
rproc = r ;
get_device ( & rproc - > dev ) ;
break ;
}
}
2020-03-23 22:29:01 -07:00
rcu_read_unlock ( ) ;
2015-05-22 15:45:27 -05:00
of_node_put ( np ) ;
return rproc ;
}
2015-06-18 11:44:41 +03:00
# else
struct rproc * rproc_get_by_phandle ( phandle phandle )
{
return NULL ;
}
# endif
2015-05-22 15:45:27 -05:00
EXPORT_SYMBOL ( rproc_get_by_phandle ) ;
2020-11-20 21:20:42 -06:00
/**
* rproc_set_firmware ( ) - assign a new firmware
* @ rproc : rproc handle to which the new firmware is being assigned
* @ fw_name : new firmware name to be assigned
*
* This function allows remoteproc drivers or clients to configure a custom
* firmware name that is different from the default name used during remoteproc
* registration . The function does not trigger a remote processor boot ,
* only sets the firmware name used for a subsequent boot . This function
* should also be called only when the remote processor is offline .
*
* This allows either the userspace to configure a different name through
* sysfs or a kernel - level remoteproc or a remoteproc client driver to set
* a specific firmware when it is controlling the boot and shutdown of the
* remote processor .
*
* Return : 0 on success or a negative value upon failure
*/
int rproc_set_firmware ( struct rproc * rproc , const char * fw_name )
{
struct device * dev ;
int ret , len ;
char * p ;
if ( ! rproc | | ! fw_name )
return - EINVAL ;
dev = rproc - > dev . parent ;
ret = mutex_lock_interruptible ( & rproc - > lock ) ;
if ( ret ) {
dev_err ( dev , " can't lock rproc %s: %d \n " , rproc - > name , ret ) ;
return - EINVAL ;
}
if ( rproc - > state ! = RPROC_OFFLINE ) {
dev_err ( dev , " can't change firmware while running \n " ) ;
ret = - EBUSY ;
goto out ;
}
len = strcspn ( fw_name , " \n " ) ;
if ( ! len ) {
dev_err ( dev , " can't provide empty string for firmware name \n " ) ;
ret = - EINVAL ;
goto out ;
}
p = kstrndup ( fw_name , len , GFP_KERNEL ) ;
if ( ! p ) {
ret = - ENOMEM ;
goto out ;
}
2021-01-18 16:59:04 +00:00
kfree_const ( rproc - > firmware ) ;
2020-11-20 21:20:42 -06:00
rproc - > firmware = p ;
out :
mutex_unlock ( & rproc - > lock ) ;
return ret ;
}
EXPORT_SYMBOL ( rproc_set_firmware ) ;
2020-07-14 13:50:31 -06:00
static int rproc_validate ( struct rproc * rproc )
{
switch ( rproc - > state ) {
case RPROC_OFFLINE :
/*
* An offline processor without a start ( )
* function makes no sense .
*/
if ( ! rproc - > ops - > start )
return - EINVAL ;
break ;
case RPROC_DETACHED :
/*
* A remote processor in a detached state without an
* attach ( ) function makes not sense .
*/
if ( ! rproc - > ops - > attach )
return - EINVAL ;
/*
* When attaching to a remote processor the device memory
* is already available and as such there is no need to have a
* cached table .
*/
if ( rproc - > cached_table )
return - EINVAL ;
break ;
default :
/*
* When adding a remote processor , the state of the device
* can be offline or detached , nothing else .
*/
return - EINVAL ;
}
return 0 ;
}
2011-10-20 16:52:46 +02:00
/**
2012-07-04 16:25:06 +03:00
* rproc_add ( ) - register a remote processor
2011-10-20 16:52:46 +02:00
* @ rproc : the remote processor handle to register
*
* Registers @ rproc with the remoteproc framework , after it has been
* allocated with rproc_alloc ( ) .
*
* This is called by the platform - specific rproc implementation , whenever
* a new remote processor device is probed .
*
* Returns 0 on success and an appropriate error code otherwise .
*
* Note : this function initiates an asynchronous firmware loading
* context , which will look for virtio devices supported by the rproc ' s
* firmware .
*
* If found , those virtio devices will be created and added , so as a result
2012-02-13 22:30:39 +01:00
* of registering this remote processor , additional virtio drivers might be
2011-10-20 16:52:46 +02:00
* probed .
*/
2012-07-04 16:25:06 +03:00
int rproc_add ( struct rproc * rproc )
2011-10-20 16:52:46 +02:00
{
remoteproc: maintain a generic child device for each rproc
For each registered rproc, maintain a generic remoteproc device whose
parent is the low level platform-specific device (commonly a pdev, but
it may certainly be any other type of device too).
With this in hand, the resulting device hierarchy might then look like:
omap-rproc.0
|
- remoteproc0 <---- new !
|
- virtio0
|
- virtio1
|
- rpmsg0
|
- rpmsg1
|
- rpmsg2
Where:
- omap-rproc.0 is the low level device that's bound to the
driver which invokes rproc_register()
- remoteproc0 is the result of this patch, and will be added by the
remoteproc framework when rproc_register() is invoked
- virtio0 and virtio1 are vdevs that are registered by remoteproc
when it realizes that they are supported by the firmware
of the physical remote processor represented by omap-rproc.0
- rpmsg0, rpmsg1 and rpmsg2 are rpmsg devices that represent rpmsg
channels, and are registerd by the rpmsg bus when it gets notified
about their existence
Technically, this patch:
- changes 'struct rproc' to contain this generic remoteproc.x device
- creates a new "remoteproc" type, to which this new generic remoteproc.x
device belong to.
- adds a super simple enumeration method for the indices of the
remoteproc.x devices
- updates all dev_* messaging to use the generic remoteproc.x device
instead of the low level platform-specific device
- updates all dma_* allocations to use the parent of remoteproc.x (where
the platform-specific memory pools, most commonly CMA, are to be found)
Adding this generic device has several merits:
- we can now add remoteproc runtime PM support simply by hooking onto the
new "remoteproc" type
- all remoteproc log messages will now carry a common name prefix
instead of having a platform-specific one
- having a device as part of the rproc struct makes it possible to simplify
refcounting (see subsequent patch)
Thanks to Stephen Boyd <sboyd@codeaurora.org> for suggesting and
discussing these ideas in one of the remoteproc review threads and
to Fernando Guzman Lugo <fernando.lugo@ti.com> for trying them out
with the (upcoming) runtime PM support for remoteproc.
Cc: Fernando Guzman Lugo <fernando.lugo@ti.com>
Reviewed-by: Stephen Boyd <sboyd@codeaurora.org>
Signed-off-by: Ohad Ben-Cohen <ohad@wizery.com>
2012-05-30 22:01:25 +03:00
struct device * dev = & rproc - > dev ;
2012-08-30 13:26:13 -05:00
int ret ;
2011-10-20 16:52:46 +02:00
remoteproc: maintain a generic child device for each rproc
For each registered rproc, maintain a generic remoteproc device whose
parent is the low level platform-specific device (commonly a pdev, but
it may certainly be any other type of device too).
With this in hand, the resulting device hierarchy might then look like:
omap-rproc.0
|
- remoteproc0 <---- new !
|
- virtio0
|
- virtio1
|
- rpmsg0
|
- rpmsg1
|
- rpmsg2
Where:
- omap-rproc.0 is the low level device that's bound to the
driver which invokes rproc_register()
- remoteproc0 is the result of this patch, and will be added by the
remoteproc framework when rproc_register() is invoked
- virtio0 and virtio1 are vdevs that are registered by remoteproc
when it realizes that they are supported by the firmware
of the physical remote processor represented by omap-rproc.0
- rpmsg0, rpmsg1 and rpmsg2 are rpmsg devices that represent rpmsg
channels, and are registerd by the rpmsg bus when it gets notified
about their existence
Technically, this patch:
- changes 'struct rproc' to contain this generic remoteproc.x device
- creates a new "remoteproc" type, to which this new generic remoteproc.x
device belong to.
- adds a super simple enumeration method for the indices of the
remoteproc.x devices
- updates all dev_* messaging to use the generic remoteproc.x device
instead of the low level platform-specific device
- updates all dma_* allocations to use the parent of remoteproc.x (where
the platform-specific memory pools, most commonly CMA, are to be found)
Adding this generic device has several merits:
- we can now add remoteproc runtime PM support simply by hooking onto the
new "remoteproc" type
- all remoteproc log messages will now carry a common name prefix
instead of having a platform-specific one
- having a device as part of the rproc struct makes it possible to simplify
refcounting (see subsequent patch)
Thanks to Stephen Boyd <sboyd@codeaurora.org> for suggesting and
discussing these ideas in one of the remoteproc review threads and
to Fernando Guzman Lugo <fernando.lugo@ti.com> for trying them out
with the (upcoming) runtime PM support for remoteproc.
Cc: Fernando Guzman Lugo <fernando.lugo@ti.com>
Reviewed-by: Stephen Boyd <sboyd@codeaurora.org>
Signed-off-by: Ohad Ben-Cohen <ohad@wizery.com>
2012-05-30 22:01:25 +03:00
ret = device_add ( dev ) ;
if ( ret < 0 )
return ret ;
2011-10-20 16:52:46 +02:00
2020-07-14 13:50:31 -06:00
ret = rproc_validate ( rproc ) ;
if ( ret < 0 )
return ret ;
remoteproc: maintain a generic child device for each rproc
For each registered rproc, maintain a generic remoteproc device whose
parent is the low level platform-specific device (commonly a pdev, but
it may certainly be any other type of device too).
With this in hand, the resulting device hierarchy might then look like:
omap-rproc.0
|
- remoteproc0 <---- new !
|
- virtio0
|
- virtio1
|
- rpmsg0
|
- rpmsg1
|
- rpmsg2
Where:
- omap-rproc.0 is the low level device that's bound to the
driver which invokes rproc_register()
- remoteproc0 is the result of this patch, and will be added by the
remoteproc framework when rproc_register() is invoked
- virtio0 and virtio1 are vdevs that are registered by remoteproc
when it realizes that they are supported by the firmware
of the physical remote processor represented by omap-rproc.0
- rpmsg0, rpmsg1 and rpmsg2 are rpmsg devices that represent rpmsg
channels, and are registerd by the rpmsg bus when it gets notified
about their existence
Technically, this patch:
- changes 'struct rproc' to contain this generic remoteproc.x device
- creates a new "remoteproc" type, to which this new generic remoteproc.x
device belong to.
- adds a super simple enumeration method for the indices of the
remoteproc.x devices
- updates all dev_* messaging to use the generic remoteproc.x device
instead of the low level platform-specific device
- updates all dma_* allocations to use the parent of remoteproc.x (where
the platform-specific memory pools, most commonly CMA, are to be found)
Adding this generic device has several merits:
- we can now add remoteproc runtime PM support simply by hooking onto the
new "remoteproc" type
- all remoteproc log messages will now carry a common name prefix
instead of having a platform-specific one
- having a device as part of the rproc struct makes it possible to simplify
refcounting (see subsequent patch)
Thanks to Stephen Boyd <sboyd@codeaurora.org> for suggesting and
discussing these ideas in one of the remoteproc review threads and
to Fernando Guzman Lugo <fernando.lugo@ti.com> for trying them out
with the (upcoming) runtime PM support for remoteproc.
Cc: Fernando Guzman Lugo <fernando.lugo@ti.com>
Reviewed-by: Stephen Boyd <sboyd@codeaurora.org>
Signed-off-by: Ohad Ben-Cohen <ohad@wizery.com>
2012-05-30 22:01:25 +03:00
dev_info ( dev , " %s is available \n " , rproc - > name ) ;
2011-10-20 16:52:46 +02:00
/* create debugfs entries */
rproc_create_debug_dir ( rproc ) ;
2017-01-24 15:13:00 -08:00
2020-07-29 10:40:01 -07:00
/* add char device for this remoteproc */
ret = rproc_char_device_add ( rproc ) ;
if ( ret < 0 )
return ret ;
2020-07-14 13:50:35 -06:00
/*
* Remind ourselves the remote processor has been attached to rather
* than booted by the remoteproc core . This is important because the
* RPROC_DETACHED state will be lost as soon as the remote processor
* has been attached to . Used in firmware_show ( ) and reset in
* rproc_stop ( ) .
*/
if ( rproc - > state = = RPROC_DETACHED )
rproc - > autonomous = true ;
2017-01-24 15:13:00 -08:00
/* if rproc is marked always-on, request it to boot */
if ( rproc - > auto_boot ) {
2017-01-24 15:13:01 -08:00
ret = rproc_trigger_auto_boot ( rproc ) ;
2017-01-24 15:13:00 -08:00
if ( ret < 0 )
return ret ;
}
2011-10-20 16:52:46 +02:00
2016-05-25 15:41:28 -05:00
/* expose to rproc_get_by_phandle users */
mutex_lock ( & rproc_list_mutex ) ;
2020-03-23 22:29:01 -07:00
list_add_rcu ( & rproc - > node , & rproc_list ) ;
2016-05-25 15:41:28 -05:00
mutex_unlock ( & rproc_list_mutex ) ;
return 0 ;
2011-10-20 16:52:46 +02:00
}
2012-07-04 16:25:06 +03:00
EXPORT_SYMBOL ( rproc_add ) ;
2011-10-20 16:52:46 +02:00
2020-04-17 19:00:37 +02:00
static void devm_rproc_remove ( void * rproc )
{
rproc_del ( rproc ) ;
}
/**
* devm_rproc_add ( ) - resource managed rproc_add ( )
* @ dev : the underlying device
* @ rproc : the remote processor handle to register
*
* This function performs like rproc_add ( ) but the registered rproc device will
* automatically be removed on driver detach .
*
* Returns : 0 on success , negative errno on failure
*/
int devm_rproc_add ( struct device * dev , struct rproc * rproc )
{
int err ;
err = rproc_add ( rproc ) ;
if ( err )
return err ;
return devm_add_action_or_reset ( dev , devm_rproc_remove , rproc ) ;
}
EXPORT_SYMBOL ( devm_rproc_add ) ;
remoteproc: maintain a generic child device for each rproc
For each registered rproc, maintain a generic remoteproc device whose
parent is the low level platform-specific device (commonly a pdev, but
it may certainly be any other type of device too).
With this in hand, the resulting device hierarchy might then look like:
omap-rproc.0
|
- remoteproc0 <---- new !
|
- virtio0
|
- virtio1
|
- rpmsg0
|
- rpmsg1
|
- rpmsg2
Where:
- omap-rproc.0 is the low level device that's bound to the
driver which invokes rproc_register()
- remoteproc0 is the result of this patch, and will be added by the
remoteproc framework when rproc_register() is invoked
- virtio0 and virtio1 are vdevs that are registered by remoteproc
when it realizes that they are supported by the firmware
of the physical remote processor represented by omap-rproc.0
- rpmsg0, rpmsg1 and rpmsg2 are rpmsg devices that represent rpmsg
channels, and are registerd by the rpmsg bus when it gets notified
about their existence
Technically, this patch:
- changes 'struct rproc' to contain this generic remoteproc.x device
- creates a new "remoteproc" type, to which this new generic remoteproc.x
device belong to.
- adds a super simple enumeration method for the indices of the
remoteproc.x devices
- updates all dev_* messaging to use the generic remoteproc.x device
instead of the low level platform-specific device
- updates all dma_* allocations to use the parent of remoteproc.x (where
the platform-specific memory pools, most commonly CMA, are to be found)
Adding this generic device has several merits:
- we can now add remoteproc runtime PM support simply by hooking onto the
new "remoteproc" type
- all remoteproc log messages will now carry a common name prefix
instead of having a platform-specific one
- having a device as part of the rproc struct makes it possible to simplify
refcounting (see subsequent patch)
Thanks to Stephen Boyd <sboyd@codeaurora.org> for suggesting and
discussing these ideas in one of the remoteproc review threads and
to Fernando Guzman Lugo <fernando.lugo@ti.com> for trying them out
with the (upcoming) runtime PM support for remoteproc.
Cc: Fernando Guzman Lugo <fernando.lugo@ti.com>
Reviewed-by: Stephen Boyd <sboyd@codeaurora.org>
Signed-off-by: Ohad Ben-Cohen <ohad@wizery.com>
2012-05-30 22:01:25 +03:00
/**
* rproc_type_release ( ) - release a remote processor instance
* @ dev : the rproc ' s device
*
* This function should _never_ be called directly .
*
* It will be called by the driver core when no one holds a valid pointer
* to @ dev anymore .
*/
static void rproc_type_release ( struct device * dev )
{
struct rproc * rproc = container_of ( dev , struct rproc , dev ) ;
2012-05-30 22:02:24 +03:00
dev_info ( & rproc - > dev , " releasing %s \n " , rproc - > name ) ;
remoteproc: maintain a generic child device for each rproc
For each registered rproc, maintain a generic remoteproc device whose
parent is the low level platform-specific device (commonly a pdev, but
it may certainly be any other type of device too).
With this in hand, the resulting device hierarchy might then look like:
omap-rproc.0
|
- remoteproc0 <---- new !
|
- virtio0
|
- virtio1
|
- rpmsg0
|
- rpmsg1
|
- rpmsg2
Where:
- omap-rproc.0 is the low level device that's bound to the
driver which invokes rproc_register()
- remoteproc0 is the result of this patch, and will be added by the
remoteproc framework when rproc_register() is invoked
- virtio0 and virtio1 are vdevs that are registered by remoteproc
when it realizes that they are supported by the firmware
of the physical remote processor represented by omap-rproc.0
- rpmsg0, rpmsg1 and rpmsg2 are rpmsg devices that represent rpmsg
channels, and are registerd by the rpmsg bus when it gets notified
about their existence
Technically, this patch:
- changes 'struct rproc' to contain this generic remoteproc.x device
- creates a new "remoteproc" type, to which this new generic remoteproc.x
device belong to.
- adds a super simple enumeration method for the indices of the
remoteproc.x devices
- updates all dev_* messaging to use the generic remoteproc.x device
instead of the low level platform-specific device
- updates all dma_* allocations to use the parent of remoteproc.x (where
the platform-specific memory pools, most commonly CMA, are to be found)
Adding this generic device has several merits:
- we can now add remoteproc runtime PM support simply by hooking onto the
new "remoteproc" type
- all remoteproc log messages will now carry a common name prefix
instead of having a platform-specific one
- having a device as part of the rproc struct makes it possible to simplify
refcounting (see subsequent patch)
Thanks to Stephen Boyd <sboyd@codeaurora.org> for suggesting and
discussing these ideas in one of the remoteproc review threads and
to Fernando Guzman Lugo <fernando.lugo@ti.com> for trying them out
with the (upcoming) runtime PM support for remoteproc.
Cc: Fernando Guzman Lugo <fernando.lugo@ti.com>
Reviewed-by: Stephen Boyd <sboyd@codeaurora.org>
Signed-off-by: Ohad Ben-Cohen <ohad@wizery.com>
2012-05-30 22:01:25 +03:00
idr_destroy ( & rproc - > notifyids ) ;
if ( rproc - > index > = 0 )
ida_simple_remove ( & rproc_dev_index , rproc - > index ) ;
2020-04-20 17:15:58 -06:00
kfree_const ( rproc - > firmware ) ;
2020-04-16 19:20:36 -05:00
kfree_const ( rproc - > name ) ;
2018-01-05 15:58:00 -08:00
kfree ( rproc - > ops ) ;
remoteproc: maintain a generic child device for each rproc
For each registered rproc, maintain a generic remoteproc device whose
parent is the low level platform-specific device (commonly a pdev, but
it may certainly be any other type of device too).
With this in hand, the resulting device hierarchy might then look like:
omap-rproc.0
|
- remoteproc0 <---- new !
|
- virtio0
|
- virtio1
|
- rpmsg0
|
- rpmsg1
|
- rpmsg2
Where:
- omap-rproc.0 is the low level device that's bound to the
driver which invokes rproc_register()
- remoteproc0 is the result of this patch, and will be added by the
remoteproc framework when rproc_register() is invoked
- virtio0 and virtio1 are vdevs that are registered by remoteproc
when it realizes that they are supported by the firmware
of the physical remote processor represented by omap-rproc.0
- rpmsg0, rpmsg1 and rpmsg2 are rpmsg devices that represent rpmsg
channels, and are registerd by the rpmsg bus when it gets notified
about their existence
Technically, this patch:
- changes 'struct rproc' to contain this generic remoteproc.x device
- creates a new "remoteproc" type, to which this new generic remoteproc.x
device belong to.
- adds a super simple enumeration method for the indices of the
remoteproc.x devices
- updates all dev_* messaging to use the generic remoteproc.x device
instead of the low level platform-specific device
- updates all dma_* allocations to use the parent of remoteproc.x (where
the platform-specific memory pools, most commonly CMA, are to be found)
Adding this generic device has several merits:
- we can now add remoteproc runtime PM support simply by hooking onto the
new "remoteproc" type
- all remoteproc log messages will now carry a common name prefix
instead of having a platform-specific one
- having a device as part of the rproc struct makes it possible to simplify
refcounting (see subsequent patch)
Thanks to Stephen Boyd <sboyd@codeaurora.org> for suggesting and
discussing these ideas in one of the remoteproc review threads and
to Fernando Guzman Lugo <fernando.lugo@ti.com> for trying them out
with the (upcoming) runtime PM support for remoteproc.
Cc: Fernando Guzman Lugo <fernando.lugo@ti.com>
Reviewed-by: Stephen Boyd <sboyd@codeaurora.org>
Signed-off-by: Ohad Ben-Cohen <ohad@wizery.com>
2012-05-30 22:01:25 +03:00
kfree ( rproc ) ;
}
2017-08-19 13:52:22 +05:30
static const struct device_type rproc_type = {
remoteproc: maintain a generic child device for each rproc
For each registered rproc, maintain a generic remoteproc device whose
parent is the low level platform-specific device (commonly a pdev, but
it may certainly be any other type of device too).
With this in hand, the resulting device hierarchy might then look like:
omap-rproc.0
|
- remoteproc0 <---- new !
|
- virtio0
|
- virtio1
|
- rpmsg0
|
- rpmsg1
|
- rpmsg2
Where:
- omap-rproc.0 is the low level device that's bound to the
driver which invokes rproc_register()
- remoteproc0 is the result of this patch, and will be added by the
remoteproc framework when rproc_register() is invoked
- virtio0 and virtio1 are vdevs that are registered by remoteproc
when it realizes that they are supported by the firmware
of the physical remote processor represented by omap-rproc.0
- rpmsg0, rpmsg1 and rpmsg2 are rpmsg devices that represent rpmsg
channels, and are registerd by the rpmsg bus when it gets notified
about their existence
Technically, this patch:
- changes 'struct rproc' to contain this generic remoteproc.x device
- creates a new "remoteproc" type, to which this new generic remoteproc.x
device belong to.
- adds a super simple enumeration method for the indices of the
remoteproc.x devices
- updates all dev_* messaging to use the generic remoteproc.x device
instead of the low level platform-specific device
- updates all dma_* allocations to use the parent of remoteproc.x (where
the platform-specific memory pools, most commonly CMA, are to be found)
Adding this generic device has several merits:
- we can now add remoteproc runtime PM support simply by hooking onto the
new "remoteproc" type
- all remoteproc log messages will now carry a common name prefix
instead of having a platform-specific one
- having a device as part of the rproc struct makes it possible to simplify
refcounting (see subsequent patch)
Thanks to Stephen Boyd <sboyd@codeaurora.org> for suggesting and
discussing these ideas in one of the remoteproc review threads and
to Fernando Guzman Lugo <fernando.lugo@ti.com> for trying them out
with the (upcoming) runtime PM support for remoteproc.
Cc: Fernando Guzman Lugo <fernando.lugo@ti.com>
Reviewed-by: Stephen Boyd <sboyd@codeaurora.org>
Signed-off-by: Ohad Ben-Cohen <ohad@wizery.com>
2012-05-30 22:01:25 +03:00
. name = " remoteproc " ,
. release = rproc_type_release ,
} ;
2011-10-20 16:52:46 +02:00
2020-04-15 14:48:53 -06:00
static int rproc_alloc_firmware ( struct rproc * rproc ,
const char * name , const char * firmware )
{
2020-04-20 17:15:58 -06:00
const char * p ;
2020-04-15 14:48:53 -06:00
2020-04-20 17:15:59 -06:00
/*
* Allocate a firmware name if the caller gave us one to work
* with . Otherwise construct a new one using a default pattern .
*/
if ( firmware )
2020-04-20 17:15:58 -06:00
p = kstrdup_const ( firmware , GFP_KERNEL ) ;
2020-04-20 17:15:59 -06:00
else
p = kasprintf ( GFP_KERNEL , " rproc-%s-fw " , name ) ;
2020-04-15 14:48:54 -06:00
if ( ! p )
return - ENOMEM ;
2020-04-15 14:48:53 -06:00
rproc - > firmware = p ;
return 0 ;
}
2020-04-20 17:16:00 -06:00
static int rproc_alloc_ops ( struct rproc * rproc , const struct rproc_ops * ops )
{
rproc - > ops = kmemdup ( ops , sizeof ( * ops ) , GFP_KERNEL ) ;
if ( ! rproc - > ops )
return - ENOMEM ;
2020-11-19 13:05:32 -08:00
/* Default to rproc_coredump if no coredump function is specified */
if ( ! rproc - > ops - > coredump )
rproc - > ops - > coredump = rproc_coredump ;
2020-04-20 17:16:00 -06:00
if ( rproc - > ops - > load )
return 0 ;
/* Default to ELF loader if no load function is specified */
rproc - > ops - > load = rproc_elf_load_segments ;
rproc - > ops - > parse_fw = rproc_elf_load_rsc_table ;
rproc - > ops - > find_loaded_rsc_table = rproc_elf_find_loaded_rsc_table ;
2020-04-22 11:30:17 +02:00
rproc - > ops - > sanity_check = rproc_elf_sanity_check ;
2020-04-20 17:16:00 -06:00
rproc - > ops - > get_boot_addr = rproc_elf_get_boot_addr ;
return 0 ;
}
2011-10-20 16:52:46 +02:00
/**
* rproc_alloc ( ) - allocate a remote processor handle
* @ dev : the underlying device
* @ name : name of this remote processor
* @ ops : platform - specific handlers ( mainly start / stop )
2013-03-28 18:41:44 -07:00
* @ firmware : name of firmware file to load , can be NULL
2011-10-20 16:52:46 +02:00
* @ len : length of private data needed by the rproc driver ( in bytes )
*
* Allocates a new remote processor handle , but does not register
2013-03-28 18:41:44 -07:00
* it yet . if @ firmware is NULL , a default name is used .
2011-10-20 16:52:46 +02:00
*
* This function should be used by rproc implementations during initialization
* of the remote processor .
*
* After creating an rproc handle using this function , and when ready ,
2012-07-04 16:25:06 +03:00
* implementations should then call rproc_add ( ) to complete
2011-10-20 16:52:46 +02:00
* the registration of the remote processor .
*
* On success the new rproc is returned , and on failure , NULL .
*
* Note : _never_ directly deallocate @ rproc , even if it was not registered
2016-10-02 17:46:38 -07:00
* yet . Instead , when you need to unroll rproc_alloc ( ) , use rproc_free ( ) .
2011-10-20 16:52:46 +02:00
*/
struct rproc * rproc_alloc ( struct device * dev , const char * name ,
2016-08-12 18:42:20 -05:00
const struct rproc_ops * ops ,
const char * firmware , int len )
2011-10-20 16:52:46 +02:00
{
struct rproc * rproc ;
if ( ! dev | | ! name | | ! ops )
return NULL ;
2016-10-17 16:48:58 +01:00
rproc = kzalloc ( sizeof ( struct rproc ) + len , GFP_KERNEL ) ;
2020-04-15 14:48:53 -06:00
if ( ! rproc )
2016-10-17 16:48:58 +01:00
return NULL ;
2020-04-15 14:48:53 -06:00
2011-10-20 16:52:46 +02:00
rproc - > priv = & rproc [ 1 ] ;
2016-08-11 14:52:50 -07:00
rproc - > auto_boot = true ;
2020-04-10 12:24:32 +02:00
rproc - > elf_class = ELFCLASSNONE ;
rproc - > elf_machine = EM_NONE ;
2011-10-20 16:52:46 +02:00
remoteproc: maintain a generic child device for each rproc
For each registered rproc, maintain a generic remoteproc device whose
parent is the low level platform-specific device (commonly a pdev, but
it may certainly be any other type of device too).
With this in hand, the resulting device hierarchy might then look like:
omap-rproc.0
|
- remoteproc0 <---- new !
|
- virtio0
|
- virtio1
|
- rpmsg0
|
- rpmsg1
|
- rpmsg2
Where:
- omap-rproc.0 is the low level device that's bound to the
driver which invokes rproc_register()
- remoteproc0 is the result of this patch, and will be added by the
remoteproc framework when rproc_register() is invoked
- virtio0 and virtio1 are vdevs that are registered by remoteproc
when it realizes that they are supported by the firmware
of the physical remote processor represented by omap-rproc.0
- rpmsg0, rpmsg1 and rpmsg2 are rpmsg devices that represent rpmsg
channels, and are registerd by the rpmsg bus when it gets notified
about their existence
Technically, this patch:
- changes 'struct rproc' to contain this generic remoteproc.x device
- creates a new "remoteproc" type, to which this new generic remoteproc.x
device belong to.
- adds a super simple enumeration method for the indices of the
remoteproc.x devices
- updates all dev_* messaging to use the generic remoteproc.x device
instead of the low level platform-specific device
- updates all dma_* allocations to use the parent of remoteproc.x (where
the platform-specific memory pools, most commonly CMA, are to be found)
Adding this generic device has several merits:
- we can now add remoteproc runtime PM support simply by hooking onto the
new "remoteproc" type
- all remoteproc log messages will now carry a common name prefix
instead of having a platform-specific one
- having a device as part of the rproc struct makes it possible to simplify
refcounting (see subsequent patch)
Thanks to Stephen Boyd <sboyd@codeaurora.org> for suggesting and
discussing these ideas in one of the remoteproc review threads and
to Fernando Guzman Lugo <fernando.lugo@ti.com> for trying them out
with the (upcoming) runtime PM support for remoteproc.
Cc: Fernando Guzman Lugo <fernando.lugo@ti.com>
Reviewed-by: Stephen Boyd <sboyd@codeaurora.org>
Signed-off-by: Ohad Ben-Cohen <ohad@wizery.com>
2012-05-30 22:01:25 +03:00
device_initialize ( & rproc - > dev ) ;
rproc - > dev . parent = dev ;
rproc - > dev . type = & rproc_type ;
2016-10-19 13:05:47 +01:00
rproc - > dev . class = & rproc_class ;
2017-08-27 22:34:53 -07:00
rproc - > dev . driver_data = rproc ;
2020-04-15 14:48:52 -06:00
idr_init ( & rproc - > notifyids ) ;
remoteproc: maintain a generic child device for each rproc
For each registered rproc, maintain a generic remoteproc device whose
parent is the low level platform-specific device (commonly a pdev, but
it may certainly be any other type of device too).
With this in hand, the resulting device hierarchy might then look like:
omap-rproc.0
|
- remoteproc0 <---- new !
|
- virtio0
|
- virtio1
|
- rpmsg0
|
- rpmsg1
|
- rpmsg2
Where:
- omap-rproc.0 is the low level device that's bound to the
driver which invokes rproc_register()
- remoteproc0 is the result of this patch, and will be added by the
remoteproc framework when rproc_register() is invoked
- virtio0 and virtio1 are vdevs that are registered by remoteproc
when it realizes that they are supported by the firmware
of the physical remote processor represented by omap-rproc.0
- rpmsg0, rpmsg1 and rpmsg2 are rpmsg devices that represent rpmsg
channels, and are registerd by the rpmsg bus when it gets notified
about their existence
Technically, this patch:
- changes 'struct rproc' to contain this generic remoteproc.x device
- creates a new "remoteproc" type, to which this new generic remoteproc.x
device belong to.
- adds a super simple enumeration method for the indices of the
remoteproc.x devices
- updates all dev_* messaging to use the generic remoteproc.x device
instead of the low level platform-specific device
- updates all dma_* allocations to use the parent of remoteproc.x (where
the platform-specific memory pools, most commonly CMA, are to be found)
Adding this generic device has several merits:
- we can now add remoteproc runtime PM support simply by hooking onto the
new "remoteproc" type
- all remoteproc log messages will now carry a common name prefix
instead of having a platform-specific one
- having a device as part of the rproc struct makes it possible to simplify
refcounting (see subsequent patch)
Thanks to Stephen Boyd <sboyd@codeaurora.org> for suggesting and
discussing these ideas in one of the remoteproc review threads and
to Fernando Guzman Lugo <fernando.lugo@ti.com> for trying them out
with the (upcoming) runtime PM support for remoteproc.
Cc: Fernando Guzman Lugo <fernando.lugo@ti.com>
Reviewed-by: Stephen Boyd <sboyd@codeaurora.org>
Signed-off-by: Ohad Ben-Cohen <ohad@wizery.com>
2012-05-30 22:01:25 +03:00
2020-04-16 19:20:36 -05:00
rproc - > name = kstrdup_const ( name , GFP_KERNEL ) ;
if ( ! rproc - > name )
goto put_device ;
2020-04-20 17:16:01 -06:00
if ( rproc_alloc_firmware ( rproc , name , firmware ) )
goto put_device ;
if ( rproc_alloc_ops ( rproc , ops ) )
goto put_device ;
remoteproc: maintain a generic child device for each rproc
For each registered rproc, maintain a generic remoteproc device whose
parent is the low level platform-specific device (commonly a pdev, but
it may certainly be any other type of device too).
With this in hand, the resulting device hierarchy might then look like:
omap-rproc.0
|
- remoteproc0 <---- new !
|
- virtio0
|
- virtio1
|
- rpmsg0
|
- rpmsg1
|
- rpmsg2
Where:
- omap-rproc.0 is the low level device that's bound to the
driver which invokes rproc_register()
- remoteproc0 is the result of this patch, and will be added by the
remoteproc framework when rproc_register() is invoked
- virtio0 and virtio1 are vdevs that are registered by remoteproc
when it realizes that they are supported by the firmware
of the physical remote processor represented by omap-rproc.0
- rpmsg0, rpmsg1 and rpmsg2 are rpmsg devices that represent rpmsg
channels, and are registerd by the rpmsg bus when it gets notified
about their existence
Technically, this patch:
- changes 'struct rproc' to contain this generic remoteproc.x device
- creates a new "remoteproc" type, to which this new generic remoteproc.x
device belong to.
- adds a super simple enumeration method for the indices of the
remoteproc.x devices
- updates all dev_* messaging to use the generic remoteproc.x device
instead of the low level platform-specific device
- updates all dma_* allocations to use the parent of remoteproc.x (where
the platform-specific memory pools, most commonly CMA, are to be found)
Adding this generic device has several merits:
- we can now add remoteproc runtime PM support simply by hooking onto the
new "remoteproc" type
- all remoteproc log messages will now carry a common name prefix
instead of having a platform-specific one
- having a device as part of the rproc struct makes it possible to simplify
refcounting (see subsequent patch)
Thanks to Stephen Boyd <sboyd@codeaurora.org> for suggesting and
discussing these ideas in one of the remoteproc review threads and
to Fernando Guzman Lugo <fernando.lugo@ti.com> for trying them out
with the (upcoming) runtime PM support for remoteproc.
Cc: Fernando Guzman Lugo <fernando.lugo@ti.com>
Reviewed-by: Stephen Boyd <sboyd@codeaurora.org>
Signed-off-by: Ohad Ben-Cohen <ohad@wizery.com>
2012-05-30 22:01:25 +03:00
/* Assign a unique device index and name */
rproc - > index = ida_simple_get ( & rproc_dev_index , 0 , 0 , GFP_KERNEL ) ;
if ( rproc - > index < 0 ) {
dev_err ( dev , " ida_simple_get failed: %d \n " , rproc - > index ) ;
2020-04-20 17:16:01 -06:00
goto put_device ;
remoteproc: maintain a generic child device for each rproc
For each registered rproc, maintain a generic remoteproc device whose
parent is the low level platform-specific device (commonly a pdev, but
it may certainly be any other type of device too).
With this in hand, the resulting device hierarchy might then look like:
omap-rproc.0
|
- remoteproc0 <---- new !
|
- virtio0
|
- virtio1
|
- rpmsg0
|
- rpmsg1
|
- rpmsg2
Where:
- omap-rproc.0 is the low level device that's bound to the
driver which invokes rproc_register()
- remoteproc0 is the result of this patch, and will be added by the
remoteproc framework when rproc_register() is invoked
- virtio0 and virtio1 are vdevs that are registered by remoteproc
when it realizes that they are supported by the firmware
of the physical remote processor represented by omap-rproc.0
- rpmsg0, rpmsg1 and rpmsg2 are rpmsg devices that represent rpmsg
channels, and are registerd by the rpmsg bus when it gets notified
about their existence
Technically, this patch:
- changes 'struct rproc' to contain this generic remoteproc.x device
- creates a new "remoteproc" type, to which this new generic remoteproc.x
device belong to.
- adds a super simple enumeration method for the indices of the
remoteproc.x devices
- updates all dev_* messaging to use the generic remoteproc.x device
instead of the low level platform-specific device
- updates all dma_* allocations to use the parent of remoteproc.x (where
the platform-specific memory pools, most commonly CMA, are to be found)
Adding this generic device has several merits:
- we can now add remoteproc runtime PM support simply by hooking onto the
new "remoteproc" type
- all remoteproc log messages will now carry a common name prefix
instead of having a platform-specific one
- having a device as part of the rproc struct makes it possible to simplify
refcounting (see subsequent patch)
Thanks to Stephen Boyd <sboyd@codeaurora.org> for suggesting and
discussing these ideas in one of the remoteproc review threads and
to Fernando Guzman Lugo <fernando.lugo@ti.com> for trying them out
with the (upcoming) runtime PM support for remoteproc.
Cc: Fernando Guzman Lugo <fernando.lugo@ti.com>
Reviewed-by: Stephen Boyd <sboyd@codeaurora.org>
Signed-off-by: Ohad Ben-Cohen <ohad@wizery.com>
2012-05-30 22:01:25 +03:00
}
dev_set_name ( & rproc - > dev , " remoteproc%d " , rproc - > index ) ;
2011-10-20 16:52:46 +02:00
atomic_set ( & rproc - > power , 0 ) ;
mutex_init ( & rproc - > lock ) ;
INIT_LIST_HEAD ( & rproc - > carveouts ) ;
INIT_LIST_HEAD ( & rproc - > mappings ) ;
INIT_LIST_HEAD ( & rproc - > traces ) ;
2012-02-13 22:30:39 +01:00
INIT_LIST_HEAD ( & rproc - > rvdevs ) ;
2016-10-19 19:40:02 -07:00
INIT_LIST_HEAD ( & rproc - > subdevs ) ;
2018-01-05 16:04:17 -08:00
INIT_LIST_HEAD ( & rproc - > dump_segments ) ;
2011-10-20 16:52:46 +02:00
2012-08-30 13:26:12 -05:00
INIT_WORK ( & rproc - > crash_handler , rproc_crash_handler_work ) ;
2011-10-20 16:52:46 +02:00
rproc - > state = RPROC_OFFLINE ;
return rproc ;
2020-04-15 14:48:53 -06:00
2020-04-20 17:16:01 -06:00
put_device :
put_device ( & rproc - > dev ) ;
2020-04-15 14:48:53 -06:00
return NULL ;
2011-10-20 16:52:46 +02:00
}
EXPORT_SYMBOL ( rproc_alloc ) ;
/**
2016-10-02 17:46:38 -07:00
* rproc_free ( ) - unroll rproc_alloc ( )
* @ rproc : the remote processor handle
*
* This function decrements the rproc dev refcount .
*
* If no one holds any reference to rproc anymore , then its refcount would
* now drop to zero , and it would be freed .
*/
void rproc_free ( struct rproc * rproc )
{
put_device ( & rproc - > dev ) ;
}
EXPORT_SYMBOL ( rproc_free ) ;
/**
* rproc_put ( ) - release rproc reference
2011-10-20 16:52:46 +02:00
* @ rproc : the remote processor handle
*
2012-07-02 11:41:16 +03:00
* This function decrements the rproc dev refcount .
2011-10-20 16:52:46 +02:00
*
2012-07-02 11:41:16 +03:00
* If no one holds any reference to rproc anymore , then its refcount would
* now drop to zero , and it would be freed .
2011-10-20 16:52:46 +02:00
*/
2012-07-04 16:25:06 +03:00
void rproc_put ( struct rproc * rproc )
2011-10-20 16:52:46 +02:00
{
2016-10-02 17:46:39 -07:00
module_put ( rproc - > dev . parent - > driver - > owner ) ;
remoteproc: maintain a generic child device for each rproc
For each registered rproc, maintain a generic remoteproc device whose
parent is the low level platform-specific device (commonly a pdev, but
it may certainly be any other type of device too).
With this in hand, the resulting device hierarchy might then look like:
omap-rproc.0
|
- remoteproc0 <---- new !
|
- virtio0
|
- virtio1
|
- rpmsg0
|
- rpmsg1
|
- rpmsg2
Where:
- omap-rproc.0 is the low level device that's bound to the
driver which invokes rproc_register()
- remoteproc0 is the result of this patch, and will be added by the
remoteproc framework when rproc_register() is invoked
- virtio0 and virtio1 are vdevs that are registered by remoteproc
when it realizes that they are supported by the firmware
of the physical remote processor represented by omap-rproc.0
- rpmsg0, rpmsg1 and rpmsg2 are rpmsg devices that represent rpmsg
channels, and are registerd by the rpmsg bus when it gets notified
about their existence
Technically, this patch:
- changes 'struct rproc' to contain this generic remoteproc.x device
- creates a new "remoteproc" type, to which this new generic remoteproc.x
device belong to.
- adds a super simple enumeration method for the indices of the
remoteproc.x devices
- updates all dev_* messaging to use the generic remoteproc.x device
instead of the low level platform-specific device
- updates all dma_* allocations to use the parent of remoteproc.x (where
the platform-specific memory pools, most commonly CMA, are to be found)
Adding this generic device has several merits:
- we can now add remoteproc runtime PM support simply by hooking onto the
new "remoteproc" type
- all remoteproc log messages will now carry a common name prefix
instead of having a platform-specific one
- having a device as part of the rproc struct makes it possible to simplify
refcounting (see subsequent patch)
Thanks to Stephen Boyd <sboyd@codeaurora.org> for suggesting and
discussing these ideas in one of the remoteproc review threads and
to Fernando Guzman Lugo <fernando.lugo@ti.com> for trying them out
with the (upcoming) runtime PM support for remoteproc.
Cc: Fernando Guzman Lugo <fernando.lugo@ti.com>
Reviewed-by: Stephen Boyd <sboyd@codeaurora.org>
Signed-off-by: Ohad Ben-Cohen <ohad@wizery.com>
2012-05-30 22:01:25 +03:00
put_device ( & rproc - > dev ) ;
2011-10-20 16:52:46 +02:00
}
2012-07-04 16:25:06 +03:00
EXPORT_SYMBOL ( rproc_put ) ;
2011-10-20 16:52:46 +02:00
/**
2012-07-04 16:25:06 +03:00
* rproc_del ( ) - unregister a remote processor
2011-10-20 16:52:46 +02:00
* @ rproc : rproc handle to unregister
*
* This function should be called when the platform specific rproc
* implementation decides to remove the rproc device . it should
2012-07-04 16:25:06 +03:00
* _only_ be called if a previous invocation of rproc_add ( )
2011-10-20 16:52:46 +02:00
* has completed successfully .
*
2012-07-04 16:25:06 +03:00
* After rproc_del ( ) returns , @ rproc isn ' t freed yet , because
2012-07-02 11:41:16 +03:00
* of the outstanding reference created by rproc_alloc . To decrement that
2016-10-02 17:46:38 -07:00
* one last refcount , one still needs to call rproc_free ( ) .
2011-10-20 16:52:46 +02:00
*
* Returns 0 on success and - EINVAL if @ rproc isn ' t valid .
*/
2012-07-04 16:25:06 +03:00
int rproc_del ( struct rproc * rproc )
2011-10-20 16:52:46 +02:00
{
if ( ! rproc )
return - EINVAL ;
2016-08-11 14:52:50 -07:00
/* TODO: make sure this works with rproc->power > 1 */
2021-03-12 09:24:37 -07:00
rproc_shutdown ( rproc ) ;
2016-08-11 14:52:50 -07:00
2017-01-23 17:53:19 -08:00
mutex_lock ( & rproc - > lock ) ;
rproc - > state = RPROC_DELETED ;
mutex_unlock ( & rproc - > lock ) ;
2017-01-23 17:48:48 -08:00
rproc_delete_debug_dir ( rproc ) ;
2020-07-29 10:40:01 -07:00
rproc_char_device_remove ( rproc ) ;
2017-01-23 17:48:48 -08:00
2015-05-22 15:45:27 -05:00
/* the rproc is downref'ed as soon as it's removed from the klist */
mutex_lock ( & rproc_list_mutex ) ;
2020-03-23 22:29:01 -07:00
list_del_rcu ( & rproc - > node ) ;
2015-05-22 15:45:27 -05:00
mutex_unlock ( & rproc_list_mutex ) ;
2020-03-23 22:29:01 -07:00
/* Ensure that no readers of rproc_list are still active */
synchronize_rcu ( ) ;
remoteproc: maintain a generic child device for each rproc
For each registered rproc, maintain a generic remoteproc device whose
parent is the low level platform-specific device (commonly a pdev, but
it may certainly be any other type of device too).
With this in hand, the resulting device hierarchy might then look like:
omap-rproc.0
|
- remoteproc0 <---- new !
|
- virtio0
|
- virtio1
|
- rpmsg0
|
- rpmsg1
|
- rpmsg2
Where:
- omap-rproc.0 is the low level device that's bound to the
driver which invokes rproc_register()
- remoteproc0 is the result of this patch, and will be added by the
remoteproc framework when rproc_register() is invoked
- virtio0 and virtio1 are vdevs that are registered by remoteproc
when it realizes that they are supported by the firmware
of the physical remote processor represented by omap-rproc.0
- rpmsg0, rpmsg1 and rpmsg2 are rpmsg devices that represent rpmsg
channels, and are registerd by the rpmsg bus when it gets notified
about their existence
Technically, this patch:
- changes 'struct rproc' to contain this generic remoteproc.x device
- creates a new "remoteproc" type, to which this new generic remoteproc.x
device belong to.
- adds a super simple enumeration method for the indices of the
remoteproc.x devices
- updates all dev_* messaging to use the generic remoteproc.x device
instead of the low level platform-specific device
- updates all dma_* allocations to use the parent of remoteproc.x (where
the platform-specific memory pools, most commonly CMA, are to be found)
Adding this generic device has several merits:
- we can now add remoteproc runtime PM support simply by hooking onto the
new "remoteproc" type
- all remoteproc log messages will now carry a common name prefix
instead of having a platform-specific one
- having a device as part of the rproc struct makes it possible to simplify
refcounting (see subsequent patch)
Thanks to Stephen Boyd <sboyd@codeaurora.org> for suggesting and
discussing these ideas in one of the remoteproc review threads and
to Fernando Guzman Lugo <fernando.lugo@ti.com> for trying them out
with the (upcoming) runtime PM support for remoteproc.
Cc: Fernando Guzman Lugo <fernando.lugo@ti.com>
Reviewed-by: Stephen Boyd <sboyd@codeaurora.org>
Signed-off-by: Ohad Ben-Cohen <ohad@wizery.com>
2012-05-30 22:01:25 +03:00
device_del ( & rproc - > dev ) ;
2011-10-20 16:52:46 +02:00
return 0 ;
}
2012-07-04 16:25:06 +03:00
EXPORT_SYMBOL ( rproc_del ) ;
2011-10-20 16:52:46 +02:00
2020-04-17 19:00:37 +02:00
static void devm_rproc_free ( struct device * dev , void * res )
{
rproc_free ( * ( struct rproc * * ) res ) ;
}
/**
* devm_rproc_alloc ( ) - resource managed rproc_alloc ( )
* @ dev : the underlying device
* @ name : name of this remote processor
* @ ops : platform - specific handlers ( mainly start / stop )
* @ firmware : name of firmware file to load , can be NULL
* @ len : length of private data needed by the rproc driver ( in bytes )
*
* This function performs like rproc_alloc ( ) but the acquired rproc device will
* automatically be released on driver detach .
*
* Returns : new rproc instance , or NULL on failure
*/
struct rproc * devm_rproc_alloc ( struct device * dev , const char * name ,
const struct rproc_ops * ops ,
const char * firmware , int len )
{
struct rproc * * ptr , * rproc ;
ptr = devres_alloc ( devm_rproc_free , sizeof ( * ptr ) , GFP_KERNEL ) ;
if ( ! ptr )
2020-05-20 15:07:05 +03:00
return NULL ;
2020-04-17 19:00:37 +02:00
rproc = rproc_alloc ( dev , name , ops , firmware , len ) ;
if ( rproc ) {
* ptr = rproc ;
devres_add ( dev , ptr ) ;
} else {
devres_free ( ptr ) ;
}
return rproc ;
}
EXPORT_SYMBOL ( devm_rproc_alloc ) ;
2016-10-19 19:40:02 -07:00
/**
* rproc_add_subdev ( ) - add a subdevice to a remoteproc
* @ rproc : rproc handle to add the subdevice to
* @ subdev : subdev handle to register
2018-06-26 07:11:57 -05:00
*
* Caller is responsible for populating optional subdevice function pointers .
2016-10-19 19:40:02 -07:00
*/
2018-06-26 07:11:57 -05:00
void rproc_add_subdev ( struct rproc * rproc , struct rproc_subdev * subdev )
2016-10-19 19:40:02 -07:00
{
list_add_tail ( & subdev - > node , & rproc - > subdevs ) ;
}
EXPORT_SYMBOL ( rproc_add_subdev ) ;
/**
* rproc_remove_subdev ( ) - remove a subdevice from a remoteproc
* @ rproc : rproc handle to remove the subdevice from
* @ subdev : subdev handle , previously registered with rproc_add_subdev ( )
*/
void rproc_remove_subdev ( struct rproc * rproc , struct rproc_subdev * subdev )
{
list_del ( & subdev - > node ) ;
}
EXPORT_SYMBOL ( rproc_remove_subdev ) ;
2017-08-27 22:34:53 -07:00
/**
* rproc_get_by_child ( ) - acquire rproc handle of @ dev ' s ancestor
* @ dev : child device to find ancestor of
*
* Returns the ancestor rproc instance , or NULL if not found .
*/
struct rproc * rproc_get_by_child ( struct device * dev )
{
for ( dev = dev - > parent ; dev ; dev = dev - > parent ) {
if ( dev - > type = = & rproc_type )
return dev - > driver_data ;
}
return NULL ;
}
EXPORT_SYMBOL ( rproc_get_by_child ) ;
2012-08-30 13:26:12 -05:00
/**
* rproc_report_crash ( ) - rproc crash reporter function
* @ rproc : remote processor
* @ type : crash type
*
* This function must be called every time a crash is detected by the low - level
* drivers implementing a specific remoteproc . This should not be called from a
* non - remoteproc driver .
*
* This function can be called from atomic / interrupt context .
*/
void rproc_report_crash ( struct rproc * rproc , enum rproc_crash_type type )
{
if ( ! rproc ) {
pr_err ( " NULL rproc pointer \n " ) ;
return ;
}
2020-04-29 11:04:42 -07:00
/* Prevent suspend while the remoteproc is being recovered */
pm_stay_awake ( rproc - > dev . parent ) ;
2012-08-30 13:26:12 -05:00
dev_err ( & rproc - > dev , " crash detected in %s: type %s \n " ,
rproc - > name , rproc_crash_to_string ( type ) ) ;
/* create a new task to handle the error */
schedule_work ( & rproc - > crash_handler ) ;
}
EXPORT_SYMBOL ( rproc_report_crash ) ;
2020-03-23 22:29:02 -07:00
static int rproc_panic_handler ( struct notifier_block * nb , unsigned long event ,
void * ptr )
{
unsigned int longest = 0 ;
struct rproc * rproc ;
unsigned int d ;
rcu_read_lock ( ) ;
list_for_each_entry_rcu ( rproc , & rproc_list , node ) {
if ( ! rproc - > ops - > panic | | rproc - > state ! = RPROC_RUNNING )
continue ;
d = rproc - > ops - > panic ( rproc ) ;
longest = max ( longest , d ) ;
}
rcu_read_unlock ( ) ;
/*
* Delay for the longest requested duration before returning . This can
* be used by the remoteproc drivers to give the remote processor time
* to perform any requested operations ( such as flush caches ) , when
* it ' s not possible to signal the Linux side due to the panic .
*/
mdelay ( longest ) ;
return NOTIFY_DONE ;
}
static void __init rproc_init_panic ( void )
{
rproc_panic_nb . notifier_call = rproc_panic_handler ;
atomic_notifier_chain_register ( & panic_notifier_list , & rproc_panic_nb ) ;
}
static void __exit rproc_exit_panic ( void )
{
atomic_notifier_chain_unregister ( & panic_notifier_list , & rproc_panic_nb ) ;
}
2011-10-20 16:52:46 +02:00
static int __init remoteproc_init ( void )
{
2016-10-19 13:05:47 +01:00
rproc_init_sysfs ( ) ;
2011-10-20 16:52:46 +02:00
rproc_init_debugfs ( ) ;
2020-07-29 10:40:01 -07:00
rproc_init_cdev ( ) ;
2020-03-23 22:29:02 -07:00
rproc_init_panic ( ) ;
remoteproc: maintain a generic child device for each rproc
For each registered rproc, maintain a generic remoteproc device whose
parent is the low level platform-specific device (commonly a pdev, but
it may certainly be any other type of device too).
With this in hand, the resulting device hierarchy might then look like:
omap-rproc.0
|
- remoteproc0 <---- new !
|
- virtio0
|
- virtio1
|
- rpmsg0
|
- rpmsg1
|
- rpmsg2
Where:
- omap-rproc.0 is the low level device that's bound to the
driver which invokes rproc_register()
- remoteproc0 is the result of this patch, and will be added by the
remoteproc framework when rproc_register() is invoked
- virtio0 and virtio1 are vdevs that are registered by remoteproc
when it realizes that they are supported by the firmware
of the physical remote processor represented by omap-rproc.0
- rpmsg0, rpmsg1 and rpmsg2 are rpmsg devices that represent rpmsg
channels, and are registerd by the rpmsg bus when it gets notified
about their existence
Technically, this patch:
- changes 'struct rproc' to contain this generic remoteproc.x device
- creates a new "remoteproc" type, to which this new generic remoteproc.x
device belong to.
- adds a super simple enumeration method for the indices of the
remoteproc.x devices
- updates all dev_* messaging to use the generic remoteproc.x device
instead of the low level platform-specific device
- updates all dma_* allocations to use the parent of remoteproc.x (where
the platform-specific memory pools, most commonly CMA, are to be found)
Adding this generic device has several merits:
- we can now add remoteproc runtime PM support simply by hooking onto the
new "remoteproc" type
- all remoteproc log messages will now carry a common name prefix
instead of having a platform-specific one
- having a device as part of the rproc struct makes it possible to simplify
refcounting (see subsequent patch)
Thanks to Stephen Boyd <sboyd@codeaurora.org> for suggesting and
discussing these ideas in one of the remoteproc review threads and
to Fernando Guzman Lugo <fernando.lugo@ti.com> for trying them out
with the (upcoming) runtime PM support for remoteproc.
Cc: Fernando Guzman Lugo <fernando.lugo@ti.com>
Reviewed-by: Stephen Boyd <sboyd@codeaurora.org>
Signed-off-by: Ohad Ben-Cohen <ohad@wizery.com>
2012-05-30 22:01:25 +03:00
2011-10-20 16:52:46 +02:00
return 0 ;
}
2019-05-30 17:52:23 -05:00
subsys_initcall ( remoteproc_init ) ;
2011-10-20 16:52:46 +02:00
static void __exit remoteproc_exit ( void )
{
2015-09-16 19:29:18 -05:00
ida_destroy ( & rproc_dev_index ) ;
2020-03-23 22:29:02 -07:00
rproc_exit_panic ( ) ;
2011-10-20 16:52:46 +02:00
rproc_exit_debugfs ( ) ;
2016-10-19 13:05:47 +01:00
rproc_exit_sysfs ( ) ;
2011-10-20 16:52:46 +02:00
}
module_exit ( remoteproc_exit ) ;
MODULE_LICENSE ( " GPL v2 " ) ;
MODULE_DESCRIPTION ( " Generic Remote Processor Framework " ) ;