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, §or_offset, §or_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, §or_offset, §or_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