| // SPDX-License-Identifier: GPL-2.0 | 
 | /* | 
 |  * Parse command line, get partition information | 
 |  * | 
 |  * Written by Cai Zhiyong <[email protected]> | 
 |  * | 
 |  */ | 
 | #include <linux/export.h> | 
 | #include <linux/cmdline-parser.h> | 
 |  | 
 | static int parse_subpart(struct cmdline_subpart **subpart, char *partdef) | 
 | { | 
 | 	int ret = 0; | 
 | 	struct cmdline_subpart *new_subpart; | 
 |  | 
 | 	*subpart = NULL; | 
 |  | 
 | 	new_subpart = kzalloc(sizeof(struct cmdline_subpart), GFP_KERNEL); | 
 | 	if (!new_subpart) | 
 | 		return -ENOMEM; | 
 |  | 
 | 	if (*partdef == '-') { | 
 | 		new_subpart->size = (sector_t)(~0ULL); | 
 | 		partdef++; | 
 | 	} else { | 
 | 		new_subpart->size = (sector_t)memparse(partdef, &partdef); | 
 | 		if (new_subpart->size < (sector_t)PAGE_SIZE) { | 
 | 			pr_warn("cmdline partition size is invalid."); | 
 | 			ret = -EINVAL; | 
 | 			goto fail; | 
 | 		} | 
 | 	} | 
 |  | 
 | 	if (*partdef == '@') { | 
 | 		partdef++; | 
 | 		new_subpart->from = (sector_t)memparse(partdef, &partdef); | 
 | 	} else { | 
 | 		new_subpart->from = (sector_t)(~0ULL); | 
 | 	} | 
 |  | 
 | 	if (*partdef == '(') { | 
 | 		int length; | 
 | 		char *next = strchr(++partdef, ')'); | 
 |  | 
 | 		if (!next) { | 
 | 			pr_warn("cmdline partition format is invalid."); | 
 | 			ret = -EINVAL; | 
 | 			goto fail; | 
 | 		} | 
 |  | 
 | 		length = min_t(int, next - partdef, | 
 | 			       sizeof(new_subpart->name) - 1); | 
 | 		strncpy(new_subpart->name, partdef, length); | 
 | 		new_subpart->name[length] = '\0'; | 
 |  | 
 | 		partdef = ++next; | 
 | 	} else | 
 | 		new_subpart->name[0] = '\0'; | 
 |  | 
 | 	new_subpart->flags = 0; | 
 |  | 
 | 	if (!strncmp(partdef, "ro", 2)) { | 
 | 		new_subpart->flags |= PF_RDONLY; | 
 | 		partdef += 2; | 
 | 	} | 
 |  | 
 | 	if (!strncmp(partdef, "lk", 2)) { | 
 | 		new_subpart->flags |= PF_POWERUP_LOCK; | 
 | 		partdef += 2; | 
 | 	} | 
 |  | 
 | 	*subpart = new_subpart; | 
 | 	return 0; | 
 | fail: | 
 | 	kfree(new_subpart); | 
 | 	return ret; | 
 | } | 
 |  | 
 | static void free_subpart(struct cmdline_parts *parts) | 
 | { | 
 | 	struct cmdline_subpart *subpart; | 
 |  | 
 | 	while (parts->subpart) { | 
 | 		subpart = parts->subpart; | 
 | 		parts->subpart = subpart->next_subpart; | 
 | 		kfree(subpart); | 
 | 	} | 
 | } | 
 |  | 
 | static int parse_parts(struct cmdline_parts **parts, const char *bdevdef) | 
 | { | 
 | 	int ret = -EINVAL; | 
 | 	char *next; | 
 | 	int length; | 
 | 	struct cmdline_subpart **next_subpart; | 
 | 	struct cmdline_parts *newparts; | 
 | 	char buf[BDEVNAME_SIZE + 32 + 4]; | 
 |  | 
 | 	*parts = NULL; | 
 |  | 
 | 	newparts = kzalloc(sizeof(struct cmdline_parts), GFP_KERNEL); | 
 | 	if (!newparts) | 
 | 		return -ENOMEM; | 
 |  | 
 | 	next = strchr(bdevdef, ':'); | 
 | 	if (!next) { | 
 | 		pr_warn("cmdline partition has no block device."); | 
 | 		goto fail; | 
 | 	} | 
 |  | 
 | 	length = min_t(int, next - bdevdef, sizeof(newparts->name) - 1); | 
 | 	strncpy(newparts->name, bdevdef, length); | 
 | 	newparts->name[length] = '\0'; | 
 | 	newparts->nr_subparts = 0; | 
 |  | 
 | 	next_subpart = &newparts->subpart; | 
 |  | 
 | 	while (next && *(++next)) { | 
 | 		bdevdef = next; | 
 | 		next = strchr(bdevdef, ','); | 
 |  | 
 | 		length = (!next) ? (sizeof(buf) - 1) : | 
 | 			min_t(int, next - bdevdef, sizeof(buf) - 1); | 
 |  | 
 | 		strncpy(buf, bdevdef, length); | 
 | 		buf[length] = '\0'; | 
 |  | 
 | 		ret = parse_subpart(next_subpart, buf); | 
 | 		if (ret) | 
 | 			goto fail; | 
 |  | 
 | 		newparts->nr_subparts++; | 
 | 		next_subpart = &(*next_subpart)->next_subpart; | 
 | 	} | 
 |  | 
 | 	if (!newparts->subpart) { | 
 | 		pr_warn("cmdline partition has no valid partition."); | 
 | 		ret = -EINVAL; | 
 | 		goto fail; | 
 | 	} | 
 |  | 
 | 	*parts = newparts; | 
 |  | 
 | 	return 0; | 
 | fail: | 
 | 	free_subpart(newparts); | 
 | 	kfree(newparts); | 
 | 	return ret; | 
 | } | 
 |  | 
 | void cmdline_parts_free(struct cmdline_parts **parts) | 
 | { | 
 | 	struct cmdline_parts *next_parts; | 
 |  | 
 | 	while (*parts) { | 
 | 		next_parts = (*parts)->next_parts; | 
 | 		free_subpart(*parts); | 
 | 		kfree(*parts); | 
 | 		*parts = next_parts; | 
 | 	} | 
 | } | 
 | EXPORT_SYMBOL(cmdline_parts_free); | 
 |  | 
 | int cmdline_parts_parse(struct cmdline_parts **parts, const char *cmdline) | 
 | { | 
 | 	int ret; | 
 | 	char *buf; | 
 | 	char *pbuf; | 
 | 	char *next; | 
 | 	struct cmdline_parts **next_parts; | 
 |  | 
 | 	*parts = NULL; | 
 |  | 
 | 	next = pbuf = buf = kstrdup(cmdline, GFP_KERNEL); | 
 | 	if (!buf) | 
 | 		return -ENOMEM; | 
 |  | 
 | 	next_parts = parts; | 
 |  | 
 | 	while (next && *pbuf) { | 
 | 		next = strchr(pbuf, ';'); | 
 | 		if (next) | 
 | 			*next = '\0'; | 
 |  | 
 | 		ret = parse_parts(next_parts, pbuf); | 
 | 		if (ret) | 
 | 			goto fail; | 
 |  | 
 | 		if (next) | 
 | 			pbuf = ++next; | 
 |  | 
 | 		next_parts = &(*next_parts)->next_parts; | 
 | 	} | 
 |  | 
 | 	if (!*parts) { | 
 | 		pr_warn("cmdline partition has no valid partition."); | 
 | 		ret = -EINVAL; | 
 | 		goto fail; | 
 | 	} | 
 |  | 
 | 	ret = 0; | 
 | done: | 
 | 	kfree(buf); | 
 | 	return ret; | 
 |  | 
 | fail: | 
 | 	cmdline_parts_free(parts); | 
 | 	goto done; | 
 | } | 
 | EXPORT_SYMBOL(cmdline_parts_parse); | 
 |  | 
 | struct cmdline_parts *cmdline_parts_find(struct cmdline_parts *parts, | 
 | 					 const char *bdev) | 
 | { | 
 | 	while (parts && strncmp(bdev, parts->name, sizeof(parts->name))) | 
 | 		parts = parts->next_parts; | 
 | 	return parts; | 
 | } | 
 | EXPORT_SYMBOL(cmdline_parts_find); | 
 |  | 
 | /* | 
 |  *  add_part() | 
 |  *    0 success. | 
 |  *    1 can not add so many partitions. | 
 |  */ | 
 | int cmdline_parts_set(struct cmdline_parts *parts, sector_t disk_size, | 
 | 		      int slot, | 
 | 		      int (*add_part)(int, struct cmdline_subpart *, void *), | 
 | 		      void *param) | 
 | { | 
 | 	sector_t from = 0; | 
 | 	struct cmdline_subpart *subpart; | 
 |  | 
 | 	for (subpart = parts->subpart; subpart; | 
 | 	     subpart = subpart->next_subpart, slot++) { | 
 | 		if (subpart->from == (sector_t)(~0ULL)) | 
 | 			subpart->from = from; | 
 | 		else | 
 | 			from = subpart->from; | 
 |  | 
 | 		if (from >= disk_size) | 
 | 			break; | 
 |  | 
 | 		if (subpart->size > (disk_size - from)) | 
 | 			subpart->size = disk_size - from; | 
 |  | 
 | 		from += subpart->size; | 
 |  | 
 | 		if (add_part(slot, subpart, param)) | 
 | 			break; | 
 | 	} | 
 |  | 
 | 	return slot; | 
 | } | 
 | EXPORT_SYMBOL(cmdline_parts_set); |