/* * Copyright (c) 2010 Broadcom Corporation * * Permission to use, copy, modify, and/or distribute this software for any * purpose with or without fee is hereby granted, provided that the above * copyright notice and this permission notice appear in all copies. * * THE SOFTWARE IS PROVIDED "AS IS" AND THE AUTHOR DISCLAIMS ALL WARRANTIES * WITH REGARD TO THIS SOFTWARE INCLUDING ALL IMPLIED WARRANTIES OF * MERCHANTABILITY AND FITNESS. IN NO EVENT SHALL THE AUTHOR BE LIABLE FOR ANY * SPECIAL, DIRECT, INDIRECT, OR CONSEQUENTIAL DAMAGES OR ANY DAMAGES * WHATSOEVER RESULTING FROM LOSS OF USE, DATA OR PROFITS, WHETHER IN AN ACTION * OF CONTRACT, NEGLIGENCE OR OTHER TORTIOUS ACTION, ARISING OUT OF OR IN * CONNECTION WITH THE USE OR PERFORMANCE OF THIS SOFTWARE. */ #include #include #include #include #include #include #include #include #include #include #include #include #include static wlc_pub_t *wlc_pub_malloc(osl_t *osh, uint unit, uint *err, uint devid); static void wlc_pub_mfree(osl_t *osh, wlc_pub_t *pub); static void wlc_tunables_init(wlc_tunables_t *tunables, uint devid); void *wlc_calloc(osl_t *osh, uint unit, uint size) { void *item; item = MALLOC(osh, size); if (item == NULL) WL_ERROR(("wl%d: %s: out of memory, malloced %d bytes\n", unit, __func__, MALLOCED(osh))); else bzero((char *)item, size); return item; } void BCMATTACHFN(wlc_tunables_init) (wlc_tunables_t *tunables, uint devid) { tunables->ntxd = NTXD; tunables->nrxd = NRXD; tunables->rxbufsz = RXBUFSZ; tunables->nrxbufpost = NRXBUFPOST; tunables->maxscb = MAXSCB; tunables->ampdunummpdu = AMPDU_NUM_MPDU; tunables->maxpktcb = MAXPKTCB; tunables->maxucodebss = WLC_MAX_UCODE_BSS; tunables->maxucodebss4 = WLC_MAX_UCODE_BSS4; tunables->maxbss = MAXBSS; tunables->datahiwat = WLC_DATAHIWAT; tunables->ampdudatahiwat = WLC_AMPDUDATAHIWAT; tunables->rxbnd = RXBND; tunables->txsbnd = TXSBND; #if defined(WLC_HIGH_ONLY) && defined(NTXD_USB_4319) if (devid == BCM4319_CHIP_ID) { tunables->ntxd = NTXD_USB_4319; } #endif /* WLC_HIGH_ONLY */ } static wlc_pub_t *BCMATTACHFN(wlc_pub_malloc) (osl_t *osh, uint unit, uint *err, uint devid) { wlc_pub_t *pub; pub = (wlc_pub_t *) wlc_calloc(osh, unit, sizeof(wlc_pub_t)); if (pub == NULL) { *err = 1001; goto fail; } pub->tunables = (wlc_tunables_t *)wlc_calloc(osh, unit, sizeof(wlc_tunables_t)); if (pub->tunables == NULL) { *err = 1028; goto fail; } /* need to init the tunables now */ wlc_tunables_init(pub->tunables, devid); pub->multicast = (struct ether_addr *)wlc_calloc(osh, unit, (sizeof(struct ether_addr) * MAXMULTILIST)); if (pub->multicast == NULL) { *err = 1003; goto fail; } return pub; fail: wlc_pub_mfree(osh, pub); return NULL; } static void BCMATTACHFN(wlc_pub_mfree) (osl_t *osh, wlc_pub_t *pub) { if (pub == NULL) return; if (pub->multicast) MFREE(osh, pub->multicast, (sizeof(struct ether_addr) * MAXMULTILIST)); if (pub->tunables) { MFREE(osh, pub->tunables, sizeof(wlc_tunables_t)); pub->tunables = NULL; } MFREE(osh, pub, sizeof(wlc_pub_t)); } wlc_bsscfg_t *wlc_bsscfg_malloc(osl_t *osh, uint unit) { wlc_bsscfg_t *cfg; cfg = (wlc_bsscfg_t *) wlc_calloc(osh, unit, sizeof(wlc_bsscfg_t)); if (cfg == NULL) goto fail; cfg->current_bss = (wlc_bss_info_t *)wlc_calloc(osh, unit, sizeof(wlc_bss_info_t)); if (cfg->current_bss == NULL) goto fail; return cfg; fail: wlc_bsscfg_mfree(osh, cfg); return NULL; } void wlc_bsscfg_mfree(osl_t *osh, wlc_bsscfg_t *cfg) { if (cfg == NULL) return; if (cfg->maclist) { MFREE(osh, cfg->maclist, (int)(OFFSETOF(struct maclist, ea) + cfg->nmac * ETHER_ADDR_LEN)); cfg->maclist = NULL; } if (cfg->current_bss != NULL) { wlc_bss_info_t *current_bss = cfg->current_bss; if (current_bss->bcn_prb != NULL) MFREE(osh, current_bss->bcn_prb, current_bss->bcn_prb_len); MFREE(osh, current_bss, sizeof(wlc_bss_info_t)); cfg->current_bss = NULL; } MFREE(osh, cfg, sizeof(wlc_bsscfg_t)); } void wlc_bsscfg_ID_assign(wlc_info_t *wlc, wlc_bsscfg_t *bsscfg) { bsscfg->ID = wlc->next_bsscfg_ID; wlc->next_bsscfg_ID++; } /* * The common driver entry routine. Error codes should be unique */ wlc_info_t *BCMATTACHFN(wlc_attach_malloc) (osl_t *osh, uint unit, uint *err, uint devid) { wlc_info_t *wlc; wlc = (wlc_info_t *) wlc_calloc(osh, unit, sizeof(wlc_info_t)); if (wlc == NULL) { *err = 1002; goto fail; } wlc->hwrxoff = WL_HWRXOFF; /* allocate wlc_pub_t state structure */ wlc->pub = wlc_pub_malloc(osh, unit, err, devid); if (wlc->pub == NULL) { *err = 1003; goto fail; } wlc->pub->wlc = wlc; /* allocate wlc_hw_info_t state structure */ wlc->hw = (wlc_hw_info_t *)wlc_calloc(osh, unit, sizeof(wlc_hw_info_t)); if (wlc->hw == NULL) { *err = 1005; goto fail; } wlc->hw->wlc = wlc; #ifdef WLC_LOW wlc->hw->bandstate[0] = (wlc_hwband_t *)wlc_calloc(osh, unit, (sizeof(wlc_hwband_t) * MAXBANDS)); if (wlc->hw->bandstate[0] == NULL) { *err = 1006; goto fail; } else { int i; for (i = 1; i < MAXBANDS; i++) { wlc->hw->bandstate[i] = (wlc_hwband_t *) ((uintptr) wlc->hw->bandstate[0] + (sizeof(wlc_hwband_t) * i)); } } #endif /* WLC_LOW */ wlc->modulecb = (modulecb_t *)wlc_calloc(osh, unit, sizeof(modulecb_t) * WLC_MAXMODULES); if (wlc->modulecb == NULL) { *err = 1009; goto fail; } wlc->default_bss = (wlc_bss_info_t *)wlc_calloc(osh, unit, sizeof(wlc_bss_info_t)); if (wlc->default_bss == NULL) { *err = 1010; goto fail; } wlc->cfg = wlc_bsscfg_malloc(osh, unit); if (wlc->cfg == NULL) { *err = 1011; goto fail; } wlc_bsscfg_ID_assign(wlc, wlc->cfg); wlc->pkt_callback = (pkt_cb_t *)wlc_calloc(osh, unit, (sizeof(pkt_cb_t) * (wlc->pub->tunables->maxpktcb + 1))); if (wlc->pkt_callback == NULL) { *err = 1013; goto fail; } wlc->wsec_def_keys[0] = (wsec_key_t *)wlc_calloc(osh, unit, (sizeof(wsec_key_t) * WLC_DEFAULT_KEYS)); if (wlc->wsec_def_keys[0] == NULL) { *err = 1015; goto fail; } else { int i; for (i = 1; i < WLC_DEFAULT_KEYS; i++) { wlc->wsec_def_keys[i] = (wsec_key_t *) ((uintptr) wlc->wsec_def_keys[0] + (sizeof(wsec_key_t) * i)); } } wlc->protection = (wlc_protection_t *)wlc_calloc(osh, unit, sizeof(wlc_protection_t)); if (wlc->protection == NULL) { *err = 1016; goto fail; } wlc->stf = (wlc_stf_t *)wlc_calloc(osh, unit, sizeof(wlc_stf_t)); if (wlc->stf == NULL) { *err = 1017; goto fail; } wlc->bandstate[0] = (wlcband_t *)wlc_calloc(osh, unit, (sizeof(wlcband_t) * MAXBANDS)); if (wlc->bandstate[0] == NULL) { *err = 1025; goto fail; } else { int i; for (i = 1; i < MAXBANDS; i++) { wlc->bandstate[i] = (wlcband_t *) ((uintptr) wlc->bandstate[0] + (sizeof(wlcband_t) * i)); } } wlc->corestate = (wlccore_t *)wlc_calloc(osh, unit, sizeof(wlccore_t)); if (wlc->corestate == NULL) { *err = 1026; goto fail; } wlc->corestate->macstat_snapshot = (macstat_t *)wlc_calloc(osh, unit, sizeof(macstat_t)); if (wlc->corestate->macstat_snapshot == NULL) { *err = 1027; goto fail; } return wlc; fail: wlc_detach_mfree(wlc, osh); return NULL; } void BCMATTACHFN(wlc_detach_mfree) (wlc_info_t *wlc, osl_t *osh) { if (wlc == NULL) return; if (wlc->modulecb) { MFREE(osh, wlc->modulecb, sizeof(modulecb_t) * WLC_MAXMODULES); wlc->modulecb = NULL; } if (wlc->default_bss) { MFREE(osh, wlc->default_bss, sizeof(wlc_bss_info_t)); wlc->default_bss = NULL; } if (wlc->cfg) { wlc_bsscfg_mfree(osh, wlc->cfg); wlc->cfg = NULL; } if (wlc->pkt_callback && wlc->pub && wlc->pub->tunables) { MFREE(osh, wlc->pkt_callback, sizeof(pkt_cb_t) * (wlc->pub->tunables->maxpktcb + 1)); wlc->pkt_callback = NULL; } if (wlc->wsec_def_keys[0]) MFREE(osh, wlc->wsec_def_keys[0], (sizeof(wsec_key_t) * WLC_DEFAULT_KEYS)); if (wlc->protection) { MFREE(osh, wlc->protection, sizeof(wlc_protection_t)); wlc->protection = NULL; } if (wlc->stf) { MFREE(osh, wlc->stf, sizeof(wlc_stf_t)); wlc->stf = NULL; } if (wlc->bandstate[0]) MFREE(osh, wlc->bandstate[0], (sizeof(wlcband_t) * MAXBANDS)); if (wlc->corestate) { if (wlc->corestate->macstat_snapshot) { MFREE(osh, wlc->corestate->macstat_snapshot, sizeof(macstat_t)); wlc->corestate->macstat_snapshot = NULL; } MFREE(osh, wlc->corestate, sizeof(wlccore_t)); wlc->corestate = NULL; } if (wlc->pub) { /* free pub struct */ wlc_pub_mfree(osh, wlc->pub); wlc->pub = NULL; } if (wlc->hw) { #ifdef WLC_LOW if (wlc->hw->bandstate[0]) { MFREE(osh, wlc->hw->bandstate[0], (sizeof(wlc_hwband_t) * MAXBANDS)); wlc->hw->bandstate[0] = NULL; } #endif /* free hw struct */ MFREE(osh, wlc->hw, sizeof(wlc_hw_info_t)); wlc->hw = NULL; } /* free the wlc */ MFREE(osh, wlc, sizeof(wlc_info_t)); wlc = NULL; }