1 /*
2  * Copyright (c) 2018 Intel Corporation.
3  * Copyright 2024 NXP
4  *
5  * SPDX-License-Identifier: Apache-2.0
6  */
7 
8 #include <string.h>
9 #include <zephyr/types.h>
10 #include <zephyr/sys/__assert.h>
11 #include <zephyr/sys/util.h>
12 #include <zephyr/init.h>
13 #include <zephyr/storage/disk_access.h>
14 #include <errno.h>
15 #include <zephyr/device.h>
16 
17 #define LOG_LEVEL CONFIG_DISK_LOG_LEVEL
18 #include <zephyr/logging/log.h>
19 LOG_MODULE_REGISTER(disk);
20 
21 /* list of mounted file systems */
22 static sys_dlist_t disk_access_list = SYS_DLIST_STATIC_INIT(&disk_access_list);
23 
24 /* lock to protect storage layer registration */
25 static struct k_spinlock lock;
26 
disk_access_get_di(const char * name)27 struct disk_info *disk_access_get_di(const char *name)
28 {
29 	struct disk_info *disk = NULL, *itr;
30 	size_t name_len = strlen(name);
31 	sys_dnode_t *node;
32 	k_spinlock_key_t spinlock_key = k_spin_lock(&lock);
33 
34 	SYS_DLIST_FOR_EACH_NODE(&disk_access_list, node) {
35 		itr = CONTAINER_OF(node, struct disk_info, node);
36 
37 		/*
38 		 * Move to next node if mount point length is
39 		 * shorter than longest_match match or if path
40 		 * name is shorter than the mount point name.
41 		 */
42 		if (strlen(itr->name) != name_len) {
43 			continue;
44 		}
45 
46 		/* Check for disk name match */
47 		if (strncmp(name, itr->name, name_len) == 0) {
48 			disk = itr;
49 			break;
50 		}
51 	}
52 	k_spin_unlock(&lock, spinlock_key);
53 
54 	return disk;
55 }
56 
disk_access_init(const char * pdrv)57 int disk_access_init(const char *pdrv)
58 {
59 	struct disk_info *disk = disk_access_get_di(pdrv);
60 	int rc = -EINVAL;
61 
62 	if ((disk != NULL) && (disk->refcnt == 0U)) {
63 		/* Disk has not been initialized, start it */
64 		if ((disk->ops != NULL) && (disk->ops->init != NULL)) {
65 			rc = disk->ops->init(disk);
66 			if (rc == 0) {
67 				/* Increment reference count */
68 				disk->refcnt++;
69 			}
70 		}
71 	} else if ((disk != NULL) && (disk->refcnt < UINT16_MAX)) {
72 		/* Disk reference count is nonzero, simply increment it */
73 		disk->refcnt++;
74 		rc = 0;
75 	}
76 
77 	return rc;
78 }
79 
disk_access_status(const char * pdrv)80 int disk_access_status(const char *pdrv)
81 {
82 	struct disk_info *disk = disk_access_get_di(pdrv);
83 	int rc = -EINVAL;
84 
85 	if ((disk != NULL) && (disk->ops != NULL) &&
86 				(disk->ops->status != NULL)) {
87 		rc = disk->ops->status(disk);
88 	}
89 
90 	return rc;
91 }
92 
disk_access_read(const char * pdrv,uint8_t * data_buf,uint32_t start_sector,uint32_t num_sector)93 int disk_access_read(const char *pdrv, uint8_t *data_buf,
94 		     uint32_t start_sector, uint32_t num_sector)
95 {
96 	struct disk_info *disk = disk_access_get_di(pdrv);
97 	int rc = -EINVAL;
98 
99 	if ((disk != NULL) && (disk->ops != NULL) &&
100 				(disk->ops->read != NULL)) {
101 		rc = disk->ops->read(disk, data_buf, start_sector, num_sector);
102 	}
103 
104 	return rc;
105 }
106 
disk_access_write(const char * pdrv,const uint8_t * data_buf,uint32_t start_sector,uint32_t num_sector)107 int disk_access_write(const char *pdrv, const uint8_t *data_buf,
108 		      uint32_t start_sector, uint32_t num_sector)
109 {
110 	struct disk_info *disk = disk_access_get_di(pdrv);
111 	int rc = -EINVAL;
112 
113 	if ((disk != NULL) && (disk->ops != NULL) &&
114 				(disk->ops->write != NULL)) {
115 		rc = disk->ops->write(disk, data_buf, start_sector, num_sector);
116 	}
117 
118 	return rc;
119 }
120 
disk_access_ioctl(const char * pdrv,uint8_t cmd,void * buf)121 int disk_access_ioctl(const char *pdrv, uint8_t cmd, void *buf)
122 {
123 	struct disk_info *disk = disk_access_get_di(pdrv);
124 	int rc = -EINVAL;
125 
126 	if ((disk != NULL) && (disk->ops != NULL) &&
127 				(disk->ops->ioctl != NULL)) {
128 		switch (cmd) {
129 		case DISK_IOCTL_CTRL_INIT:
130 			if (disk->refcnt == 0U) {
131 				rc = disk->ops->ioctl(disk, cmd, buf);
132 				if (rc == 0) {
133 					disk->refcnt++;
134 				}
135 			} else if (disk->refcnt < UINT16_MAX) {
136 				disk->refcnt++;
137 				rc = 0;
138 			} else {
139 				LOG_ERR("Disk reference count at max value");
140 			}
141 			break;
142 		case DISK_IOCTL_CTRL_DEINIT:
143 			if ((buf != NULL) && (*((bool *)buf))) {
144 				/* Force deinit disk */
145 				disk->refcnt = 0U;
146 				disk->ops->ioctl(disk, cmd, buf);
147 				rc = 0;
148 			} else if (disk->refcnt == 1U) {
149 				rc = disk->ops->ioctl(disk, cmd, buf);
150 				if (rc == 0) {
151 					disk->refcnt--;
152 				}
153 			} else if (disk->refcnt > 0) {
154 				disk->refcnt--;
155 				rc = 0;
156 			} else {
157 				LOG_WRN("Disk is already deinitialized");
158 			}
159 			break;
160 		default:
161 			rc = disk->ops->ioctl(disk, cmd, buf);
162 		}
163 	}
164 
165 	return rc;
166 }
167 
disk_access_register(struct disk_info * disk)168 int disk_access_register(struct disk_info *disk)
169 {
170 	k_spinlock_key_t spinlock_key;
171 
172 	if ((disk == NULL) || (disk->name == NULL)) {
173 		LOG_ERR("invalid disk interface!!");
174 		return -EINVAL;
175 	}
176 
177 	if (disk_access_get_di(disk->name) != NULL) {
178 		LOG_ERR("disk interface already registered!!");
179 		return -EINVAL;
180 	}
181 
182 	/* Initialize reference count to zero */
183 	disk->refcnt = 0U;
184 
185 	spinlock_key = k_spin_lock(&lock);
186 	/*  append to the disk list */
187 	sys_dlist_append(&disk_access_list, &disk->node);
188 	LOG_DBG("disk interface(%s) registered", disk->name);
189 	k_spin_unlock(&lock, spinlock_key);
190 	return 0;
191 }
192 
disk_access_unregister(struct disk_info * disk)193 int disk_access_unregister(struct disk_info *disk)
194 {
195 	k_spinlock_key_t spinlock_key;
196 
197 	if ((disk == NULL) || (disk->name == NULL)) {
198 		LOG_ERR("invalid disk interface!!");
199 		return -EINVAL;
200 	}
201 
202 	if (disk_access_get_di(disk->name) == NULL) {
203 		LOG_ERR("disk interface not registered!!");
204 		return -EINVAL;
205 	}
206 
207 	spinlock_key = k_spin_lock(&lock);
208 	/* remove disk node from the list */
209 	sys_dlist_remove(&disk->node);
210 	k_spin_unlock(&lock, spinlock_key);
211 	LOG_DBG("disk interface(%s) unregistered", disk->name);
212 	return 0;
213 }
214