forked from torvalds/linux
-
Notifications
You must be signed in to change notification settings - Fork 0
Commit
This commit does not belong to any branch on this repository, and may belong to a fork outside of the repository.
Merge tag 'arm-ffa-5.14' of git://git.kernel.org/pub/scm/linux/kernel…
…/git/sudeep.holla/linux into arm/drivers Arm Firmware Framework for ARMv8-A(FFA) interface driver The Arm FFA specification describes a software architecture to leverages the virtualization extension to isolate software images provided by an ecosystem of vendors from each other and describes interfaces that standardize communication between the various software images including communication between images in the Secure world and Normal world. Any Hypervisor could use the FFA interfaces to enable communication between VMs it manages. The Hypervisor a.k.a Partition managers in FFA terminology can assign system resources(Memory regions, Devices, CPU cycles) to the partitions and manage isolation amongst them. This is the initial and minimal support for the FFA interface to enable communication between secure partitions and the normal world OS. * tag 'arm-ffa-5.14' of git://git.kernel.org/pub/scm/linux/kernel/git/sudeep.holla/linux: firmware: arm_ffa: Add support for MEM_* interfaces firmware: arm_ffa: Setup in-kernel users of FFA partitions firmware: arm_ffa: Add support for SMCCC as transport to FFA driver firmware: arm_ffa: Add initial Arm FFA driver support firmware: arm_ffa: Add initial FFA bus support for device enumeration arm64: smccc: Add support for SMCCCv1.2 extended input/output registers Link: https://lore.kernel.org/r/20210601095838.GA838783@bogus Signed-off-by: Olof Johansson <[email protected]>
- Loading branch information
Showing
13 changed files
with
1,435 additions
and
0 deletions.
There are no files selected for viewing
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
|
@@ -7077,6 +7077,13 @@ F: include/linux/firewire.h | |
F: include/uapi/linux/firewire*.h | ||
F: tools/firewire/ | ||
|
||
FIRMWARE FRAMEWORK FOR ARMV8-A | ||
M: Sudeep Holla <[email protected]> | ||
L: [email protected] | ||
S: Maintained | ||
F: drivers/firmware/arm_ffa/ | ||
F: include/linux/arm_ffa.h | ||
|
||
FIRMWARE LOADER (request_firmware) | ||
M: Luis Chamberlain <[email protected]> | ||
L: [email protected] | ||
|
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,21 @@ | ||
# SPDX-License-Identifier: GPL-2.0-only | ||
config ARM_FFA_TRANSPORT | ||
tristate "Arm Firmware Framework for Armv8-A" | ||
depends on OF | ||
depends on ARM64 | ||
default n | ||
help | ||
This Firmware Framework(FF) for Arm A-profile processors describes | ||
interfaces that standardize communication between the various | ||
software images which includes communication between images in | ||
the Secure world and Normal world. It also leverages the | ||
virtualization extension to isolate software images provided | ||
by an ecosystem of vendors from each other. | ||
|
||
This driver provides interface for all the client drivers making | ||
use of the features offered by ARM FF-A. | ||
|
||
config ARM_FFA_SMCCC | ||
bool | ||
default ARM_FFA_TRANSPORT | ||
depends on ARM64 && HAVE_ARM_SMCCC_DISCOVERY |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,6 @@ | ||
# SPDX-License-Identifier: GPL-2.0-only | ||
ffa-bus-y = bus.o | ||
ffa-driver-y = driver.o | ||
ffa-transport-$(CONFIG_ARM_FFA_SMCCC) += smccc.o | ||
ffa-module-objs := $(ffa-bus-y) $(ffa-driver-y) $(ffa-transport-y) | ||
obj-$(CONFIG_ARM_FFA_TRANSPORT) = ffa-module.o |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,210 @@ | ||
// SPDX-License-Identifier: GPL-2.0 | ||
/* | ||
* Copyright (C) 2021 ARM Ltd. | ||
*/ | ||
|
||
#define pr_fmt(fmt) KBUILD_MODNAME ": " fmt | ||
|
||
#include <linux/arm_ffa.h> | ||
#include <linux/device.h> | ||
#include <linux/fs.h> | ||
#include <linux/kernel.h> | ||
#include <linux/module.h> | ||
#include <linux/slab.h> | ||
#include <linux/types.h> | ||
|
||
#include "common.h" | ||
|
||
static int ffa_device_match(struct device *dev, struct device_driver *drv) | ||
{ | ||
const struct ffa_device_id *id_table; | ||
struct ffa_device *ffa_dev; | ||
|
||
id_table = to_ffa_driver(drv)->id_table; | ||
ffa_dev = to_ffa_dev(dev); | ||
|
||
while (!uuid_is_null(&id_table->uuid)) { | ||
/* | ||
* FF-A v1.0 doesn't provide discovery of UUIDs, just the | ||
* partition IDs, so fetch the partitions IDs for this | ||
* id_table UUID and assign the UUID to the device if the | ||
* partition ID matches | ||
*/ | ||
if (uuid_is_null(&ffa_dev->uuid)) | ||
ffa_device_match_uuid(ffa_dev, &id_table->uuid); | ||
|
||
if (uuid_equal(&ffa_dev->uuid, &id_table->uuid)) | ||
return 1; | ||
id_table++; | ||
} | ||
|
||
return 0; | ||
} | ||
|
||
static int ffa_device_probe(struct device *dev) | ||
{ | ||
struct ffa_driver *ffa_drv = to_ffa_driver(dev->driver); | ||
struct ffa_device *ffa_dev = to_ffa_dev(dev); | ||
|
||
if (!ffa_device_match(dev, dev->driver)) | ||
return -ENODEV; | ||
|
||
return ffa_drv->probe(ffa_dev); | ||
} | ||
|
||
static int ffa_device_uevent(struct device *dev, struct kobj_uevent_env *env) | ||
{ | ||
struct ffa_device *ffa_dev = to_ffa_dev(dev); | ||
|
||
return add_uevent_var(env, "MODALIAS=arm_ffa:%04x:%pUb", | ||
ffa_dev->vm_id, &ffa_dev->uuid); | ||
} | ||
|
||
static ssize_t partition_id_show(struct device *dev, | ||
struct device_attribute *attr, char *buf) | ||
{ | ||
struct ffa_device *ffa_dev = to_ffa_dev(dev); | ||
|
||
return sprintf(buf, "0x%04x\n", ffa_dev->vm_id); | ||
} | ||
static DEVICE_ATTR_RO(partition_id); | ||
|
||
static ssize_t uuid_show(struct device *dev, struct device_attribute *attr, | ||
char *buf) | ||
{ | ||
struct ffa_device *ffa_dev = to_ffa_dev(dev); | ||
|
||
return sprintf(buf, "%pUb\n", &ffa_dev->uuid); | ||
} | ||
static DEVICE_ATTR_RO(uuid); | ||
|
||
static struct attribute *ffa_device_attributes_attrs[] = { | ||
&dev_attr_partition_id.attr, | ||
&dev_attr_uuid.attr, | ||
NULL, | ||
}; | ||
ATTRIBUTE_GROUPS(ffa_device_attributes); | ||
|
||
struct bus_type ffa_bus_type = { | ||
.name = "arm_ffa", | ||
.match = ffa_device_match, | ||
.probe = ffa_device_probe, | ||
.uevent = ffa_device_uevent, | ||
.dev_groups = ffa_device_attributes_groups, | ||
}; | ||
EXPORT_SYMBOL_GPL(ffa_bus_type); | ||
|
||
int ffa_driver_register(struct ffa_driver *driver, struct module *owner, | ||
const char *mod_name) | ||
{ | ||
int ret; | ||
|
||
driver->driver.bus = &ffa_bus_type; | ||
driver->driver.name = driver->name; | ||
driver->driver.owner = owner; | ||
driver->driver.mod_name = mod_name; | ||
|
||
ret = driver_register(&driver->driver); | ||
if (!ret) | ||
pr_debug("registered new ffa driver %s\n", driver->name); | ||
|
||
return ret; | ||
} | ||
EXPORT_SYMBOL_GPL(ffa_driver_register); | ||
|
||
void ffa_driver_unregister(struct ffa_driver *driver) | ||
{ | ||
driver_unregister(&driver->driver); | ||
} | ||
EXPORT_SYMBOL_GPL(ffa_driver_unregister); | ||
|
||
static void ffa_release_device(struct device *dev) | ||
{ | ||
struct ffa_device *ffa_dev = to_ffa_dev(dev); | ||
|
||
kfree(ffa_dev); | ||
} | ||
|
||
static int __ffa_devices_unregister(struct device *dev, void *data) | ||
{ | ||
ffa_release_device(dev); | ||
|
||
return 0; | ||
} | ||
|
||
static void ffa_devices_unregister(void) | ||
{ | ||
bus_for_each_dev(&ffa_bus_type, NULL, NULL, | ||
__ffa_devices_unregister); | ||
} | ||
|
||
bool ffa_device_is_valid(struct ffa_device *ffa_dev) | ||
{ | ||
bool valid = false; | ||
struct device *dev = NULL; | ||
struct ffa_device *tmp_dev; | ||
|
||
do { | ||
dev = bus_find_next_device(&ffa_bus_type, dev); | ||
tmp_dev = to_ffa_dev(dev); | ||
if (tmp_dev == ffa_dev) { | ||
valid = true; | ||
break; | ||
} | ||
put_device(dev); | ||
} while (dev); | ||
|
||
put_device(dev); | ||
|
||
return valid; | ||
} | ||
|
||
struct ffa_device *ffa_device_register(const uuid_t *uuid, int vm_id) | ||
{ | ||
int ret; | ||
struct device *dev; | ||
struct ffa_device *ffa_dev; | ||
|
||
ffa_dev = kzalloc(sizeof(*ffa_dev), GFP_KERNEL); | ||
if (!ffa_dev) | ||
return NULL; | ||
|
||
dev = &ffa_dev->dev; | ||
dev->bus = &ffa_bus_type; | ||
dev->release = ffa_release_device; | ||
dev_set_name(&ffa_dev->dev, "arm-ffa-%04x", vm_id); | ||
|
||
ffa_dev->vm_id = vm_id; | ||
uuid_copy(&ffa_dev->uuid, uuid); | ||
|
||
ret = device_register(&ffa_dev->dev); | ||
if (ret) { | ||
dev_err(dev, "unable to register device %s err=%d\n", | ||
dev_name(dev), ret); | ||
put_device(dev); | ||
return NULL; | ||
} | ||
|
||
return ffa_dev; | ||
} | ||
EXPORT_SYMBOL_GPL(ffa_device_register); | ||
|
||
void ffa_device_unregister(struct ffa_device *ffa_dev) | ||
{ | ||
if (!ffa_dev) | ||
return; | ||
|
||
device_unregister(&ffa_dev->dev); | ||
} | ||
EXPORT_SYMBOL_GPL(ffa_device_unregister); | ||
|
||
int arm_ffa_bus_init(void) | ||
{ | ||
return bus_register(&ffa_bus_type); | ||
} | ||
|
||
void arm_ffa_bus_exit(void) | ||
{ | ||
ffa_devices_unregister(); | ||
bus_unregister(&ffa_bus_type); | ||
} |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,31 @@ | ||
/* SPDX-License-Identifier: GPL-2.0 */ | ||
/* | ||
* Copyright (C) 2021 ARM Ltd. | ||
*/ | ||
|
||
#ifndef _FFA_COMMON_H | ||
#define _FFA_COMMON_H | ||
|
||
#include <linux/arm_ffa.h> | ||
#include <linux/arm-smccc.h> | ||
#include <linux/err.h> | ||
|
||
typedef struct arm_smccc_1_2_regs ffa_value_t; | ||
|
||
typedef void (ffa_fn)(ffa_value_t, ffa_value_t *); | ||
|
||
int arm_ffa_bus_init(void); | ||
void arm_ffa_bus_exit(void); | ||
bool ffa_device_is_valid(struct ffa_device *ffa_dev); | ||
void ffa_device_match_uuid(struct ffa_device *ffa_dev, const uuid_t *uuid); | ||
|
||
#ifdef CONFIG_ARM_FFA_SMCCC | ||
int __init ffa_transport_init(ffa_fn **invoke_ffa_fn); | ||
#else | ||
static inline int __init ffa_transport_init(ffa_fn **invoke_ffa_fn) | ||
{ | ||
return -EOPNOTSUPP; | ||
} | ||
#endif | ||
|
||
#endif /* _FFA_COMMON_H */ |
Oops, something went wrong.