/**************************************************************************** * drivers/wireless/ieee80211/bcm43xxx/bcmf_core.c * * Licensed to the Apache Software Foundation (ASF) under one or more * contributor license agreements. See the NOTICE file distributed with * this work for additional information regarding copyright ownership. The * ASF licenses this file to you under the Apache License, Version 2.0 (the * "License"); you may not use this file except in compliance with the * License. You may obtain a copy of the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, WITHOUT * WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the * License for the specific language governing permissions and limitations * under the License. * ****************************************************************************/ /**************************************************************************** * Included Files ****************************************************************************/ #include #include #include #include #include #include #include #include #include #include #include #include #include #include "bcmf_core.h" #include "bcmf_sdio.h" #include "bcmf_sdio_regs.h" /**************************************************************************** * Pre-processor Definitions ****************************************************************************/ /* Agent registers (common for every core) */ #define BCMA_IOCTL 0x0408 /* IO control */ #define BCMA_IOST 0x0500 /* IO status */ #define BCMA_RESET_CTL 0x0800 /* Reset control */ #define BCMA_RESET_ST 0x0804 #define BCMA_IOCTL_CLK 0x0001 #define BCMA_IOCTL_FGC 0x0002 #define BCMA_IOCTL_CORE_BITS 0x3FFC #define BCMA_IOCTL_PME_EN 0x4000 #define BCMA_IOCTL_BIST_EN 0x8000 /* ARM CR4 core specific control flag bits */ #define ARMCR4_BCMA_IOCTL_CPUHALT 0x0020 #define BCMA_IOST_CORE_BITS 0x0FFF #define BCMA_IOST_DMA64 0x1000 #define BCMA_IOST_GATED_CLK 0x2000 #define BCMA_IOST_BIST_ERROR 0x4000 #define BCMA_IOST_BIST_DONE 0x8000 #define BCMA_RESET_CTL_RESET 0x0001 /* SOCSRAM core registers */ #define SOCSRAM_BANKX_INDEX ((uint32_t) (0x18004000 + 0x10) ) #define SOCSRAM_BANKX_PDA ((uint32_t) (0x18004000 + 0x44) ) /* Transfer size properties */ #define BCMF_UPLOAD_TRANSFER_SIZE (64 * 256) /* Define this to validate uploaded materials */ /* #define DBG_VALIDATE_UPLOAD */ /**************************************************************************** * Private Types ****************************************************************************/ #ifdef DBG_VALIDATE_UPLOAD static uint8_t compare_buffer[BCMF_UPLOAD_TRANSFER_SIZE]; #endif /**************************************************************************** * Private Function Prototypes ****************************************************************************/ static int bcmf_core_set_backplane_window(FAR struct bcmf_sdio_dev_s *sbus, uint32_t address); static int bcmf_upload_binary(FAR struct bcmf_sdio_dev_s *sbusv, uint32_t address, uint8_t *buf, unsigned int len); static int bcmf_upload_nvram(FAR struct bcmf_sdio_dev_s *sbus); /**************************************************************************** * Private Functions ****************************************************************************/ int bcmf_core_set_backplane_window(FAR struct bcmf_sdio_dev_s *sbus, uint32_t address) { int ret; int i; address &= ~SBSDIO_SB_OFT_ADDR_MASK; for (i = 1; i < 4; i++) { uint8_t addr_part = (address >> (8*i)) & 0xff; uint8_t cur_addr_part = (sbus->backplane_current_addr >> (8*i)) & 0xff; if (addr_part != cur_addr_part) { /* Update current backplane base address */ ret = bcmf_write_reg(sbus, 1, SBSDIO_FUNC1_SBADDRLOW + i - 1, addr_part); if (ret != OK) { return ret; } sbus->backplane_current_addr &= ~(0xff << (8*i)); sbus->backplane_current_addr |= addr_part << (8*i); } } return OK; } int bcmf_upload_binary(FAR struct bcmf_sdio_dev_s *sbus, uint32_t address, uint8_t *buf, unsigned int len) { unsigned int size; #ifdef DBG_VALIDATE_UPLOAD uint32_t validate_address = address; uint8_t *validate_buffer = buf; unsigned int validate_len = len; #endif while (len > 0) { /* Set the backplane window to include the start address */ int ret = bcmf_core_set_backplane_window(sbus, address); if (ret != OK) { wlerr("Backplane setting failed at %08" PRIx32 "\n", address); return ret; } if (len > BCMF_UPLOAD_TRANSFER_SIZE) { size = BCMF_UPLOAD_TRANSFER_SIZE; } else { size = len; } /* Transfer firmware data */ ret = bcmf_transfer_bytes(sbus, true, 1, address & SBSDIO_SB_OFT_ADDR_MASK, buf, size); if (ret != OK) { wlerr("transfer failed %d %" PRIx32 " %d\n", ret, address, size); return ret; } len -= size; address += size; buf += size; } #ifdef DBG_VALIDATE_UPLOAD wlwarn("Validating....\n"); while (validate_len > 0) { /* Set the backplane window to include the start address */ int ret = bcmf_core_set_backplane_window(sbus, validate_address); if (ret != OK) { wlerr("Backplane setting failed at %08x\n", validate_address); return ret; } if (validate_len > BCMF_UPLOAD_TRANSFER_SIZE) { size = BCMF_UPLOAD_TRANSFER_SIZE; } else { size = validate_len; } /* Transfer firmware data */ ret = bcmf_transfer_bytes(sbus, false, 1, validate_address & SBSDIO_SB_OFT_ADDR_MASK, compare_buffer, size); if (ret != OK) { wlerr("validate transfer failed %d %x %d\n", ret, validate_address, size); return ret; } if (memcmp(validate_buffer, compare_buffer, size)) { wlerr("Match failed at address base %08x\n", validate_address); return -EILSEQ; } validate_len -= size; validate_address += size; validate_buffer += size; } wlwarn("Validation passed\n"); #endif return OK; } #ifdef CONFIG_IEEE80211_BROADCOM_FWFILES int bcmf_upload_file(FAR struct bcmf_sdio_dev_s *sbus, uint32_t address, FAR const char *path) { struct file finfo; FAR uint8_t *buf; size_t total_read; ssize_t nread; int ret; /* Open the file in the detached state */ ret = file_open(&finfo, path, O_RDONLY); if (ret < 0) { wlerr("ERROR: Failed to open the FILE MTD file %s: %d\n", path, ret); return ret; } /* Allocate an I/O buffer */ buf = (FAR uint8_t *)kmm_malloc(BCMF_UPLOAD_TRANSFER_SIZE); if (buf == NULL) { wlerr("ERROR: Failed allocate an I/O buffer\n"); ret = -ENOMEM; goto errout_with_file; } /* Loop until the firmware has been loaded */ do { /* Set the backplane window to include the start address */ nread = file_read(&finfo, buf, BCMF_UPLOAD_TRANSFER_SIZE); if (nread < 0) { ret = (int)nread; wlerr("ERROR: Failed to read file: %d\n", ret); goto errout_with_buf; } if (nread == 0) { break; } wlinfo("Read %ld bytes\n", (long)nread); ret = bcmf_core_set_backplane_window(sbus, address); if (ret < 0) { wlerr("ERROR: bcmf_core_set_backplane_window() failed: %d\n", ret); goto errout_with_buf; } total_read = nread; /* Transfer firmware data */ ret = bcmf_transfer_bytes(sbus, true, 1, address & SBSDIO_SB_OFT_ADDR_MASK, buf, total_read); if (ret < 0) { wlerr("ERROR: Transfer failed address=%lx total_read=%lu: %d\n", (unsigned long)address, (unsigned long)total_read, ret); goto errout_with_buf; } address += total_read; } while (nread == BCMF_UPLOAD_TRANSFER_SIZE); file_close(&finfo); kmm_free(buf); wlinfo("Upload complete\n"); return OK; errout_with_buf: kmm_free(buf); errout_with_file: file_close(&finfo); return ret; } #endif int bcmf_upload_nvram(FAR struct bcmf_sdio_dev_s *sbus) { FAR uint8_t *nvram_buf = sbus->chip->nvram_image; uint32_t nvram_sz = *sbus->chip->nvram_image_size; uint32_t token; int ret; #ifdef CONFIG_IEEE80211_BROADCOM_FWFILES FAR const char *nvfile = CONFIG_IEEE80211_BROADCOM_NVFILENAME; bool skipline = false; struct file finfo; struct stat stat; FAR uint8_t *buf; int i; if (strlen(nvfile) <= 0) { goto out; } ret = file_open(&finfo, nvfile, O_RDONLY); if (ret < 0) { goto out; } ret = file_fstat(&finfo, &stat); if (ret < 0 || stat.st_size <= 0) { goto out; } /* Round up the ram buffer size */ stat.st_size = (stat.st_size + 63) & (~63); buf = (FAR uint8_t *)kmm_malloc(stat.st_size); if (buf == NULL) { goto out; } /* Convert text pattern: * 1. Remove the comment line (Prefix with '#') * 2. Convert LF('\n') to NULL'\0' */ ret = file_read(&finfo, buf, stat.st_size); if (ret <= 0) { kmm_free(buf); goto out; } nvram_buf = buf; for (i = 0; i < ret; i++) { if (nvram_buf[i] == '\n') { if (buf != nvram_buf && *(buf - 1) != '\0') { *buf++ = '\0'; } skipline = false; } else if (nvram_buf[i] == '#') { skipline = true; } else if (!skipline && (nvram_buf + i) != buf) { *buf++ = nvram_buf[i]; } } nvram_sz = buf - nvram_buf; out: file_close(&finfo); #endif /* Round up the size of the image */ nvram_sz = (nvram_sz + 63) & (~63); wlinfo("nvram size is %" PRId32 " bytes\n", nvram_sz); /* Write image */ ret = bcmf_upload_binary(sbus, sbus->chip->ram_size - 4 - nvram_sz + sbus->chip->ram_base, nvram_buf, nvram_sz); #ifdef CONFIG_IEEE80211_BROADCOM_FWFILES if (nvram_buf != sbus->chip->nvram_image) { kmm_free(nvram_buf); } #endif if (ret != OK) { return ret; } /* Generate length token */ token = nvram_sz / 4; token = (~token << 16) | (token & 0x0000ffff); /* Write the length token to the last word */ return bcmf_write_sbreg(sbus, sbus->chip->ram_size - 4 + sbus->chip->ram_base, (FAR uint8_t *)&token, 4); } /**************************************************************************** * Public Functions ****************************************************************************/ /**************************************************************************** * Name: bcmf_read_sbreg ****************************************************************************/ int bcmf_read_sbreg(FAR struct bcmf_sdio_dev_s *sbus, uint32_t address, FAR uint8_t *reg, unsigned int len) { int ret = bcmf_core_set_backplane_window(sbus, address); if (ret != OK) { return ret; } address &= SBSDIO_SB_OFT_ADDR_MASK; /* Map to 32-bit access if len == 4 */ if (len == 4) { address |= SBSDIO_SB_ACCESS_2_4B_FLAG; } return bcmf_transfer_bytes(sbus, false, 1, address, reg, len); } /**************************************************************************** * Name: bcmf_write_sbreg ****************************************************************************/ int bcmf_write_sbreg(FAR struct bcmf_sdio_dev_s *sbus, uint32_t address, FAR uint8_t *reg, unsigned int len) { int ret = bcmf_core_set_backplane_window(sbus, address); if (ret != OK) { return ret; } address &= SBSDIO_SB_OFT_ADDR_MASK; /* Map to 32-bit access if len == 4 */ if (len == 4) { address |= SBSDIO_SB_ACCESS_2_4B_FLAG; } return bcmf_transfer_bytes(sbus, true, 1, address, reg, len); } /**************************************************************************** * Name: bcmf_core_upload_firmware ****************************************************************************/ int bcmf_core_upload_firmware(FAR struct bcmf_sdio_dev_s *sbus) { int ret; #if defined(CONFIG_IEEE80211_BROADCOM_BCM43455) uint32_t base; uint32_t value; #endif wlinfo("upload firmware\n"); switch (sbus->cur_chip_id) { #if defined(CONFIG_IEEE80211_BROADCOM_BCM4301X) || \ defined(CONFIG_IEEE80211_BROADCOM_BCM43362) || \ defined(CONFIG_IEEE80211_BROADCOM_BCM43438) case SDIO_DEVICE_ID_BROADCOM_43012: case SDIO_DEVICE_ID_BROADCOM_43013: case SDIO_DEVICE_ID_BROADCOM_43362: case SDIO_DEVICE_ID_BROADCOM_43430: /* Disable ARMCM3 core and reset SOCRAM core to set device in * firmware upload mode */ bcmf_core_disable(sbus, WLAN_ARMCM3_CORE_ID, 0, 0); bcmf_core_reset(sbus, SOCSRAM_CORE_ID, 0, 0, 0); #ifdef CONFIG_IEEE80211_BROADCOM_BCM43438 if (sbus->cur_chip_id == SDIO_DEVICE_ID_BROADCOM_43430) { /* Disable remap for SRAM_3. Only for 4343x */ bcmf_write_sbregw(sbus, SOCSRAM_BANKX_INDEX, 0x3); bcmf_write_sbregw(sbus, SOCSRAM_BANKX_PDA, 0); } #endif break; #endif #if defined(CONFIG_IEEE80211_BROADCOM_BCM43455) case SDIO_DEVICE_ID_BROADCOM_43455: /* Clear all IOCTL bits except HALT bit */ base = sbus->chip->core_base[WLAN_ARMCR4_CORE_ID]; bcmf_read_sbregw(sbus, base + BCMA_IOCTL, &value); value &= ARMCR4_BCMA_IOCTL_CPUHALT; bcmf_core_reset(sbus, WLAN_ARMCR4_CORE_ID, value, ARMCR4_BCMA_IOCTL_CPUHALT, ARMCR4_BCMA_IOCTL_CPUHALT); break; #endif default: DEBUGPANIC(); } nxsig_usleep(50 * 1000); /* Flash chip firmware */ #ifdef CONFIG_IEEE80211_BROADCOM_FWFILES ret = bcmf_upload_file(sbus, sbus->chip->ram_base, CONFIG_IEEE80211_BROADCOM_FWFILENAME); #else wlinfo("firmware size is %d bytes\n", *sbus->chip->firmware_image_size); ret = bcmf_upload_binary(sbus, sbus->chip->ram_base, sbus->chip->firmware_image, *sbus->chip->firmware_image_size); #endif if (ret < 0) { wlerr("ERROR: Failed to upload firmware\n"); return ret; } /* Flash NVRAM configuration file */ wlinfo("upload nvram configuration\n"); ret = bcmf_upload_nvram(sbus); if (ret < 0) { wlerr("ERROR: Failed to upload NVRAM\n"); return ret; } /* Firmware upload done, restart ARM CM3/CR4 core */ switch (sbus->cur_chip_id) { #if defined(CONFIG_IEEE80211_BROADCOM_BCM4301X) || \ defined(CONFIG_IEEE80211_BROADCOM_BCM43362) || \ defined(CONFIG_IEEE80211_BROADCOM_BCM43438) case SDIO_DEVICE_ID_BROADCOM_43012: case SDIO_DEVICE_ID_BROADCOM_43013: case SDIO_DEVICE_ID_BROADCOM_43362: case SDIO_DEVICE_ID_BROADCOM_43430: nxsig_usleep(10 * 1000); bcmf_core_reset(sbus, WLAN_ARMCM3_CORE_ID, 0, 0, 0); /* Check ARMCM3 core is running */ nxsig_usleep(10 * 1000); if (!bcmf_core_isup(sbus, WLAN_ARMCM3_CORE_ID)) { wlerr("Cannot start ARMCM3 core\n"); return -ETIMEDOUT; } break; #endif #if defined(CONFIG_IEEE80211_BROADCOM_BCM43455) case SDIO_DEVICE_ID_BROADCOM_43455: /* Clear all interrupts */ bcmf_write_sbregw( sbus, CORE_BUS_REG(sbus->chip->core_base[SDIOD_CORE_ID], intstatus), 0xffffffff); /* Write reset vector to address 0 */ ret = bcmf_upload_binary(sbus, 0, sbus->chip->firmware_image, 4); if (ret < 0) { return ret; } bcmf_core_reset(sbus, WLAN_ARMCR4_CORE_ID, ARMCR4_BCMA_IOCTL_CPUHALT, 0, 0); /* Check ARMCR4 core is running */ nxsig_usleep(10 * 1000); if (!bcmf_core_isup(sbus, WLAN_ARMCR4_CORE_ID)) { wlerr("Cannot start ARMCR4 core\n"); return -ETIMEDOUT; } break; #endif default: DEBUGPANIC(); } return OK; } bool bcmf_core_isup(FAR struct bcmf_sdio_dev_s *sbus, unsigned int core) { uint32_t value = 0; uint32_t base; if (core >= MAX_CORE_ID) { wlerr("Invalid core id %d\n", core); return false; } base = sbus->chip->core_base[core]; bcmf_read_sbregw(sbus, base + BCMA_IOCTL, &value); if ((value & (BCMA_IOCTL_FGC | BCMA_IOCTL_CLK)) != BCMA_IOCTL_CLK) { return false; } bcmf_read_sbregw(sbus, base + BCMA_RESET_CTL, &value); return (value & BCMA_RESET_CTL_RESET) == 0; } void bcmf_core_disable(FAR struct bcmf_sdio_dev_s *sbus, unsigned int core, uint32_t prereset, uint32_t reset) { uint32_t value; if (core >= MAX_CORE_ID) { wlerr("Invalid core id %d\n", core); return; } uint32_t base = sbus->chip->core_base[core]; /* Check if core is already in reset state. * If core is already in reset state, skip reset. */ bcmf_read_sbregw(sbus, base + BCMA_RESET_CTL, &value); if ((value & BCMA_RESET_CTL_RESET) == 0) { /* Core is not in reset state */ /* Ensure no backplane operation is pending */ nxsig_usleep(10 * 1000); bcmf_write_sbregw(sbus, base + BCMA_IOCTL, prereset | BCMA_IOCTL_FGC | BCMA_IOCTL_CLK); bcmf_read_sbregw(sbus, base + BCMA_IOCTL, &value); /* Set core in reset state */ bcmf_write_sbregw(sbus, base + BCMA_RESET_CTL, BCMA_RESET_CTL_RESET); up_udelay(1); } bcmf_write_sbregw(sbus, base + BCMA_IOCTL, reset | BCMA_IOCTL_FGC | BCMA_IOCTL_CLK); bcmf_read_sbregw(sbus, base + BCMA_IOCTL, &value); up_udelay(10); } void bcmf_core_reset(FAR struct bcmf_sdio_dev_s *sbus, unsigned int core, uint32_t prereset, uint32_t reset, uint32_t postreset) { uint32_t value; uint32_t base; if (core >= MAX_CORE_ID) { wlerr("Invalid core id %d\n", core); return; } base = sbus->chip->core_base[core]; /* Put core in reset state */ bcmf_core_disable(sbus, core, prereset, reset); /* Run initialization sequence */ bcmf_write_sbregw(sbus, base + BCMA_RESET_CTL, 0); bcmf_read_sbregw(sbus, base + BCMA_RESET_CTL, &value); up_udelay(1); bcmf_write_sbregw(sbus, base + BCMA_IOCTL, postreset | BCMA_IOCTL_CLK); bcmf_read_sbregw(sbus, base + BCMA_IOCTL, &value); up_udelay(1); }