xref: /openbmc/u-boot/drivers/dfu/dfu_ram.c (revision 83d290c56fab2d38cd1ab4c4cc7099559c1d5046)
1*83d290c5STom Rini // SPDX-License-Identifier: GPL-2.0+
2a9479f04SAfzal Mohammed /*
3a9479f04SAfzal Mohammed  * (C) Copyright 2013
4a9479f04SAfzal Mohammed  * Afzal Mohammed <afzal.mohd.ma@gmail.com>
5a9479f04SAfzal Mohammed  *
6a9479f04SAfzal Mohammed  * Reference: dfu_mmc.c
7a9479f04SAfzal Mohammed  * Copyright (C) 2012 Samsung Electronics
8a9479f04SAfzal Mohammed  * author: Lukasz Majewski <l.majewski@samsung.com>
9a9479f04SAfzal Mohammed  */
10a9479f04SAfzal Mohammed 
11a9479f04SAfzal Mohammed #include <common.h>
12a9479f04SAfzal Mohammed #include <malloc.h>
13a9479f04SAfzal Mohammed #include <errno.h>
14a9479f04SAfzal Mohammed #include <dfu.h>
15a9479f04SAfzal Mohammed 
dfu_transfer_medium_ram(enum dfu_op op,struct dfu_entity * dfu,u64 offset,void * buf,long * len)16a9479f04SAfzal Mohammed static int dfu_transfer_medium_ram(enum dfu_op op, struct dfu_entity *dfu,
17a9479f04SAfzal Mohammed 				   u64 offset, void *buf, long *len)
18a9479f04SAfzal Mohammed {
19a9479f04SAfzal Mohammed 	if (dfu->layout != DFU_RAM_ADDR) {
209b643e31SMasahiro Yamada 		pr_err("unsupported layout: %s\n", dfu_get_layout(dfu->layout));
21a9479f04SAfzal Mohammed 		return  -EINVAL;
22a9479f04SAfzal Mohammed 	}
23a9479f04SAfzal Mohammed 
24a9479f04SAfzal Mohammed 	if (offset > dfu->data.ram.size) {
259b643e31SMasahiro Yamada 		pr_err("request exceeds allowed area\n");
26a9479f04SAfzal Mohammed 		return -EINVAL;
27a9479f04SAfzal Mohammed 	}
28a9479f04SAfzal Mohammed 
29a9479f04SAfzal Mohammed 	if (op == DFU_OP_WRITE)
30a9479f04SAfzal Mohammed 		memcpy(dfu->data.ram.start + offset, buf, *len);
31a9479f04SAfzal Mohammed 	else
32a9479f04SAfzal Mohammed 		memcpy(buf, dfu->data.ram.start + offset, *len);
33a9479f04SAfzal Mohammed 
34a9479f04SAfzal Mohammed 	return 0;
35a9479f04SAfzal Mohammed }
36a9479f04SAfzal Mohammed 
dfu_write_medium_ram(struct dfu_entity * dfu,u64 offset,void * buf,long * len)37a9479f04SAfzal Mohammed static int dfu_write_medium_ram(struct dfu_entity *dfu, u64 offset,
38a9479f04SAfzal Mohammed 				void *buf, long *len)
39a9479f04SAfzal Mohammed {
40a9479f04SAfzal Mohammed 	return dfu_transfer_medium_ram(DFU_OP_WRITE, dfu, offset, buf, len);
41a9479f04SAfzal Mohammed }
42a9479f04SAfzal Mohammed 
dfu_get_medium_size_ram(struct dfu_entity * dfu,u64 * size)4315970d87SPatrick Delaunay int dfu_get_medium_size_ram(struct dfu_entity *dfu, u64 *size)
440e285b50SStephen Warren {
454de51201SPatrick Delaunay 	*size = dfu->data.ram.size;
464de51201SPatrick Delaunay 
474de51201SPatrick Delaunay 	return 0;
480e285b50SStephen Warren }
490e285b50SStephen Warren 
dfu_read_medium_ram(struct dfu_entity * dfu,u64 offset,void * buf,long * len)50a9479f04SAfzal Mohammed static int dfu_read_medium_ram(struct dfu_entity *dfu, u64 offset,
51a9479f04SAfzal Mohammed 			       void *buf, long *len)
52a9479f04SAfzal Mohammed {
53a9479f04SAfzal Mohammed 	return dfu_transfer_medium_ram(DFU_OP_READ, dfu, offset, buf, len);
54a9479f04SAfzal Mohammed }
55a9479f04SAfzal Mohammed 
dfu_fill_entity_ram(struct dfu_entity * dfu,char * devstr,char * s)56dd64827eSStephen Warren int dfu_fill_entity_ram(struct dfu_entity *dfu, char *devstr, char *s)
57a9479f04SAfzal Mohammed {
58e1b0f6feSMugunthan V N 	const char *argv[3];
59e1b0f6feSMugunthan V N 	const char **parg = argv;
60e1b0f6feSMugunthan V N 
61e1b0f6feSMugunthan V N 	for (; parg < argv + sizeof(argv) / sizeof(*argv); ++parg) {
62e1b0f6feSMugunthan V N 		*parg = strsep(&s, " ");
63e1b0f6feSMugunthan V N 		if (*parg == NULL) {
649b643e31SMasahiro Yamada 			pr_err("Invalid number of arguments.\n");
65e1b0f6feSMugunthan V N 			return -ENODEV;
66e1b0f6feSMugunthan V N 		}
67e1b0f6feSMugunthan V N 	}
68a9479f04SAfzal Mohammed 
69a9479f04SAfzal Mohammed 	dfu->dev_type = DFU_DEV_RAM;
70e1b0f6feSMugunthan V N 	if (strcmp(argv[0], "ram")) {
719b643e31SMasahiro Yamada 		pr_err("unsupported device: %s\n", argv[0]);
72a9479f04SAfzal Mohammed 		return -ENODEV;
73a9479f04SAfzal Mohammed 	}
74a9479f04SAfzal Mohammed 
75a9479f04SAfzal Mohammed 	dfu->layout = DFU_RAM_ADDR;
763517de6dSStephen Warren 	dfu->data.ram.start = (void *)simple_strtoul(argv[1], NULL, 16);
773517de6dSStephen Warren 	dfu->data.ram.size = simple_strtoul(argv[2], NULL, 16);
78a9479f04SAfzal Mohammed 
79a9479f04SAfzal Mohammed 	dfu->write_medium = dfu_write_medium_ram;
800e285b50SStephen Warren 	dfu->get_medium_size = dfu_get_medium_size_ram;
81a9479f04SAfzal Mohammed 	dfu->read_medium = dfu_read_medium_ram;
82a9479f04SAfzal Mohammed 
83a9479f04SAfzal Mohammed 	dfu->inited = 0;
84a9479f04SAfzal Mohammed 
85a9479f04SAfzal Mohammed 	return 0;
86a9479f04SAfzal Mohammed }
87