xref: /dpdk/drivers/common/sfc_efx/base/efx_bootcfg.c (revision 672386c1e9e1f64f7aa3b1360ad22dc737ea8d72)
15e111ed8SAndrew Rybchenko /* SPDX-License-Identifier: BSD-3-Clause
25e111ed8SAndrew Rybchenko  *
3*672386c1SAndrew Rybchenko  * Copyright(c) 2019-2021 Xilinx, Inc.
45e111ed8SAndrew Rybchenko  * Copyright(c) 2009-2019 Solarflare Communications Inc.
55e111ed8SAndrew Rybchenko  */
65e111ed8SAndrew Rybchenko 
75e111ed8SAndrew Rybchenko #include "efx.h"
85e111ed8SAndrew Rybchenko #include "efx_impl.h"
95e111ed8SAndrew Rybchenko 
105e111ed8SAndrew Rybchenko #if EFSYS_OPT_BOOTCFG
115e111ed8SAndrew Rybchenko 
125e111ed8SAndrew Rybchenko /*
135e111ed8SAndrew Rybchenko  * Maximum size of BOOTCFG block across all nics as understood by SFCgPXE.
145e111ed8SAndrew Rybchenko  * NOTE: This is larger than the Medford per-PF bootcfg sector.
155e111ed8SAndrew Rybchenko  */
165e111ed8SAndrew Rybchenko #define	BOOTCFG_MAX_SIZE 0x1000
175e111ed8SAndrew Rybchenko 
185e111ed8SAndrew Rybchenko /* Medford per-PF bootcfg sector */
195e111ed8SAndrew Rybchenko #define	BOOTCFG_PER_PF   0x800
205e111ed8SAndrew Rybchenko #define	BOOTCFG_PF_COUNT 16
215e111ed8SAndrew Rybchenko 
225e111ed8SAndrew Rybchenko #define	DHCP_OPT_HAS_VALUE(opt) \
235e111ed8SAndrew Rybchenko 	(((opt) > EFX_DHCP_PAD) && ((opt) < EFX_DHCP_END))
245e111ed8SAndrew Rybchenko 
255e111ed8SAndrew Rybchenko #define	DHCP_MAX_VALUE 255
265e111ed8SAndrew Rybchenko 
275e111ed8SAndrew Rybchenko #define	DHCP_ENCAPSULATOR(encap_opt) ((encap_opt) >> 8)
285e111ed8SAndrew Rybchenko #define	DHCP_ENCAPSULATED(encap_opt) ((encap_opt) & 0xff)
295e111ed8SAndrew Rybchenko #define	DHCP_IS_ENCAP_OPT(opt) DHCP_OPT_HAS_VALUE(DHCP_ENCAPSULATOR(opt))
305e111ed8SAndrew Rybchenko 
315e111ed8SAndrew Rybchenko typedef struct efx_dhcp_tag_hdr_s {
325e111ed8SAndrew Rybchenko 	uint8_t		tag;
335e111ed8SAndrew Rybchenko 	uint8_t		length;
345e111ed8SAndrew Rybchenko } efx_dhcp_tag_hdr_t;
355e111ed8SAndrew Rybchenko 
365e111ed8SAndrew Rybchenko /*
375e111ed8SAndrew Rybchenko  * Length calculations for tags with value field. PAD and END
385e111ed8SAndrew Rybchenko  * have a fixed length of 1, with no length or value field.
395e111ed8SAndrew Rybchenko  */
405e111ed8SAndrew Rybchenko #define	DHCP_FULL_TAG_LENGTH(hdr) \
415e111ed8SAndrew Rybchenko 	(sizeof (efx_dhcp_tag_hdr_t) + (hdr)->length)
425e111ed8SAndrew Rybchenko 
435e111ed8SAndrew Rybchenko #define	DHCP_NEXT_TAG(hdr) \
445e111ed8SAndrew Rybchenko 	((efx_dhcp_tag_hdr_t *)(((uint8_t *)(hdr)) + \
455e111ed8SAndrew Rybchenko 	DHCP_FULL_TAG_LENGTH((hdr))))
465e111ed8SAndrew Rybchenko 
475e111ed8SAndrew Rybchenko #define	DHCP_CALC_TAG_LENGTH(payload_len) \
485e111ed8SAndrew Rybchenko 	((payload_len) + sizeof (efx_dhcp_tag_hdr_t))
495e111ed8SAndrew Rybchenko 
505e111ed8SAndrew Rybchenko 
515e111ed8SAndrew Rybchenko /* Report the layout of bootcfg sectors in NVRAM partition. */
525e111ed8SAndrew Rybchenko 	__checkReturn		efx_rc_t
efx_bootcfg_sector_info(__in efx_nic_t * enp,__in uint32_t pf,__out_opt uint32_t * sector_countp,__out size_t * offsetp,__out size_t * max_sizep)535e111ed8SAndrew Rybchenko efx_bootcfg_sector_info(
545e111ed8SAndrew Rybchenko 	__in			efx_nic_t *enp,
555e111ed8SAndrew Rybchenko 	__in			uint32_t pf,
565e111ed8SAndrew Rybchenko 	__out_opt		uint32_t *sector_countp,
575e111ed8SAndrew Rybchenko 	__out			size_t *offsetp,
585e111ed8SAndrew Rybchenko 	__out			size_t *max_sizep)
595e111ed8SAndrew Rybchenko {
605e111ed8SAndrew Rybchenko 	uint32_t count;
615e111ed8SAndrew Rybchenko 	size_t max_size;
625e111ed8SAndrew Rybchenko 	size_t offset;
635e111ed8SAndrew Rybchenko 	int rc;
645e111ed8SAndrew Rybchenko 
655e111ed8SAndrew Rybchenko 	switch (enp->en_family) {
665e111ed8SAndrew Rybchenko #if EFSYS_OPT_SIENA
675e111ed8SAndrew Rybchenko 	case EFX_FAMILY_SIENA:
685e111ed8SAndrew Rybchenko 		max_size = BOOTCFG_MAX_SIZE;
695e111ed8SAndrew Rybchenko 		offset = 0;
705e111ed8SAndrew Rybchenko 		count = 1;
715e111ed8SAndrew Rybchenko 		break;
725e111ed8SAndrew Rybchenko #endif /* EFSYS_OPT_SIENA */
735e111ed8SAndrew Rybchenko 
745e111ed8SAndrew Rybchenko #if EFSYS_OPT_HUNTINGTON
755e111ed8SAndrew Rybchenko 	case EFX_FAMILY_HUNTINGTON:
765e111ed8SAndrew Rybchenko 		max_size = BOOTCFG_MAX_SIZE;
775e111ed8SAndrew Rybchenko 		offset = 0;
785e111ed8SAndrew Rybchenko 		count = 1;
795e111ed8SAndrew Rybchenko 		break;
805e111ed8SAndrew Rybchenko #endif /* EFSYS_OPT_HUNTINGTON */
815e111ed8SAndrew Rybchenko 
825e111ed8SAndrew Rybchenko #if EFSYS_OPT_MEDFORD
835e111ed8SAndrew Rybchenko 	case EFX_FAMILY_MEDFORD: {
845e111ed8SAndrew Rybchenko 		/* Shared partition (array indexed by PF) */
855e111ed8SAndrew Rybchenko 		max_size = BOOTCFG_PER_PF;
865e111ed8SAndrew Rybchenko 		count = BOOTCFG_PF_COUNT;
875e111ed8SAndrew Rybchenko 		if (pf >= count) {
885e111ed8SAndrew Rybchenko 			rc = EINVAL;
895e111ed8SAndrew Rybchenko 			goto fail2;
905e111ed8SAndrew Rybchenko 		}
915e111ed8SAndrew Rybchenko 		offset = max_size * pf;
925e111ed8SAndrew Rybchenko 		break;
935e111ed8SAndrew Rybchenko 	}
945e111ed8SAndrew Rybchenko #endif /* EFSYS_OPT_MEDFORD */
955e111ed8SAndrew Rybchenko 
965e111ed8SAndrew Rybchenko #if EFSYS_OPT_MEDFORD2
975e111ed8SAndrew Rybchenko 	case EFX_FAMILY_MEDFORD2: {
985e111ed8SAndrew Rybchenko 		/* Shared partition (array indexed by PF) */
995e111ed8SAndrew Rybchenko 		max_size = BOOTCFG_PER_PF;
1005e111ed8SAndrew Rybchenko 		count = BOOTCFG_PF_COUNT;
1015e111ed8SAndrew Rybchenko 		if (pf >= count) {
1025e111ed8SAndrew Rybchenko 			rc = EINVAL;
1035e111ed8SAndrew Rybchenko 			goto fail3;
1045e111ed8SAndrew Rybchenko 		}
1055e111ed8SAndrew Rybchenko 		offset = max_size * pf;
1065e111ed8SAndrew Rybchenko 		break;
1075e111ed8SAndrew Rybchenko 	}
1085e111ed8SAndrew Rybchenko #endif /* EFSYS_OPT_MEDFORD2 */
1095e111ed8SAndrew Rybchenko 
1105e111ed8SAndrew Rybchenko 	default:
1115e111ed8SAndrew Rybchenko 		EFSYS_ASSERT(0);
1125e111ed8SAndrew Rybchenko 		rc = ENOTSUP;
1135e111ed8SAndrew Rybchenko 		goto fail1;
1145e111ed8SAndrew Rybchenko 	}
1155e111ed8SAndrew Rybchenko 	EFSYS_ASSERT3U(max_size, <=, BOOTCFG_MAX_SIZE);
1165e111ed8SAndrew Rybchenko 
1175e111ed8SAndrew Rybchenko 	if (sector_countp != NULL)
1185e111ed8SAndrew Rybchenko 		*sector_countp = count;
1195e111ed8SAndrew Rybchenko 	*offsetp = offset;
1205e111ed8SAndrew Rybchenko 	*max_sizep = max_size;
1215e111ed8SAndrew Rybchenko 
1225e111ed8SAndrew Rybchenko 	return (0);
1235e111ed8SAndrew Rybchenko 
1245e111ed8SAndrew Rybchenko #if EFSYS_OPT_MEDFORD2
1255e111ed8SAndrew Rybchenko fail3:
1265e111ed8SAndrew Rybchenko 	EFSYS_PROBE(fail3);
1275e111ed8SAndrew Rybchenko #endif
1285e111ed8SAndrew Rybchenko #if EFSYS_OPT_MEDFORD
1295e111ed8SAndrew Rybchenko fail2:
1305e111ed8SAndrew Rybchenko 	EFSYS_PROBE(fail2);
1315e111ed8SAndrew Rybchenko #endif
1325e111ed8SAndrew Rybchenko fail1:
1335e111ed8SAndrew Rybchenko 	EFSYS_PROBE1(fail1, efx_rc_t, rc);
1345e111ed8SAndrew Rybchenko 	return (rc);
1355e111ed8SAndrew Rybchenko }
1365e111ed8SAndrew Rybchenko 
1375e111ed8SAndrew Rybchenko 
1385e111ed8SAndrew Rybchenko 	__checkReturn		uint8_t
efx_dhcp_csum(__in_bcount (size)uint8_t const * data,__in size_t size)1395e111ed8SAndrew Rybchenko efx_dhcp_csum(
1405e111ed8SAndrew Rybchenko 	__in_bcount(size)	uint8_t const *data,
1415e111ed8SAndrew Rybchenko 	__in			size_t size)
1425e111ed8SAndrew Rybchenko {
1435e111ed8SAndrew Rybchenko 	unsigned int pos;
1445e111ed8SAndrew Rybchenko 	uint8_t checksum = 0;
1455e111ed8SAndrew Rybchenko 
1465e111ed8SAndrew Rybchenko 	for (pos = 0; pos < size; pos++)
1475e111ed8SAndrew Rybchenko 		checksum += data[pos];
1485e111ed8SAndrew Rybchenko 	return (checksum);
1495e111ed8SAndrew Rybchenko }
1505e111ed8SAndrew Rybchenko 
1515e111ed8SAndrew Rybchenko 	__checkReturn		efx_rc_t
efx_dhcp_verify(__in_bcount (size)uint8_t const * data,__in size_t size,__out_opt size_t * usedp)1525e111ed8SAndrew Rybchenko efx_dhcp_verify(
1535e111ed8SAndrew Rybchenko 	__in_bcount(size)	uint8_t const *data,
1545e111ed8SAndrew Rybchenko 	__in			size_t size,
1555e111ed8SAndrew Rybchenko 	__out_opt		size_t *usedp)
1565e111ed8SAndrew Rybchenko {
1575e111ed8SAndrew Rybchenko 	size_t offset = 0;
1585e111ed8SAndrew Rybchenko 	size_t used = 0;
1595e111ed8SAndrew Rybchenko 	efx_rc_t rc;
1605e111ed8SAndrew Rybchenko 
1615e111ed8SAndrew Rybchenko 	/* Start parsing tags immediately after the checksum */
1625e111ed8SAndrew Rybchenko 	for (offset = 1; offset < size; ) {
1635e111ed8SAndrew Rybchenko 		uint8_t tag;
1645e111ed8SAndrew Rybchenko 		uint8_t length;
1655e111ed8SAndrew Rybchenko 
1665e111ed8SAndrew Rybchenko 		/* Consume tag */
1675e111ed8SAndrew Rybchenko 		tag = data[offset];
1685e111ed8SAndrew Rybchenko 		if (tag == EFX_DHCP_END) {
1695e111ed8SAndrew Rybchenko 			offset++;
1705e111ed8SAndrew Rybchenko 			used = offset;
1715e111ed8SAndrew Rybchenko 			break;
1725e111ed8SAndrew Rybchenko 		}
1735e111ed8SAndrew Rybchenko 		if (tag == EFX_DHCP_PAD) {
1745e111ed8SAndrew Rybchenko 			offset++;
1755e111ed8SAndrew Rybchenko 			continue;
1765e111ed8SAndrew Rybchenko 		}
1775e111ed8SAndrew Rybchenko 
1785e111ed8SAndrew Rybchenko 		/* Consume length */
1795e111ed8SAndrew Rybchenko 		if (offset + 1 >= size) {
1805e111ed8SAndrew Rybchenko 			rc = ENOSPC;
1815e111ed8SAndrew Rybchenko 			goto fail1;
1825e111ed8SAndrew Rybchenko 		}
1835e111ed8SAndrew Rybchenko 		length = data[offset + 1];
1845e111ed8SAndrew Rybchenko 
1855e111ed8SAndrew Rybchenko 		/* Consume *length */
1865e111ed8SAndrew Rybchenko 		if (offset + 1 + length >= size) {
1875e111ed8SAndrew Rybchenko 			rc = ENOSPC;
1885e111ed8SAndrew Rybchenko 			goto fail2;
1895e111ed8SAndrew Rybchenko 		}
1905e111ed8SAndrew Rybchenko 
1915e111ed8SAndrew Rybchenko 		offset += 2 + length;
1925e111ed8SAndrew Rybchenko 		used = offset;
1935e111ed8SAndrew Rybchenko 	}
1945e111ed8SAndrew Rybchenko 
1955e111ed8SAndrew Rybchenko 	/* Checksum the entire sector, including bytes after any EFX_DHCP_END */
1965e111ed8SAndrew Rybchenko 	if (efx_dhcp_csum(data, size) != 0) {
1975e111ed8SAndrew Rybchenko 		rc = EINVAL;
1985e111ed8SAndrew Rybchenko 		goto fail3;
1995e111ed8SAndrew Rybchenko 	}
2005e111ed8SAndrew Rybchenko 
2015e111ed8SAndrew Rybchenko 	if (usedp != NULL)
2025e111ed8SAndrew Rybchenko 		*usedp = used;
2035e111ed8SAndrew Rybchenko 
2045e111ed8SAndrew Rybchenko 	return (0);
2055e111ed8SAndrew Rybchenko 
2065e111ed8SAndrew Rybchenko fail3:
2075e111ed8SAndrew Rybchenko 	EFSYS_PROBE(fail3);
2085e111ed8SAndrew Rybchenko fail2:
2095e111ed8SAndrew Rybchenko 	EFSYS_PROBE(fail2);
2105e111ed8SAndrew Rybchenko fail1:
2115e111ed8SAndrew Rybchenko 	EFSYS_PROBE1(fail1, efx_rc_t, rc);
2125e111ed8SAndrew Rybchenko 
2135e111ed8SAndrew Rybchenko 	return (rc);
2145e111ed8SAndrew Rybchenko }
2155e111ed8SAndrew Rybchenko 
2165e111ed8SAndrew Rybchenko /*
2175e111ed8SAndrew Rybchenko  * Walk the entire tag set looking for option. The sought option may be
2185e111ed8SAndrew Rybchenko  * encapsulated. ENOENT indicates the walk completed without finding the
2195e111ed8SAndrew Rybchenko  * option. If we run out of buffer during the walk the function will return
2205e111ed8SAndrew Rybchenko  * ENOSPC.
2215e111ed8SAndrew Rybchenko  */
2225e111ed8SAndrew Rybchenko static	efx_rc_t
efx_dhcp_walk_tags(__deref_inout uint8_t ** tagpp,__inout size_t * buffer_sizep,__in uint16_t opt)2235e111ed8SAndrew Rybchenko efx_dhcp_walk_tags(
2245e111ed8SAndrew Rybchenko 	__deref_inout	uint8_t **tagpp,
2255e111ed8SAndrew Rybchenko 	__inout		size_t *buffer_sizep,
2265e111ed8SAndrew Rybchenko 	__in		uint16_t opt)
2275e111ed8SAndrew Rybchenko {
2285e111ed8SAndrew Rybchenko 	efx_rc_t rc = 0;
2295e111ed8SAndrew Rybchenko 	boolean_t is_encap = B_FALSE;
2305e111ed8SAndrew Rybchenko 
2315e111ed8SAndrew Rybchenko 	if (DHCP_IS_ENCAP_OPT(opt)) {
2325e111ed8SAndrew Rybchenko 		/*
2335e111ed8SAndrew Rybchenko 		 * Look for the encapsulator and, if found, limit ourselves
2345e111ed8SAndrew Rybchenko 		 * to its payload. If it's not found then the entire tag
2355e111ed8SAndrew Rybchenko 		 * cannot be found, so the encapsulated opt search is
2365e111ed8SAndrew Rybchenko 		 * skipped.
2375e111ed8SAndrew Rybchenko 		 */
2385e111ed8SAndrew Rybchenko 		rc = efx_dhcp_walk_tags(tagpp, buffer_sizep,
2395e111ed8SAndrew Rybchenko 		    DHCP_ENCAPSULATOR(opt));
2405e111ed8SAndrew Rybchenko 		if (rc == 0) {
2415e111ed8SAndrew Rybchenko 			*buffer_sizep = ((efx_dhcp_tag_hdr_t *)*tagpp)->length;
2425e111ed8SAndrew Rybchenko 			(*tagpp) += sizeof (efx_dhcp_tag_hdr_t);
2435e111ed8SAndrew Rybchenko 		}
2445e111ed8SAndrew Rybchenko 		opt = DHCP_ENCAPSULATED(opt);
2455e111ed8SAndrew Rybchenko 		is_encap = B_TRUE;
2465e111ed8SAndrew Rybchenko 	}
2475e111ed8SAndrew Rybchenko 
2485e111ed8SAndrew Rybchenko 	EFSYS_ASSERT(!DHCP_IS_ENCAP_OPT(opt));
2495e111ed8SAndrew Rybchenko 
2505e111ed8SAndrew Rybchenko 	while (rc == 0) {
2515e111ed8SAndrew Rybchenko 		size_t size;
2525e111ed8SAndrew Rybchenko 
2535e111ed8SAndrew Rybchenko 		if (*buffer_sizep == 0) {
2545e111ed8SAndrew Rybchenko 			rc = ENOSPC;
2555e111ed8SAndrew Rybchenko 			goto fail1;
2565e111ed8SAndrew Rybchenko 		}
2575e111ed8SAndrew Rybchenko 
2585e111ed8SAndrew Rybchenko 		if (DHCP_ENCAPSULATED(**tagpp) == opt)
2595e111ed8SAndrew Rybchenko 			break;
2605e111ed8SAndrew Rybchenko 
2615e111ed8SAndrew Rybchenko 		if ((**tagpp) == EFX_DHCP_END) {
2625e111ed8SAndrew Rybchenko 			rc = ENOENT;
2635e111ed8SAndrew Rybchenko 			break;
2645e111ed8SAndrew Rybchenko 		} else if ((**tagpp) == EFX_DHCP_PAD) {
2655e111ed8SAndrew Rybchenko 			size = 1;
2665e111ed8SAndrew Rybchenko 		} else {
2675e111ed8SAndrew Rybchenko 			if (*buffer_sizep < sizeof (efx_dhcp_tag_hdr_t)) {
2685e111ed8SAndrew Rybchenko 				rc = ENOSPC;
2695e111ed8SAndrew Rybchenko 				goto fail2;
2705e111ed8SAndrew Rybchenko 			}
2715e111ed8SAndrew Rybchenko 
2725e111ed8SAndrew Rybchenko 			size =
2735e111ed8SAndrew Rybchenko 			    DHCP_FULL_TAG_LENGTH((efx_dhcp_tag_hdr_t *)*tagpp);
2745e111ed8SAndrew Rybchenko 		}
2755e111ed8SAndrew Rybchenko 
2765e111ed8SAndrew Rybchenko 		if (size > *buffer_sizep) {
2775e111ed8SAndrew Rybchenko 			rc = ENOSPC;
2785e111ed8SAndrew Rybchenko 			goto fail3;
2795e111ed8SAndrew Rybchenko 		}
2805e111ed8SAndrew Rybchenko 
2815e111ed8SAndrew Rybchenko 		(*tagpp) += size;
2825e111ed8SAndrew Rybchenko 		(*buffer_sizep) -= size;
2835e111ed8SAndrew Rybchenko 
2845e111ed8SAndrew Rybchenko 		if ((*buffer_sizep == 0) && is_encap) {
2855e111ed8SAndrew Rybchenko 			/* Search within encapulator tag finished */
2865e111ed8SAndrew Rybchenko 			rc = ENOENT;
2875e111ed8SAndrew Rybchenko 			break;
2885e111ed8SAndrew Rybchenko 		}
2895e111ed8SAndrew Rybchenko 	}
2905e111ed8SAndrew Rybchenko 
2915e111ed8SAndrew Rybchenko 	/*
2925e111ed8SAndrew Rybchenko 	 * Returns 0 if found otherwise ENOENT indicating search finished
2935e111ed8SAndrew Rybchenko 	 * correctly
2945e111ed8SAndrew Rybchenko 	 */
2955e111ed8SAndrew Rybchenko 	return (rc);
2965e111ed8SAndrew Rybchenko 
2975e111ed8SAndrew Rybchenko fail3:
2985e111ed8SAndrew Rybchenko 	EFSYS_PROBE(fail3);
2995e111ed8SAndrew Rybchenko fail2:
3005e111ed8SAndrew Rybchenko 	EFSYS_PROBE(fail2);
3015e111ed8SAndrew Rybchenko fail1:
3025e111ed8SAndrew Rybchenko 	EFSYS_PROBE1(fail1, efx_rc_t, rc);
3035e111ed8SAndrew Rybchenko 
3045e111ed8SAndrew Rybchenko 	return (rc);
3055e111ed8SAndrew Rybchenko }
3065e111ed8SAndrew Rybchenko 
3075e111ed8SAndrew Rybchenko /*
3085e111ed8SAndrew Rybchenko  * Locate value buffer for option in the given buffer.
3095e111ed8SAndrew Rybchenko  * Returns 0 if found, ENOENT indicating search finished
3105e111ed8SAndrew Rybchenko  * correctly, otherwise search failed before completion.
3115e111ed8SAndrew Rybchenko  */
3125e111ed8SAndrew Rybchenko 	__checkReturn	efx_rc_t
efx_dhcp_find_tag(__in_bcount (buffer_length)uint8_t * bufferp,__in size_t buffer_length,__in uint16_t opt,__deref_out uint8_t ** valuepp,__out size_t * value_lengthp)3135e111ed8SAndrew Rybchenko efx_dhcp_find_tag(
3145e111ed8SAndrew Rybchenko 	__in_bcount(buffer_length)	uint8_t *bufferp,
3155e111ed8SAndrew Rybchenko 	__in				size_t buffer_length,
3165e111ed8SAndrew Rybchenko 	__in				uint16_t opt,
3175e111ed8SAndrew Rybchenko 	__deref_out			uint8_t **valuepp,
3185e111ed8SAndrew Rybchenko 	__out				size_t *value_lengthp)
3195e111ed8SAndrew Rybchenko {
3205e111ed8SAndrew Rybchenko 	efx_rc_t rc;
3215e111ed8SAndrew Rybchenko 	uint8_t *tagp = bufferp;
3225e111ed8SAndrew Rybchenko 	size_t len = buffer_length;
3235e111ed8SAndrew Rybchenko 
3245e111ed8SAndrew Rybchenko 	rc = efx_dhcp_walk_tags(&tagp, &len, opt);
3255e111ed8SAndrew Rybchenko 	if (rc == 0) {
3265e111ed8SAndrew Rybchenko 		efx_dhcp_tag_hdr_t *hdrp;
3275e111ed8SAndrew Rybchenko 
3285e111ed8SAndrew Rybchenko 		hdrp = (efx_dhcp_tag_hdr_t *)tagp;
3295e111ed8SAndrew Rybchenko 		*valuepp = (uint8_t *)(&hdrp[1]);
3305e111ed8SAndrew Rybchenko 		*value_lengthp = hdrp->length;
3315e111ed8SAndrew Rybchenko 	} else if (rc != ENOENT) {
3325e111ed8SAndrew Rybchenko 		goto fail1;
3335e111ed8SAndrew Rybchenko 	}
3345e111ed8SAndrew Rybchenko 
3355e111ed8SAndrew Rybchenko 	return (rc);
3365e111ed8SAndrew Rybchenko 
3375e111ed8SAndrew Rybchenko fail1:
3385e111ed8SAndrew Rybchenko 	EFSYS_PROBE1(fail1, efx_rc_t, rc);
3395e111ed8SAndrew Rybchenko 
3405e111ed8SAndrew Rybchenko 	return (rc);
3415e111ed8SAndrew Rybchenko }
3425e111ed8SAndrew Rybchenko 
3435e111ed8SAndrew Rybchenko /*
3445e111ed8SAndrew Rybchenko  * Locate the end tag in the given buffer.
3455e111ed8SAndrew Rybchenko  * Returns 0 if found, ENOENT indicating search finished
3465e111ed8SAndrew Rybchenko  * correctly but end tag was not found; otherwise search
3475e111ed8SAndrew Rybchenko  * failed before completion.
3485e111ed8SAndrew Rybchenko  */
3495e111ed8SAndrew Rybchenko 	__checkReturn	efx_rc_t
efx_dhcp_find_end(__in_bcount (buffer_length)uint8_t * bufferp,__in size_t buffer_length,__deref_out uint8_t ** endpp)3505e111ed8SAndrew Rybchenko efx_dhcp_find_end(
3515e111ed8SAndrew Rybchenko 	__in_bcount(buffer_length)	uint8_t *bufferp,
3525e111ed8SAndrew Rybchenko 	__in				size_t buffer_length,
3535e111ed8SAndrew Rybchenko 	__deref_out			uint8_t **endpp)
3545e111ed8SAndrew Rybchenko {
3555e111ed8SAndrew Rybchenko 	efx_rc_t rc;
3565e111ed8SAndrew Rybchenko 	uint8_t *endp = bufferp;
3575e111ed8SAndrew Rybchenko 	size_t len = buffer_length;
3585e111ed8SAndrew Rybchenko 
3595e111ed8SAndrew Rybchenko 	rc = efx_dhcp_walk_tags(&endp, &len, EFX_DHCP_END);
3605e111ed8SAndrew Rybchenko 	if (rc == 0)
3615e111ed8SAndrew Rybchenko 		*endpp = endp;
3625e111ed8SAndrew Rybchenko 	else if (rc != ENOENT)
3635e111ed8SAndrew Rybchenko 		goto fail1;
3645e111ed8SAndrew Rybchenko 
3655e111ed8SAndrew Rybchenko 	return (rc);
3665e111ed8SAndrew Rybchenko 
3675e111ed8SAndrew Rybchenko fail1:
3685e111ed8SAndrew Rybchenko 	EFSYS_PROBE1(fail1, efx_rc_t, rc);
3695e111ed8SAndrew Rybchenko 
3705e111ed8SAndrew Rybchenko 	return (rc);
3715e111ed8SAndrew Rybchenko }
3725e111ed8SAndrew Rybchenko 
3735e111ed8SAndrew Rybchenko 
3745e111ed8SAndrew Rybchenko /*
3755e111ed8SAndrew Rybchenko  * Delete the given tag from anywhere in the buffer. Copes with
3765e111ed8SAndrew Rybchenko  * encapsulated tags, and updates or deletes the encapsulating opt as
3775e111ed8SAndrew Rybchenko  * necessary.
3785e111ed8SAndrew Rybchenko  */
3795e111ed8SAndrew Rybchenko 	__checkReturn	efx_rc_t
efx_dhcp_delete_tag(__inout_bcount (buffer_length)uint8_t * bufferp,__in size_t buffer_length,__in uint16_t opt)3805e111ed8SAndrew Rybchenko efx_dhcp_delete_tag(
3815e111ed8SAndrew Rybchenko 	__inout_bcount(buffer_length)	uint8_t *bufferp,
3825e111ed8SAndrew Rybchenko 	__in				size_t buffer_length,
3835e111ed8SAndrew Rybchenko 	__in				uint16_t opt)
3845e111ed8SAndrew Rybchenko {
3855e111ed8SAndrew Rybchenko 	efx_rc_t rc;
3865e111ed8SAndrew Rybchenko 	efx_dhcp_tag_hdr_t *hdrp;
3875e111ed8SAndrew Rybchenko 	size_t len;
3885e111ed8SAndrew Rybchenko 	uint8_t *startp;
3895e111ed8SAndrew Rybchenko 	uint8_t *endp;
3905e111ed8SAndrew Rybchenko 
3915e111ed8SAndrew Rybchenko 	len = buffer_length;
3925e111ed8SAndrew Rybchenko 	startp = bufferp;
3935e111ed8SAndrew Rybchenko 
3945e111ed8SAndrew Rybchenko 	if (!DHCP_OPT_HAS_VALUE(DHCP_ENCAPSULATED(opt))) {
3955e111ed8SAndrew Rybchenko 		rc = EINVAL;
3965e111ed8SAndrew Rybchenko 		goto fail1;
3975e111ed8SAndrew Rybchenko 	}
3985e111ed8SAndrew Rybchenko 
3995e111ed8SAndrew Rybchenko 	rc = efx_dhcp_walk_tags(&startp, &len, opt);
4005e111ed8SAndrew Rybchenko 	if (rc != 0)
4015e111ed8SAndrew Rybchenko 		goto fail1;
4025e111ed8SAndrew Rybchenko 
4035e111ed8SAndrew Rybchenko 	hdrp = (efx_dhcp_tag_hdr_t *)startp;
4045e111ed8SAndrew Rybchenko 
4055e111ed8SAndrew Rybchenko 	if (DHCP_IS_ENCAP_OPT(opt)) {
4065e111ed8SAndrew Rybchenko 		uint8_t tag_length = DHCP_FULL_TAG_LENGTH(hdrp);
4075e111ed8SAndrew Rybchenko 		uint8_t *encapp = bufferp;
4085e111ed8SAndrew Rybchenko 		efx_dhcp_tag_hdr_t *encap_hdrp;
4095e111ed8SAndrew Rybchenko 
4105e111ed8SAndrew Rybchenko 		len = buffer_length;
4115e111ed8SAndrew Rybchenko 		rc = efx_dhcp_walk_tags(&encapp, &len,
4125e111ed8SAndrew Rybchenko 		    DHCP_ENCAPSULATOR(opt));
4135e111ed8SAndrew Rybchenko 		if (rc != 0)
4145e111ed8SAndrew Rybchenko 			goto fail2;
4155e111ed8SAndrew Rybchenko 
4165e111ed8SAndrew Rybchenko 		encap_hdrp = (efx_dhcp_tag_hdr_t *)encapp;
4175e111ed8SAndrew Rybchenko 		if (encap_hdrp->length > tag_length) {
4185e111ed8SAndrew Rybchenko 			encap_hdrp->length = (uint8_t)(
4195e111ed8SAndrew Rybchenko 			    (size_t)encap_hdrp->length - tag_length);
4205e111ed8SAndrew Rybchenko 		} else {
4215e111ed8SAndrew Rybchenko 			/* delete the encapsulating tag */
4225e111ed8SAndrew Rybchenko 			hdrp = encap_hdrp;
4235e111ed8SAndrew Rybchenko 		}
4245e111ed8SAndrew Rybchenko 	}
4255e111ed8SAndrew Rybchenko 
4265e111ed8SAndrew Rybchenko 	startp = (uint8_t *)hdrp;
4275e111ed8SAndrew Rybchenko 	endp = (uint8_t *)DHCP_NEXT_TAG(hdrp);
4285e111ed8SAndrew Rybchenko 
4295e111ed8SAndrew Rybchenko 	if (startp < bufferp) {
4305e111ed8SAndrew Rybchenko 		rc = EINVAL;
4315e111ed8SAndrew Rybchenko 		goto fail3;
4325e111ed8SAndrew Rybchenko 	}
4335e111ed8SAndrew Rybchenko 
4345e111ed8SAndrew Rybchenko 	if (endp > &bufferp[buffer_length]) {
4355e111ed8SAndrew Rybchenko 		rc = EINVAL;
4365e111ed8SAndrew Rybchenko 		goto fail4;
4375e111ed8SAndrew Rybchenko 	}
4385e111ed8SAndrew Rybchenko 
4395e111ed8SAndrew Rybchenko 	memmove(startp, endp,
4405e111ed8SAndrew Rybchenko 		buffer_length - (endp - bufferp));
4415e111ed8SAndrew Rybchenko 
4425e111ed8SAndrew Rybchenko 	return (0);
4435e111ed8SAndrew Rybchenko 
4445e111ed8SAndrew Rybchenko fail4:
4455e111ed8SAndrew Rybchenko 	EFSYS_PROBE(fail4);
4465e111ed8SAndrew Rybchenko fail3:
4475e111ed8SAndrew Rybchenko 	EFSYS_PROBE(fail3);
4485e111ed8SAndrew Rybchenko fail2:
4495e111ed8SAndrew Rybchenko 	EFSYS_PROBE(fail2);
4505e111ed8SAndrew Rybchenko fail1:
4515e111ed8SAndrew Rybchenko 	EFSYS_PROBE1(fail1, efx_rc_t, rc);
4525e111ed8SAndrew Rybchenko 
4535e111ed8SAndrew Rybchenko 	return (rc);
4545e111ed8SAndrew Rybchenko }
4555e111ed8SAndrew Rybchenko 
4565e111ed8SAndrew Rybchenko /*
4575e111ed8SAndrew Rybchenko  * Write the tag header into write_pointp and optionally copies the payload
4585e111ed8SAndrew Rybchenko  * into the space following.
4595e111ed8SAndrew Rybchenko  */
4605e111ed8SAndrew Rybchenko static	void
efx_dhcp_write_tag(__in uint8_t * write_pointp,__in uint16_t opt,__in_bcount_opt (value_length)uint8_t * valuep,__in size_t value_length)4615e111ed8SAndrew Rybchenko efx_dhcp_write_tag(
4625e111ed8SAndrew Rybchenko 	__in		uint8_t *write_pointp,
4635e111ed8SAndrew Rybchenko 	__in		uint16_t opt,
4645e111ed8SAndrew Rybchenko 	__in_bcount_opt(value_length)
4655e111ed8SAndrew Rybchenko 			uint8_t *valuep,
4665e111ed8SAndrew Rybchenko 	__in		size_t value_length)
4675e111ed8SAndrew Rybchenko {
4685e111ed8SAndrew Rybchenko 	efx_dhcp_tag_hdr_t *hdrp = (efx_dhcp_tag_hdr_t *)write_pointp;
4695e111ed8SAndrew Rybchenko 	hdrp->tag = DHCP_ENCAPSULATED(opt);
4705e111ed8SAndrew Rybchenko 	hdrp->length = (uint8_t)value_length;
4715e111ed8SAndrew Rybchenko 	if ((value_length > 0) && (valuep != NULL))
4725e111ed8SAndrew Rybchenko 		memcpy(&hdrp[1], valuep, value_length);
4735e111ed8SAndrew Rybchenko }
4745e111ed8SAndrew Rybchenko 
4755e111ed8SAndrew Rybchenko /*
4765e111ed8SAndrew Rybchenko  * Add the given tag to the end of the buffer. Copes with creating an
4775e111ed8SAndrew Rybchenko  * encapsulated tag, and updates or creates the encapsulating opt as
4785e111ed8SAndrew Rybchenko  * necessary.
4795e111ed8SAndrew Rybchenko  */
4805e111ed8SAndrew Rybchenko 	__checkReturn	efx_rc_t
efx_dhcp_add_tag(__inout_bcount (buffer_length)uint8_t * bufferp,__in size_t buffer_length,__in uint16_t opt,__in_bcount_opt (value_length)uint8_t * valuep,__in size_t value_length)4815e111ed8SAndrew Rybchenko efx_dhcp_add_tag(
4825e111ed8SAndrew Rybchenko 	__inout_bcount(buffer_length)	uint8_t *bufferp,
4835e111ed8SAndrew Rybchenko 	__in				size_t buffer_length,
4845e111ed8SAndrew Rybchenko 	__in				uint16_t opt,
4855e111ed8SAndrew Rybchenko 	__in_bcount_opt(value_length)	uint8_t *valuep,
4865e111ed8SAndrew Rybchenko 	__in				size_t value_length)
4875e111ed8SAndrew Rybchenko {
4885e111ed8SAndrew Rybchenko 	efx_rc_t rc;
4895e111ed8SAndrew Rybchenko 	efx_dhcp_tag_hdr_t *encap_hdrp = NULL;
4905e111ed8SAndrew Rybchenko 	uint8_t *insert_pointp = NULL;
4915e111ed8SAndrew Rybchenko 	uint8_t *endp;
4925e111ed8SAndrew Rybchenko 	size_t available_space;
4935e111ed8SAndrew Rybchenko 	size_t added_length;
4945e111ed8SAndrew Rybchenko 	size_t search_size;
4955e111ed8SAndrew Rybchenko 	uint8_t *searchp;
4965e111ed8SAndrew Rybchenko 
4975e111ed8SAndrew Rybchenko 	if (!DHCP_OPT_HAS_VALUE(DHCP_ENCAPSULATED(opt))) {
4985e111ed8SAndrew Rybchenko 		rc = EINVAL;
4995e111ed8SAndrew Rybchenko 		goto fail1;
5005e111ed8SAndrew Rybchenko 	}
5015e111ed8SAndrew Rybchenko 
5025e111ed8SAndrew Rybchenko 	if (value_length > DHCP_MAX_VALUE) {
5035e111ed8SAndrew Rybchenko 		rc = EINVAL;
5045e111ed8SAndrew Rybchenko 		goto fail2;
5055e111ed8SAndrew Rybchenko 	}
5065e111ed8SAndrew Rybchenko 
5075e111ed8SAndrew Rybchenko 	if ((value_length > 0) && (valuep == NULL)) {
5085e111ed8SAndrew Rybchenko 		rc = EINVAL;
5095e111ed8SAndrew Rybchenko 		goto fail3;
5105e111ed8SAndrew Rybchenko 	}
5115e111ed8SAndrew Rybchenko 
5125e111ed8SAndrew Rybchenko 	endp = bufferp;
5135e111ed8SAndrew Rybchenko 	available_space = buffer_length;
5145e111ed8SAndrew Rybchenko 	rc = efx_dhcp_walk_tags(&endp, &available_space, EFX_DHCP_END);
5155e111ed8SAndrew Rybchenko 	if (rc != 0)
5165e111ed8SAndrew Rybchenko 		goto fail4;
5175e111ed8SAndrew Rybchenko 
5185e111ed8SAndrew Rybchenko 	searchp = bufferp;
5195e111ed8SAndrew Rybchenko 	search_size = buffer_length;
5205e111ed8SAndrew Rybchenko 	if (DHCP_IS_ENCAP_OPT(opt)) {
5215e111ed8SAndrew Rybchenko 		rc = efx_dhcp_walk_tags(&searchp, &search_size,
5225e111ed8SAndrew Rybchenko 		    DHCP_ENCAPSULATOR(opt));
5235e111ed8SAndrew Rybchenko 		if (rc == 0) {
5245e111ed8SAndrew Rybchenko 			encap_hdrp = (efx_dhcp_tag_hdr_t *)searchp;
5255e111ed8SAndrew Rybchenko 
5265e111ed8SAndrew Rybchenko 			/* Check encapsulated tag is not present */
5275e111ed8SAndrew Rybchenko 			search_size = encap_hdrp->length;
5285e111ed8SAndrew Rybchenko 			rc = efx_dhcp_walk_tags(&searchp, &search_size,
5295e111ed8SAndrew Rybchenko 			    opt);
5305e111ed8SAndrew Rybchenko 			if (rc != ENOENT) {
5315e111ed8SAndrew Rybchenko 				rc = EINVAL;
5325e111ed8SAndrew Rybchenko 				goto fail5;
5335e111ed8SAndrew Rybchenko 			}
5345e111ed8SAndrew Rybchenko 
5355e111ed8SAndrew Rybchenko 			/* Check encapsulator will not overflow */
5365e111ed8SAndrew Rybchenko 			if (((size_t)encap_hdrp->length +
5375e111ed8SAndrew Rybchenko 			    DHCP_CALC_TAG_LENGTH(value_length)) >
5385e111ed8SAndrew Rybchenko 			    DHCP_MAX_VALUE) {
5395e111ed8SAndrew Rybchenko 				rc = E2BIG;
5405e111ed8SAndrew Rybchenko 				goto fail6;
5415e111ed8SAndrew Rybchenko 			}
5425e111ed8SAndrew Rybchenko 
5435e111ed8SAndrew Rybchenko 			/* Insert at start of existing encapsulator */
5445e111ed8SAndrew Rybchenko 			insert_pointp = (uint8_t *)&encap_hdrp[1];
5455e111ed8SAndrew Rybchenko 			opt = DHCP_ENCAPSULATED(opt);
5465e111ed8SAndrew Rybchenko 		} else if (rc == ENOENT) {
5475e111ed8SAndrew Rybchenko 			encap_hdrp = NULL;
5485e111ed8SAndrew Rybchenko 		} else {
5495e111ed8SAndrew Rybchenko 			goto fail7;
5505e111ed8SAndrew Rybchenko 		}
5515e111ed8SAndrew Rybchenko 	} else {
5525e111ed8SAndrew Rybchenko 		/* Check unencapsulated tag is not present */
5535e111ed8SAndrew Rybchenko 		rc = efx_dhcp_walk_tags(&searchp, &search_size,
5545e111ed8SAndrew Rybchenko 		    opt);
5555e111ed8SAndrew Rybchenko 		if (rc != ENOENT) {
5565e111ed8SAndrew Rybchenko 			rc = EINVAL;
5575e111ed8SAndrew Rybchenko 			goto fail8;
5585e111ed8SAndrew Rybchenko 		}
5595e111ed8SAndrew Rybchenko 	}
5605e111ed8SAndrew Rybchenko 
5615e111ed8SAndrew Rybchenko 	if (insert_pointp == NULL) {
5625e111ed8SAndrew Rybchenko 		/* Insert at end of existing tags */
5635e111ed8SAndrew Rybchenko 		insert_pointp = endp;
5645e111ed8SAndrew Rybchenko 	}
5655e111ed8SAndrew Rybchenko 
5665e111ed8SAndrew Rybchenko 	/* Includes the new encapsulator tag hdr if required */
5675e111ed8SAndrew Rybchenko 	added_length = DHCP_CALC_TAG_LENGTH(value_length) +
5685e111ed8SAndrew Rybchenko 	    (DHCP_IS_ENCAP_OPT(opt) ? sizeof (efx_dhcp_tag_hdr_t) : 0);
5695e111ed8SAndrew Rybchenko 
5705e111ed8SAndrew Rybchenko 	if (available_space <= added_length) {
5715e111ed8SAndrew Rybchenko 		rc = ENOMEM;
5725e111ed8SAndrew Rybchenko 		goto fail9;
5735e111ed8SAndrew Rybchenko 	}
5745e111ed8SAndrew Rybchenko 
5755e111ed8SAndrew Rybchenko 	memmove(insert_pointp + added_length, insert_pointp,
5765e111ed8SAndrew Rybchenko 	    available_space - added_length);
5775e111ed8SAndrew Rybchenko 
5785e111ed8SAndrew Rybchenko 	if (DHCP_IS_ENCAP_OPT(opt)) {
5795e111ed8SAndrew Rybchenko 		/* Create new encapsulator header */
5805e111ed8SAndrew Rybchenko 		added_length -= sizeof (efx_dhcp_tag_hdr_t);
5815e111ed8SAndrew Rybchenko 		efx_dhcp_write_tag(insert_pointp,
5825e111ed8SAndrew Rybchenko 		    DHCP_ENCAPSULATOR(opt), NULL, added_length);
5835e111ed8SAndrew Rybchenko 		insert_pointp += sizeof (efx_dhcp_tag_hdr_t);
5845e111ed8SAndrew Rybchenko 	} else if (encap_hdrp)
5855e111ed8SAndrew Rybchenko 		/* Modify existing encapsulator header */
5865e111ed8SAndrew Rybchenko 		encap_hdrp->length +=
5875e111ed8SAndrew Rybchenko 		    ((uint8_t)DHCP_CALC_TAG_LENGTH(value_length));
5885e111ed8SAndrew Rybchenko 
5895e111ed8SAndrew Rybchenko 	efx_dhcp_write_tag(insert_pointp, opt, valuep, value_length);
5905e111ed8SAndrew Rybchenko 
5915e111ed8SAndrew Rybchenko 	return (0);
5925e111ed8SAndrew Rybchenko 
5935e111ed8SAndrew Rybchenko fail9:
5945e111ed8SAndrew Rybchenko 	EFSYS_PROBE(fail9);
5955e111ed8SAndrew Rybchenko fail8:
5965e111ed8SAndrew Rybchenko 	EFSYS_PROBE(fail8);
5975e111ed8SAndrew Rybchenko fail7:
5985e111ed8SAndrew Rybchenko 	EFSYS_PROBE(fail7);
5995e111ed8SAndrew Rybchenko fail6:
6005e111ed8SAndrew Rybchenko 	EFSYS_PROBE(fail6);
6015e111ed8SAndrew Rybchenko fail5:
6025e111ed8SAndrew Rybchenko 	EFSYS_PROBE(fail5);
6035e111ed8SAndrew Rybchenko fail4:
6045e111ed8SAndrew Rybchenko 	EFSYS_PROBE(fail4);
6055e111ed8SAndrew Rybchenko fail3:
6065e111ed8SAndrew Rybchenko 	EFSYS_PROBE(fail3);
6075e111ed8SAndrew Rybchenko fail2:
6085e111ed8SAndrew Rybchenko 	EFSYS_PROBE(fail2);
6095e111ed8SAndrew Rybchenko fail1:
6105e111ed8SAndrew Rybchenko 	EFSYS_PROBE1(fail1, efx_rc_t, rc);
6115e111ed8SAndrew Rybchenko 
6125e111ed8SAndrew Rybchenko 	return (rc);
6135e111ed8SAndrew Rybchenko }
6145e111ed8SAndrew Rybchenko 
6155e111ed8SAndrew Rybchenko /*
6165e111ed8SAndrew Rybchenko  * Update an existing tag to the new value. Copes with encapsulated
6175e111ed8SAndrew Rybchenko  * tags, and updates the encapsulating opt as necessary.
6185e111ed8SAndrew Rybchenko  */
6195e111ed8SAndrew Rybchenko 	__checkReturn	efx_rc_t
efx_dhcp_update_tag(__inout_bcount (buffer_length)uint8_t * bufferp,__in size_t buffer_length,__in uint16_t opt,__in uint8_t * value_locationp,__in_bcount_opt (value_length)uint8_t * valuep,__in size_t value_length)6205e111ed8SAndrew Rybchenko efx_dhcp_update_tag(
6215e111ed8SAndrew Rybchenko 	__inout_bcount(buffer_length)	uint8_t *bufferp,
6225e111ed8SAndrew Rybchenko 	__in				size_t buffer_length,
6235e111ed8SAndrew Rybchenko 	__in				uint16_t opt,
6245e111ed8SAndrew Rybchenko 	__in				uint8_t *value_locationp,
6255e111ed8SAndrew Rybchenko 	__in_bcount_opt(value_length)	uint8_t *valuep,
6265e111ed8SAndrew Rybchenko 	__in				size_t value_length)
6275e111ed8SAndrew Rybchenko {
6285e111ed8SAndrew Rybchenko 	efx_rc_t rc;
6295e111ed8SAndrew Rybchenko 	uint8_t *write_pointp = value_locationp - sizeof (efx_dhcp_tag_hdr_t);
6305e111ed8SAndrew Rybchenko 	efx_dhcp_tag_hdr_t *hdrp = (efx_dhcp_tag_hdr_t *)write_pointp;
6315e111ed8SAndrew Rybchenko 	efx_dhcp_tag_hdr_t *encap_hdrp = NULL;
6325e111ed8SAndrew Rybchenko 	size_t old_length;
6335e111ed8SAndrew Rybchenko 
6345e111ed8SAndrew Rybchenko 	if (!DHCP_OPT_HAS_VALUE(DHCP_ENCAPSULATED(opt))) {
6355e111ed8SAndrew Rybchenko 		rc = EINVAL;
6365e111ed8SAndrew Rybchenko 		goto fail1;
6375e111ed8SAndrew Rybchenko 	}
6385e111ed8SAndrew Rybchenko 
6395e111ed8SAndrew Rybchenko 	if (value_length > DHCP_MAX_VALUE) {
6405e111ed8SAndrew Rybchenko 		rc = EINVAL;
6415e111ed8SAndrew Rybchenko 		goto fail2;
6425e111ed8SAndrew Rybchenko 	}
6435e111ed8SAndrew Rybchenko 
6445e111ed8SAndrew Rybchenko 	if ((value_length > 0) && (valuep == NULL)) {
6455e111ed8SAndrew Rybchenko 		rc = EINVAL;
6465e111ed8SAndrew Rybchenko 		goto fail3;
6475e111ed8SAndrew Rybchenko 	}
6485e111ed8SAndrew Rybchenko 
6495e111ed8SAndrew Rybchenko 	old_length = hdrp->length;
6505e111ed8SAndrew Rybchenko 
6515e111ed8SAndrew Rybchenko 	if (old_length < value_length) {
6525e111ed8SAndrew Rybchenko 		uint8_t *endp = bufferp;
6535e111ed8SAndrew Rybchenko 		size_t available_space = buffer_length;
6545e111ed8SAndrew Rybchenko 
6555e111ed8SAndrew Rybchenko 		rc = efx_dhcp_walk_tags(&endp, &available_space,
6565e111ed8SAndrew Rybchenko 		    EFX_DHCP_END);
6575e111ed8SAndrew Rybchenko 		if (rc != 0)
6585e111ed8SAndrew Rybchenko 			goto fail4;
6595e111ed8SAndrew Rybchenko 
6605e111ed8SAndrew Rybchenko 		if (available_space < (value_length - old_length)) {
6615e111ed8SAndrew Rybchenko 			rc = EINVAL;
6625e111ed8SAndrew Rybchenko 			goto fail5;
6635e111ed8SAndrew Rybchenko 		}
6645e111ed8SAndrew Rybchenko 	}
6655e111ed8SAndrew Rybchenko 
6665e111ed8SAndrew Rybchenko 	if (DHCP_IS_ENCAP_OPT(opt)) {
6675e111ed8SAndrew Rybchenko 		uint8_t *encapp = bufferp;
6685e111ed8SAndrew Rybchenko 		size_t following_encap = buffer_length;
6695e111ed8SAndrew Rybchenko 		size_t new_length;
6705e111ed8SAndrew Rybchenko 
6715e111ed8SAndrew Rybchenko 		rc = efx_dhcp_walk_tags(&encapp, &following_encap,
6725e111ed8SAndrew Rybchenko 		    DHCP_ENCAPSULATOR(opt));
6735e111ed8SAndrew Rybchenko 		if (rc != 0)
6745e111ed8SAndrew Rybchenko 			goto fail6;
6755e111ed8SAndrew Rybchenko 
6765e111ed8SAndrew Rybchenko 		encap_hdrp = (efx_dhcp_tag_hdr_t *)encapp;
6775e111ed8SAndrew Rybchenko 
6785e111ed8SAndrew Rybchenko 		new_length = ((size_t)encap_hdrp->length +
6795e111ed8SAndrew Rybchenko 		    value_length - old_length);
6805e111ed8SAndrew Rybchenko 		/* Check encapsulator will not overflow */
6815e111ed8SAndrew Rybchenko 		if (new_length > DHCP_MAX_VALUE) {
6825e111ed8SAndrew Rybchenko 			rc = E2BIG;
6835e111ed8SAndrew Rybchenko 			goto fail7;
6845e111ed8SAndrew Rybchenko 		}
6855e111ed8SAndrew Rybchenko 
6865e111ed8SAndrew Rybchenko 		encap_hdrp->length = (uint8_t)new_length;
6875e111ed8SAndrew Rybchenko 	}
6885e111ed8SAndrew Rybchenko 
6895e111ed8SAndrew Rybchenko 	/*
6905e111ed8SAndrew Rybchenko 	 * Move the following data up/down to accomodate the new payload
6915e111ed8SAndrew Rybchenko 	 * length.
6925e111ed8SAndrew Rybchenko 	 */
6935e111ed8SAndrew Rybchenko 	if (old_length != value_length) {
6945e111ed8SAndrew Rybchenko 		uint8_t *destp = (uint8_t *)DHCP_NEXT_TAG(hdrp) +
6955e111ed8SAndrew Rybchenko 		    value_length - old_length;
6965e111ed8SAndrew Rybchenko 		size_t count = &bufferp[buffer_length] -
6975e111ed8SAndrew Rybchenko 		    (uint8_t *)DHCP_NEXT_TAG(hdrp);
6985e111ed8SAndrew Rybchenko 
6995e111ed8SAndrew Rybchenko 		memmove(destp, DHCP_NEXT_TAG(hdrp), count);
7005e111ed8SAndrew Rybchenko 	}
7015e111ed8SAndrew Rybchenko 
7025e111ed8SAndrew Rybchenko 	EFSYS_ASSERT(hdrp->tag == DHCP_ENCAPSULATED(opt));
7035e111ed8SAndrew Rybchenko 	efx_dhcp_write_tag(write_pointp, opt, valuep, value_length);
7045e111ed8SAndrew Rybchenko 
7055e111ed8SAndrew Rybchenko 	return (0);
7065e111ed8SAndrew Rybchenko 
7075e111ed8SAndrew Rybchenko fail7:
7085e111ed8SAndrew Rybchenko 	EFSYS_PROBE(fail7);
7095e111ed8SAndrew Rybchenko fail6:
7105e111ed8SAndrew Rybchenko 	EFSYS_PROBE(fail6);
7115e111ed8SAndrew Rybchenko fail5:
7125e111ed8SAndrew Rybchenko 	EFSYS_PROBE(fail5);
7135e111ed8SAndrew Rybchenko fail4:
7145e111ed8SAndrew Rybchenko 	EFSYS_PROBE(fail4);
7155e111ed8SAndrew Rybchenko fail3:
7165e111ed8SAndrew Rybchenko 	EFSYS_PROBE(fail3);
7175e111ed8SAndrew Rybchenko fail2:
7185e111ed8SAndrew Rybchenko 	EFSYS_PROBE(fail2);
7195e111ed8SAndrew Rybchenko fail1:
7205e111ed8SAndrew Rybchenko 	EFSYS_PROBE1(fail1, efx_rc_t, rc);
7215e111ed8SAndrew Rybchenko 
7225e111ed8SAndrew Rybchenko 	return (rc);
7235e111ed8SAndrew Rybchenko }
7245e111ed8SAndrew Rybchenko 
7255e111ed8SAndrew Rybchenko 
7265e111ed8SAndrew Rybchenko /*
7275e111ed8SAndrew Rybchenko  * Copy bootcfg sector data to a target buffer which may differ in size.
7285e111ed8SAndrew Rybchenko  * Optionally corrects format errors in source buffer.
7295e111ed8SAndrew Rybchenko  */
7305e111ed8SAndrew Rybchenko 				efx_rc_t
efx_bootcfg_copy_sector(__in efx_nic_t * enp,__inout_bcount (sector_length)uint8_t * sector,__in size_t sector_length,__out_bcount (data_size)uint8_t * data,__in size_t data_size,__in boolean_t handle_format_errors)7315e111ed8SAndrew Rybchenko efx_bootcfg_copy_sector(
7325e111ed8SAndrew Rybchenko 	__in			efx_nic_t *enp,
7335e111ed8SAndrew Rybchenko 	__inout_bcount(sector_length)
7345e111ed8SAndrew Rybchenko 				uint8_t *sector,
7355e111ed8SAndrew Rybchenko 	__in			size_t sector_length,
7365e111ed8SAndrew Rybchenko 	__out_bcount(data_size)	uint8_t *data,
7375e111ed8SAndrew Rybchenko 	__in			size_t data_size,
7385e111ed8SAndrew Rybchenko 	__in			boolean_t handle_format_errors)
7395e111ed8SAndrew Rybchenko {
7405e111ed8SAndrew Rybchenko 	_NOTE(ARGUNUSED(enp))
7415e111ed8SAndrew Rybchenko 
7425e111ed8SAndrew Rybchenko 	size_t used_bytes;
7435e111ed8SAndrew Rybchenko 	efx_rc_t rc;
7445e111ed8SAndrew Rybchenko 
7455e111ed8SAndrew Rybchenko 	/* Minimum buffer is checksum byte and EFX_DHCP_END terminator */
7465e111ed8SAndrew Rybchenko 	if (data_size < 2) {
7475e111ed8SAndrew Rybchenko 		rc = ENOSPC;
7485e111ed8SAndrew Rybchenko 		goto fail1;
7495e111ed8SAndrew Rybchenko 	}
7505e111ed8SAndrew Rybchenko 
7515e111ed8SAndrew Rybchenko 	/* Verify that the area is correctly formatted and checksummed */
7525e111ed8SAndrew Rybchenko 	rc = efx_dhcp_verify(sector, sector_length,
7535e111ed8SAndrew Rybchenko 				    &used_bytes);
7545e111ed8SAndrew Rybchenko 
7555e111ed8SAndrew Rybchenko 	if (!handle_format_errors) {
7565e111ed8SAndrew Rybchenko 		if (rc != 0)
7575e111ed8SAndrew Rybchenko 			goto fail2;
7585e111ed8SAndrew Rybchenko 
7595e111ed8SAndrew Rybchenko 		if ((used_bytes < 2) ||
7605e111ed8SAndrew Rybchenko 		    (sector[used_bytes - 1] != EFX_DHCP_END)) {
7615e111ed8SAndrew Rybchenko 			/* Block too short, or EFX_DHCP_END missing */
7625e111ed8SAndrew Rybchenko 			rc = ENOENT;
7635e111ed8SAndrew Rybchenko 			goto fail3;
7645e111ed8SAndrew Rybchenko 		}
7655e111ed8SAndrew Rybchenko 	}
7665e111ed8SAndrew Rybchenko 
7675e111ed8SAndrew Rybchenko 	/* Synthesize empty format on verification failure */
7685e111ed8SAndrew Rybchenko 	if (rc != 0 || used_bytes == 0) {
7695e111ed8SAndrew Rybchenko 		sector[0] = 0;
7705e111ed8SAndrew Rybchenko 		sector[1] = EFX_DHCP_END;
7715e111ed8SAndrew Rybchenko 		used_bytes = 2;
7725e111ed8SAndrew Rybchenko 	}
7735e111ed8SAndrew Rybchenko 	EFSYS_ASSERT(used_bytes >= 2);	/* checksum and EFX_DHCP_END */
7745e111ed8SAndrew Rybchenko 	EFSYS_ASSERT(used_bytes <= sector_length);
7755e111ed8SAndrew Rybchenko 	EFSYS_ASSERT(sector_length >= 2);
7765e111ed8SAndrew Rybchenko 
7775e111ed8SAndrew Rybchenko 	/*
7785e111ed8SAndrew Rybchenko 	 * Legacy bootcfg sectors don't terminate with an EFX_DHCP_END
7795e111ed8SAndrew Rybchenko 	 * character. Modify the returned payload so it does.
7805e111ed8SAndrew Rybchenko 	 * Reinitialise the sector if there isn't room for the character.
7815e111ed8SAndrew Rybchenko 	 */
7825e111ed8SAndrew Rybchenko 	if (sector[used_bytes - 1] != EFX_DHCP_END) {
7835e111ed8SAndrew Rybchenko 		if (used_bytes >= sector_length) {
7845e111ed8SAndrew Rybchenko 			sector[0] = 0;
7855e111ed8SAndrew Rybchenko 			used_bytes = 1;
7865e111ed8SAndrew Rybchenko 		}
7875e111ed8SAndrew Rybchenko 		sector[used_bytes] = EFX_DHCP_END;
7885e111ed8SAndrew Rybchenko 		++used_bytes;
7895e111ed8SAndrew Rybchenko 	}
7905e111ed8SAndrew Rybchenko 
7915e111ed8SAndrew Rybchenko 	/*
7925e111ed8SAndrew Rybchenko 	 * Verify that the target buffer is large enough for the
7935e111ed8SAndrew Rybchenko 	 * entire used bootcfg area, then copy into the target buffer.
7945e111ed8SAndrew Rybchenko 	 */
7955e111ed8SAndrew Rybchenko 	if (used_bytes > data_size) {
7965e111ed8SAndrew Rybchenko 		rc = ENOSPC;
7975e111ed8SAndrew Rybchenko 		goto fail4;
7985e111ed8SAndrew Rybchenko 	}
7995e111ed8SAndrew Rybchenko 
8005e111ed8SAndrew Rybchenko 	data[0] = 0; /* checksum, updated below */
8015e111ed8SAndrew Rybchenko 
8025e111ed8SAndrew Rybchenko 	/* Copy all after the checksum to the target buffer */
8035e111ed8SAndrew Rybchenko 	memcpy(data + 1, sector + 1, used_bytes - 1);
8045e111ed8SAndrew Rybchenko 
8055e111ed8SAndrew Rybchenko 	/* Zero out the unused portion of the target buffer */
8065e111ed8SAndrew Rybchenko 	if (used_bytes < data_size)
8075e111ed8SAndrew Rybchenko 		(void) memset(data + used_bytes, 0, data_size - used_bytes);
8085e111ed8SAndrew Rybchenko 
8095e111ed8SAndrew Rybchenko 	/*
8105e111ed8SAndrew Rybchenko 	 * The checksum includes trailing data after any EFX_DHCP_END
8115e111ed8SAndrew Rybchenko 	 * character, which we've just modified (by truncation or appending
8125e111ed8SAndrew Rybchenko 	 * EFX_DHCP_END).
8135e111ed8SAndrew Rybchenko 	 */
8145e111ed8SAndrew Rybchenko 	data[0] -= efx_dhcp_csum(data, data_size);
8155e111ed8SAndrew Rybchenko 
8165e111ed8SAndrew Rybchenko 	return (0);
8175e111ed8SAndrew Rybchenko 
8185e111ed8SAndrew Rybchenko fail4:
8195e111ed8SAndrew Rybchenko 	EFSYS_PROBE(fail4);
8205e111ed8SAndrew Rybchenko fail3:
8215e111ed8SAndrew Rybchenko 	EFSYS_PROBE(fail3);
8225e111ed8SAndrew Rybchenko fail2:
8235e111ed8SAndrew Rybchenko 	EFSYS_PROBE(fail2);
8245e111ed8SAndrew Rybchenko fail1:
8255e111ed8SAndrew Rybchenko 	EFSYS_PROBE1(fail1, efx_rc_t, rc);
8265e111ed8SAndrew Rybchenko 
8275e111ed8SAndrew Rybchenko 	return (rc);
8285e111ed8SAndrew Rybchenko }
8295e111ed8SAndrew Rybchenko 
8305e111ed8SAndrew Rybchenko 				efx_rc_t
efx_bootcfg_read(__in efx_nic_t * enp,__out_bcount (size)uint8_t * data,__in size_t size)8315e111ed8SAndrew Rybchenko efx_bootcfg_read(
8325e111ed8SAndrew Rybchenko 	__in			efx_nic_t *enp,
8335e111ed8SAndrew Rybchenko 	__out_bcount(size)	uint8_t *data,
8345e111ed8SAndrew Rybchenko 	__in			size_t size)
8355e111ed8SAndrew Rybchenko {
8365e111ed8SAndrew Rybchenko 	uint8_t *payload = NULL;
8375e111ed8SAndrew Rybchenko 	size_t used_bytes;
8385e111ed8SAndrew Rybchenko 	size_t partn_length;
8395e111ed8SAndrew Rybchenko 	size_t sector_length;
8405e111ed8SAndrew Rybchenko 	size_t sector_offset;
8415e111ed8SAndrew Rybchenko 	efx_rc_t rc;
8425e111ed8SAndrew Rybchenko 	uint32_t sector_number;
8435e111ed8SAndrew Rybchenko 
8445e111ed8SAndrew Rybchenko 	/* Minimum buffer is checksum byte and EFX_DHCP_END terminator */
8455e111ed8SAndrew Rybchenko 	if (size < 2) {
8465e111ed8SAndrew Rybchenko 		rc = ENOSPC;
8475e111ed8SAndrew Rybchenko 		goto fail1;
8485e111ed8SAndrew Rybchenko 	}
8495e111ed8SAndrew Rybchenko 
8505e111ed8SAndrew Rybchenko #if EFX_OPTS_EF10()
8515e111ed8SAndrew Rybchenko 	sector_number = enp->en_nic_cfg.enc_pf;
8525e111ed8SAndrew Rybchenko #else
8535e111ed8SAndrew Rybchenko 	sector_number = 0;
8545e111ed8SAndrew Rybchenko #endif
8555e111ed8SAndrew Rybchenko 	rc = efx_nvram_size(enp, EFX_NVRAM_BOOTROM_CFG, &partn_length);
8565e111ed8SAndrew Rybchenko 	if (rc != 0)
8575e111ed8SAndrew Rybchenko 		goto fail2;
8585e111ed8SAndrew Rybchenko 
8595e111ed8SAndrew Rybchenko 	/* The bootcfg sector may be stored in a (larger) shared partition */
8605e111ed8SAndrew Rybchenko 	rc = efx_bootcfg_sector_info(enp, sector_number,
8615e111ed8SAndrew Rybchenko 	    NULL, &sector_offset, &sector_length);
8625e111ed8SAndrew Rybchenko 	if (rc != 0)
8635e111ed8SAndrew Rybchenko 		goto fail3;
8645e111ed8SAndrew Rybchenko 
8655e111ed8SAndrew Rybchenko 	if (sector_length < 2) {
8665e111ed8SAndrew Rybchenko 		rc = EINVAL;
8675e111ed8SAndrew Rybchenko 		goto fail4;
8685e111ed8SAndrew Rybchenko 	}
8695e111ed8SAndrew Rybchenko 
8705e111ed8SAndrew Rybchenko 	if (sector_length > BOOTCFG_MAX_SIZE)
8715e111ed8SAndrew Rybchenko 		sector_length = BOOTCFG_MAX_SIZE;
8725e111ed8SAndrew Rybchenko 
8735e111ed8SAndrew Rybchenko 	if (sector_offset + sector_length > partn_length) {
8745e111ed8SAndrew Rybchenko 		/* Partition is too small */
8755e111ed8SAndrew Rybchenko 		rc = EFBIG;
8765e111ed8SAndrew Rybchenko 		goto fail5;
8775e111ed8SAndrew Rybchenko 	}
8785e111ed8SAndrew Rybchenko 
8795e111ed8SAndrew Rybchenko 	/*
8805e111ed8SAndrew Rybchenko 	 * We need to read the entire BOOTCFG sector to ensure we read all
8815e111ed8SAndrew Rybchenko 	 * tags, because legacy bootcfg sectors are not guaranteed to end
8825e111ed8SAndrew Rybchenko 	 * with an EFX_DHCP_END character. If the user hasn't supplied a
8835e111ed8SAndrew Rybchenko 	 * sufficiently large buffer then use our own buffer.
8845e111ed8SAndrew Rybchenko 	 */
8855e111ed8SAndrew Rybchenko 	if (sector_length > size) {
8865e111ed8SAndrew Rybchenko 		EFSYS_KMEM_ALLOC(enp->en_esip, sector_length, payload);
8875e111ed8SAndrew Rybchenko 		if (payload == NULL) {
8885e111ed8SAndrew Rybchenko 			rc = ENOMEM;
8895e111ed8SAndrew Rybchenko 			goto fail6;
8905e111ed8SAndrew Rybchenko 		}
8915e111ed8SAndrew Rybchenko 	} else
8925e111ed8SAndrew Rybchenko 		payload = (uint8_t *)data;
8935e111ed8SAndrew Rybchenko 
8945e111ed8SAndrew Rybchenko 	if ((rc = efx_nvram_rw_start(enp, EFX_NVRAM_BOOTROM_CFG, NULL)) != 0)
8955e111ed8SAndrew Rybchenko 		goto fail7;
8965e111ed8SAndrew Rybchenko 
8975e111ed8SAndrew Rybchenko 	if ((rc = efx_nvram_read_chunk(enp, EFX_NVRAM_BOOTROM_CFG,
8985e111ed8SAndrew Rybchenko 	    sector_offset, (caddr_t)payload, sector_length)) != 0) {
8995e111ed8SAndrew Rybchenko 		(void) efx_nvram_rw_finish(enp, EFX_NVRAM_BOOTROM_CFG, NULL);
9005e111ed8SAndrew Rybchenko 		goto fail8;
9015e111ed8SAndrew Rybchenko 	}
9025e111ed8SAndrew Rybchenko 
9035e111ed8SAndrew Rybchenko 	if ((rc = efx_nvram_rw_finish(enp, EFX_NVRAM_BOOTROM_CFG, NULL)) != 0)
9045e111ed8SAndrew Rybchenko 		goto fail9;
9055e111ed8SAndrew Rybchenko 
9065e111ed8SAndrew Rybchenko 	/* Verify that the area is correctly formatted and checksummed */
9075e111ed8SAndrew Rybchenko 	rc = efx_dhcp_verify(payload, sector_length,
9085e111ed8SAndrew Rybchenko 	    &used_bytes);
9095e111ed8SAndrew Rybchenko 	if (rc != 0 || used_bytes == 0) {
9105e111ed8SAndrew Rybchenko 		payload[0] = 0;
9115e111ed8SAndrew Rybchenko 		payload[1] = EFX_DHCP_END;
9125e111ed8SAndrew Rybchenko 		used_bytes = 2;
9135e111ed8SAndrew Rybchenko 	}
9145e111ed8SAndrew Rybchenko 
9155e111ed8SAndrew Rybchenko 	EFSYS_ASSERT(used_bytes >= 2);	/* checksum and EFX_DHCP_END */
9165e111ed8SAndrew Rybchenko 	EFSYS_ASSERT(used_bytes <= sector_length);
9175e111ed8SAndrew Rybchenko 
9185e111ed8SAndrew Rybchenko 	/*
9195e111ed8SAndrew Rybchenko 	 * Legacy bootcfg sectors don't terminate with an EFX_DHCP_END
9205e111ed8SAndrew Rybchenko 	 * character. Modify the returned payload so it does.
9215e111ed8SAndrew Rybchenko 	 * BOOTCFG_MAX_SIZE is by definition large enough for any valid
9225e111ed8SAndrew Rybchenko 	 * (per-port) bootcfg sector, so reinitialise the sector if there
9235e111ed8SAndrew Rybchenko 	 * isn't room for the character.
9245e111ed8SAndrew Rybchenko 	 */
9255e111ed8SAndrew Rybchenko 	if (payload[used_bytes - 1] != EFX_DHCP_END) {
9265e111ed8SAndrew Rybchenko 		if (used_bytes >= sector_length)
9275e111ed8SAndrew Rybchenko 			used_bytes = 1;
9285e111ed8SAndrew Rybchenko 
9295e111ed8SAndrew Rybchenko 		payload[used_bytes] = EFX_DHCP_END;
9305e111ed8SAndrew Rybchenko 		++used_bytes;
9315e111ed8SAndrew Rybchenko 	}
9325e111ed8SAndrew Rybchenko 
9335e111ed8SAndrew Rybchenko 	/*
9345e111ed8SAndrew Rybchenko 	 * Verify that the user supplied buffer is large enough for the
9355e111ed8SAndrew Rybchenko 	 * entire used bootcfg area, then copy into the user supplied buffer.
9365e111ed8SAndrew Rybchenko 	 */
9375e111ed8SAndrew Rybchenko 	if (used_bytes > size) {
9385e111ed8SAndrew Rybchenko 		rc = ENOSPC;
9395e111ed8SAndrew Rybchenko 		goto fail10;
9405e111ed8SAndrew Rybchenko 	}
9415e111ed8SAndrew Rybchenko 
9425e111ed8SAndrew Rybchenko 	data[0] = 0; /* checksum, updated below */
9435e111ed8SAndrew Rybchenko 
9445e111ed8SAndrew Rybchenko 	if (sector_length > size) {
9455e111ed8SAndrew Rybchenko 		/* Copy all after the checksum to the target buffer */
9465e111ed8SAndrew Rybchenko 		memcpy(data + 1, payload + 1, used_bytes - 1);
9475e111ed8SAndrew Rybchenko 		EFSYS_KMEM_FREE(enp->en_esip, sector_length, payload);
9485e111ed8SAndrew Rybchenko 	}
9495e111ed8SAndrew Rybchenko 
9505e111ed8SAndrew Rybchenko 	/* Zero out the unused portion of the user buffer */
9515e111ed8SAndrew Rybchenko 	if (used_bytes < size)
9525e111ed8SAndrew Rybchenko 		(void) memset(data + used_bytes, 0, size - used_bytes);
9535e111ed8SAndrew Rybchenko 
9545e111ed8SAndrew Rybchenko 	/*
9555e111ed8SAndrew Rybchenko 	 * The checksum includes trailing data after any EFX_DHCP_END character,
9565e111ed8SAndrew Rybchenko 	 * which we've just modified (by truncation or appending EFX_DHCP_END).
9575e111ed8SAndrew Rybchenko 	 */
9585e111ed8SAndrew Rybchenko 	data[0] -= efx_dhcp_csum(data, size);
9595e111ed8SAndrew Rybchenko 
9605e111ed8SAndrew Rybchenko 	return (0);
9615e111ed8SAndrew Rybchenko 
9625e111ed8SAndrew Rybchenko fail10:
9635e111ed8SAndrew Rybchenko 	EFSYS_PROBE(fail10);
9645e111ed8SAndrew Rybchenko fail9:
9655e111ed8SAndrew Rybchenko 	EFSYS_PROBE(fail9);
9665e111ed8SAndrew Rybchenko fail8:
9675e111ed8SAndrew Rybchenko 	EFSYS_PROBE(fail8);
9685e111ed8SAndrew Rybchenko fail7:
9695e111ed8SAndrew Rybchenko 	EFSYS_PROBE(fail7);
9705e111ed8SAndrew Rybchenko 	if (sector_length > size)
9715e111ed8SAndrew Rybchenko 		EFSYS_KMEM_FREE(enp->en_esip, sector_length, payload);
9725e111ed8SAndrew Rybchenko fail6:
9735e111ed8SAndrew Rybchenko 	EFSYS_PROBE(fail6);
9745e111ed8SAndrew Rybchenko fail5:
9755e111ed8SAndrew Rybchenko 	EFSYS_PROBE(fail5);
9765e111ed8SAndrew Rybchenko fail4:
9775e111ed8SAndrew Rybchenko 	EFSYS_PROBE(fail4);
9785e111ed8SAndrew Rybchenko fail3:
9795e111ed8SAndrew Rybchenko 	EFSYS_PROBE(fail3);
9805e111ed8SAndrew Rybchenko fail2:
9815e111ed8SAndrew Rybchenko 	EFSYS_PROBE(fail2);
9825e111ed8SAndrew Rybchenko fail1:
9835e111ed8SAndrew Rybchenko 	EFSYS_PROBE1(fail1, efx_rc_t, rc);
9845e111ed8SAndrew Rybchenko 
9855e111ed8SAndrew Rybchenko 	return (rc);
9865e111ed8SAndrew Rybchenko }
9875e111ed8SAndrew Rybchenko 
9885e111ed8SAndrew Rybchenko 				efx_rc_t
efx_bootcfg_write(__in efx_nic_t * enp,__in_bcount (size)uint8_t * data,__in size_t size)9895e111ed8SAndrew Rybchenko efx_bootcfg_write(
9905e111ed8SAndrew Rybchenko 	__in			efx_nic_t *enp,
9915e111ed8SAndrew Rybchenko 	__in_bcount(size)	uint8_t *data,
9925e111ed8SAndrew Rybchenko 	__in			size_t size)
9935e111ed8SAndrew Rybchenko {
9945e111ed8SAndrew Rybchenko 	uint8_t *partn_data;
9955e111ed8SAndrew Rybchenko 	uint8_t checksum;
9965e111ed8SAndrew Rybchenko 	size_t partn_length;
9975e111ed8SAndrew Rybchenko 	size_t sector_length;
9985e111ed8SAndrew Rybchenko 	size_t sector_offset;
9995e111ed8SAndrew Rybchenko 	size_t used_bytes;
10005e111ed8SAndrew Rybchenko 	efx_rc_t rc;
10015e111ed8SAndrew Rybchenko 	uint32_t sector_number;
10025e111ed8SAndrew Rybchenko 
10035e111ed8SAndrew Rybchenko #if EFX_OPTS_EF10()
10045e111ed8SAndrew Rybchenko 	sector_number = enp->en_nic_cfg.enc_pf;
10055e111ed8SAndrew Rybchenko #else
10065e111ed8SAndrew Rybchenko 	sector_number = 0;
10075e111ed8SAndrew Rybchenko #endif
10085e111ed8SAndrew Rybchenko 
10095e111ed8SAndrew Rybchenko 	rc = efx_nvram_size(enp, EFX_NVRAM_BOOTROM_CFG, &partn_length);
10105e111ed8SAndrew Rybchenko 	if (rc != 0)
10115e111ed8SAndrew Rybchenko 		goto fail1;
10125e111ed8SAndrew Rybchenko 
10135e111ed8SAndrew Rybchenko 	/* The bootcfg sector may be stored in a (larger) shared partition */
10145e111ed8SAndrew Rybchenko 	rc = efx_bootcfg_sector_info(enp, sector_number,
10155e111ed8SAndrew Rybchenko 	    NULL, &sector_offset, &sector_length);
10165e111ed8SAndrew Rybchenko 	if (rc != 0)
10175e111ed8SAndrew Rybchenko 		goto fail2;
10185e111ed8SAndrew Rybchenko 
10195e111ed8SAndrew Rybchenko 	if (sector_length > BOOTCFG_MAX_SIZE)
10205e111ed8SAndrew Rybchenko 		sector_length = BOOTCFG_MAX_SIZE;
10215e111ed8SAndrew Rybchenko 
10225e111ed8SAndrew Rybchenko 	if (sector_offset + sector_length > partn_length) {
10235e111ed8SAndrew Rybchenko 		/* Partition is too small */
10245e111ed8SAndrew Rybchenko 		rc = EFBIG;
10255e111ed8SAndrew Rybchenko 		goto fail3;
10265e111ed8SAndrew Rybchenko 	}
10275e111ed8SAndrew Rybchenko 
10285e111ed8SAndrew Rybchenko 	if ((rc = efx_dhcp_verify(data, size, &used_bytes)) != 0)
10295e111ed8SAndrew Rybchenko 		goto fail4;
10305e111ed8SAndrew Rybchenko 
10315e111ed8SAndrew Rybchenko 	/*
10325e111ed8SAndrew Rybchenko 	 * The caller *must* terminate their block with a EFX_DHCP_END
10335e111ed8SAndrew Rybchenko 	 * character
10345e111ed8SAndrew Rybchenko 	 */
10355e111ed8SAndrew Rybchenko 	if ((used_bytes < 2) || ((uint8_t)data[used_bytes - 1] !=
10365e111ed8SAndrew Rybchenko 	    EFX_DHCP_END)) {
10375e111ed8SAndrew Rybchenko 		/* Block too short or EFX_DHCP_END missing */
10385e111ed8SAndrew Rybchenko 		rc = ENOENT;
10395e111ed8SAndrew Rybchenko 		goto fail5;
10405e111ed8SAndrew Rybchenko 	}
10415e111ed8SAndrew Rybchenko 
10425e111ed8SAndrew Rybchenko 	/* Check that the hardware has support for this much data */
10435e111ed8SAndrew Rybchenko 	if (used_bytes > MIN(sector_length, BOOTCFG_MAX_SIZE)) {
10445e111ed8SAndrew Rybchenko 		rc = ENOSPC;
10455e111ed8SAndrew Rybchenko 		goto fail6;
10465e111ed8SAndrew Rybchenko 	}
10475e111ed8SAndrew Rybchenko 
10485e111ed8SAndrew Rybchenko 	/*
10495e111ed8SAndrew Rybchenko 	 * If the BOOTCFG sector is stored in a shared partition, then we must
10505e111ed8SAndrew Rybchenko 	 * read the whole partition and insert the updated bootcfg sector at the
10515e111ed8SAndrew Rybchenko 	 * correct offset.
10525e111ed8SAndrew Rybchenko 	 */
10535e111ed8SAndrew Rybchenko 	EFSYS_KMEM_ALLOC(enp->en_esip, partn_length, partn_data);
10545e111ed8SAndrew Rybchenko 	if (partn_data == NULL) {
10555e111ed8SAndrew Rybchenko 		rc = ENOMEM;
10565e111ed8SAndrew Rybchenko 		goto fail7;
10575e111ed8SAndrew Rybchenko 	}
10585e111ed8SAndrew Rybchenko 
10595e111ed8SAndrew Rybchenko 	rc = efx_nvram_rw_start(enp, EFX_NVRAM_BOOTROM_CFG, NULL);
10605e111ed8SAndrew Rybchenko 	if (rc != 0)
10615e111ed8SAndrew Rybchenko 		goto fail8;
10625e111ed8SAndrew Rybchenko 
10635e111ed8SAndrew Rybchenko 	/* Read the entire partition */
10645e111ed8SAndrew Rybchenko 	rc = efx_nvram_read_chunk(enp, EFX_NVRAM_BOOTROM_CFG, 0,
10655e111ed8SAndrew Rybchenko 				    (caddr_t)partn_data, partn_length);
10665e111ed8SAndrew Rybchenko 	if (rc != 0)
10675e111ed8SAndrew Rybchenko 		goto fail9;
10685e111ed8SAndrew Rybchenko 
10695e111ed8SAndrew Rybchenko 	/*
10705e111ed8SAndrew Rybchenko 	 * Insert the BOOTCFG sector into the partition, Zero out all data
10715e111ed8SAndrew Rybchenko 	 * after the EFX_DHCP_END tag, and adjust the checksum.
10725e111ed8SAndrew Rybchenko 	 */
10735e111ed8SAndrew Rybchenko 	(void) memset(partn_data + sector_offset, 0x0, sector_length);
10745e111ed8SAndrew Rybchenko 	(void) memcpy(partn_data + sector_offset, data, used_bytes);
10755e111ed8SAndrew Rybchenko 
10765e111ed8SAndrew Rybchenko 	checksum = efx_dhcp_csum(data, used_bytes);
10775e111ed8SAndrew Rybchenko 	partn_data[sector_offset] -= checksum;
10785e111ed8SAndrew Rybchenko 
10795e111ed8SAndrew Rybchenko 	if ((rc = efx_nvram_erase(enp, EFX_NVRAM_BOOTROM_CFG)) != 0)
10805e111ed8SAndrew Rybchenko 		goto fail10;
10815e111ed8SAndrew Rybchenko 
10825e111ed8SAndrew Rybchenko 	if ((rc = efx_nvram_write_chunk(enp, EFX_NVRAM_BOOTROM_CFG,
10835e111ed8SAndrew Rybchenko 		    0, (caddr_t)partn_data, partn_length)) != 0)
10845e111ed8SAndrew Rybchenko 		goto fail11;
10855e111ed8SAndrew Rybchenko 
10865e111ed8SAndrew Rybchenko 	if ((rc = efx_nvram_rw_finish(enp, EFX_NVRAM_BOOTROM_CFG, NULL)) != 0)
10875e111ed8SAndrew Rybchenko 		goto fail12;
10885e111ed8SAndrew Rybchenko 
10895e111ed8SAndrew Rybchenko 	EFSYS_KMEM_FREE(enp->en_esip, partn_length, partn_data);
10905e111ed8SAndrew Rybchenko 
10915e111ed8SAndrew Rybchenko 	return (0);
10925e111ed8SAndrew Rybchenko 
10935e111ed8SAndrew Rybchenko fail12:
10945e111ed8SAndrew Rybchenko 	EFSYS_PROBE(fail12);
10955e111ed8SAndrew Rybchenko fail11:
10965e111ed8SAndrew Rybchenko 	EFSYS_PROBE(fail11);
10975e111ed8SAndrew Rybchenko fail10:
10985e111ed8SAndrew Rybchenko 	EFSYS_PROBE(fail10);
10995e111ed8SAndrew Rybchenko fail9:
11005e111ed8SAndrew Rybchenko 	EFSYS_PROBE(fail9);
11015e111ed8SAndrew Rybchenko 
11025e111ed8SAndrew Rybchenko 	(void) efx_nvram_rw_finish(enp, EFX_NVRAM_BOOTROM_CFG, NULL);
11035e111ed8SAndrew Rybchenko fail8:
11045e111ed8SAndrew Rybchenko 	EFSYS_PROBE(fail8);
11055e111ed8SAndrew Rybchenko 
11065e111ed8SAndrew Rybchenko 	EFSYS_KMEM_FREE(enp->en_esip, partn_length, partn_data);
11075e111ed8SAndrew Rybchenko fail7:
11085e111ed8SAndrew Rybchenko 	EFSYS_PROBE(fail7);
11095e111ed8SAndrew Rybchenko fail6:
11105e111ed8SAndrew Rybchenko 	EFSYS_PROBE(fail6);
11115e111ed8SAndrew Rybchenko fail5:
11125e111ed8SAndrew Rybchenko 	EFSYS_PROBE(fail5);
11135e111ed8SAndrew Rybchenko fail4:
11145e111ed8SAndrew Rybchenko 	EFSYS_PROBE(fail4);
11155e111ed8SAndrew Rybchenko fail3:
11165e111ed8SAndrew Rybchenko 	EFSYS_PROBE(fail3);
11175e111ed8SAndrew Rybchenko fail2:
11185e111ed8SAndrew Rybchenko 	EFSYS_PROBE(fail2);
11195e111ed8SAndrew Rybchenko fail1:
11205e111ed8SAndrew Rybchenko 	EFSYS_PROBE1(fail1, efx_rc_t, rc);
11215e111ed8SAndrew Rybchenko 
11225e111ed8SAndrew Rybchenko 	return (rc);
11235e111ed8SAndrew Rybchenko }
11245e111ed8SAndrew Rybchenko 
11255e111ed8SAndrew Rybchenko #endif	/* EFSYS_OPT_BOOTCFG */
1126