accel/habanalabs: register compute device as an accel device

Register the compute device as an accel device, and remove the creation
of the habanalabs compute char device.

The IOCTLs in this patch are still handled by the current driver
handler. Moving to DRM IOCTL handling requires moving the IOCTLs
numbers to a specific range, so it will be handled in subsequent
patches.

Signed-off-by: Tomer Tayar <ttayar@habana.ai>
Reviewed-by: Oded Gabbay <ogabbay@kernel.org>
Signed-off-by: Oded Gabbay <ogabbay@kernel.org>
This commit is contained in:
Tomer Tayar 2023-02-19 11:58:46 +02:00 committed by Oded Gabbay
parent a8ab1a81cc
commit fe77368c0f
6 changed files with 161 additions and 250 deletions

View File

@ -18,8 +18,6 @@
#define MMU_KBUF_SIZE (MMU_ADDR_BUF_SIZE + MMU_ASID_BUF_SIZE) #define MMU_KBUF_SIZE (MMU_ADDR_BUF_SIZE + MMU_ASID_BUF_SIZE)
#define I2C_MAX_TRANSACTION_LEN 8 #define I2C_MAX_TRANSACTION_LEN 8
static struct dentry *hl_debug_root;
static int hl_debugfs_i2c_read(struct hl_device *hdev, u8 i2c_bus, u8 i2c_addr, static int hl_debugfs_i2c_read(struct hl_device *hdev, u8 i2c_bus, u8 i2c_addr,
u8 i2c_reg, u8 i2c_len, u64 *val) u8 i2c_reg, u8 i2c_len, u64 *val)
{ {
@ -1788,20 +1786,14 @@ void hl_debugfs_add_device(struct hl_device *hdev)
{ {
struct hl_dbg_device_entry *dev_entry = &hdev->hl_debugfs; struct hl_dbg_device_entry *dev_entry = &hdev->hl_debugfs;
dev_entry->root = debugfs_create_dir(dev_name(hdev->dev), hl_debug_root); dev_entry->root = hdev->drm.accel->debugfs_root;
add_files_to_device(hdev, dev_entry, dev_entry->root); add_files_to_device(hdev, dev_entry, dev_entry->root);
if (!hdev->asic_prop.fw_security_enabled) if (!hdev->asic_prop.fw_security_enabled)
add_secured_nodes(dev_entry, dev_entry->root); add_secured_nodes(dev_entry, dev_entry->root);
} }
void hl_debugfs_remove_device(struct hl_device *hdev)
{
struct hl_dbg_device_entry *entry = &hdev->hl_debugfs;
debugfs_remove_recursive(entry->root);
}
void hl_debugfs_add_file(struct hl_fpriv *hpriv) void hl_debugfs_add_file(struct hl_fpriv *hpriv)
{ {
struct hl_dbg_device_entry *dev_entry = &hpriv->hdev->hl_debugfs; struct hl_dbg_device_entry *dev_entry = &hpriv->hdev->hl_debugfs;
@ -1932,13 +1924,3 @@ void hl_debugfs_set_state_dump(struct hl_device *hdev, char *data,
up_write(&dev_entry->state_dump_sem); up_write(&dev_entry->state_dump_sem);
} }
void __init hl_debugfs_init(void)
{
hl_debug_root = debugfs_create_dir("habanalabs", NULL);
}
void hl_debugfs_fini(void)
{
debugfs_remove_recursive(hl_debug_root);
}

View File

@ -14,6 +14,9 @@
#include <linux/hwmon.h> #include <linux/hwmon.h>
#include <linux/vmalloc.h> #include <linux/vmalloc.h>
#include <drm/drm_accel.h>
#include <drm/drm_drv.h>
#include <trace/events/habanalabs.h> #include <trace/events/habanalabs.h>
#define HL_RESET_DELAY_USEC 10000 /* 10ms */ #define HL_RESET_DELAY_USEC 10000 /* 10ms */
@ -520,24 +523,20 @@ static void print_device_in_use_info(struct hl_device *hdev, const char *message
} }
/* /*
* hl_device_release - release function for habanalabs device * hl_device_release() - release function for habanalabs device.
* * @ddev: pointer to DRM device structure.
* @inode: pointer to inode structure * @file: pointer to DRM file private data structure.
* @filp: pointer to file structure
* *
* Called when process closes an habanalabs device * Called when process closes an habanalabs device
*/ */
static int hl_device_release(struct inode *inode, struct file *filp) void hl_device_release(struct drm_device *ddev, struct drm_file *file_priv)
{ {
struct hl_fpriv *hpriv = filp->private_data; struct hl_fpriv *hpriv = file_priv->driver_priv;
struct hl_device *hdev = hpriv->hdev; struct hl_device *hdev = to_hl_device(ddev);
filp->private_data = NULL;
if (!hdev) { if (!hdev) {
pr_crit("Closing FD after device was removed. Memory leak will occur and it is advised to reboot.\n"); pr_crit("Closing FD after device was removed. Memory leak will occur and it is advised to reboot.\n");
put_pid(hpriv->taskpid); put_pid(hpriv->taskpid);
return 0;
} }
hl_ctx_mgr_fini(hdev, &hpriv->ctx_mgr); hl_ctx_mgr_fini(hdev, &hpriv->ctx_mgr);
@ -555,8 +554,6 @@ static int hl_device_release(struct inode *inode, struct file *filp)
} }
hdev->last_open_session_duration_jif = jiffies - hdev->last_successful_open_jif; hdev->last_open_session_duration_jif = jiffies - hdev->last_successful_open_jif;
return 0;
} }
static int hl_device_release_ctrl(struct inode *inode, struct file *filp) static int hl_device_release_ctrl(struct inode *inode, struct file *filp)
@ -587,18 +584,8 @@ out:
return 0; return 0;
} }
/* static int __hl_mmap(struct hl_fpriv *hpriv, struct vm_area_struct *vma)
* hl_mmap - mmap function for habanalabs device
*
* @*filp: pointer to file structure
* @*vma: pointer to vm_area_struct of the process
*
* Called when process does an mmap on habanalabs device. Call the relevant mmap
* function at the end of the common code.
*/
static int hl_mmap(struct file *filp, struct vm_area_struct *vma)
{ {
struct hl_fpriv *hpriv = filp->private_data;
struct hl_device *hdev = hpriv->hdev; struct hl_device *hdev = hpriv->hdev;
unsigned long vm_pgoff; unsigned long vm_pgoff;
@ -621,14 +608,22 @@ static int hl_mmap(struct file *filp, struct vm_area_struct *vma)
return -EINVAL; return -EINVAL;
} }
static const struct file_operations hl_ops = { /*
.owner = THIS_MODULE, * hl_mmap - mmap function for habanalabs device
.open = hl_device_open, *
.release = hl_device_release, * @*filp: pointer to file structure
.mmap = hl_mmap, * @*vma: pointer to vm_area_struct of the process
.unlocked_ioctl = hl_ioctl, *
.compat_ioctl = hl_ioctl * Called when process does an mmap on habanalabs device. Call the relevant mmap
}; * function at the end of the common code.
*/
int hl_mmap(struct file *filp, struct vm_area_struct *vma)
{
struct drm_file *file_priv = filp->private_data;
struct hl_fpriv *hpriv = file_priv->driver_priv;
return __hl_mmap(hpriv, vma);
}
static const struct file_operations hl_ctrl_ops = { static const struct file_operations hl_ctrl_ops = {
.owner = THIS_MODULE, .owner = THIS_MODULE,
@ -656,7 +651,7 @@ static void device_release_func(struct device *dev)
* *
* Initialize a cdev and a Linux device for habanalabs's device. * Initialize a cdev and a Linux device for habanalabs's device.
*/ */
static int device_init_cdev(struct hl_device *hdev, struct class *class, static int device_init_cdev(struct hl_device *hdev, const struct class *class,
int minor, const struct file_operations *fops, int minor, const struct file_operations *fops,
char *name, struct cdev *cdev, char *name, struct cdev *cdev,
struct device **dev) struct device **dev)
@ -680,23 +675,26 @@ static int device_init_cdev(struct hl_device *hdev, struct class *class,
static int cdev_sysfs_debugfs_add(struct hl_device *hdev) static int cdev_sysfs_debugfs_add(struct hl_device *hdev)
{ {
const struct class *accel_class = hdev->drm.accel->kdev->class;
char name[32];
int rc; int rc;
rc = cdev_device_add(&hdev->cdev, hdev->dev); hdev->cdev_idx = hdev->drm.accel->index;
if (rc) {
dev_err(hdev->dev, /* Initialize cdev and device structures for the control device */
"failed to add a char device to the system\n"); snprintf(name, sizeof(name), "accel_controlD%d", hdev->cdev_idx);
rc = device_init_cdev(hdev, accel_class, hdev->cdev_idx, &hl_ctrl_ops, name,
&hdev->cdev_ctrl, &hdev->dev_ctrl);
if (rc)
return rc; return rc;
}
rc = cdev_device_add(&hdev->cdev_ctrl, hdev->dev_ctrl); rc = cdev_device_add(&hdev->cdev_ctrl, hdev->dev_ctrl);
if (rc) { if (rc) {
dev_err(hdev->dev, dev_err(hdev->dev_ctrl,
"failed to add a control char device to the system\n"); "failed to add an accel control char device to the system\n");
goto delete_cdev_device; goto free_ctrl_device;
} }
/* hl_sysfs_init() must be done after adding the device to the system */
rc = hl_sysfs_init(hdev); rc = hl_sysfs_init(hdev);
if (rc) { if (rc) {
dev_err(hdev->dev, "failed to initialize sysfs\n"); dev_err(hdev->dev, "failed to initialize sysfs\n");
@ -711,23 +709,19 @@ static int cdev_sysfs_debugfs_add(struct hl_device *hdev)
delete_ctrl_cdev_device: delete_ctrl_cdev_device:
cdev_device_del(&hdev->cdev_ctrl, hdev->dev_ctrl); cdev_device_del(&hdev->cdev_ctrl, hdev->dev_ctrl);
delete_cdev_device: free_ctrl_device:
cdev_device_del(&hdev->cdev, hdev->dev); put_device(hdev->dev_ctrl);
return rc; return rc;
} }
static void cdev_sysfs_debugfs_remove(struct hl_device *hdev) static void cdev_sysfs_debugfs_remove(struct hl_device *hdev)
{ {
if (!hdev->cdev_sysfs_debugfs_created) if (!hdev->cdev_sysfs_debugfs_created)
goto put_devices; return;
hl_debugfs_remove_device(hdev);
hl_sysfs_fini(hdev); hl_sysfs_fini(hdev);
cdev_device_del(&hdev->cdev_ctrl, hdev->dev_ctrl);
cdev_device_del(&hdev->cdev, hdev->dev);
put_devices: cdev_device_del(&hdev->cdev_ctrl, hdev->dev_ctrl);
put_device(hdev->dev);
put_device(hdev->dev_ctrl); put_device(hdev->dev_ctrl);
} }
@ -2011,51 +2005,6 @@ void hl_notifier_event_send_all(struct hl_device *hdev, u64 event_mask)
mutex_unlock(&hdev->fpriv_ctrl_list_lock); mutex_unlock(&hdev->fpriv_ctrl_list_lock);
} }
static int create_cdev(struct hl_device *hdev)
{
char *name;
int rc;
hdev->cdev_idx = hdev->id / 2;
name = kasprintf(GFP_KERNEL, "hl%d", hdev->cdev_idx);
if (!name) {
rc = -ENOMEM;
goto out_err;
}
/* Initialize cdev and device structures */
rc = device_init_cdev(hdev, hdev->hclass, hdev->id, &hl_ops, name,
&hdev->cdev, &hdev->dev);
kfree(name);
if (rc)
goto out_err;
name = kasprintf(GFP_KERNEL, "hl_controlD%d", hdev->cdev_idx);
if (!name) {
rc = -ENOMEM;
goto free_dev;
}
/* Initialize cdev and device structures for control device */
rc = device_init_cdev(hdev, hdev->hclass, hdev->id_control, &hl_ctrl_ops,
name, &hdev->cdev_ctrl, &hdev->dev_ctrl);
kfree(name);
if (rc)
goto free_dev;
return 0;
free_dev:
put_device(hdev->dev);
out_err:
return rc;
}
/* /*
* hl_device_init - main initialization function for habanalabs device * hl_device_init - main initialization function for habanalabs device
* *
@ -2070,14 +2019,10 @@ int hl_device_init(struct hl_device *hdev)
int i, rc, cq_cnt, user_interrupt_cnt, cq_ready_cnt; int i, rc, cq_cnt, user_interrupt_cnt, cq_ready_cnt;
bool expose_interfaces_on_err = false; bool expose_interfaces_on_err = false;
rc = create_cdev(hdev);
if (rc)
goto out_disabled;
/* Initialize ASIC function pointers and perform early init */ /* Initialize ASIC function pointers and perform early init */
rc = device_early_init(hdev); rc = device_early_init(hdev);
if (rc) if (rc)
goto free_dev; goto out_disabled;
user_interrupt_cnt = hdev->asic_prop.user_dec_intr_count + user_interrupt_cnt = hdev->asic_prop.user_dec_intr_count +
hdev->asic_prop.user_interrupt_count; hdev->asic_prop.user_interrupt_count;
@ -2264,6 +2209,14 @@ int hl_device_init(struct hl_device *hdev)
* From here there is no need to expose them in case of an error. * From here there is no need to expose them in case of an error.
*/ */
expose_interfaces_on_err = false; expose_interfaces_on_err = false;
rc = drm_dev_register(&hdev->drm, 0);
if (rc) {
dev_err(hdev->dev, "Failed to register DRM device, rc %d\n", rc);
rc = 0;
goto out_disabled;
}
rc = cdev_sysfs_debugfs_add(hdev); rc = cdev_sysfs_debugfs_add(hdev);
if (rc) { if (rc) {
dev_err(hdev->dev, "Failed to add char devices and sysfs/debugfs files\n"); dev_err(hdev->dev, "Failed to add char devices and sysfs/debugfs files\n");
@ -2332,15 +2285,14 @@ free_usr_intr_mem:
kfree(hdev->user_interrupt); kfree(hdev->user_interrupt);
early_fini: early_fini:
device_early_fini(hdev); device_early_fini(hdev);
free_dev:
put_device(hdev->dev_ctrl);
put_device(hdev->dev);
out_disabled: out_disabled:
hdev->disabled = true; hdev->disabled = true;
if (expose_interfaces_on_err) if (expose_interfaces_on_err) {
drm_dev_register(&hdev->drm, 0);
cdev_sysfs_debugfs_add(hdev); cdev_sysfs_debugfs_add(hdev);
dev_err(&hdev->pdev->dev, }
"Failed to initialize hl%d. Device %s is NOT usable !\n",
pr_err("Failed to initialize accel%d. Device %s is NOT usable!\n",
hdev->cdev_idx, dev_name(&hdev->pdev->dev)); hdev->cdev_idx, dev_name(&hdev->pdev->dev));
return rc; return rc;
@ -2486,6 +2438,7 @@ void hl_device_fini(struct hl_device *hdev)
/* Hide devices and sysfs/debugfs files from user */ /* Hide devices and sysfs/debugfs files from user */
cdev_sysfs_debugfs_remove(hdev); cdev_sysfs_debugfs_remove(hdev);
drm_dev_unregister(&hdev->drm);
hl_debugfs_device_fini(hdev); hl_debugfs_device_fini(hdev);

View File

@ -29,6 +29,9 @@
#include <linux/coresight.h> #include <linux/coresight.h>
#include <linux/dma-buf.h> #include <linux/dma-buf.h>
#include <drm/drm_device.h>
#include <drm/drm_file.h>
#include "security.h" #include "security.h"
#define HL_NAME "habanalabs" #define HL_NAME "habanalabs"
@ -2258,7 +2261,7 @@ struct hl_notifier_event {
/** /**
* struct hl_fpriv - process information stored in FD private data. * struct hl_fpriv - process information stored in FD private data.
* @hdev: habanalabs device structure. * @hdev: habanalabs device structure.
* @filp: pointer to the given file structure. * @filp: pointer to the DRM file private data structure.
* @taskpid: current process ID. * @taskpid: current process ID.
* @ctx: current executing context. TODO: remove for multiple ctx per process * @ctx: current executing context. TODO: remove for multiple ctx per process
* @ctx_mgr: context manager to handle multiple context for this FD. * @ctx_mgr: context manager to handle multiple context for this FD.
@ -2273,7 +2276,7 @@ struct hl_notifier_event {
*/ */
struct hl_fpriv { struct hl_fpriv {
struct hl_device *hdev; struct hl_device *hdev;
struct file *filp; struct drm_file *file_priv;
struct pid *taskpid; struct pid *taskpid;
struct hl_ctx *ctx; struct hl_ctx *ctx;
struct hl_ctx_mgr ctx_mgr; struct hl_ctx_mgr ctx_mgr;
@ -3141,8 +3144,7 @@ struct hl_reset_info {
* (required only for PCI address match mode) * (required only for PCI address match mode)
* @pcie_bar: array of available PCIe bars virtual addresses. * @pcie_bar: array of available PCIe bars virtual addresses.
* @rmmio: configuration area address on SRAM. * @rmmio: configuration area address on SRAM.
* @hclass: pointer to the habanalabs class. * @drm: related DRM device.
* @cdev: related char device.
* @cdev_ctrl: char device for control operations only (INFO IOCTL) * @cdev_ctrl: char device for control operations only (INFO IOCTL)
* @dev: related kernel basic device structure. * @dev: related kernel basic device structure.
* @dev_ctrl: related kernel device structure for the control device * @dev_ctrl: related kernel device structure for the control device
@ -3269,8 +3271,7 @@ struct hl_reset_info {
* @rotator_binning: contains mask of rotators engines that is received from the f/w * @rotator_binning: contains mask of rotators engines that is received from the f/w
* which indicates which rotator engines are binned-out(Gaudi3 and above). * which indicates which rotator engines are binned-out(Gaudi3 and above).
* @id: device minor. * @id: device minor.
* @id_control: minor of the control device. * @cdev_idx: char device index.
* @cdev_idx: char device index. Used for setting its name.
* @cpu_pci_msb_addr: 50-bit extension bits for the device CPU's 40-bit * @cpu_pci_msb_addr: 50-bit extension bits for the device CPU's 40-bit
* addresses. * addresses.
* @is_in_dram_scrub: true if dram scrub operation is on going. * @is_in_dram_scrub: true if dram scrub operation is on going.
@ -3332,8 +3333,7 @@ struct hl_device {
u64 pcie_bar_phys[HL_PCI_NUM_BARS]; u64 pcie_bar_phys[HL_PCI_NUM_BARS];
void __iomem *pcie_bar[HL_PCI_NUM_BARS]; void __iomem *pcie_bar[HL_PCI_NUM_BARS];
void __iomem *rmmio; void __iomem *rmmio;
struct class *hclass; struct drm_device drm;
struct cdev cdev;
struct cdev cdev_ctrl; struct cdev cdev_ctrl;
struct device *dev; struct device *dev;
struct device *dev_ctrl; struct device *dev_ctrl;
@ -3442,7 +3442,6 @@ struct hl_device {
u32 device_release_watchdog_timeout_sec; u32 device_release_watchdog_timeout_sec;
u32 rotator_binning; u32 rotator_binning;
u16 id; u16 id;
u16 id_control;
u16 cdev_idx; u16 cdev_idx;
u16 cpu_pci_msb_addr; u16 cpu_pci_msb_addr;
u8 is_in_dram_scrub; u8 is_in_dram_scrub;
@ -3606,6 +3605,11 @@ static inline bool hl_mem_area_inside_range(u64 address, u64 size,
return false; return false;
} }
static inline struct hl_device *to_hl_device(struct drm_device *ddev)
{
return container_of(ddev, struct hl_device, drm);
}
/** /**
* hl_mem_area_crosses_range() - Checks whether address+size crossing a range. * hl_mem_area_crosses_range() - Checks whether address+size crossing a range.
* @address: The start address of the area we want to validate. * @address: The start address of the area we want to validate.
@ -3644,7 +3648,12 @@ int hl_access_cfg_region(struct hl_device *hdev, u64 addr, u64 *val,
enum debugfs_access_type acc_type); enum debugfs_access_type acc_type);
int hl_access_dev_mem(struct hl_device *hdev, enum pci_region region_type, int hl_access_dev_mem(struct hl_device *hdev, enum pci_region region_type,
u64 addr, u64 *val, enum debugfs_access_type acc_type); u64 addr, u64 *val, enum debugfs_access_type acc_type);
int hl_device_open(struct inode *inode, struct file *filp);
int hl_mmap(struct file *filp, struct vm_area_struct *vma);
int hl_device_open(struct drm_device *drm, struct drm_file *file_priv);
void hl_device_release(struct drm_device *ddev, struct drm_file *file_priv);
int hl_device_open_ctrl(struct inode *inode, struct file *filp); int hl_device_open_ctrl(struct inode *inode, struct file *filp);
bool hl_device_operational(struct hl_device *hdev, bool hl_device_operational(struct hl_device *hdev,
enum hl_device_status *status); enum hl_device_status *status);
@ -3973,12 +3982,9 @@ void hl_enable_err_info_capture(struct hl_error_info *captured_err_info);
#ifdef CONFIG_DEBUG_FS #ifdef CONFIG_DEBUG_FS
void hl_debugfs_init(void);
void hl_debugfs_fini(void);
int hl_debugfs_device_init(struct hl_device *hdev); int hl_debugfs_device_init(struct hl_device *hdev);
void hl_debugfs_device_fini(struct hl_device *hdev); void hl_debugfs_device_fini(struct hl_device *hdev);
void hl_debugfs_add_device(struct hl_device *hdev); void hl_debugfs_add_device(struct hl_device *hdev);
void hl_debugfs_remove_device(struct hl_device *hdev);
void hl_debugfs_add_file(struct hl_fpriv *hpriv); void hl_debugfs_add_file(struct hl_fpriv *hpriv);
void hl_debugfs_remove_file(struct hl_fpriv *hpriv); void hl_debugfs_remove_file(struct hl_fpriv *hpriv);
void hl_debugfs_add_cb(struct hl_cb *cb); void hl_debugfs_add_cb(struct hl_cb *cb);
@ -3997,31 +4003,10 @@ void hl_debugfs_set_state_dump(struct hl_device *hdev, char *data,
#else #else
static inline void __init hl_debugfs_init(void)
{
}
static inline void hl_debugfs_fini(void)
{
}
static inline int hl_debugfs_device_init(struct hl_device *hdev)
{
return 0;
}
static inline void hl_debugfs_device_fini(struct hl_device *hdev)
{
}
static inline void hl_debugfs_add_device(struct hl_device *hdev) static inline void hl_debugfs_add_device(struct hl_device *hdev)
{ {
} }
static inline void hl_debugfs_remove_device(struct hl_device *hdev)
{
}
static inline void hl_debugfs_add_file(struct hl_fpriv *hpriv) static inline void hl_debugfs_add_file(struct hl_fpriv *hpriv)
{ {
} }

View File

@ -14,6 +14,10 @@
#include <linux/pci.h> #include <linux/pci.h>
#include <linux/module.h> #include <linux/module.h>
#include <linux/vmalloc.h> #include <linux/vmalloc.h>
#include <linux/version.h>
#include <drm/drm_accel.h>
#include <drm/drm_drv.h>
#define CREATE_TRACE_POINTS #define CREATE_TRACE_POINTS
#include <trace/events/habanalabs.h> #include <trace/events/habanalabs.h>
@ -27,7 +31,6 @@ MODULE_DESCRIPTION(HL_DRIVER_DESC);
MODULE_LICENSE("GPL v2"); MODULE_LICENSE("GPL v2");
static int hl_major; static int hl_major;
static struct class *hl_class;
static DEFINE_IDR(hl_devs_idr); static DEFINE_IDR(hl_devs_idr);
static DEFINE_MUTEX(hl_devs_idr_lock); static DEFINE_MUTEX(hl_devs_idr_lock);
@ -70,6 +73,31 @@ static const struct pci_device_id ids[] = {
}; };
MODULE_DEVICE_TABLE(pci, ids); MODULE_DEVICE_TABLE(pci, ids);
static const struct file_operations hl_fops = {
.owner = THIS_MODULE,
.open = accel_open,
.release = drm_release,
.unlocked_ioctl = hl_ioctl,
.compat_ioctl = hl_ioctl,
.llseek = noop_llseek,
.mmap = hl_mmap
};
static const struct drm_driver hl_driver = {
.driver_features = DRIVER_COMPUTE_ACCEL,
.name = HL_NAME,
.desc = HL_DRIVER_DESC,
.major = LINUX_VERSION_MAJOR,
.minor = LINUX_VERSION_PATCHLEVEL,
.patchlevel = LINUX_VERSION_SUBLEVEL,
.date = "20190505",
.fops = &hl_fops,
.open = hl_device_open,
.postclose = hl_device_release
};
/* /*
* get_asic_type - translate device id to asic type * get_asic_type - translate device id to asic type
* *
@ -123,43 +151,28 @@ static bool is_asic_secured(enum hl_asic_type asic_type)
} }
/* /*
* hl_device_open - open function for habanalabs device * hl_device_open() - open function for habanalabs device.
* * @ddev: pointer to DRM device structure.
* @inode: pointer to inode structure * @file: pointer to DRM file private data structure.
* @filp: pointer to file structure
* *
* Called when process opens an habanalabs device. * Called when process opens an habanalabs device.
*/ */
int hl_device_open(struct inode *inode, struct file *filp) int hl_device_open(struct drm_device *ddev, struct drm_file *file_priv)
{ {
struct hl_device *hdev = to_hl_device(ddev);
enum hl_device_status status; enum hl_device_status status;
struct hl_device *hdev;
struct hl_fpriv *hpriv; struct hl_fpriv *hpriv;
int rc; int rc;
mutex_lock(&hl_devs_idr_lock);
hdev = idr_find(&hl_devs_idr, iminor(inode));
mutex_unlock(&hl_devs_idr_lock);
if (!hdev) {
pr_err("Couldn't find device %d:%d\n",
imajor(inode), iminor(inode));
return -ENXIO;
}
hpriv = kzalloc(sizeof(*hpriv), GFP_KERNEL); hpriv = kzalloc(sizeof(*hpriv), GFP_KERNEL);
if (!hpriv) if (!hpriv)
return -ENOMEM; return -ENOMEM;
hpriv->hdev = hdev; hpriv->hdev = hdev;
filp->private_data = hpriv;
hpriv->filp = filp;
mutex_init(&hpriv->notifier_event.lock); mutex_init(&hpriv->notifier_event.lock);
mutex_init(&hpriv->restore_phase_mutex); mutex_init(&hpriv->restore_phase_mutex);
mutex_init(&hpriv->ctx_lock); mutex_init(&hpriv->ctx_lock);
kref_init(&hpriv->refcount); kref_init(&hpriv->refcount);
nonseekable_open(inode, filp);
hl_ctx_mgr_init(&hpriv->ctx_mgr); hl_ctx_mgr_init(&hpriv->ctx_mgr);
hl_mem_mgr_init(hpriv->hdev->dev, &hpriv->mem_mgr); hl_mem_mgr_init(hpriv->hdev->dev, &hpriv->mem_mgr);
@ -225,6 +238,9 @@ int hl_device_open(struct inode *inode, struct file *filp)
hdev->last_successful_open_jif = jiffies; hdev->last_successful_open_jif = jiffies;
hdev->last_successful_open_ktime = ktime_get(); hdev->last_successful_open_ktime = ktime_get();
file_priv->driver_priv = hpriv;
hpriv->file_priv = file_priv;
return 0; return 0;
out_err: out_err:
@ -232,7 +248,6 @@ out_err:
hl_mem_mgr_fini(&hpriv->mem_mgr); hl_mem_mgr_fini(&hpriv->mem_mgr);
hl_mem_mgr_idr_destroy(&hpriv->mem_mgr); hl_mem_mgr_idr_destroy(&hpriv->mem_mgr);
hl_ctx_mgr_fini(hpriv->hdev, &hpriv->ctx_mgr); hl_ctx_mgr_fini(hpriv->hdev, &hpriv->ctx_mgr);
filp->private_data = NULL;
mutex_destroy(&hpriv->ctx_lock); mutex_destroy(&hpriv->ctx_lock);
mutex_destroy(&hpriv->restore_phase_mutex); mutex_destroy(&hpriv->restore_phase_mutex);
mutex_destroy(&hpriv->notifier_event.lock); mutex_destroy(&hpriv->notifier_event.lock);
@ -268,7 +283,6 @@ int hl_device_open_ctrl(struct inode *inode, struct file *filp)
*/ */
hpriv->hdev = hdev; hpriv->hdev = hdev;
filp->private_data = hpriv; filp->private_data = hpriv;
hpriv->filp = filp;
mutex_init(&hpriv->notifier_event.lock); mutex_init(&hpriv->notifier_event.lock);
nonseekable_open(inode, filp); nonseekable_open(inode, filp);
@ -317,7 +331,6 @@ static void copy_kernel_module_params_to_device(struct hl_device *hdev)
hdev->asic_prop.fw_security_enabled = is_asic_secured(hdev->asic_type); hdev->asic_prop.fw_security_enabled = is_asic_secured(hdev->asic_type);
hdev->major = hl_major; hdev->major = hl_major;
hdev->hclass = hl_class;
hdev->memory_scrub = memory_scrub; hdev->memory_scrub = memory_scrub;
hdev->reset_on_lockup = reset_on_lockup; hdev->reset_on_lockup = reset_on_lockup;
hdev->boot_error_status_mask = boot_error_status_mask; hdev->boot_error_status_mask = boot_error_status_mask;
@ -383,6 +396,31 @@ static int fixup_device_params(struct hl_device *hdev)
return 0; return 0;
} }
static int allocate_device_id(struct hl_device *hdev)
{
int id;
mutex_lock(&hl_devs_idr_lock);
id = idr_alloc(&hl_devs_idr, hdev, 0, HL_MAX_MINORS, GFP_KERNEL);
mutex_unlock(&hl_devs_idr_lock);
if (id < 0) {
if (id == -ENOSPC)
pr_err("too many devices in the system\n");
return -EBUSY;
}
hdev->id = id;
/*
* Firstly initialized with the internal device ID.
* Will be updated later after the DRM device registration to hold the minor ID.
*/
hdev->cdev_idx = hdev->id;
return 0;
}
/** /**
* create_hdev - create habanalabs device instance * create_hdev - create habanalabs device instance
* *
@ -395,14 +433,16 @@ static int fixup_device_params(struct hl_device *hdev)
*/ */
static int create_hdev(struct hl_device **dev, struct pci_dev *pdev) static int create_hdev(struct hl_device **dev, struct pci_dev *pdev)
{ {
int main_id, ctrl_id = 0, rc = 0;
struct hl_device *hdev; struct hl_device *hdev;
int rc;
*dev = NULL; *dev = NULL;
hdev = kzalloc(sizeof(*hdev), GFP_KERNEL); hdev = devm_drm_dev_alloc(&pdev->dev, &hl_driver, struct hl_device, drm);
if (!hdev) if (IS_ERR(hdev))
return -ENOMEM; return PTR_ERR(hdev);
hdev->dev = hdev->drm.dev;
/* Will be NULL in case of simulator device */ /* Will be NULL in case of simulator device */
hdev->pdev = pdev; hdev->pdev = pdev;
@ -425,7 +465,7 @@ static int create_hdev(struct hl_device **dev, struct pci_dev *pdev)
if (hdev->asic_type == ASIC_INVALID) { if (hdev->asic_type == ASIC_INVALID) {
dev_err(&pdev->dev, "Unsupported ASIC\n"); dev_err(&pdev->dev, "Unsupported ASIC\n");
rc = -ENODEV; rc = -ENODEV;
goto free_hdev; goto out_err;
} }
copy_kernel_module_params_to_device(hdev); copy_kernel_module_params_to_device(hdev);
@ -434,42 +474,15 @@ static int create_hdev(struct hl_device **dev, struct pci_dev *pdev)
fixup_device_params(hdev); fixup_device_params(hdev);
mutex_lock(&hl_devs_idr_lock); rc = allocate_device_id(hdev);
if (rc)
/* Always save 2 numbers, 1 for main device and 1 for control. goto out_err;
* They must be consecutive
*/
main_id = idr_alloc(&hl_devs_idr, hdev, 0, HL_MAX_MINORS, GFP_KERNEL);
if (main_id >= 0)
ctrl_id = idr_alloc(&hl_devs_idr, hdev, main_id + 1,
main_id + 2, GFP_KERNEL);
mutex_unlock(&hl_devs_idr_lock);
if ((main_id < 0) || (ctrl_id < 0)) {
if ((main_id == -ENOSPC) || (ctrl_id == -ENOSPC))
pr_err("too many devices in the system\n");
if (main_id >= 0) {
mutex_lock(&hl_devs_idr_lock);
idr_remove(&hl_devs_idr, main_id);
mutex_unlock(&hl_devs_idr_lock);
}
rc = -EBUSY;
goto free_hdev;
}
hdev->id = main_id;
hdev->id_control = ctrl_id;
*dev = hdev; *dev = hdev;
return 0; return 0;
free_hdev: out_err:
kfree(hdev);
return rc; return rc;
} }
@ -484,10 +497,8 @@ static void destroy_hdev(struct hl_device *hdev)
/* Remove device from the device list */ /* Remove device from the device list */
mutex_lock(&hl_devs_idr_lock); mutex_lock(&hl_devs_idr_lock);
idr_remove(&hl_devs_idr, hdev->id); idr_remove(&hl_devs_idr, hdev->id);
idr_remove(&hl_devs_idr, hdev->id_control);
mutex_unlock(&hl_devs_idr_lock); mutex_unlock(&hl_devs_idr_lock);
kfree(hdev);
} }
static int hl_pmops_suspend(struct device *dev) static int hl_pmops_suspend(struct device *dev)
@ -691,28 +702,16 @@ static int __init hl_init(void)
hl_major = MAJOR(dev); hl_major = MAJOR(dev);
hl_class = class_create(HL_NAME);
if (IS_ERR(hl_class)) {
pr_err("failed to allocate class\n");
rc = PTR_ERR(hl_class);
goto remove_major;
}
hl_debugfs_init();
rc = pci_register_driver(&hl_pci_driver); rc = pci_register_driver(&hl_pci_driver);
if (rc) { if (rc) {
pr_err("failed to register pci device\n"); pr_err("failed to register pci device\n");
goto remove_debugfs; goto remove_major;
} }
pr_debug("driver loaded\n"); pr_debug("driver loaded\n");
return 0; return 0;
remove_debugfs:
hl_debugfs_fini();
class_destroy(hl_class);
remove_major: remove_major:
unregister_chrdev_region(MKDEV(hl_major, 0), HL_MAX_MINORS); unregister_chrdev_region(MKDEV(hl_major, 0), HL_MAX_MINORS);
return rc; return rc;
@ -725,14 +724,6 @@ static void __exit hl_exit(void)
{ {
pci_unregister_driver(&hl_pci_driver); pci_unregister_driver(&hl_pci_driver);
/*
* Removing debugfs must be after all devices or simulator devices
* have been removed because otherwise we get a bug in the
* debugfs module for referencing NULL objects
*/
hl_debugfs_fini();
class_destroy(hl_class);
unregister_chrdev_region(MKDEV(hl_major, 0), HL_MAX_MINORS); unregister_chrdev_region(MKDEV(hl_major, 0), HL_MAX_MINORS);
idr_destroy(&hl_devs_idr); idr_destroy(&hl_devs_idr);

View File

@ -1166,10 +1166,9 @@ static const struct hl_ioctl_desc hl_ioctls_control[] = {
HL_IOCTL_DEF(HL_IOCTL_INFO, hl_info_ioctl_control) HL_IOCTL_DEF(HL_IOCTL_INFO, hl_info_ioctl_control)
}; };
static long _hl_ioctl(struct file *filep, unsigned int cmd, unsigned long arg, static long _hl_ioctl(struct hl_fpriv *hpriv, unsigned int cmd, unsigned long arg,
const struct hl_ioctl_desc *ioctl, struct device *dev) const struct hl_ioctl_desc *ioctl, struct device *dev)
{ {
struct hl_fpriv *hpriv = filep->private_data;
unsigned int nr = _IOC_NR(cmd); unsigned int nr = _IOC_NR(cmd);
char stack_kdata[128] = {0}; char stack_kdata[128] = {0};
char *kdata = NULL; char *kdata = NULL;
@ -1235,7 +1234,8 @@ out_err:
long hl_ioctl(struct file *filep, unsigned int cmd, unsigned long arg) long hl_ioctl(struct file *filep, unsigned int cmd, unsigned long arg)
{ {
struct hl_fpriv *hpriv = filep->private_data; struct drm_file *file_priv = filep->private_data;
struct hl_fpriv *hpriv = file_priv->driver_priv;
struct hl_device *hdev = hpriv->hdev; struct hl_device *hdev = hpriv->hdev;
const struct hl_ioctl_desc *ioctl = NULL; const struct hl_ioctl_desc *ioctl = NULL;
unsigned int nr = _IOC_NR(cmd); unsigned int nr = _IOC_NR(cmd);
@ -1256,7 +1256,7 @@ long hl_ioctl(struct file *filep, unsigned int cmd, unsigned long arg)
return -ENOTTY; return -ENOTTY;
} }
return _hl_ioctl(filep, cmd, arg, ioctl, hdev->dev); return _hl_ioctl(hpriv, cmd, arg, ioctl, hdev->dev);
} }
long hl_ioctl_control(struct file *filep, unsigned int cmd, unsigned long arg) long hl_ioctl_control(struct file *filep, unsigned int cmd, unsigned long arg)
@ -1282,5 +1282,5 @@ long hl_ioctl_control(struct file *filep, unsigned int cmd, unsigned long arg)
return -ENOTTY; return -ENOTTY;
} }
return _hl_ioctl(filep, cmd, arg, ioctl, hdev->dev_ctrl); return _hl_ioctl(hpriv, cmd, arg, ioctl, hdev->dev_ctrl);
} }

View File

@ -1818,7 +1818,7 @@ static void hl_release_dmabuf(struct dma_buf *dmabuf)
hl_ctx_put(ctx); hl_ctx_put(ctx);
/* Paired with get_file() in export_dmabuf() */ /* Paired with get_file() in export_dmabuf() */
fput(ctx->hpriv->filp); fput(ctx->hpriv->file_priv->filp);
kfree(hl_dmabuf); kfree(hl_dmabuf);
} }
@ -1864,7 +1864,7 @@ static int export_dmabuf(struct hl_ctx *ctx,
* released first and only then the compute device. * released first and only then the compute device.
* Paired with fput() in hl_release_dmabuf(). * Paired with fput() in hl_release_dmabuf().
*/ */
get_file(ctx->hpriv->filp); get_file(ctx->hpriv->file_priv->filp);
*dmabuf_fd = fd; *dmabuf_fd = fd;