Files
linux-nv-oot/drivers/virt/tegra/hvc_sysfs.c
Manish Bhardwaj 61f67509a8 hvc_sysfs: delete kernel object
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>
2023-06-24 07:26:30 -07:00

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");