| // SPDX-License-Identifier: GPL-2.0-only | 
| /* | 
|  * Copyright (C) 2004 IBM Corporation | 
|  * Authors: | 
|  * Leendert van Doorn <leendert@watson.ibm.com> | 
|  * Dave Safford <safford@watson.ibm.com> | 
|  * Reiner Sailer <sailer@watson.ibm.com> | 
|  * Kylene Hall <kjhall@us.ibm.com> | 
|  * | 
|  * Copyright (C) 2013 Obsidian Research Corp | 
|  * Jason Gunthorpe <jgunthorpe@obsidianresearch.com> | 
|  * | 
|  * Device file system interface to the TPM | 
|  */ | 
| #include <linux/slab.h> | 
| #include "tpm-dev.h" | 
|   | 
| static int tpm_open(struct inode *inode, struct file *file) | 
| { | 
|     struct tpm_chip *chip; | 
|     struct file_priv *priv; | 
|   | 
|     chip = container_of(inode->i_cdev, struct tpm_chip, cdev); | 
|   | 
|     /* It's assured that the chip will be opened just once, | 
|      * by the check of is_open variable, which is protected | 
|      * by driver_lock. */ | 
|     if (test_and_set_bit(0, &chip->is_open)) { | 
|         dev_dbg(&chip->dev, "Another process owns this TPM\n"); | 
|         return -EBUSY; | 
|     } | 
|   | 
|     priv = kzalloc(sizeof(*priv), GFP_KERNEL); | 
|     if (priv == NULL) | 
|         goto out; | 
|   | 
|     tpm_common_open(file, chip, priv, NULL); | 
|   | 
|     return 0; | 
|   | 
|  out: | 
|     clear_bit(0, &chip->is_open); | 
|     return -ENOMEM; | 
| } | 
|   | 
| /* | 
|  * Called on file close | 
|  */ | 
| static int tpm_release(struct inode *inode, struct file *file) | 
| { | 
|     struct file_priv *priv = file->private_data; | 
|   | 
|     tpm_common_release(file, priv); | 
|     clear_bit(0, &priv->chip->is_open); | 
|     kfree(priv); | 
|   | 
|     return 0; | 
| } | 
|   | 
| const struct file_operations tpm_fops = { | 
|     .owner = THIS_MODULE, | 
|     .llseek = no_llseek, | 
|     .open = tpm_open, | 
|     .read = tpm_common_read, | 
|     .write = tpm_common_write, | 
|     .poll = tpm_common_poll, | 
|     .release = tpm_release, | 
| }; |