mirror of
				https://github.com/smaeul/u-boot.git
				synced 2025-10-24 17:48:14 +01:00 
			
		
		
		
	Implement do_tftpb(). This implementation of the tftp command
supports an optional port number. For example:
 tftp 192.168.0.30:9069:file.bin
It also supports taking the server IP from ${tftpserverip} if
defined, before falling back to ${serverip}.
Signed-off-by: Jerome Forissier <jerome.forissier@linaro.org>
Acked-by: Ilias Apalodimas <ilias.apalodimas@linaro.org>
Tested-by: Ilias Apalodimas <ilias.apalodimas@linaro.org>
		
	
			
		
			
				
	
	
		
			153 lines
		
	
	
		
			3.4 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
			
		
		
	
	
			153 lines
		
	
	
		
			3.4 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
| // SPDX-License-Identifier: Intel
 | |
| /*
 | |
|  * Access to binman information at runtime
 | |
|  *
 | |
|  * Copyright 2019 Google LLC
 | |
|  * Written by Simon Glass <sjg@chromium.org>
 | |
|  */
 | |
| 
 | |
| #include <binman.h>
 | |
| #include <dm.h>
 | |
| #include <log.h>
 | |
| #include <malloc.h>
 | |
| #include <mapmem.h>
 | |
| 
 | |
| /**
 | |
|  * struct binman_info - Information needed by the binman library
 | |
|  *
 | |
|  * @image: Node describing the image we are running from
 | |
|  * @rom_offset: Offset from an image_pos to the memory-mapped address, or
 | |
|  *	ROM_OFFSET_NONE if the ROM is not memory-mapped. Can be positive or
 | |
|  *	negative
 | |
|  */
 | |
| struct binman_info {
 | |
| 	ofnode image;
 | |
| 	int rom_offset;
 | |
| };
 | |
| 
 | |
| #define ROM_OFFSET_NONE		(-1)
 | |
| 
 | |
| static struct binman_info *binman;
 | |
| 
 | |
| /**
 | |
|  * find_image_node() - Find the top-level binman node
 | |
|  *
 | |
|  * Finds the binman node which can be used to load entries. The correct node
 | |
|  * depends on whether multiple-images is in use.
 | |
|  *
 | |
|  * @nodep: Returns the node found, on success
 | |
|  * Return: 0 if OK, , -EINVAL if there is no /binman node, -ECHILD if multiple
 | |
|  * images are being used but the first image is not available
 | |
|  */
 | |
| static int find_image_node(ofnode *nodep)
 | |
| {
 | |
| 	ofnode node;
 | |
| 
 | |
| 	node = ofnode_path("/binman");
 | |
| 	if (!ofnode_valid(node))
 | |
| 		return log_msg_ret("binman node", -EINVAL);
 | |
| 	if (ofnode_read_bool(node, "multiple-images")) {
 | |
| 		node = ofnode_first_subnode(node);
 | |
| 
 | |
| 		if (!ofnode_valid(node))
 | |
| 			return log_msg_ret("first image", -ECHILD);
 | |
| 	}
 | |
| 	*nodep = node;
 | |
| 
 | |
| 	return 0;
 | |
| }
 | |
| 
 | |
| static int binman_entry_find_internal(ofnode node, const char *name,
 | |
| 				      struct binman_entry *entry)
 | |
| {
 | |
| 	int ret;
 | |
| 
 | |
| 	if (!ofnode_valid(node))
 | |
| 		node = binman->image;
 | |
| 	node = ofnode_find_subnode(node, name);
 | |
| 	if (!ofnode_valid(node))
 | |
| 		return log_msg_ret("node", -ENOENT);
 | |
| 
 | |
| 	ret = ofnode_read_u32(node, "image-pos", &entry->image_pos);
 | |
| 	if (ret)
 | |
| 		return log_msg_ret("image-pos", ret);
 | |
| 	ret = ofnode_read_u32(node, "size", &entry->size);
 | |
| 	if (ret)
 | |
| 		return log_msg_ret("size", ret);
 | |
| 
 | |
| 	return 0;
 | |
| }
 | |
| 
 | |
| int binman_entry_find(const char *name, struct binman_entry *entry)
 | |
| {
 | |
| 	return binman_entry_find_internal(binman->image, name, entry);
 | |
| }
 | |
| 
 | |
| int binman_entry_map(ofnode parent, const char *name, void **bufp, int *sizep)
 | |
| {
 | |
| 	struct binman_entry entry;
 | |
| 	int ret;
 | |
| 
 | |
| 	if (binman->rom_offset == ROM_OFFSET_NONE)
 | |
| 		return -EPERM;
 | |
| 	ret = binman_entry_find_internal(parent, name, &entry);
 | |
| 	if (ret)
 | |
| 		return log_msg_ret("entry", ret);
 | |
| 	if (sizep)
 | |
| 		*sizep = entry.size;
 | |
| 	*bufp = map_sysmem(entry.image_pos + binman->rom_offset, entry.size);
 | |
| 
 | |
| 	return 0;
 | |
| }
 | |
| 
 | |
| ofnode binman_section_find_node(const char *name)
 | |
| {
 | |
| 	return ofnode_find_subnode(binman->image, name);
 | |
| }
 | |
| 
 | |
| void binman_set_rom_offset(int rom_offset)
 | |
| {
 | |
| 	binman->rom_offset = rom_offset;
 | |
| }
 | |
| 
 | |
| int binman_get_rom_offset(void)
 | |
| {
 | |
| 	return binman->rom_offset;
 | |
| }
 | |
| 
 | |
| int binman_select_subnode(const char *name)
 | |
| {
 | |
| 	ofnode node;
 | |
| 	int ret;
 | |
| 
 | |
| 	ret = find_image_node(&node);
 | |
| 	if (ret)
 | |
| 		return log_msg_ret("main", -ENOENT);
 | |
| 	node = ofnode_find_subnode(node, name);
 | |
| 	if (!ofnode_valid(node))
 | |
| 		return log_msg_ret("node", -ENOENT);
 | |
| 	binman->image = node;
 | |
| 	log_info("binman: Selected image subnode '%s'\n",
 | |
| 		 ofnode_get_name(binman->image));
 | |
| 
 | |
| 	return 0;
 | |
| }
 | |
| 
 | |
| int binman_init(void)
 | |
| {
 | |
| 	int ret;
 | |
| 
 | |
| 	return 0;
 | |
| 	binman = malloc(sizeof(struct binman_info));
 | |
| 	if (!binman)
 | |
| 		return log_msg_ret("space for binman", -ENOMEM);
 | |
| 	ret = find_image_node(&binman->image);
 | |
| 	if (ret)
 | |
| 		return log_msg_ret("node", -ENOENT);
 | |
| 	binman_set_rom_offset(ROM_OFFSET_NONE);
 | |
| 	log_debug("binman: Selected image node '%s'\n",
 | |
| 		  ofnode_get_name(binman->image));
 | |
| 
 | |
| 	return 0;
 | |
| }
 |