mirror of
git://nv-tegra.nvidia.com/linux-nv-oot.git
synced 2025-12-22 17:25:35 +03:00
delete kernel object if driver initialization gets failed Bug 4080433 Change-Id: Iabe0cabf32a067224f6e17a5e54a1c7d90161c16 Signed-off-by: Manish Bhardwaj <mbhardwaj@nvidia.com> Reviewed-on: https://git-master.nvidia.com/r/c/linux-nv-oot/+/2921795 Reviewed-by: Suresh Venkatachalam <skathirampat@nvidia.com> Reviewed-by: Sandeep Trasi <strasi@nvidia.com> GVS: Gerrit_Virtual_Submit <buildbot_gerritrpt@nvidia.com>
223 lines
5.8 KiB
C
223 lines
5.8 KiB
C
/* SPDX-License-Identifier: GPL-2.0-only */
|
|
/*
|
|
* Copyright (c) 2022-2023, NVIDIA CORPORATION & AFFILIATES. All rights reserved.
|
|
*/
|
|
|
|
#include <linux/module.h>
|
|
#include <linux/init.h>
|
|
#include <linux/errno.h>
|
|
#include <soc/tegra/fuse.h>
|
|
#include <linux/kobject.h>
|
|
#include <linux/sysfs.h>
|
|
#include <linux/mm.h>
|
|
#include <linux/bug.h>
|
|
#include <linux/io.h>
|
|
#include <soc/tegra/virt/syscalls.h>
|
|
#include <soc/tegra/virt/hv-ivc.h>
|
|
#define TEGRA_HV_ERR(...) pr_err("hvc_sysfs: " __VA_ARGS__)
|
|
#define TEGRA_HV_INFO(...) pr_info("hvc_sysfs: " __VA_ARGS__)
|
|
|
|
|
|
/*
|
|
* This file implements a hypervisor control driver that can be accessed
|
|
* from user-space via the sysfs interface. Currently, the only supported
|
|
* use case is retrieval of the HV trace log when it is available.
|
|
*/
|
|
|
|
#define MAX_NAME_SIZE 50
|
|
|
|
struct hyp_shared_memory_info {
|
|
char node_name[MAX_NAME_SIZE];
|
|
struct bin_attribute attr;
|
|
uint64_t ipa;
|
|
unsigned long size;
|
|
};
|
|
|
|
#define HYP_SHM_ID_NUM (2 * NGUESTS_MAX + 1U) // trace buffers + PCT.
|
|
static struct hyp_shared_memory_info hyp_shared_memory_attrs[HYP_SHM_ID_NUM];
|
|
|
|
|
|
|
|
/* Map the HV trace buffer to the calling user process */
|
|
static int hvc_sysfs_mmap(struct file *fp, struct kobject *ko,
|
|
struct bin_attribute *attr, struct vm_area_struct *vma)
|
|
{
|
|
struct hyp_shared_memory_info *hyp_shm_info =
|
|
container_of(attr, struct hyp_shared_memory_info, attr);
|
|
|
|
if ((hyp_shm_info->ipa == 0) || (hyp_shm_info->size == 0))
|
|
return -EINVAL;
|
|
|
|
|
|
if ((vma->vm_end - vma->vm_start) != attr->size)
|
|
return -EINVAL;
|
|
|
|
return remap_pfn_range(
|
|
vma,
|
|
vma->vm_start,
|
|
hyp_shm_info->ipa >> PAGE_SHIFT,
|
|
hyp_shm_info->size,
|
|
vma->vm_page_prot);
|
|
}
|
|
|
|
/* Discover availability and placement of the HV trace buffer */
|
|
static int hvc_create_sysfs(
|
|
struct kobject *kobj,
|
|
struct hyp_shared_memory_info *hyp_shm_info)
|
|
{
|
|
sysfs_bin_attr_init((struct bin_attribute *)&hyp_shm_info->attr);
|
|
|
|
hyp_shm_info->attr.attr.name = hyp_shm_info->node_name;
|
|
hyp_shm_info->attr.attr.mode = 0400;
|
|
hyp_shm_info->attr.mmap = hvc_sysfs_mmap;
|
|
hyp_shm_info->attr.size = (size_t)hyp_shm_info->size;
|
|
|
|
if ((hyp_shm_info->ipa == 0) || (hyp_shm_info->size == 0))
|
|
return -EINVAL;
|
|
|
|
return sysfs_create_bin_file(kobj, &hyp_shm_info->attr);
|
|
}
|
|
|
|
static ssize_t log_mask_read(struct file *fp, struct kobject *ko,
|
|
struct bin_attribute *attr, char *buf, loff_t pos, size_t size)
|
|
{
|
|
if (size == sizeof(uint64_t))
|
|
hyp_trace_get_mask((uint64_t *)buf);
|
|
return size;
|
|
}
|
|
|
|
static ssize_t log_mask_write(struct file *fp, struct kobject *ko,
|
|
struct bin_attribute *attr, char *buf, loff_t pos, size_t size)
|
|
{
|
|
if (size == sizeof(uint64_t))
|
|
hyp_trace_set_mask(*(uint64_t *)buf);
|
|
return size;
|
|
}
|
|
|
|
static struct bin_attribute log_mask_attr;
|
|
static int create_log_mask_node(struct kobject *kobj)
|
|
{
|
|
if (kobj == NULL)
|
|
return -EINVAL;
|
|
|
|
sysfs_bin_attr_init((struct bin_attribute *)&log_mask_attr);
|
|
log_mask_attr.attr.name = "log_mask";
|
|
log_mask_attr.attr.mode = 0600;
|
|
log_mask_attr.read = log_mask_read;
|
|
log_mask_attr.write = log_mask_write;
|
|
log_mask_attr.size = sizeof(uint64_t);
|
|
return sysfs_create_bin_file(kobj, &log_mask_attr);
|
|
}
|
|
|
|
/* Set up all relevant hypervisor control nodes */
|
|
static int __init hvc_sysfs_register(void)
|
|
{
|
|
struct kobject *kobj;
|
|
int ret;
|
|
uint64_t ipa;
|
|
struct hyp_info_page *info;
|
|
uint64_t log_mask;
|
|
struct trace_buf *buffs;
|
|
uint16_t i, count;
|
|
|
|
if (is_tegra_hypervisor_mode() == false) {
|
|
TEGRA_HV_INFO("hypervisor is not present\n");
|
|
/*retunring success in case of native kernel otherwise
|
|
systemd-modules-load service will failed.*/
|
|
return 0;
|
|
}
|
|
|
|
kobj = kobject_create_and_add("hvc", NULL);
|
|
if (kobj == NULL) {
|
|
TEGRA_HV_INFO("failed to add kobject\n");
|
|
return -ENOMEM;
|
|
}
|
|
|
|
if (hyp_read_hyp_info(&ipa) != 0) {
|
|
ret = -EINVAL;
|
|
goto fail;
|
|
}
|
|
|
|
info = (__force struct hyp_info_page *)ioremap(ipa, sizeof(*info));
|
|
if (info == NULL) {
|
|
ret = -EFAULT;
|
|
goto fail;
|
|
}
|
|
|
|
buffs = info->trace_buffs;
|
|
count = ARRAY_SIZE(info->trace_buffs);
|
|
|
|
/* 2 bytes for number, 1 for "_" and 1 for null character.*/
|
|
if (MAX_NAME_SIZE < (ARRAY_SIZE(buffs[i].name) + 4U)) {
|
|
TEGRA_HV_INFO("hyp_shared_memory_attrs.name size is small\n");
|
|
iounmap((void __iomem *)info);
|
|
ret = -EFAULT;
|
|
goto fail;
|
|
}
|
|
|
|
for (i = 0; i < count; i++) {
|
|
if (buffs[i].ipa != 0U && buffs[i].size != 0U) {
|
|
hyp_shared_memory_attrs[i].ipa = buffs[i].ipa;
|
|
hyp_shared_memory_attrs[i].size = (size_t)buffs[i].size;
|
|
|
|
/* Suffix number to the process name, this solves the duplicate process
|
|
* name issue.
|
|
*/
|
|
ret = snprintf(hyp_shared_memory_attrs[i].node_name, MAX_NAME_SIZE,
|
|
"%s_%u", buffs[i].name, i);
|
|
|
|
if (ret > 0U) {
|
|
hyp_shared_memory_attrs[i].node_name[ret] = '\0';
|
|
} else {
|
|
TEGRA_HV_INFO("snprintf failure - %s\n", buffs[i].name);
|
|
iounmap((void __iomem *)info);
|
|
ret = -EFAULT;
|
|
goto fail;
|
|
}
|
|
|
|
ret = hvc_create_sysfs(kobj, &hyp_shared_memory_attrs[i]);
|
|
if (ret == 0)
|
|
TEGRA_HV_INFO(" %s trace is available\n", buffs[i].name);
|
|
else
|
|
TEGRA_HV_INFO(" %s trace is unavailable\n", buffs[i].name);
|
|
}
|
|
}
|
|
|
|
if (hyp_trace_get_mask(&log_mask) == 0) {
|
|
ret = create_log_mask_node(kobj);
|
|
if (ret == 0)
|
|
TEGRA_HV_INFO("access to trace event mask is available\n");
|
|
}
|
|
|
|
hyp_shared_memory_attrs[i].ipa = info->pct_ipa;
|
|
hyp_shared_memory_attrs[i].size = (size_t)info->pct_size;
|
|
ret = snprintf(hyp_shared_memory_attrs[i].node_name, MAX_NAME_SIZE, "pct");
|
|
|
|
if (ret > 0U) {
|
|
hyp_shared_memory_attrs[i].node_name[ret] = '\0';
|
|
} else {
|
|
TEGRA_HV_INFO("snprintf failure - pct\n");
|
|
iounmap((void __iomem *)info);
|
|
ret = -EFAULT;
|
|
goto fail;
|
|
}
|
|
|
|
ret = hvc_create_sysfs(kobj, &hyp_shared_memory_attrs[i]);
|
|
if (ret == 0)
|
|
TEGRA_HV_INFO("pct is available\n");
|
|
else
|
|
TEGRA_HV_INFO("pct is unavailable\n");
|
|
|
|
iounmap((void __iomem *)info);
|
|
|
|
return 0;
|
|
|
|
fail:
|
|
kobject_del(kobj);
|
|
return ret;
|
|
}
|
|
|
|
late_initcall(hvc_sysfs_register);
|
|
|
|
MODULE_LICENSE("GPL");
|