drivers/vpd: Add VPD support
VPD reference: https://chromium.googlesource.com/chromiumos/platform/vpd/+/master/README.md Copy ChromeOS VPD driver to add support for VPD without CROMEOS. Possible use case: * Storing calibration data * Storing MAC address * Storing serial * Storing boot options + Now it's possible to define the VPD space by choosing one of the following enums: VPD_ANY, VPD_RW, VPD_RO. + CHROMEOS selects now VPD as part of it. + VPD is implemented as driver. Change-Id: Id9263bd39bf25d024e93daa57053fefcb1adc53a Signed-off-by: Patrick Rudolph <patrick.rudolph@9elements.com> Reviewed-on: https://review.coreboot.org/25046 Reviewed-by: David Hendricks <david.hendricks@gmail.com> Tested-by: build bot (Jenkins) <no-reply@coreboot.org>
This commit is contained in:
		
				
					committed by
					
						 Philipp Deppenwiese
						Philipp Deppenwiese
					
				
			
			
				
	
			
			
			
						parent
						
							c4b0fd0a86
						
					
				
				
					commit
					28cee59ca2
				
			
							
								
								
									
										113
									
								
								src/drivers/vpd/lib_vpd.c
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										113
									
								
								src/drivers/vpd/lib_vpd.c
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,113 @@ | ||||
| /* | ||||
|  * Copyright (c) 2014 The Chromium OS Authors. All rights reserved. | ||||
|  * Use of this source code is governed by a BSD-style license that can be | ||||
|  * found in the LICENSE file. | ||||
|  * | ||||
|  */ | ||||
| #include <assert.h> | ||||
| #include "lib_vpd.h" | ||||
|  | ||||
| /* Given an encoded string, this functions decodes the length field which varies | ||||
|  * from 1 byte to many bytes. | ||||
|  * | ||||
|  * The in points the actual byte going to be decoded. The *length returns | ||||
|  * the decoded length field. The number of consumed bytes will be stroed in | ||||
|  * decoded_len. | ||||
|  * | ||||
|  * Returns VPD_FAIL if more bit is 1, but actually reaches the end of string. | ||||
|  */ | ||||
| int decodeLen(const int32_t max_len, | ||||
| 	      const uint8_t *in, | ||||
| 	      int32_t *length, | ||||
| 	      int32_t *decoded_len) | ||||
| { | ||||
| 	uint8_t more; | ||||
| 	int i = 0; | ||||
|  | ||||
| 	assert(length); | ||||
| 	assert(decoded_len); | ||||
|  | ||||
| 	*length = 0; | ||||
| 	do { | ||||
| 		if (i >= max_len) | ||||
| 			return VPD_FAIL; | ||||
|  | ||||
| 		more = in[i] & 0x80; | ||||
| 		*length <<= 7; | ||||
| 		*length |= in[i] & 0x7f; | ||||
| 		++i; | ||||
| 	} while (more); | ||||
|  | ||||
| 	*decoded_len = i; | ||||
|  | ||||
| 	return VPD_OK; | ||||
| } | ||||
|  | ||||
| /* Given the encoded string, this function invokes callback with extracted | ||||
|  * (key, value). The *consumed will be plused the number of bytes consumed in | ||||
|  * this function. | ||||
|  * | ||||
|  * The input_buf points to the first byte of the input buffer. | ||||
|  * | ||||
|  * The *consumed starts from 0, which is actually the next byte to be decoded. | ||||
|  * It can be non-zero to be used in multiple calls. | ||||
|  * | ||||
|  * If one entry is successfully decoded, sends it to callback and returns the | ||||
|  * result. | ||||
|  */ | ||||
| int decodeVpdString(const int32_t max_len, | ||||
| 		    const uint8_t *input_buf, | ||||
| 		    int32_t *consumed, | ||||
| 		    VpdDecodeCallback callback, | ||||
| 		    void *callback_arg) | ||||
| { | ||||
| 	int type; | ||||
| 	int32_t key_len, value_len; | ||||
| 	int32_t decoded_len; | ||||
| 	const uint8_t *key, *value; | ||||
|  | ||||
| 	/* type */ | ||||
| 	if (*consumed >= max_len) | ||||
| 		return VPD_FAIL; | ||||
|  | ||||
| 	type = input_buf[*consumed]; | ||||
| 	switch (type) { | ||||
| 	case VPD_TYPE_INFO: | ||||
| 	case VPD_TYPE_STRING: | ||||
| 		(*consumed)++; | ||||
| 		/* key */ | ||||
| 		if (VPD_OK != decodeLen(max_len - *consumed, | ||||
| 					&input_buf[*consumed], &key_len, | ||||
| 					&decoded_len) || | ||||
| 				*consumed + decoded_len >= max_len) { | ||||
| 				return VPD_FAIL; | ||||
| 		} | ||||
|  | ||||
| 		*consumed += decoded_len; | ||||
| 		key = &input_buf[*consumed]; | ||||
| 		*consumed += key_len; | ||||
|  | ||||
| 		/* value */ | ||||
| 		if (VPD_OK != decodeLen(max_len - *consumed, | ||||
| 					&input_buf[*consumed], | ||||
| 					&value_len, &decoded_len) || | ||||
| 					*consumed + decoded_len > max_len) { | ||||
| 					return VPD_FAIL; | ||||
| 		} | ||||
| 		*consumed += decoded_len; | ||||
| 		value = &input_buf[*consumed]; | ||||
| 		*consumed += value_len; | ||||
|  | ||||
| 		if (type == VPD_TYPE_STRING) | ||||
| 			return callback(key, key_len, value, value_len, | ||||
| 					callback_arg); | ||||
|  | ||||
| 		return VPD_OK; | ||||
|  | ||||
| 	default: | ||||
| 		return VPD_FAIL; | ||||
| 	break; | ||||
| 	} | ||||
|  | ||||
| 	return VPD_OK; | ||||
| } | ||||
		Reference in New Issue
	
	Block a user