/** @file uapcmd.c * * @brief This file contains the handling of command. * * Copyright (C) 2008-2009, Marvell International Ltd. * * This software file (the "File") is distributed by Marvell International * Ltd. under the terms of the GNU General Public License Version 2, June 1991 * (the "License"). You may use, redistribute and/or modify this File in * accordance with the terms and conditions of the License, a copy of which * is available along with the File in the gpl.txt file or by writing to * the Free Software Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA * 02111-1307 or on the worldwide web at http://www.gnu.org/licenses/gpl.txt. * * THE FILE IS DISTRIBUTED AS-IS, WITHOUT WARRANTY OF ANY KIND, AND THE * IMPLIED WARRANTIES OF MERCHANTABILITY OR FITNESS FOR A PARTICULAR PURPOSE * ARE EXPRESSLY DISCLAIMED. The License provides additional details about * this warranty disclaimer. * */ /**************************************************************************** Change log: 03/01/08: Initial creation ****************************************************************************/ /**************************************************************************** Header files ****************************************************************************/ #include #include #include #include #include #include #include #include #include #include #include #include #include #include "uaputl.h" #include "uapcmd.h" extern struct option cmd_options[]; /**************************************************************************** Local functions ****************************************************************************/ /** * @brief Show usage information for the sys_cfg_ap_mac_address * command * * $return N/A */ void print_sys_cfg_ap_mac_address_usage(void) { printf("\nUsage : sys_cfg_ap_mac_address [AP_MAC_ADDRESS]\n"); printf ("\nIf AP_MAC_ADDRESS is provided, a 'set' is performed, else a 'get' is performed.\n"); return; } /** * @brief Show usage information for the sys_cfg_ssid command * * $return N/A */ void print_sys_cfg_ssid_usage(void) { printf("\nUsage : sys_cfg_ssid [SSID]\n"); printf ("\nIf SSID is provided, a 'set' is performed, else a 'get' is performed.\n"); return; } /** * @brief Show usage information for the sys_cfg_beacon_period * command * * $return N/A */ void print_sys_cfg_beacon_period_usage(void) { printf("\nUsage : sys_cfg_beacon_period [BEACON_PERIOD]\n"); printf ("\nIf BEACON_PERIOD is provided, a 'set' is performed, else a 'get' is performed.\n"); return; } /** * @brief Show usage information for the sys_cfg_dtim_period * command * * $return N/A */ void print_sys_cfg_dtim_period_usage(void) { printf("\nUsage : sys_cfg_dtim_period [DTIM_PERIOD]\n"); printf ("\nIf DTIM_PERIOD is provided, a 'set' is performed, else a 'get' is performed.\n"); return; } /** * @brief Show usage information for the sys_cfg_channel * command * * $return N/A */ void print_sys_cfg_channel_usage(void) { printf("\nUsage : sys_cfg_channel [CHANNEL] [MODE]\n"); printf ("\nIf CHANNEL is provided, a 'set' is performed, else a 'get' is performed."); printf("\nIf MODE is provided, 0 for manual channel selection,"); printf("\nelse ACS (automatic channel selection) is performed\n"); return; } /** * @brief Show usage information for the sys_cfg_scan_channels * command * * $return N/A */ void print_sys_cfg_scan_channels_usage(void) { printf("\nUsage : sys_cfg_scan_channels [CHANNELS]\n"); printf ("\nIf CHANNELS are provided, a 'set' is performed, else a 'get' is performed.\n"); return; } /** * @brief Show usage information for the sys_cfg_rates_ext command * * $return N/A */ void print_sys_cfg_rates_ext_usage(void) { printf ("\nUsage : sys_cfg_rates_ext [rates RATES] [mbrate RATE] [urate RATE]\n"); printf ("\nIf 'Rate' provided, a 'set' is performed else a 'get' is performed"); printf ("\nRATES is provided as a set of data rates, in unit of 500 kilobits"); printf("\nA rate with MSB bit is basic rate, i.e 0x82 is basic rate.\n"); printf("\nFollowing is the list of supported rates in units of 500 Kbps:"); printf("\nDecimal: (2, 4, 11, 22, 12, 18, 24, 36, 48, 72, 96, 108)"); printf ("\nHex: (0x02, 0x04, 0x0b, 0x16, 0x0C, 0x12, 0x18, 0x24, 0x30, 0x48, 0x60, 0x6c)"); printf ("\nBasic rates: (0x82, 0x84, 0x8b, 0x96, 0x8C, 0x92, 0x98, 0xA4, 0xB0, 0xC8, 0xE0, 0xEc)\n"); printf ("\nRates 2, 4, 11 and 22 (in units of 500 Kbps) must be present in either of basic or"); printf ("\nnon-basic rates. If OFDM rates are enabled then 12, 24 and 48 (in units of 500 Kbps)"); printf("\nmust be present in either basic or non-basic rates"); printf("\nEach rate must be separated by a space."); printf("\nrates followed by RATES for setting operational rates."); printf ("\nmbrate followed by RATE for setting multicast and broadcast rate."); printf("\nurate followed by RATE for setting unicast rate.\n"); return; } /** * @brief Show usage information for the sys_cfg_rates command * * $return N/A */ void print_sys_cfg_rates_usage(void) { printf("\nUsage : sys_cfg_rates [RATES]\n"); printf ("\n[RATES] is set of data rates in unit of 500 kbps and each rate can be"); printf ("\nentered in hexadecimal or decimal format. Rates must be separated by"); printf("\nspace. Duplicate Rate fields are not allowed"); printf("\nA rate with MSB bit is basic rate, i.e 0x82 is basic rate."); printf("\nFollowing is the list of supported rates in units of 500 Kbps:"); printf("\nDecimal: (2, 4, 11, 22, 12, 18, 24, 36, 48, 72, 96, 108)"); printf ("\nHex: (0x02, 0x04, 0x0b, 0x16, 0x0C, 0x12, 0x18, 0x24, 0x30, 0x48, 0x60, 0x6c)"); printf ("\nBasic rates: (0x82, 0x84, 0x8b, 0x96, 0x8C, 0x92, 0x98, 0xA4, 0xB0, 0xC8, 0xE0, 0xEc)\n"); return; } /** * @brief Show usage information for the sys_cfg_tx_power * command * * $return N/A */ void print_sys_cfg_tx_power_usage(void) { printf("\nUsage : sys_cfg_tx_power [TX_POWER]\n"); printf ("\nIf TX_POWER is provided, a 'set' is performed, else a 'get' is performed."); printf("\nTX_POWER is represented in dBm.\n"); return; } /** * @brief Show usage information for the sys_cfg_bcast_ssid_ctl * command * * $return N/A */ void print_sys_cfg_bcast_ssid_ctl_usage(void) { printf("\nUsage : sys_cfg_bcast_ssid_ctl [0|1]\n"); printf("\nOptions: 0 - Disable SSID broadcast"); printf("\n 1 - Enable SSID broadcast"); printf("\n empty - Get current SSID broadcast setting\n"); return; } /** * @brief Show usage information for the sys_cfg_rsn_replay_prot * command * * $return N/A */ void print_sys_cfg_rsn_replay_prot_usage(void) { printf("\nUsage : sys_cfg_rsn_replay_prot [0|1]\n"); printf("\nOptions: 0 - Disable RSN replay protection"); printf("\n 1 - Enable RSN replay protection"); printf("\n empty - Get current RSN replay protection setting\n"); return; } /** * @brief Show usage information for the sys_cfg_preamble_ctl * command * * $return N/A */ void print_sys_cfg_preamble_ctl_usage(void) { printf("\nUsage : sys_cfg_preamble_ctl\n"); return; } /** * @brief Show usage information for the sys_cfg_antenna_ctl * command * * $return N/A */ void print_sys_cfg_antenna_ctl_usage(void) { printf("\nUsage : sys_cfg_antenna_ctl [MODE]\n"); printf("\nOptions: ANTENNA : 0 - Rx antenna"); printf("\n 1 - Tx antenna"); printf("\n MODE : 0 - Antenna A"); printf("\n 1 - Antenna B"); printf("\n empty - Get current antenna settings\n"); return; } /** * @brief Show usage information for the sys_cfg_rts_threshold * command * * $return N/A */ void print_sys_cfg_rts_threshold_usage(void) { printf("\nUsage : sys_cfg_rts_threshold [RTS_THRESHOLD]\n"); printf ("\nIf RTS_THRESHOLD is provided, a 'set' is performed, else a 'get' is performed.\n"); return; } /** * @brief Show usage information for the sys_cfg_frag_threshold * command * * $return N/A */ void print_sys_cfg_frag_threshold_usage(void) { printf("\nUsage : sys_cfg_frag_threshold [FRAG_THRESHOLD]\n"); printf ("\nIf FRAG_THRESHOLD is provided, a 'set' is performed, else a 'get' is performed."); printf("\nFragment threshold should between 256 and 2346.\n"); return; } /** * @brief Show usage information for the sys_cfg_radio_ctl * command * * $return N/A */ void print_sys_cfg_radio_ctl_usage(void) { printf("\nUsage : sys_cfg_radio_ctl [0|1]\n"); printf("\nOptions: 0 - Turn radio on"); printf("\n 1 - Turn radio off"); printf("\n empty - Get current radio setting\n"); return; } /** * @brief Show usage information for the sys_cfg_tx_data_rate * command * * $return N/A */ void print_sys_cfg_tx_data_rates_usage(void) { printf("\nUsage : sys_cfg_tx_data_rate [TX_DATA_RATE]\n"); printf("\nOptions: 0 - Auto rate"); printf("\n >0 - Set specified data rate"); printf("\n empty - Get current data rate"); printf("\nFollowing is the list of supported rates in units of 500 Kbps"); printf("\nDecimal: (2, 4, 11, 22, 12, 18, 24, 36, 48, 72, 96, 108)"); printf ("\nHex: (0x02, 0x04, 0x0b, 0x16, 0x0C, 0x12, 0x18, 0x24, 0x30, 0x48, 0x60, 0x6c)"); printf("\nOnly zero or rates currently configured are allowed.\n"); return; } /** * @brief Show usage information for the sys_cfg_mcbc_data_rate * command * * $return N/A */ void print_sys_cfg_mcbc_data_rates_usage(void) { printf("\nUsage : sys_cfg_mcbc_data_rate [MCBC_DATA_RATE]\n"); printf("\nOptions: 0 - Auto rate"); printf("\n >0 - Set specified MCBC data rate"); printf("\n empty - Get current MCBC data rate"); printf("\nFollowing is the list of supported rates in units of 500 Kbps"); printf("\nDecimal: (2, 4, 11, 22, 12, 18, 24, 36, 48, 72, 96, 108)"); printf ("\nHex: (0x02, 0x04, 0x0b, 0x16, 0x0C, 0x12, 0x18, 0x24, 0x30, 0x48, 0x60, 0x6c)"); printf ("\nOnly zero or one of the basic rates currently configured are allowed.\n"); return; } /** * @brief Show usage information for the sys_cfg_auth command * * $return N/A */ void print_sys_cfg_auth_usage(void) { printf("\nUsage : sys_cfg_auth [AUTHMODE]\n"); printf("\nOptions: AUTHMODE : 0 - Open authentication"); printf("\n 1 - Shared key authentication"); printf("\n empty - Get current authenticaton mode\n"); return; } /** * @brief Show usage information for the sys_cfg_pkt_fwd_ctl command * * $return N/A */ void print_sys_cfg_pkt_fwd_ctl_usage(void) { printf("\nUsage : sys_cfg_pkt_fwd_ctl [0|1]\n"); printf("\nOptions: 0 - Forward all packets to the host"); printf("\n 1 - Firmware handles intra-BSS packets"); printf("\n empty - Get current packet forwarding setting\n"); return; } /** * @brief Show usage information for the sys_cfg_sta_ageout_timer * command * * $return N/A */ void print_sys_cfg_sta_ageout_timer_usage(void) { printf("\nUsage : sys_cfg_sta_ageout_timer [STA_AGEOUT_TIMER]\n"); printf ("\nIf STA_AGEOUT_TIMER is provided, a 'set' is performed, else a 'get' is performed."); printf("\nSTA_AGEOUT_TIMER is represented in units of 100 ms."); printf("\nValue of 0 will mean that stations will never be aged out."); printf("\nThe value should between 300 and 864000.\n"); return; } /** * @brief Show usage information for the sys_cfg_protocol command * * $return N/A */ void print_sys_cfg_protocol_usage(void) { printf("\nUsage : sys_cfg_protocol [PROTOCOL]\n"); printf("\nOptions: PROTOCOL: 1 - No RSN"); printf("\n 2 - WEP Static"); printf("\n 8 - WPA"); printf("\n 32 - WPA2"); printf("\n 40 - WPA2 Mixed"); printf("\n empty - Get current protocol\n"); return; } /** * @brief Show usage information for the sys_cfg_wep_key * command * * $return N/A */ void print_sys_cfg_wep_key_usage(void) { printf("\nUsage : sys_cfg_wep_key "); printf ("[INDEX_0 IS_DEFAULT KEY_0] [INDEX_1 IS_DEFAULT KEY_1] [INDEX_2 IS_DEFAULT KEY_2] [INDEX_3 IS_DEFAULT KEY_3]\n"); printf("[Index_0] [Index_1] [Index_2] [Index_3]\n"); printf("\nOptions: INDEX_* : 0 - KeyIndex is 0"); printf("\n 1 - KeyIndex is 1"); printf("\n 2 - KeyIndex is 2"); printf("\n 3 - KeyIndex is 3"); printf("\n IS_DEFAULT : 0 - KeyIndex is not the default"); printf ("\n 1 - KeyIndex is the default transmit key"); printf("\n KEY_* : Key value"); printf("\n Index_*: 0 - Get key 0 setting"); printf("\n 1 - Get key 1 setting"); printf("\n 2 - Get key 2 setting"); printf("\n 3 - Get key 3 setting"); printf("\n empty - Get current WEP key settings\n"); return; } /** * @brief Show usage information for the sys_cfg_custom_ie * command * * $return N/A */ void print_sys_cfg_custom_ie_usage(void) { printf("\nUsage : sys_cfg_custom_ie [INDEX] [MASK] [IEBuffer]"); printf("\n empty - Get all IE settings\n"); printf("\n INDEX: 0 - Get/Set IE index 0 setting"); printf("\n 1 - Get/Set IE index 1 setting"); printf("\n 2 - Get/Set IE index 2 setting"); printf("\n 3 - Get/Set IE index 3 setting"); printf ("\n MASK : Management subtype mask value as per bit defintions"); printf("\n : Bit 0 - Association request."); printf("\n : Bit 1 - Association response."); printf("\n : Bit 2 - Reassociation request."); printf("\n : Bit 3 - Reassociation response."); printf("\n : Bit 4 - Probe request."); printf("\n : Bit 5 - Probe response."); printf("\n : Bit 8 - Beacon."); printf("\n MASK : MASK = 0 to clear the mask and the IE buffer"); printf("\n IEBuffer : IE Buffer in hex (max 256 bytes)\n\n"); return; } /* @brief Show usage information for the sys_cfg_cipher * command * * $return N/A */ void print_sys_cfg_cipher_usage(void) { printf("\nUsage : sys_cfg_cipher [PAIRWISE_CIPHER GROUP_CIPHER]\n"); printf("\nOptions: PAIRWISE_CIPHER: 0 - NONE"); printf("\n 4 - TKIP"); printf("\n 8 - AES CCMP"); printf("\n 12 - AES CCMP + TKIP"); printf("\n GROUP_CIPHER : 0 - NONE"); printf("\n 4 - TKIP"); printf("\n 8 - AES CCMP"); printf("\n empty - Get current cipher settings\n"); return; } /** * @brief Show usage information for the sys_cfg_group_rekey_timer command * * $return N/A */ void print_sys_cfg_group_rekey_timer_usage(void) { printf("\nUsage : sys_cfg_group_rekey_timer [GROUP_REKEY_TIMER]\n"); printf("\nOptions: GROUP_REKEY_TIME is represented in seconds"); printf("\n empty - Get current group re-key time\n"); return; } /** * @brief Show usage information for the sys_cfg_wpa_passphrase * command * * $return N/A */ void print_sys_cfg_wpa_passphrase_usage(void) { printf("\nUsage : sys_cfg_wpa_passphrase [PASSPHRASE]\n"); printf ("\nIf PASSPHRASE is provided, a 'set' is performed, else a 'get' is performed.\n"); return; } /** * @brief Show usage information for the sta_filter_table command * * $return N/A */ void print_sta_filter_table_usage(void) { printf("\nUsage : sta_filter_table \n"); printf("\nOptions: FILTERMODE : 0 - Disable filter table"); printf ("\n 1 - allow MAC addresses specified in the allowed list"); printf ("\n 2 - block MAC addresses specified in the banned list"); printf ("\n MACADDRESS_LIST is the list of MAC addresses to be acted upon. Each"); printf ("\n MAC address must be separated with a space. Maximum of"); printf("\n 16 MAC addresses are supported."); printf("\n empty - Get current mac filter settings\n"); return; } /** * @brief Show usage information for the sys_cfg_max_sta_num command * * $return N/A */ void print_sys_cfg_max_sta_num_usage(void) { printf("\nUsage : sys_cfg_max_sta_num [STA_NUM]\n"); printf ("\nIf STA_NUM is provided, a 'set' is performed, else a 'get' is performed."); printf("\nSTA_NUM should not bigger than 8.\n"); return; } /** * @brief Show usage information for the sys_cfg_retry_limit command * * $return N/A */ void print_sys_cfg_retry_limit_usage(void) { printf("\nUsage : sys_cfg_retry_limit [RETRY_LIMIT]\n"); printf ("\nIf RETRY_LIMIT is provided, a 'set' is performed, else a 'get' is performed."); printf("\nRETRY_LIMIT should not bigger than 14.\n"); return; } /** * @brief Show usage information for the sys_cfg_retry_limit command * * $return N/A */ void print_cfg_data_usage(void) { printf("\nUsage : cfg_data [*.conf]\n"); printf("\n type : 2 -- cal data"); printf("\n *.conf : file contain configuration data"); printf("\n empty - get current configuration data\n"); return; } /** * @brief get configured operational rates. * * @param rates Operational rates allowed are * stored at this pointer * @return number of basic rates allowed. * -1 if a failure */ int get_sys_cfg_rates(u8 * rates) { APCMDBUF_SYS_CONFIGURE *cmd_buf = NULL; TLVBUF_RATES *tlv = NULL; u8 *buffer = NULL; u16 cmd_len; int ret = UAP_FAILURE; int i = 0; int rate_cnt = 0; /* Initialize the command length */ cmd_len = sizeof(APCMDBUF_SYS_CONFIGURE) + sizeof(TLVBUF_RATES) + MAX_DATA_RATES; /* Initialize the command buffer */ buffer = (u8 *) malloc(cmd_len); if (!buffer) { printf("ERR:Cannot allocate buffer for command!\n"); return -1; } bzero((char *) buffer, cmd_len); /* Locate headers */ cmd_buf = (APCMDBUF_SYS_CONFIGURE *) buffer; tlv = (TLVBUF_RATES *) (buffer + sizeof(APCMDBUF_SYS_CONFIGURE)); /* Fill the command buffer */ cmd_buf->CmdCode = APCMD_SYS_CONFIGURE; cmd_buf->Size = cmd_len; cmd_buf->SeqNum = 0; cmd_buf->Result = 0; tlv->Tag = MRVL_RATES_TLV_ID; cmd_buf->Action = ACTION_GET; tlv->Length = MAX_DATA_RATES; endian_convert_tlv_header_out(tlv); /* Send the command */ ret = uap_ioctl((u8 *) cmd_buf, &cmd_len, cmd_len); endian_convert_tlv_header_in(tlv); /* Process response */ if (ret == UAP_SUCCESS) { /* Verify response */ if ((cmd_buf->CmdCode != (APCMD_SYS_CONFIGURE | APCMD_RESP_CHECK)) || (tlv->Tag != MRVL_RATES_TLV_ID)) { printf("ERR:Corrupted response! CmdCode=%x, Tlv->Tag=%x\n", cmd_buf->CmdCode, tlv->Tag); free(buffer); return -1; } /* copy response */ if (cmd_buf->Result == CMD_SUCCESS) { for (i = 0; i < tlv->Length; i++) { if (tlv->OperationalRates[i] != 0) { rates[rate_cnt++] = tlv->OperationalRates[i]; } } } else { printf("ERR:Could not get operational rates!\n"); } } else { printf("ERR:Command sending failed!\n"); } if (buffer) free(buffer); return rate_cnt; } /** * @brief check rate is valid or not. * * @param rate rate for check * * @return UAP_SUCCESS or UAP_FAILURE */ int is_tx_rate_valid(u8 rate) { int rate_cnt = 0; int i; u8 rates[MAX_DATA_RATES]; rate_cnt = get_sys_cfg_rates((u8 *) & rates); if (rate_cnt > 0) { for (i = 0; i < rate_cnt; i++) { if (rate == (rates[i] & ~BASIC_RATE_SET_BIT)) { return UAP_SUCCESS; } } } return UAP_FAILURE; } /** * @brief check mcbc rate is valid or not. * * @param rate rate for check * * @return UAP_SUCCESS or UAP_FAILURE */ int is_mcbc_rate_valid(u8 rate) { int rate_cnt = 0; int i; u8 rates[MAX_DATA_RATES]; rate_cnt = get_sys_cfg_rates((u8 *) & rates); if (rate_cnt > 0) { for (i = 0; i < rate_cnt; i++) { if (rates[i] & BASIC_RATE_SET_BIT) { if (rate == (rates[i] & ~BASIC_RATE_SET_BIT)) { return UAP_SUCCESS; } } } } return UAP_FAILURE; } /**************************************************************************** Global functions ****************************************************************************/ /** * @brief Creates a sys_cfg request for AP MAC address * and sends to the driver * * Usage: "sys_cfg_ap_mac_address [AP_MAC_ADDRESS]" * if AP_MAC_ADDRESS is provided, a 'set' is performed, * else a 'get' is performed. * * @param argc Number of arguments * @param argv Pointer to the arguments * @return N/A */ void apcmd_sys_cfg_ap_mac_address(int argc, char *argv[]) { APCMDBUF_SYS_CONFIGURE *cmd_buf = NULL; TLVBUF_AP_MAC_ADDRESS *tlv = NULL; u8 *buffer = NULL; u16 cmd_len; int ret = UAP_FAILURE; int opt; while ((opt = getopt_long(argc, argv, "+", cmd_options, NULL)) != -1) { switch (opt) { default: print_sys_cfg_ap_mac_address_usage(); return; } } argc -= optind; argv += optind; /* Check arguments */ if (argc > 1) { printf("ERR:Too many arguments.\n"); print_sys_cfg_ap_mac_address_usage(); return; } /* Initialize the command length */ cmd_len = sizeof(APCMDBUF_SYS_CONFIGURE) + sizeof(TLVBUF_AP_MAC_ADDRESS); /* Initialize the command buffer */ buffer = (u8 *) malloc(cmd_len); if (!buffer) { printf("ERR:Cannot allocate buffer for command!\n"); return; } bzero((char *) buffer, cmd_len); /* Locate headers */ cmd_buf = (APCMDBUF_SYS_CONFIGURE *) buffer; tlv = (TLVBUF_AP_MAC_ADDRESS *) (buffer + sizeof(APCMDBUF_SYS_CONFIGURE)); /* Fill the command buffer */ cmd_buf->CmdCode = APCMD_SYS_CONFIGURE; cmd_buf->Size = cmd_len; cmd_buf->SeqNum = 0; cmd_buf->Result = 0; tlv->Tag = MRVL_AP_MAC_ADDRESS_TLV_ID; tlv->Length = ETH_ALEN; if (argc == 0) { cmd_buf->Action = ACTION_GET; } else { cmd_buf->Action = ACTION_SET; if ((ret = mac2raw(argv[0], tlv->ApMacAddr)) != UAP_SUCCESS) { printf("ERR: %s Address \n", ret == UAP_FAILURE ? "Invalid MAC" : ret == UAP_RET_MAC_BROADCAST ? "Broadcast" : "Multicast"); free(buffer); return; } } endian_convert_tlv_header_out(tlv); /* Send the command */ ret = uap_ioctl((u8 *) cmd_buf, &cmd_len, cmd_len); endian_convert_tlv_header_in(tlv); /* Process response */ if (ret == UAP_SUCCESS) { /* Verify response */ if ((cmd_buf->CmdCode != (APCMD_SYS_CONFIGURE | APCMD_RESP_CHECK)) || (tlv->Tag != MRVL_AP_MAC_ADDRESS_TLV_ID)) { printf("ERR:Corrupted response! CmdCode=%x, Tlv->Tag=%x\n", cmd_buf->CmdCode, tlv->Tag); free(buffer); return; } /* Print response */ if (cmd_buf->Result == CMD_SUCCESS) { if (argc == 0) { printf("AP MAC address = "); print_mac(tlv->ApMacAddr); printf("\n"); } else { printf("AP MAC address setting successful\n"); } } else { if (argc == 0) { printf("ERR:Could not get AP MAC address!\n"); } else { printf("ERR:Could not set AP MAC address!\n"); } } } else { printf("ERR:Command sending failed!\n"); } if (buffer) free(buffer); return; } /** * @brief Creates a sys_cfg request for SSID * and sends to the driver * * Usage: "sys_cfg_ssid [SSID]" * if SSID is provided, a 'set' is performed * else a 'get' is performed * * @param argc Number of arguments * @param argv Pointer to the arguments * @return N/A */ void apcmd_sys_cfg_ssid(int argc, char *argv[]) { APCMDBUF_SYS_CONFIGURE *cmd_buf = NULL; TLVBUF_SSID *tlv = NULL; u8 *buffer = NULL; u16 cmd_len; int ret = UAP_FAILURE; int opt; u8 ssid[33]; while ((opt = getopt_long(argc, argv, "+", cmd_options, NULL)) != -1) { switch (opt) { default: print_sys_cfg_ssid_usage(); return; } } argc -= optind; argv += optind; /* Check arguments */ if (argc > 1) { printf("ERR:Too many arguments.\n"); print_sys_cfg_ssid_usage(); return; } if (argc == 0) { /* Initialize the command length */ cmd_len = sizeof(APCMDBUF_SYS_CONFIGURE) + sizeof(TLVBUF_SSID) + MAX_SSID_LENGTH; } else { if (strlen(argv[0]) > MAX_SSID_LENGTH) { printf("ERR:SSID too long.\n"); return; } /* Initialize the command length */ if (argv[0][1] == '"') { argv[0]++; } if (argv[0][strlen(argv[0])] == '"') { argv[0][strlen(argv[0])] = '\0'; } if (!strlen(argv[0])) { printf("ERR:NULL SSID not allowed.\n"); return; } cmd_len = sizeof(APCMDBUF_SYS_CONFIGURE) + sizeof(TLVBUF_SSID) + strlen(argv[0]); } /* Initialize the command buffer */ buffer = (u8 *) malloc(cmd_len); if (!buffer) { printf("ERR:Cannot allocate buffer for command!\n"); return; } bzero((char *) buffer, cmd_len); /* Locate headers */ cmd_buf = (APCMDBUF_SYS_CONFIGURE *) buffer; tlv = (TLVBUF_SSID *) (buffer + sizeof(APCMDBUF_SYS_CONFIGURE)); /* Fill the command buffer */ cmd_buf->CmdCode = APCMD_SYS_CONFIGURE; cmd_buf->Size = cmd_len; cmd_buf->SeqNum = 0; cmd_buf->Result = 0; tlv->Tag = MRVL_SSID_TLV_ID; if (argc == 0) { cmd_buf->Action = ACTION_GET; tlv->Length = MAX_SSID_LENGTH; } else { cmd_buf->Action = ACTION_SET; tlv->Length = strlen(argv[0]); memcpy(tlv->Ssid, argv[0], tlv->Length); } endian_convert_tlv_header_out(tlv); /* Send the command */ ret = uap_ioctl((u8 *) cmd_buf, &cmd_len, cmd_len); endian_convert_tlv_header_in(tlv); /* Process response */ if (ret == UAP_SUCCESS) { /* Verify response */ if ((cmd_buf->CmdCode != (APCMD_SYS_CONFIGURE | APCMD_RESP_CHECK)) || (tlv->Tag != MRVL_SSID_TLV_ID)) { printf("ERR:Corrupted response! CmdCode=%x, Tlv->Tag=%x\n", cmd_buf->CmdCode, tlv->Tag); free(buffer); return; } /* Print response */ if (cmd_buf->Result == CMD_SUCCESS) { if (argc == 0) { memset(ssid, 0, sizeof(ssid)); memcpy(ssid, tlv->Ssid, tlv->Length); printf("SSID = %s\n", ssid); } else { printf("SSID setting successful\n"); } } else { if (argc == 0) { printf("ERR:Could not get SSID!\n"); } else { printf("ERR:Could not set SSID!\n"); } } } else { printf("ERR:Command sending failed!\n"); } if (buffer) free(buffer); return; } /** * @brief Creates a sys_cfg request for beacon period * and sends to the driver * * Usage: "sys_cfg_beacon_period [BEACON_PERIOD]" * if BEACON_PERIOD is provided, a 'set' is performed * else a 'get' is performed. * * BEACON_PERIOD is represented in ms * * @param argc Number of arguments * @param argv Pointer to the arguments * @return N/A */ void apcmd_sys_cfg_beacon_period(int argc, char *argv[]) { APCMDBUF_SYS_CONFIGURE *cmd_buf = NULL; TLVBUF_BEACON_PERIOD *tlv = NULL; u8 *buffer = NULL; u16 cmd_len; int ret = UAP_FAILURE; int opt; while ((opt = getopt_long(argc, argv, "+", cmd_options, NULL)) != -1) { switch (opt) { default: print_sys_cfg_beacon_period_usage(); return; } } argc -= optind; argv += optind; /* Check arguments */ if (argc && is_input_valid(BEACONPERIOD, argc, argv) != UAP_SUCCESS) { print_sys_cfg_beacon_period_usage(); return; } /* Initialize the command length */ cmd_len = sizeof(APCMDBUF_SYS_CONFIGURE) + sizeof(TLVBUF_BEACON_PERIOD); /* Initialize the command buffer */ buffer = (u8 *) malloc(cmd_len); if (!buffer) { printf("ERR:Cannot allocate buffer for command!\n"); return; } bzero((char *) buffer, cmd_len); /* Locate headers */ cmd_buf = (APCMDBUF_SYS_CONFIGURE *) buffer; tlv = (TLVBUF_BEACON_PERIOD *) (buffer + sizeof(APCMDBUF_SYS_CONFIGURE)); /* Fill the command buffer */ cmd_buf->CmdCode = APCMD_SYS_CONFIGURE; cmd_buf->Size = cmd_len; cmd_buf->SeqNum = 0; cmd_buf->Result = 0; tlv->Tag = MRVL_BEACON_PERIOD_TLV_ID; tlv->Length = 2; if (argc == 0) { cmd_buf->Action = ACTION_GET; } else { cmd_buf->Action = ACTION_SET; tlv->BeaconPeriod_ms = (u16) atoi(argv[0]); } endian_convert_tlv_header_out(tlv); tlv->BeaconPeriod_ms = uap_cpu_to_le16(tlv->BeaconPeriod_ms); /* Send the command */ ret = uap_ioctl((u8 *) cmd_buf, &cmd_len, cmd_len); endian_convert_tlv_header_in(tlv); tlv->BeaconPeriod_ms = uap_le16_to_cpu(tlv->BeaconPeriod_ms); /* Process response */ if (ret == UAP_SUCCESS) { /* Verify response */ if ((cmd_buf->CmdCode != (APCMD_SYS_CONFIGURE | APCMD_RESP_CHECK)) || (tlv->Tag != MRVL_BEACON_PERIOD_TLV_ID)) { printf("ERR:Corrupted response! CmdCode=%x, Tlv->Tag=%x\n", cmd_buf->CmdCode, tlv->Tag); free(buffer); return; } /* Print response */ if (cmd_buf->Result == CMD_SUCCESS) { if (argc == 0) { printf("Beacon period = %d\n", tlv->BeaconPeriod_ms); } else { printf("Beacon period setting successful\n"); } } else { if (argc == 0) { printf("ERR:Could not get beacon period!\n"); } else { printf("ERR:Could not set beacon period!\n"); } } } else { printf("ERR:Command sending failed!\n"); } if (buffer) free(buffer); return; } /** * @brief Creates a sys_cfg request for DTIM period * and sends to the driver * * Usage: "sys_cfg_dtim_period [DTIM_PERIOD]" * if DTIM_PERIOD is provided, a 'set' is performed * else a 'get' is performed * * @param argc Number of arguments * @param argv Pointer to the arguments * @return N/A */ void apcmd_sys_cfg_dtim_period(int argc, char *argv[]) { APCMDBUF_SYS_CONFIGURE *cmd_buf = NULL; TLVBUF_DTIM_PERIOD *tlv = NULL; u8 *buffer = NULL; u16 cmd_len; int ret = UAP_FAILURE; int opt; while ((opt = getopt_long(argc, argv, "+", cmd_options, NULL)) != -1) { switch (opt) { default: print_sys_cfg_dtim_period_usage(); return; } } argc -= optind; argv += optind; /* Check arguments */ if (argc && (is_input_valid(DTIMPERIOD, argc, argv) != UAP_SUCCESS)) { print_sys_cfg_dtim_period_usage(); return; } /* Initialize the command length */ cmd_len = sizeof(APCMDBUF_SYS_CONFIGURE) + sizeof(TLVBUF_DTIM_PERIOD); /* Initialize the command buffer */ buffer = (u8 *) malloc(cmd_len); if (!buffer) { printf("ERR:Cannot allocate buffer for command!\n"); return; } bzero((char *) buffer, cmd_len); /* Locate headers */ cmd_buf = (APCMDBUF_SYS_CONFIGURE *) buffer; tlv = (TLVBUF_DTIM_PERIOD *) (buffer + sizeof(APCMDBUF_SYS_CONFIGURE)); /* Fill the command buffer */ cmd_buf->CmdCode = APCMD_SYS_CONFIGURE; cmd_buf->Size = cmd_len; cmd_buf->SeqNum = 0; cmd_buf->Result = 0; tlv->Tag = MRVL_DTIM_PERIOD_TLV_ID; tlv->Length = 1; if (argc == 0) { cmd_buf->Action = ACTION_GET; } else { cmd_buf->Action = ACTION_SET; tlv->DtimPeriod = (u8) atoi(argv[0]); } endian_convert_tlv_header_out(tlv); /* Send the command */ ret = uap_ioctl((u8 *) cmd_buf, &cmd_len, cmd_len); endian_convert_tlv_header_in(tlv); /* Process response */ if (ret == UAP_SUCCESS) { /* Verify response */ if ((cmd_buf->CmdCode != (APCMD_SYS_CONFIGURE | APCMD_RESP_CHECK)) || (tlv->Tag != MRVL_DTIM_PERIOD_TLV_ID)) { printf("ERR:Corrupted response! CmdCode=%x, Tlv->Tag=%x\n", cmd_buf->CmdCode, tlv->Tag); free(buffer); return; } /* Print response */ if (cmd_buf->Result == CMD_SUCCESS) { if (argc == 0) { printf("DTIM period = %d\n", tlv->DtimPeriod); } else { printf("DTIM period setting successful\n"); } } else { if (argc == 0) { printf("ERR:Could not get DTIM period!\n"); } else { printf("ERR:Could not set DTIM period!\n"); } } } else { printf("ERR:Command sending failed!\n"); } if (buffer) free(buffer); return; } /** * @brief Creates a sys_cfg request for channel * and sends to the driver * * Usage: "sys_cfg_channel [CHANNEL] [MODE]" * if CHANNEL is provided, a 'set' is performed * else a 'get' is performed * if MODE is provided, a 'set' is performed with * 0 as manual channel selection * 1 as automatic channel selection. * * @param argc Number of arguments * @param argv Pointer to the arguments * @return N/A */ void apcmd_sys_cfg_channel(int argc, char *argv[]) { APCMDBUF_SYS_CONFIGURE *cmd_buf = NULL; TLVBUF_CHANNEL_CONFIG *tlv = NULL; u8 *buffer = NULL; u16 cmd_len; int ret = UAP_FAILURE; int opt; while ((opt = getopt_long(argc, argv, "+", cmd_options, NULL)) != -1) { switch (opt) { default: print_sys_cfg_channel_usage(); return; } } argc -= optind; argv += optind; /* Check arguments */ if (argc && is_input_valid(CHANNEL, argc, argv) != UAP_SUCCESS) { print_sys_cfg_channel_usage(); return; } /* Initialize the command length */ cmd_len = sizeof(APCMDBUF_SYS_CONFIGURE) + sizeof(TLVBUF_CHANNEL_CONFIG); /* Initialize the command buffer */ buffer = (u8 *) malloc(cmd_len); if (!buffer) { printf("ERR:Cannot allocate buffer for command!\n"); return; } bzero((char *) buffer, cmd_len); /* Locate headers */ cmd_buf = (APCMDBUF_SYS_CONFIGURE *) buffer; tlv = (TLVBUF_CHANNEL_CONFIG *) (buffer + sizeof(APCMDBUF_SYS_CONFIGURE)); /* Fill the command buffer */ cmd_buf->CmdCode = APCMD_SYS_CONFIGURE; cmd_buf->Size = cmd_len; cmd_buf->SeqNum = 0; cmd_buf->Result = 0; tlv->Tag = MRVL_CHANNELCONFIG_TLV_ID; tlv->Length = 2; if (argc == 0) { cmd_buf->Action = ACTION_GET; } else { cmd_buf->Action = ACTION_SET; if (argc == 1) { tlv->ChanNumber = (u8) atoi(argv[0]); tlv->BandConfigType = 0; } else { tlv->BandConfigType = atoi(argv[1]) ? BAND_CONFIG_ACS_MODE : 0; tlv->ChanNumber = (u8) atoi(argv[0]); } } endian_convert_tlv_header_out(tlv); /* Send the command */ ret = uap_ioctl((u8 *) cmd_buf, &cmd_len, cmd_len); endian_convert_tlv_header_in(tlv); /* Process response */ if (ret == UAP_SUCCESS) { /* Verify response */ if ((cmd_buf->CmdCode != (APCMD_SYS_CONFIGURE | APCMD_RESP_CHECK)) || (tlv->Tag != MRVL_CHANNELCONFIG_TLV_ID)) { printf("ERR:Corrupted response! CmdCode=%x, Tlv->Tag=%x\n", cmd_buf->CmdCode, tlv->Tag); free(buffer); return; } /* Print response */ if (cmd_buf->Result == CMD_SUCCESS) { if (argc == 0) { printf("Mode = %s\n", (tlv->BandConfigType == 0) ? "Manual" : "ACS"); printf("Channel = %d\n", tlv->ChanNumber); } else { printf("Channel setting successful\n"); } } else { if (argc == 0) { printf("ERR:Could not get channel!\n"); } else { printf("ERR:Could not set channel!\n"); } } } else { printf("ERR:Command sending failed!\n"); } if (buffer) free(buffer); return; } /** * @brief Creates a sys_cfg request for channel list * and sends to the driver * * Usage: "sys_cfg_scan_channels [CHANNELS]" * if CHANNELS are provided, a 'set' is performed * else a 'get' is performed * * @param argc Number of arguments * @param argv Pointer to the arguments * @return N/A */ void apcmd_sys_cfg_scan_channels(int argc, char *argv[]) { APCMDBUF_SYS_CONFIGURE *cmd_buf = NULL; TLVBUF_CHANNEL_LIST *tlv = NULL; CHANNEL_LIST *pChanList = NULL; u8 *buffer = NULL; u16 cmd_len; int ret = UAP_FAILURE; int opt; int i; while ((opt = getopt_long(argc, argv, "+", cmd_options, NULL)) != -1) { switch (opt) { default: print_sys_cfg_scan_channels_usage(); return; } } argc -= optind; argv += optind; /* Check arguments */ if (argc && is_input_valid(SCANCHANNELS, argc, argv) != UAP_SUCCESS) { print_sys_cfg_scan_channels_usage(); return; } /* Initialize the command length */ if (argc == 0) cmd_len = sizeof(APCMDBUF_SYS_CONFIGURE) + sizeof(TLVBUF_CHANNEL_LIST) + sizeof(CHANNEL_LIST) * MAX_CHANNELS; else cmd_len = sizeof(APCMDBUF_SYS_CONFIGURE) + sizeof(TLVBUF_CHANNEL_LIST) + sizeof(CHANNEL_LIST) * argc; /* Initialize the command buffer */ buffer = (u8 *) malloc(cmd_len); if (!buffer) { printf("ERR:Can not allocate buffer for command!\n"); return; } bzero((char *) buffer, cmd_len); /* LOCATE HEADERS */ cmd_buf = (APCMDBUF_SYS_CONFIGURE *) buffer; tlv = (TLVBUF_CHANNEL_LIST *) (buffer + sizeof(APCMDBUF_SYS_CONFIGURE)); /* Fill the command buffer */ cmd_buf->CmdCode = APCMD_SYS_CONFIGURE; cmd_buf->Size = cmd_len; cmd_buf->SeqNum = 0; cmd_buf->Result = 0; tlv->Tag = MRVL_CHANNELLIST_TLV_ID; if (argc == 0) { cmd_buf->Action = ACTION_GET; tlv->Length = sizeof(CHANNEL_LIST) * MAX_CHANNELS; } else { cmd_buf->Action = ACTION_SET; tlv->Length = sizeof(CHANNEL_LIST) * argc; pChanList = tlv->ChanList; for (i = 0; i < argc; i++) { pChanList->ChanNumber = (u8) atoi(argv[i]); pChanList->BandConfigType = 0; pChanList++; } } endian_convert_tlv_header_out(tlv); /* Send the command */ ret = uap_ioctl((u8 *) cmd_buf, &cmd_len, cmd_len); endian_convert_tlv_header_in(tlv); /* Process response */ if (ret == UAP_SUCCESS) { /* Verify response */ if ((cmd_buf->CmdCode != (APCMD_SYS_CONFIGURE | APCMD_RESP_CHECK)) || (tlv->Tag != MRVL_CHANNELLIST_TLV_ID)) { printf("ERR:Corrupted response! CmdCode=%x, Tlv->Tag=%x\n", cmd_buf->CmdCode, tlv->Tag); free(buffer); return; } /* Print response */ if (cmd_buf->Result == CMD_SUCCESS) { if (argc == 0) { printf("Channels List = "); if (tlv->Length % sizeof(CHANNEL_LIST)) { printf("Error: Length mismatch\n"); free(buffer); return; } pChanList = tlv->ChanList; for (i = 0; i < (tlv->Length / sizeof(CHANNEL_LIST)); i++) { printf("%d ", pChanList->ChanNumber); pChanList++; } printf("\n"); } else { printf("Scan Channel List setting successful\n"); } } else { printf("ERR:Could not %s scan channel list!\n", argc ? "SET" : "GET"); } } else { printf("ERR:Command sending failed!\n"); } if (buffer) free(buffer); return; } /** * @brief parser for sys_cfg_rates_ext input * * @param argc Number of arguments * @param argv Pointer to the arguments * @param output stores indexes for "rates, mbrate and urate" * arguments * * e.g., * * "rates 0x82 4 16 22 0x30 mbrate 2 urate 16" * * will have output array as * * start_index end_index * rates 0 5 * mbrate 6 7 * urate 8 9 * * @return NA * */ void parse_input(int argc, char **argv, int output[3][2]) { int i, j, k = 0; char *keywords[3] = { "rates", "mbrate", "urate" }; for (i = 0; i < 3; i++) output[i][0] = -1; for (i = 0; i < argc; i++) { for (j = 0; j < 3; j++) { if (strcmp(argv[i], keywords[j]) == 0) { output[j][1] = output[j][0] = i; k = j; break; } } output[k][1] += 1; } } /** * @brief Creates a sys_cfg request for setting data_rates, MCBC and * TX data rates. * * Usage: "sys_cfg_rates_ext [RATES]" * * @param argc Number of arguments * @param argv Pointer to the arguments * @return N/A */ void apcmd_sys_cfg_rates_ext(int argc, char *argv[]) { int i, j = 0; int opt; int rflag = 0, mflag = 0, uflag = 0; char *argv_rate[14]; int argc_rate = 0; char *argv_mrate[1]; char *argv_urate[1]; int mrate_found = UAP_FAILURE; int urate_found = UAP_FAILURE; u8 *tlv_buf = NULL; TLVBUF_TX_DATA_RATE *tlv_urate = NULL; TLVBUF_MCBC_DATA_RATE *tlv_mrate = NULL; TLVBUF_RATES *tlv_rate = NULL; u8 *buffer = NULL; APCMDBUF_SYS_CONFIGURE *cmd_buf = NULL; int ret = UAP_FAILURE; u16 cmd_len; int output[3][2]; while ((opt = getopt_long(argc, argv, "+", cmd_options, NULL)) != -1) { switch (opt) { default: print_sys_cfg_rates_ext_usage(); return; } } argc -= optind; argv += optind; if (argc) { /* * SET */ parse_input(argc, argv, output); /* * Rate */ if ((output[0][0] != -1) && (output[0][1] > output[0][0])) { rflag = 1; i = 0; for (j = (output[0][0] + 1); j < output[0][1]; j++) { argv_rate[i] = (char *) malloc(sizeof(char) * (strlen(argv[j]) + 1)); strcpy(argv_rate[i], argv[j]); argc_rate = ++i; } } /* * mrate */ if ((output[1][0] != -1) && (output[1][1] > output[1][0])) { if ((output[1][1] - output[1][0]) != 2) { printf("\nERR: Invalid mrate input"); print_sys_cfg_rates_ext_usage(); goto done; } mflag = 1; argv_mrate[0] = (char *) malloc(sizeof(char) * (strlen(argv[j]) + 1)); strcpy(argv_mrate[0], argv[output[1][0] + 1]); } /* * urate */ if ((output[2][0] != -1) && (output[2][1] > output[2][0])) { if ((output[2][1] - output[2][0]) != 2) { printf("\nERR: Invalid urate input"); print_sys_cfg_rates_ext_usage(); goto done; } uflag = 1; argv_urate[0] = (char *) malloc(sizeof(char) * (strlen(argv[j]) + 1)); strcpy(argv_urate[0], argv[output[2][0] + 1]); } if (!rflag && !mflag & !uflag) { printf("ERR: Invalid input\n"); print_sys_cfg_rates_ext_usage(); goto done; } if (rflag && is_input_valid(RATE, argc_rate, argv_rate) != UAP_SUCCESS) { printf("ERR: Invalid RATE\n"); print_sys_cfg_rates_ext_usage(); goto done; } if (mflag && is_input_valid(MCBCDATARATE, 1, argv_mrate) != UAP_SUCCESS) { printf("ERR: Invalid MCBC RATE\n"); print_sys_cfg_rates_ext_usage(); goto done; } if (uflag && is_input_valid(TXDATARATE, 1, argv_urate) != UAP_SUCCESS) { printf("ERR: Invalid TX RATE\n"); print_sys_cfg_rates_ext_usage(); goto done; } if (!rflag && (mflag || uflag)) { /* * Check mrate and urate wrt old Rates */ if (mflag && A2HEXDECIMAL(argv_mrate[0]) && is_mcbc_rate_valid(A2HEXDECIMAL(argv_mrate[0])) != UAP_SUCCESS) { printf("ERR: invalid MCBC data rate."); print_sys_cfg_rates_ext_usage(); goto done; } if (uflag && A2HEXDECIMAL(argv_urate[0]) && is_tx_rate_valid(A2HEXDECIMAL(argv_urate[0])) != UAP_SUCCESS) { printf("ERR: invalid tx data rate."); print_sys_cfg_rates_ext_usage(); goto done; } } else if (rflag && (mflag || uflag)) { /* * Check mrate and urate wrt new Rates */ for (i = 0; i < argc_rate; i++) { /* * MCBC rate must be from Basic rates */ if (mflag && !mrate_found && A2HEXDECIMAL(argv_rate[i]) & BASIC_RATE_SET_BIT) { if (A2HEXDECIMAL(argv_mrate[0]) == (A2HEXDECIMAL(argv_rate[i]) & ~BASIC_RATE_SET_BIT)) { mrate_found = UAP_SUCCESS; } } if (uflag && !urate_found && (A2HEXDECIMAL(argv_urate[0]) == ((A2HEXDECIMAL(argv_rate[i]) & ~BASIC_RATE_SET_BIT)))) { urate_found = UAP_SUCCESS; } } if (mflag && A2HEXDECIMAL(argv_mrate[0]) && !(mrate_found == UAP_SUCCESS)) { printf("ERR: mrate not valid\n"); goto done; } if (uflag && A2HEXDECIMAL(argv_urate[0]) && !(urate_found == UAP_SUCCESS)) { printf("ERR: urate not valid\n"); goto done; } } /* post-parsing */ cmd_len = sizeof(APCMDBUF_SYS_CONFIGURE); if (rflag) { cmd_len += sizeof(TLVBUF_RATES) + argc_rate; cmd_len += sizeof(TLVBUF_MCBC_DATA_RATE); cmd_len += sizeof(TLVBUF_TX_DATA_RATE); } else { if (mflag) cmd_len += sizeof(TLVBUF_MCBC_DATA_RATE); if (uflag) cmd_len += sizeof(TLVBUF_TX_DATA_RATE); } } else { /* GET */ cmd_len = sizeof(APCMDBUF_SYS_CONFIGURE) + sizeof(TLVBUF_RATES) + MAX_RATES + sizeof(TLVBUF_MCBC_DATA_RATE) + sizeof(TLVBUF_TX_DATA_RATE); } buffer = (u8 *) malloc(cmd_len); if (!buffer) { printf("ERR:Cannot allocate buffer for command!\n"); goto done; } bzero((char *) buffer, cmd_len); /* Fill the command buffer */ cmd_buf = (APCMDBUF_SYS_CONFIGURE *) buffer; cmd_buf->CmdCode = APCMD_SYS_CONFIGURE; cmd_buf->Size = cmd_len; cmd_buf->SeqNum = 0; cmd_buf->Result = 0; cmd_buf->Action = argc ? ACTION_SET : ACTION_GET; tlv_buf = buffer + sizeof(APCMDBUF_SYS_CONFIGURE); /* Locate headers */ if (rflag || (!argc)) { tlv_rate = (TLVBUF_RATES *) (buffer + sizeof(APCMDBUF_SYS_CONFIGURE)); tlv_rate->Tag = MRVL_RATES_TLV_ID; tlv_rate->Length = argc ? argc_rate : MAX_RATES; for (i = 0; i < argc_rate; i++) { tlv_rate->OperationalRates[i] = (u8) A2HEXDECIMAL(argv_rate[i]); } tlv_buf += tlv_rate->Length + sizeof(TLVBUF_RATES); endian_convert_tlv_header_out(tlv_rate); } if (rflag || mflag || (!argc)) { tlv_mrate = (TLVBUF_MCBC_DATA_RATE *) tlv_buf; tlv_mrate->Tag = MRVL_MCBC_DATA_RATE_TLV_ID; tlv_mrate->Length = 2; tlv_mrate->MCBCdatarate = 0; if (mflag) { tlv_mrate->MCBCdatarate = (u16) A2HEXDECIMAL(argv_mrate[0]) & ~BASIC_RATE_SET_BIT; tlv_mrate->MCBCdatarate = uap_cpu_to_le16(tlv_mrate->MCBCdatarate); } tlv_buf += sizeof(TLVBUF_MCBC_DATA_RATE); endian_convert_tlv_header_out(tlv_mrate); } if (rflag || uflag || (!argc)) { tlv_urate = (TLVBUF_TX_DATA_RATE *) tlv_buf; tlv_urate->Tag = MRVL_TX_DATA_RATE_TLV_ID; tlv_urate->Length = 2; tlv_urate->TxDataRate = 0; if (uflag) { tlv_urate->TxDataRate = (u16) A2HEXDECIMAL(argv_urate[0]) & ~BASIC_RATE_SET_BIT; tlv_urate->TxDataRate = uap_cpu_to_le16(tlv_urate->TxDataRate); } endian_convert_tlv_header_out(tlv_urate); } /* Send the command */ ret = uap_ioctl((u8 *) cmd_buf, &cmd_len, cmd_len); tlv_buf = buffer + sizeof(APCMDBUF_SYS_CONFIGURE); if (ret == UAP_SUCCESS) { /* Verify response */ if (cmd_buf->CmdCode != (APCMD_SYS_CONFIGURE | APCMD_RESP_CHECK)) { printf("ERR:Corrupted response! CmdCode=%x\n", cmd_buf->CmdCode); goto done; } /* Print response */ if (cmd_buf->Result == CMD_SUCCESS) { if (argc) { printf("Rates setting successful\n"); } else { print_tlv((u8 *) tlv_buf, cmd_buf->Size - sizeof(APCMDBUF_SYS_CONFIGURE) + BUF_HEADER_SIZE); } } else { printf("ERR:Could not %s operational rates!\n", argc ? "set" : "get"); if (argc) printf ("operational rates only allow to set before bss start.\n"); } } else { printf("ERR:Command sending failed!\n"); } done: if (rflag) { for (i = 0; i < argc_rate; i++) { free(argv_rate[i]); } } if (mflag) free(argv_mrate[0]); if (uflag) free(argv_urate[0]); if (buffer) free(buffer); return; } /** * @brief Creates a sys_cfg request for data rates * and sends to the driver * * Usage: "sys_cfg_rates [RATES]" * * RATES is provided as a set of data rates, in * unit of 500 kilobits/s. * Maximum 12 rates can be provided. * * if no RATE is provided, then it gets configured rates * * Each rate must be separated by a space * * @param argc Number of arguments * @param argv Pointer to the arguments * @return N/A */ void apcmd_sys_cfg_rates(int argc, char *argv[]) { APCMDBUF_SYS_CONFIGURE *cmd_buf = NULL; TLVBUF_RATES *tlv = NULL; u8 *buffer = NULL; u16 cmd_len; int ret = UAP_FAILURE; int i = 0; int opt; while ((opt = getopt_long(argc, argv, "+", cmd_options, NULL)) != -1) { switch (opt) { default: print_sys_cfg_rates_usage(); return; } } argc -= optind; argv += optind; /* Check arguments */ if (argc && is_input_valid(RATE, argc, argv) != UAP_SUCCESS) { print_sys_cfg_rates_usage(); return; } if (argc == 0) { /* Initialize the command length */ cmd_len = sizeof(APCMDBUF_SYS_CONFIGURE) + sizeof(TLVBUF_RATES) + MAX_RATES; } else { /* Initialize the command length */ cmd_len = sizeof(APCMDBUF_SYS_CONFIGURE) + sizeof(TLVBUF_RATES) + argc; } /* Initialize the command buffer */ buffer = (u8 *) malloc(cmd_len); if (!buffer) { printf("ERR:Cannot allocate buffer for command!\n"); return; } bzero((char *) buffer, cmd_len); /* Locate headers */ cmd_buf = (APCMDBUF_SYS_CONFIGURE *) buffer; tlv = (TLVBUF_RATES *) (buffer + sizeof(APCMDBUF_SYS_CONFIGURE)); /* Fill the command buffer */ cmd_buf->CmdCode = APCMD_SYS_CONFIGURE; cmd_buf->Size = cmd_len; cmd_buf->SeqNum = 0; cmd_buf->Result = 0; tlv->Tag = MRVL_RATES_TLV_ID; if (argc == 0) { cmd_buf->Action = ACTION_GET; tlv->Length = MAX_RATES; } else { cmd_buf->Action = ACTION_SET; tlv->Length = argc; for (i = 0; i < tlv->Length; i++) { tlv->OperationalRates[i] = (u8) A2HEXDECIMAL(argv[i]); } } endian_convert_tlv_header_out(tlv); /* Send the command */ ret = uap_ioctl((u8 *) cmd_buf, &cmd_len, cmd_len); endian_convert_tlv_header_in(tlv); /* Process response */ if (ret == UAP_SUCCESS) { /* Verify response */ if ((cmd_buf->CmdCode != (APCMD_SYS_CONFIGURE | APCMD_RESP_CHECK)) || (tlv->Tag != MRVL_RATES_TLV_ID)) { printf("ERR:Corrupted response! CmdCode=%x, Tlv->Tag=%x\n", cmd_buf->CmdCode, tlv->Tag); free(buffer); return; } /* Print response */ if (cmd_buf->Result == CMD_SUCCESS) { if (argc == 0) { print_rate(tlv); } else { printf("Rates setting successful\n"); } } else { if (argc == 0) { printf("ERR:Could not get operational rates!\n"); } else { printf("ERR:Could not set operational rates!\n"); } } } else { printf("ERR:Command sending failed!\n"); } if (buffer) free(buffer); return; } /** * @brief Creates a sys_cfg request for Tx power * and sends to the driver * * Usage: "sys_cfg_tx_power [TX_POWER]" * if TX_POWER is provided, a 'set' is performed * else a 'get' is performed. * * TX_POWER is represented in dBm * * @param argc Number of arguments * @param argv Pointer to the arguments * @return N/A */ void apcmd_sys_cfg_tx_power(int argc, char *argv[]) { APCMDBUF_SYS_CONFIGURE *cmd_buf = NULL; TLVBUF_TX_POWER *tlv = NULL; u8 *buffer = NULL; u16 cmd_len; int ret = UAP_FAILURE; int opt; while ((opt = getopt_long(argc, argv, "+", cmd_options, NULL)) != -1) { switch (opt) { default: print_sys_cfg_tx_power_usage(); return; } } argc -= optind; argv += optind; /* Check arguments */ if (argc && is_input_valid(TXPOWER, argc, argv) != UAP_SUCCESS) { print_sys_cfg_tx_power_usage(); return; } /* Initialize the command length */ cmd_len = sizeof(APCMDBUF_SYS_CONFIGURE) + sizeof(TLVBUF_TX_POWER); /* Initialize the command buffer */ buffer = (u8 *) malloc(cmd_len); if (!buffer) { printf("ERR:Cannot allocate buffer for command!\n"); return; } bzero((char *) buffer, cmd_len); /* Locate headers */ cmd_buf = (APCMDBUF_SYS_CONFIGURE *) buffer; tlv = (TLVBUF_TX_POWER *) (buffer + sizeof(APCMDBUF_SYS_CONFIGURE)); /* Fill the command buffer */ cmd_buf->CmdCode = APCMD_SYS_CONFIGURE; cmd_buf->Size = cmd_len; cmd_buf->SeqNum = 0; cmd_buf->Result = 0; tlv->Tag = MRVL_TX_POWER_TLV_ID; tlv->Length = 1; if (argc == 0) { cmd_buf->Action = ACTION_GET; } else { printf("Please check power calibration for board to see if this power\n" "setting is within calibrated range. Firmware may over-ride\n " "this setting if it is not within calibrated range, which can\n" "vary from board to board.\n"); cmd_buf->Action = ACTION_SET; tlv->TxPower_dBm = (u8) atoi(argv[0]); } endian_convert_tlv_header_out(tlv); /* Send the command */ ret = uap_ioctl((u8 *) cmd_buf, &cmd_len, cmd_len); endian_convert_tlv_header_in(tlv); /* Process response */ if (ret == UAP_SUCCESS) { /* Verify response */ if ((cmd_buf->CmdCode != (APCMD_SYS_CONFIGURE | APCMD_RESP_CHECK)) || (tlv->Tag != MRVL_TX_POWER_TLV_ID)) { printf("ERR:Corrupted response! CmdCode=%x, Tlv->Tag=%x\n", cmd_buf->CmdCode, tlv->Tag); free(buffer); return; } /* Print response */ if (cmd_buf->Result == CMD_SUCCESS) { if (argc == 0) { printf("Tx power = %d dBm\n", tlv->TxPower_dBm); } else { printf("Tx power setting successful\n"); } } else { if (argc == 0) { printf("ERR:Could not get tx power!\n"); } else { printf("ERR:Could not set tx power!\n"); } } } else { printf("ERR:Command sending failed!\n"); } if (buffer) free(buffer); return; } /** * @brief Creates a sys_cfg request for SSID broadcast * and sends to the driver * * Usage: "sys_cfg_bcast_ssid_ctl [0|1]" * * Options: 0 - Disable SSID broadcast * 1 - Enable SSID broadcast * empty - Get current SSID broadcast setting * * @param argc Number of arguments * @param argv Pointer to the arguments * @return N/A */ void apcmd_sys_cfg_bcast_ssid_ctl(int argc, char *argv[]) { APCMDBUF_SYS_CONFIGURE *cmd_buf = NULL; TLVBUF_BCAST_SSID_CTL *tlv = NULL; u8 *buffer = NULL; u16 cmd_len; int ret = UAP_FAILURE; int opt; while ((opt = getopt_long(argc, argv, "+", cmd_options, NULL)) != -1) { switch (opt) { default: print_sys_cfg_bcast_ssid_ctl_usage(); return; } } argc -= optind; argv += optind; /* Check arguments */ if (argc && is_input_valid(BROADCASTSSID, argc, argv) != UAP_SUCCESS) { print_sys_cfg_bcast_ssid_ctl_usage(); return; } /* Initialize the command length */ cmd_len = sizeof(APCMDBUF_SYS_CONFIGURE) + sizeof(TLVBUF_BCAST_SSID_CTL); /* Initialize the command buffer */ buffer = (u8 *) malloc(cmd_len); if (!buffer) { printf("ERR:Cannot allocate buffer for command!\n"); return; } bzero((char *) buffer, cmd_len); /* Locate headers */ cmd_buf = (APCMDBUF_SYS_CONFIGURE *) buffer; tlv = (TLVBUF_BCAST_SSID_CTL *) (buffer + sizeof(APCMDBUF_SYS_CONFIGURE)); /* Fill the command buffer */ cmd_buf->CmdCode = APCMD_SYS_CONFIGURE; cmd_buf->Size = cmd_len; cmd_buf->SeqNum = 0; cmd_buf->Result = 0; tlv->Tag = MRVL_BCAST_SSID_CTL_TLV_ID; tlv->Length = 1; if (argc == 0) { cmd_buf->Action = ACTION_GET; } else { cmd_buf->Action = ACTION_SET; tlv->BcastSsidCtl = (u8) atoi(argv[0]); } endian_convert_tlv_header_out(tlv); /* Send the command */ ret = uap_ioctl((u8 *) cmd_buf, &cmd_len, cmd_len); endian_convert_tlv_header_in(tlv); /* Process response */ if (ret == UAP_SUCCESS) { /* Verify response */ if ((cmd_buf->CmdCode != (APCMD_SYS_CONFIGURE | APCMD_RESP_CHECK)) || (tlv->Tag != MRVL_BCAST_SSID_CTL_TLV_ID)) { printf("ERR:Corrupted response! CmdCode=%x, Tlv->Tag=%x\n", cmd_buf->CmdCode, tlv->Tag); free(buffer); return; } /* Print response */ if (cmd_buf->Result == CMD_SUCCESS) { if (argc == 0) { printf("SSID broadcast is %s\n", (tlv->BcastSsidCtl == 1) ? "enabled" : "disabled"); } else { printf("SSID broadcast setting successful\n"); } } else { if (argc == 0) { printf("ERR:Could not get SSID broadcast!\n"); } else { printf("ERR:Could not set SSID broadcast!\n"); } } } else { printf("ERR:Command sending failed!\n"); } if (buffer) free(buffer); return; } /** * @brief Creates a sys_cfg request for preamble settings * and sends to the driver * * Usage: "sys_cfg_preamble_ctl" * * @param argc Number of arguments * @param argv Pointer to the arguments * @return N/A */ void apcmd_sys_cfg_preamble_ctl(int argc, char *argv[]) { APCMDBUF_SYS_CONFIGURE *cmd_buf = NULL; TLVBUF_PREAMBLE_CTL *tlv = NULL; u8 *buffer = NULL; u16 cmd_len; int ret = UAP_FAILURE; int opt; while ((opt = getopt_long(argc, argv, "+", cmd_options, NULL)) != -1) { switch (opt) { default: print_sys_cfg_preamble_ctl_usage(); return; } } argc -= optind; argv += optind; /* Check arguments */ if (argc != 0) { printf("ERR:Too many arguments.\n"); print_sys_cfg_preamble_ctl_usage(); return; } /* Initialize the command length */ cmd_len = sizeof(APCMDBUF_SYS_CONFIGURE) + sizeof(TLVBUF_PREAMBLE_CTL); /* Initialize the command buffer */ buffer = (u8 *) malloc(cmd_len); if (!buffer) { printf("ERR:Cannot allocate buffer for command!\n"); return; } bzero((char *) buffer, cmd_len); /* Locate headers */ cmd_buf = (APCMDBUF_SYS_CONFIGURE *) buffer; tlv = (TLVBUF_PREAMBLE_CTL *) (buffer + sizeof(APCMDBUF_SYS_CONFIGURE)); /* Fill the command buffer */ cmd_buf->CmdCode = APCMD_SYS_CONFIGURE; cmd_buf->Size = cmd_len; cmd_buf->SeqNum = 0; cmd_buf->Result = 0; tlv->Tag = MRVL_PREAMBLE_CTL_TLV_ID; tlv->Length = 1; cmd_buf->Action = ACTION_GET; endian_convert_tlv_header_out(tlv); /* Send the command */ ret = uap_ioctl((u8 *) cmd_buf, &cmd_len, cmd_len); endian_convert_tlv_header_in(tlv); /* Process response */ if (ret == UAP_SUCCESS) { /* Verify response */ if ((cmd_buf->CmdCode != (APCMD_SYS_CONFIGURE | APCMD_RESP_CHECK)) || (tlv->Tag != MRVL_PREAMBLE_CTL_TLV_ID)) { printf("ERR:Corrupted response! CmdCode=%x, Tlv->Tag=%x\n", cmd_buf->CmdCode, tlv->Tag); free(buffer); return; } /* Print response */ if (cmd_buf->Result == CMD_SUCCESS) printf("Preamble type is %s\n", (tlv->PreambleType == 0) ? "auto" : ((tlv->PreambleType == 1) ? "short" : "long")); else printf("ERR:Could not get preamble type!\n"); } else { printf("ERR:Command sending failed!\n"); } if (buffer) free(buffer); return; } /** * @brief Creates a sys_cfg request for antenna configuration * and sends to the driver * * Usage: "sys_cfg_antenna_ctl [MODE]" * * Options: ANTENNA : 0 - Rx antenna * 1 - Tx antenna * MODE : 0 - Antenna A * 1 - Antenna B * empty - Get current antenna settings * * @param argc Number of arguments * @param argv Pointer to the arguments * @return N/A */ void apcmd_sys_cfg_antenna_ctl(int argc, char *argv[]) { APCMDBUF_SYS_CONFIGURE *cmd_buf = NULL; TLVBUF_ANTENNA_CTL *tlv = NULL; u8 *buffer = NULL; u16 cmd_len; int ret = UAP_FAILURE; int opt; while ((opt = getopt_long(argc, argv, "+", cmd_options, NULL)) != -1) { switch (opt) { default: print_sys_cfg_antenna_ctl_usage(); return; } } argc -= optind; argv += optind; /* Check arguments */ if ((argc == 0) || (argc > 2)) { printf("ERR:wrong arguments.\n"); print_sys_cfg_antenna_ctl_usage(); return; } else if (argc == 1) { if ((ISDIGIT(argv[0]) == 0) || (atoi(argv[0]) < 0) || (atoi(argv[0]) > 1)) { printf ("ERR:Illegal ANTENNA parameter %s. Must be either '0' or '1'.\n", argv[0]); print_sys_cfg_antenna_ctl_usage(); return; } } else { if ((ISDIGIT(argv[0]) == 0) || (atoi(argv[0]) < 0) || (atoi(argv[0]) > 1)) { printf ("ERR:Illegal ANTENNA parameter %s. Must be either '0' or '1'.\n", argv[0]); print_sys_cfg_antenna_ctl_usage(); return; } if ((ISDIGIT(argv[1]) == 0) || (atoi(argv[1]) < 0) || (atoi(argv[1]) > 1)) { printf ("ERR:Illegal MODE parameter %s. Must be either '0' or '1'.\n", argv[1]); print_sys_cfg_antenna_ctl_usage(); return; } } /* Initialize the command length */ cmd_len = sizeof(APCMDBUF_SYS_CONFIGURE) + sizeof(TLVBUF_ANTENNA_CTL); /* Initialize the command buffer */ buffer = (u8 *) malloc(cmd_len); if (!buffer) { printf("ERR:Cannot allocate buffer for command!\n"); return; } bzero((char *) buffer, cmd_len); /* Locate headers */ cmd_buf = (APCMDBUF_SYS_CONFIGURE *) buffer; tlv = (TLVBUF_ANTENNA_CTL *) (buffer + sizeof(APCMDBUF_SYS_CONFIGURE)); /* Fill the command buffer */ cmd_buf->CmdCode = APCMD_SYS_CONFIGURE; cmd_buf->Size = cmd_len; cmd_buf->SeqNum = 0; cmd_buf->Result = 0; tlv->Tag = MRVL_ANTENNA_CTL_TLV_ID; tlv->Length = 2; tlv->WhichAntenna = (u8) atoi(argv[0]); if (argc == 1) { cmd_buf->Action = ACTION_GET; } else { cmd_buf->Action = ACTION_SET; tlv->AntennaMode = (u8) atoi(argv[1]); } endian_convert_tlv_header_out(tlv); /* Send the command */ ret = uap_ioctl((u8 *) cmd_buf, &cmd_len, cmd_len); endian_convert_tlv_header_in(tlv); /* Process response */ if (ret == UAP_SUCCESS) { /* Verify response */ if ((cmd_buf->CmdCode != (APCMD_SYS_CONFIGURE | APCMD_RESP_CHECK)) || (tlv->Tag != MRVL_ANTENNA_CTL_TLV_ID)) { printf("ERR:Corrupted response! CmdCode=%x, Tlv->Tag=%x\n", cmd_buf->CmdCode, tlv->Tag); free(buffer); return; } /* Print response */ if (cmd_buf->Result == CMD_SUCCESS) { if (argc == 1) { printf("%s antenna: %s\n", (tlv->WhichAntenna == 0) ? "Rx" : "Tx", (tlv->AntennaMode == 0) ? "A" : "B"); } else { printf("Antenna setting successful\n"); } } else { if (argc == 1) { printf("ERR:Could not get antenna!\n"); } else { printf("ERR:Could not set antenna!\n"); } } } else { printf("ERR:Command sending failed!\n"); } if (buffer) free(buffer); return; } /** * @brief Creates a sys_cfg request for RTS threshold * and sends to the driver * * Usage: "sys_cfg_rts_threshold [RTS_THRESHOLD]" * if RTS_THRESHOLD is provided, a 'set' is performed * else a 'get' is performed * * @param argc Number of arguments * @param argv Pointer to the arguments * @return N/A */ void apcmd_sys_cfg_rts_threshold(int argc, char *argv[]) { APCMDBUF_SYS_CONFIGURE *cmd_buf = NULL; TLVBUF_RTS_THRESHOLD *tlv = NULL; u8 *buffer = NULL; u16 cmd_len; int ret = UAP_FAILURE; int opt; while ((opt = getopt_long(argc, argv, "+", cmd_options, NULL)) != -1) { switch (opt) { default: print_sys_cfg_rts_threshold_usage(); return; } } argc -= optind; argv += optind; /* Check arguments */ if (argc && (is_input_valid(RTSTHRESH, argc, argv) != UAP_SUCCESS)) { print_sys_cfg_rts_threshold_usage(); return; } /* Initialize the command length */ cmd_len = sizeof(APCMDBUF_SYS_CONFIGURE) + sizeof(TLVBUF_RTS_THRESHOLD); /* Initialize the command buffer */ buffer = (u8 *) malloc(cmd_len); if (!buffer) { printf("ERR:Cannot allocate buffer for command!\n"); return; } bzero((char *) buffer, cmd_len); /* Locate headers */ cmd_buf = (APCMDBUF_SYS_CONFIGURE *) buffer; tlv = (TLVBUF_RTS_THRESHOLD *) (buffer + sizeof(APCMDBUF_SYS_CONFIGURE)); /* Fill the command buffer */ cmd_buf->CmdCode = APCMD_SYS_CONFIGURE; cmd_buf->Size = cmd_len; cmd_buf->SeqNum = 0; cmd_buf->Result = 0; tlv->Tag = MRVL_RTS_THRESHOLD_TLV_ID; tlv->Length = 2; if (argc == 0) { cmd_buf->Action = ACTION_GET; } else { cmd_buf->Action = ACTION_SET; tlv->RtsThreshold = (u16) atoi(argv[0]); } endian_convert_tlv_header_out(tlv); tlv->RtsThreshold = uap_cpu_to_le16(tlv->RtsThreshold); /* Send the command */ ret = uap_ioctl((u8 *) cmd_buf, &cmd_len, cmd_len); endian_convert_tlv_header_in(tlv); tlv->RtsThreshold = uap_le16_to_cpu(tlv->RtsThreshold); /* Process response */ if (ret == UAP_SUCCESS) { /* Verify response */ if ((cmd_buf->CmdCode != (APCMD_SYS_CONFIGURE | APCMD_RESP_CHECK)) || (tlv->Tag != MRVL_RTS_THRESHOLD_TLV_ID)) { printf("ERR:Corrupted response! CmdCode=%x, Tlv->Tag=%x\n", cmd_buf->CmdCode, tlv->Tag); free(buffer); return; } /* Print response */ if (cmd_buf->Result == CMD_SUCCESS) { if (argc == 0) { printf("RTS threshold = %d\n", tlv->RtsThreshold); } else { printf("RTS threshold setting successful\n"); } } else { if (argc == 0) { printf("ERR:Could not get RTS threshold!\n"); } else { printf("ERR:Could not set RTS threshold!\n"); } } } else { printf("ERR:Command sending failed!\n"); } if (buffer) free(buffer); return; } /** * @brief Creates a sys_cfg request for Fragmentation threshold * and sends to the driver * * Usage: "sys_cfg_frag_threshold [FRAG_THRESHOLD]" * if FRAG_THRESHOLD is provided, a 'set' is performed * else a 'get' is performed * * @param argc Number of arguments * @param argv Pointer to the arguments * @return N/A */ void apcmd_sys_cfg_frag_threshold(int argc, char *argv[]) { APCMDBUF_SYS_CONFIGURE *cmd_buf = NULL; TLVBUF_FRAG_THRESHOLD *tlv = NULL; u8 *buffer = NULL; u16 cmd_len; int ret = UAP_FAILURE; int opt; while ((opt = getopt_long(argc, argv, "+", cmd_options, NULL)) != -1) { switch (opt) { default: print_sys_cfg_frag_threshold_usage(); return; } } argc -= optind; argv += optind; /* Check arguments */ if (argc && (is_input_valid(FRAGTHRESH, argc, argv) != UAP_SUCCESS)) { print_sys_cfg_frag_threshold_usage(); return; } /* Initialize the command length */ cmd_len = sizeof(APCMDBUF_SYS_CONFIGURE) + sizeof(TLVBUF_FRAG_THRESHOLD); /* Initialize the command buffer */ buffer = (u8 *) malloc(cmd_len); if (!buffer) { printf("ERR:Cannot allocate buffer for command!\n"); return; } bzero((char *) buffer, cmd_len); /* Locate headers */ cmd_buf = (APCMDBUF_SYS_CONFIGURE *) buffer; tlv = (TLVBUF_FRAG_THRESHOLD *) (buffer + sizeof(APCMDBUF_SYS_CONFIGURE)); /* Fill the command buffer */ cmd_buf->CmdCode = APCMD_SYS_CONFIGURE; cmd_buf->Size = cmd_len; cmd_buf->SeqNum = 0; cmd_buf->Result = 0; tlv->Tag = MRVL_FRAG_THRESHOLD_TLV_ID; tlv->Length = 2; if (argc == 0) { cmd_buf->Action = ACTION_GET; } else { cmd_buf->Action = ACTION_SET; tlv->FragThreshold = (u16) atoi(argv[0]); } endian_convert_tlv_header_out(tlv); tlv->FragThreshold = uap_cpu_to_le16(tlv->FragThreshold); /* Send the command */ ret = uap_ioctl((u8 *) cmd_buf, &cmd_len, cmd_len); endian_convert_tlv_header_in(tlv); tlv->FragThreshold = uap_le16_to_cpu(tlv->FragThreshold); /* Process response */ if (ret == UAP_SUCCESS) { /* Verify response */ if ((cmd_buf->CmdCode != (APCMD_SYS_CONFIGURE | APCMD_RESP_CHECK)) || (tlv->Tag != MRVL_FRAG_THRESHOLD_TLV_ID)) { printf("ERR:Corrupted response! CmdCode=%x, Tlv->Tag=%x\n", cmd_buf->CmdCode, tlv->Tag); free(buffer); return; } /* Print response */ if (cmd_buf->Result == CMD_SUCCESS) { if (argc == 0) { printf("Fragmentation threshold = %d\n", tlv->FragThreshold); } else { printf("Fragmentation threshold setting successful\n"); } } else { if (argc == 1) { printf("ERR:Could not get Fragmentation threshold!\n"); } else { printf("ERR:Could not set Fragmentation threshold!\n"); } } } else { printf("ERR:Command sending failed!\n"); } if (buffer) free(buffer); return; } /** * @brief Creates a sys_cfg request for radio settings * and sends to the driver * * Usage: "sys_cfg_radio_ctl [0|1]" * * Options: 0 - Turn radio on * 1 - Turn radio off * empty - Get current radio setting * * @param argc Number of arguments * @param argv Pointer to the arguments * @return N/A */ void apcmd_sys_cfg_radio_ctl(int argc, char *argv[]) { APCMDBUF_SYS_CONFIGURE *cmd_buf = NULL; TLVBUF_RADIO_CTL *tlv = NULL; u8 *buffer = NULL; u16 cmd_len; int ret = UAP_FAILURE; int opt; while ((opt = getopt_long(argc, argv, "+", cmd_options, NULL)) != -1) { switch (opt) { default: print_sys_cfg_radio_ctl_usage(); return; } } argc -= optind; argv += optind; /* Check arguments */ if (argc && (is_input_valid(RADIOCONTROL, argc, argv) != UAP_SUCCESS)) { print_sys_cfg_radio_ctl_usage(); return; } /* Initialize the command length */ cmd_len = sizeof(APCMDBUF_SYS_CONFIGURE) + sizeof(TLVBUF_RADIO_CTL); /* Initialize the command buffer */ buffer = (u8 *) malloc(cmd_len); if (!buffer) { printf("ERR:Cannot allocate buffer for command!\n"); return; } bzero((char *) buffer, cmd_len); /* Locate headers */ cmd_buf = (APCMDBUF_SYS_CONFIGURE *) buffer; tlv = (TLVBUF_RADIO_CTL *) (buffer + sizeof(APCMDBUF_SYS_CONFIGURE)); /* Fill the command buffer */ cmd_buf->CmdCode = APCMD_SYS_CONFIGURE; cmd_buf->Size = cmd_len; cmd_buf->SeqNum = 0; cmd_buf->Result = 0; tlv->Tag = MRVL_RADIO_CTL_TLV_ID; tlv->Length = 1; if (argc == 0) { cmd_buf->Action = ACTION_GET; } else { cmd_buf->Action = ACTION_SET; tlv->RadioCtl = (u8) atoi(argv[0]); } endian_convert_tlv_header_out(tlv); /* Send the command */ ret = uap_ioctl((u8 *) cmd_buf, &cmd_len, cmd_len); endian_convert_tlv_header_in(tlv); /* Process response */ if (ret == UAP_SUCCESS) { /* Verify response */ if ((cmd_buf->CmdCode != (APCMD_SYS_CONFIGURE | APCMD_RESP_CHECK)) || (tlv->Tag != MRVL_RADIO_CTL_TLV_ID)) { printf("ERR:Corrupted response! CmdCode=%x, Tlv->Tag=%x\n", cmd_buf->CmdCode, tlv->Tag); free(buffer); return; } /* Print response */ if (cmd_buf->Result == CMD_SUCCESS) { if (argc == 0) { printf("Radio is %s\n", (tlv->RadioCtl == 0) ? "on" : "off"); } else { printf("Radio setting successful\n"); } } else { if (argc == 0) { printf("ERR:Could not get radio status!\n"); } else { printf("ERR:Could not set radio status!\n"); } } } else { printf("ERR:Command sending failed!\n"); } if (buffer) free(buffer); return; } /** * @brief Creates a sys_cfg request for RSN replay protection * and sends to the driver * * Usage: "sys_cfg_rsn_replay_prot [0|1]" * * Options: 0 - Disable RSN replay protection * 1 - Enable RSN replay protection * empty - Get current RSN replay protection setting * * @param argc Number of arguments * @param argv Pointer to the arguments * @return N/A */ void apcmd_sys_cfg_rsn_replay_prot(int argc, char *argv[]) { APCMDBUF_SYS_CONFIGURE *cmd_buf = NULL; tlvbuf_rsn_replay_prot *tlv = NULL; u8 *buffer = NULL; u16 cmd_len; int ret = UAP_FAILURE; int opt; while ((opt = getopt_long(argc, argv, "+", cmd_options, NULL)) != -1) { switch (opt) { default: print_sys_cfg_rsn_replay_prot_usage(); return; } } argc -= optind; argv += optind; /* Check arguments */ if (argc && is_input_valid(RSNREPLAYPROT, argc, argv) != UAP_SUCCESS) { print_sys_cfg_rsn_replay_prot_usage(); return; } /* Initialize the command length */ cmd_len = sizeof(APCMDBUF_SYS_CONFIGURE) + sizeof(tlvbuf_rsn_replay_prot); /* Initialize the command buffer */ buffer = (u8 *) malloc(cmd_len); if (!buffer) { printf("ERR:Cannot allocate buffer for command!\n"); return; } bzero((char *) buffer, cmd_len); /* Locate headers */ cmd_buf = (APCMDBUF_SYS_CONFIGURE *) buffer; tlv = (tlvbuf_rsn_replay_prot *) (buffer + sizeof(APCMDBUF_SYS_CONFIGURE)); /* Fill the command buffer */ cmd_buf->CmdCode = APCMD_SYS_CONFIGURE; cmd_buf->Size = cmd_len; cmd_buf->SeqNum = 0; cmd_buf->Result = 0; tlv->Tag = MRVL_RSN_REPLAY_PROT_TLV_ID; tlv->Length = 1; if (argc == 0) { cmd_buf->Action = ACTION_GET; } else { cmd_buf->Action = ACTION_SET; tlv->rsn_replay_prot = (u8) atoi(argv[0]); } endian_convert_tlv_header_out(tlv); /* Send the command */ ret = uap_ioctl((u8 *) cmd_buf, &cmd_len, cmd_len); endian_convert_tlv_header_in(tlv); /* Process response */ if (ret == UAP_SUCCESS) { /* Verify response */ if ((cmd_buf->CmdCode != (APCMD_SYS_CONFIGURE | APCMD_RESP_CHECK)) || (tlv->Tag != MRVL_RSN_REPLAY_PROT_TLV_ID)) { printf("ERR:Corrupted response! CmdCode=%x, Tlv->Tag=%x\n", cmd_buf->CmdCode, tlv->Tag); free(buffer); return; } /* Print response */ if (cmd_buf->Result == CMD_SUCCESS) { if (argc == 0) { printf("RSN replay protection is %s\n", (tlv->rsn_replay_prot == 1) ? "enabled" : "disabled"); } else { printf("RSN replay protection setting successful\n"); } } else { if (argc == 0) { printf("ERR:Could not get RSN replay protection !\n"); } else { printf("ERR:Could not set RSN replay protection !\n"); } } } else { printf("ERR:Command sending failed!\n"); } if (buffer) free(buffer); return; } /** * @brief Creates a sys_cfg request for MCBC data rates * and sends to the driver * * Usage: "sys_cfg_mcbc_data_rate [MCBC_DATA_RATE]" * * Options: 0 - Auto rate * >0 - Set specified MCBC data rate * empty - Get current MCBC data rate * * MCBC_DATA_RATE is represented in units of 500 kbps * * @param argc Number of arguments * @param argv Pointer to the arguments * @return N/A */ void apcmd_sys_cfg_mcbc_data_rate(int argc, char *argv[]) { APCMDBUF_SYS_CONFIGURE *cmd_buf = NULL; TLVBUF_MCBC_DATA_RATE *tlv = NULL; u8 *buffer = NULL; u16 cmd_len; int ret = UAP_FAILURE; int opt; while ((opt = getopt_long(argc, argv, "+", cmd_options, NULL)) != -1) { switch (opt) { default: print_sys_cfg_mcbc_data_rates_usage(); return; } } argc -= optind; argv += optind; /* Check arguments */ if (argc) { if (is_input_valid(MCBCDATARATE, argc, argv) != UAP_SUCCESS) { printf("ERR: Invalid input\n"); print_sys_cfg_mcbc_data_rates_usage(); return; } if ((A2HEXDECIMAL(argv[0]) != 0) && (is_mcbc_rate_valid(A2HEXDECIMAL(argv[0])) != UAP_SUCCESS)) { printf("ERR: invalid MCBC data rate."); print_sys_cfg_mcbc_data_rates_usage(); return; } } /* Initialize the command length */ cmd_len = sizeof(APCMDBUF_SYS_CONFIGURE) + sizeof(TLVBUF_MCBC_DATA_RATE); /* Initialize the command buffer */ buffer = (u8 *) malloc(cmd_len); if (!buffer) { printf("ERR:Cannot allocate buffer for command!\n"); return; } bzero((char *) buffer, cmd_len); /* Locate headers */ cmd_buf = (APCMDBUF_SYS_CONFIGURE *) buffer; tlv = (TLVBUF_MCBC_DATA_RATE *) (buffer + sizeof(APCMDBUF_SYS_CONFIGURE)); /* Fill the command buffer */ cmd_buf->CmdCode = APCMD_SYS_CONFIGURE; cmd_buf->Size = cmd_len; cmd_buf->SeqNum = 0; cmd_buf->Result = 0; tlv->Tag = MRVL_MCBC_DATA_RATE_TLV_ID; tlv->Length = 2; if (argc == 0) { cmd_buf->Action = ACTION_GET; } else { cmd_buf->Action = ACTION_SET; tlv->MCBCdatarate = (u16) A2HEXDECIMAL(argv[0]); } endian_convert_tlv_header_out(tlv); tlv->MCBCdatarate = uap_cpu_to_le16(tlv->MCBCdatarate); /* Send the command */ ret = uap_ioctl((u8 *) cmd_buf, &cmd_len, cmd_len); endian_convert_tlv_header_in(tlv); tlv->MCBCdatarate = uap_le16_to_cpu(tlv->MCBCdatarate); /* Process response */ if (ret == UAP_SUCCESS) { /* Verify response */ if ((cmd_buf->CmdCode != (APCMD_SYS_CONFIGURE | APCMD_RESP_CHECK)) || (tlv->Tag != MRVL_MCBC_DATA_RATE_TLV_ID)) { printf("ERR:Corrupted response! CmdCode=%x, Tlv->Tag=%x\n", cmd_buf->CmdCode, tlv->Tag); free(buffer); return; } /* Print response */ if (cmd_buf->Result == CMD_SUCCESS) { if (argc == 0) { if (tlv->MCBCdatarate == 0) { printf("MCBC data rate is auto\n"); } else { printf("MCBC data rate = 0x%x\n", tlv->MCBCdatarate); } } else { printf("MCBC data rate setting successful\n"); } } else { if (argc == 0) { printf("ERR:Could not get MCBC data rate!\n"); } else { printf("ERR:Could not set MCBC data rate!\n"); } } } else { printf("ERR:Command sending failed!\n"); } if (buffer) free(buffer); return; } /** * @brief Creates a sys_cfg request for Tx data rates * and sends to the driver * * Usage: "sys_cfg_tx_data_rate [TX_DATA_RATE]" * * Options: 0 - Auto rate * >0 - Set specified data rate * empty - Get current data rate * * TX_DATA_RATE is represented in units of 500 kbps * * @param argc Number of arguments * @param argv Pointer to the arguments * @return N/A */ void apcmd_sys_cfg_tx_data_rate(int argc, char *argv[]) { APCMDBUF_SYS_CONFIGURE *cmd_buf = NULL; TLVBUF_TX_DATA_RATE *tlv = NULL; u8 *buffer = NULL; u16 cmd_len; int ret = UAP_FAILURE; int opt; while ((opt = getopt_long(argc, argv, "+", cmd_options, NULL)) != -1) { switch (opt) { default: print_sys_cfg_tx_data_rates_usage(); return; } } argc -= optind; argv += optind; /* Check arguments */ if (argc) { if (is_input_valid(TXDATARATE, argc, argv) != UAP_SUCCESS) { printf("ERR: Invalid input\n"); print_sys_cfg_tx_data_rates_usage(); return; } else if ((A2HEXDECIMAL(argv[0]) != 0) && (is_tx_rate_valid(A2HEXDECIMAL(argv[0])) != UAP_SUCCESS)) { printf("ERR: invalid tx data rate."); print_sys_cfg_tx_data_rates_usage(); return; } } /* Initialize the command length */ cmd_len = sizeof(APCMDBUF_SYS_CONFIGURE) + sizeof(TLVBUF_TX_DATA_RATE); /* Initialize the command buffer */ buffer = (u8 *) malloc(cmd_len); if (!buffer) { printf("ERR:Cannot allocate buffer for command!\n"); return; } bzero((char *) buffer, cmd_len); /* Locate headers */ cmd_buf = (APCMDBUF_SYS_CONFIGURE *) buffer; tlv = (TLVBUF_TX_DATA_RATE *) (buffer + sizeof(APCMDBUF_SYS_CONFIGURE)); /* Fill the command buffer */ cmd_buf->CmdCode = APCMD_SYS_CONFIGURE; cmd_buf->Size = cmd_len; cmd_buf->SeqNum = 0; cmd_buf->Result = 0; tlv->Tag = MRVL_TX_DATA_RATE_TLV_ID; tlv->Length = 2; if (argc == 0) { cmd_buf->Action = ACTION_GET; } else { cmd_buf->Action = ACTION_SET; tlv->TxDataRate = (u16) A2HEXDECIMAL(argv[0]); } endian_convert_tlv_header_out(tlv); tlv->TxDataRate = uap_cpu_to_le16(tlv->TxDataRate); /* Send the command */ ret = uap_ioctl((u8 *) cmd_buf, &cmd_len, cmd_len); endian_convert_tlv_header_in(tlv); tlv->TxDataRate = uap_le16_to_cpu(tlv->TxDataRate); /* Process response */ if (ret == UAP_SUCCESS) { /* Verify response */ if ((cmd_buf->CmdCode != (APCMD_SYS_CONFIGURE | APCMD_RESP_CHECK)) || (tlv->Tag != MRVL_TX_DATA_RATE_TLV_ID)) { printf("ERR:Corrupted response! CmdCode=%x, Tlv->Tag=%x\n", cmd_buf->CmdCode, tlv->Tag); free(buffer); return; } /* Print response */ if (cmd_buf->Result == CMD_SUCCESS) { if (argc == 0) { if (tlv->TxDataRate == 0) { printf("Tx data rate is auto\n"); } else { printf("Tx data rate = 0x%x\n", tlv->TxDataRate); } } else { printf("Tx data rate setting successful\n"); } } else { if (argc == 0) { printf("ERR:Could not get tx data rate!\n"); } else { printf("ERR:Could not set tx data rate!\n"); } } } else { printf("ERR:Command sending failed!\n"); } if (buffer) free(buffer); return; } /** * @brief Creates a sys_cfg request for packet forwarding * and sends to the driver * * Usage: "sys_cfg_pkt_fwd_ctl [0|1]" * * Options: 0 - Forward all packets to the host * 1 - Firmware handles intra-BSS packets * empty - Get current packet forwarding setting * * @param argc Number of arguments * @param argv Pointer to the arguments * @return N/A */ void apcmd_sys_cfg_pkt_fwd_ctl(int argc, char *argv[]) { APCMDBUF_SYS_CONFIGURE *cmd_buf = NULL; TLVBUF_PKT_FWD_CTL *tlv = NULL; u8 *buffer = NULL; u16 cmd_len; int ret = UAP_FAILURE; int opt; while ((opt = getopt_long(argc, argv, "+", cmd_options, NULL)) != -1) { switch (opt) { default: print_sys_cfg_pkt_fwd_ctl_usage(); return; } } argc -= optind; argv += optind; /* Check arguments */ if (argc && (is_input_valid(PKTFWD, argc, argv) != UAP_SUCCESS)) { print_sys_cfg_pkt_fwd_ctl_usage(); return; } /* Initialize the command length */ cmd_len = sizeof(APCMDBUF_SYS_CONFIGURE) + sizeof(TLVBUF_PKT_FWD_CTL); /* Initialize the command buffer */ buffer = (u8 *) malloc(cmd_len); if (!buffer) { printf("ERR:Cannot allocate buffer for command!\n"); return; } bzero((char *) buffer, cmd_len); /* Locate headers */ cmd_buf = (APCMDBUF_SYS_CONFIGURE *) buffer; tlv = (TLVBUF_PKT_FWD_CTL *) (buffer + sizeof(APCMDBUF_SYS_CONFIGURE)); /* Fill the command buffer */ cmd_buf->CmdCode = APCMD_SYS_CONFIGURE; cmd_buf->Size = cmd_len; cmd_buf->SeqNum = 0; cmd_buf->Result = 0; tlv->Tag = MRVL_PKT_FWD_CTL_TLV_ID; tlv->Length = 1; if (argc == 0) { cmd_buf->Action = ACTION_GET; } else { cmd_buf->Action = ACTION_SET; tlv->PktFwdCtl = (u8) atoi(argv[0]); } endian_convert_tlv_header_out(tlv); /* Send the command */ ret = uap_ioctl((u8 *) cmd_buf, &cmd_len, cmd_len); endian_convert_tlv_header_in(tlv); /* Process response */ if (ret == UAP_SUCCESS) { /* Verify response */ if ((cmd_buf->CmdCode != (APCMD_SYS_CONFIGURE | APCMD_RESP_CHECK)) || (tlv->Tag != MRVL_PKT_FWD_CTL_TLV_ID)) { printf("ERR:Corrupted response! CmdCode=%x, Tlv->Tag=%x\n", cmd_buf->CmdCode, tlv->Tag); free(buffer); return; } /* Print response */ if (cmd_buf->Result == CMD_SUCCESS) { if (argc == 0) { printf("Firmware %s\n", (tlv->PktFwdCtl == 0) ? "forwards all packets to the host" : "handles intra-BSS packets"); } else { printf("Packet control logic setting successful\n"); } } else { if (argc == 0) { printf("ERR:Could not get packet control logic!\n"); } else { printf("ERR:Could not set packet control logic!\n"); } } } else { printf("ERR:Command sending failed!\n"); } if (buffer) free(buffer); return; } /** * @brief Creates a sys_cfg request for STA ageout timer * and sends to the driver * * Usage: "sys_cfg_sta_ageout_timer [STA_AGEOUT_TIMER]" * if STA_AGEOUT_TIMER is provided, a 'set' is performed * else a 'get' is performed. * The value should between 300 and 864000 * * STA_AGEOUT_TIMER is represented in units of 100 ms * * @param argc Number of arguments * @param argv Pointer to the arguments * @return N/A */ void apcmd_sys_cfg_sta_ageout_timer(int argc, char *argv[]) { APCMDBUF_SYS_CONFIGURE *cmd_buf = NULL; TLVBUF_STA_AGEOUT_TIMER *tlv = NULL; u8 *buffer = NULL; u16 cmd_len; int ret = UAP_FAILURE; int opt; while ((opt = getopt_long(argc, argv, "+", cmd_options, NULL)) != -1) { switch (opt) { default: print_sys_cfg_sta_ageout_timer_usage(); return; } } argc -= optind; argv += optind; /* Check arguments */ if (argc && (is_input_valid(STAAGEOUTTIMER, argc, argv) != UAP_SUCCESS)) { print_sys_cfg_sta_ageout_timer_usage(); return; } /* Initialize the command length */ cmd_len = sizeof(APCMDBUF_SYS_CONFIGURE) + sizeof(TLVBUF_STA_AGEOUT_TIMER); /* Initialize the command buffer */ buffer = (u8 *) malloc(cmd_len); if (!buffer) { printf("ERR:Cannot allocate buffer for command!\n"); return; } bzero((char *) buffer, cmd_len); /* Locate headers */ cmd_buf = (APCMDBUF_SYS_CONFIGURE *) buffer; tlv = (TLVBUF_STA_AGEOUT_TIMER *) (buffer + sizeof(APCMDBUF_SYS_CONFIGURE)); /* Fill the command buffer */ cmd_buf->CmdCode = APCMD_SYS_CONFIGURE; cmd_buf->Size = cmd_len; cmd_buf->SeqNum = 0; cmd_buf->Result = 0; tlv->Tag = MRVL_STA_AGEOUT_TIMER_TLV_ID; tlv->Length = 4; if (argc == 0) { cmd_buf->Action = ACTION_GET; } else { cmd_buf->Action = ACTION_SET; tlv->StaAgeoutTimer_ms = (u32) atoi(argv[0]); } endian_convert_tlv_header_out(tlv); tlv->StaAgeoutTimer_ms = uap_cpu_to_le32(tlv->StaAgeoutTimer_ms); /* Send the command */ ret = uap_ioctl((u8 *) cmd_buf, &cmd_len, cmd_len); endian_convert_tlv_header_in(tlv); tlv->StaAgeoutTimer_ms = uap_le32_to_cpu(tlv->StaAgeoutTimer_ms); /* Process response */ if (ret == UAP_SUCCESS) { /* Verify response */ if ((cmd_buf->CmdCode != (APCMD_SYS_CONFIGURE | APCMD_RESP_CHECK)) || (tlv->Tag != MRVL_STA_AGEOUT_TIMER_TLV_ID)) { printf("ERR:Corrupted response! CmdCode=%x, Tlv->Tag=%x\n", cmd_buf->CmdCode, tlv->Tag); free(buffer); return; } /* Print response */ if (cmd_buf->Result == CMD_SUCCESS) { if (argc == 0) { printf("STA ageout timer value = %d\n", (int) tlv->StaAgeoutTimer_ms); } else { printf("STA ageout timer setting successful\n"); } } else { if (argc == 0) { printf("ERR:Could not get STA ageout timer!\n"); } else { printf("ERR:Could not set STA ageout timer!\n"); } } } else { printf("ERR:Command sending failed!\n"); } if (buffer) free(buffer); return; } /** * @brief Creates a sys_cfg request for authentication mode * and sends to the driver * * Usage: "Usage : sys_cfg_auth [AUTHMODE]" * * Options: AUTHMODE : 0 - Open authentication * 1 - Shared key authentication * empty - Get current authentication mode * * @param argc Number of arguments * @param argv Pointer to the arguments * @return N/A */ void apcmd_sys_cfg_auth(int argc, char *argv[]) { APCMDBUF_SYS_CONFIGURE *cmd_buf = NULL; TLVBUF_AUTH_MODE *tlv = NULL; u8 *buffer = NULL; u16 cmd_len; int ret = UAP_FAILURE; int opt; while ((opt = getopt_long(argc, argv, "+", cmd_options, NULL)) != -1) { switch (opt) { default: print_sys_cfg_auth_usage(); return; } } argc -= optind; argv += optind; /* Check arguments */ if (argc && (is_input_valid(AUTHMODE, argc, argv) != UAP_SUCCESS)) { print_sys_cfg_auth_usage(); return; } /* Initialize the command length */ cmd_len = sizeof(APCMDBUF_SYS_CONFIGURE) + sizeof(TLVBUF_AUTH_MODE); /* Initialize the command buffer */ buffer = (u8 *) malloc(cmd_len); if (!buffer) { printf("ERR:Cannot allocate buffer for command!\n"); return; } bzero((char *) buffer, cmd_len); /* Locate headers */ cmd_buf = (APCMDBUF_SYS_CONFIGURE *) buffer; tlv = (TLVBUF_AUTH_MODE *) (buffer + sizeof(APCMDBUF_SYS_CONFIGURE)); /* Fill the command buffer */ cmd_buf->CmdCode = APCMD_SYS_CONFIGURE; cmd_buf->Size = cmd_len; cmd_buf->SeqNum = 0; cmd_buf->Result = 0; tlv->Tag = MRVL_AUTH_TLV_ID; tlv->Length = 1; if (argc == 0) { cmd_buf->Action = ACTION_GET; } else { cmd_buf->Action = ACTION_SET; tlv->AuthMode = (u8) atoi(argv[0]); } endian_convert_tlv_header_out(tlv); /* Send the command */ ret = uap_ioctl((u8 *) cmd_buf, &cmd_len, cmd_len); endian_convert_tlv_header_in(tlv); /* Process response */ if (ret == UAP_SUCCESS) { /* Verify response */ if ((cmd_buf->CmdCode != (APCMD_SYS_CONFIGURE | APCMD_RESP_CHECK)) || (tlv->Tag != MRVL_AUTH_TLV_ID)) { printf("ERR:Corrupted response! CmdCode=%x, Tlv->Tag=%x\n", cmd_buf->CmdCode, tlv->Tag); free(buffer); return; } /* Print response */ if (cmd_buf->Result == CMD_SUCCESS) { if (argc == 0) { print_auth(tlv); } else { printf("authentication mode setting successful\n"); } } else { if (argc == 0) { printf("ERR:Could not get authentication mode!\n"); } else { printf("ERR:Could not set authentication mode!\n"); } } } else { printf("ERR:Command sending failed!\n"); } if (buffer) free(buffer); return; } /** * @brief Creates a sys_cfg request for encryption protocol * and sends to the driver * * Usage: "Usage : sys_cfg_protocol [PROTOCOL]" * * Options: PROTOCOL Bit 0 - No RSN * Bit 1 - WEP Static * Bit 3 - WPA * Bit 3 - WPA2 * empty - Get current protocol * * @param argc Number of arguments * @param argv Pointer to the arguments * @return N/A */ void apcmd_sys_cfg_protocol(int argc, char *argv[]) { APCMDBUF_SYS_CONFIGURE *cmd_buf = NULL; TLVBUF_PROTOCOL *tlv = NULL; TLVBUF_AKMP *akmp_tlv = NULL; u8 *buffer = NULL; u16 cmd_len; int ret = UAP_FAILURE; int opt; while ((opt = getopt_long(argc, argv, "+", cmd_options, NULL)) != -1) { switch (opt) { default: print_sys_cfg_protocol_usage(); return; } } argc -= optind; argv += optind; /* Check arguments */ if (argc && is_input_valid(PROTOCOL, argc, argv) != UAP_SUCCESS) { print_sys_cfg_protocol_usage(); return; } /* Initialize the command length */ if ((argc == 1) && ((atoi(argv[0]) == PROTOCOL_NO_SECURITY) || (atoi(argv[0]) == PROTOCOL_STATIC_WEP))) { cmd_len = sizeof(APCMDBUF_SYS_CONFIGURE) + sizeof(TLVBUF_PROTOCOL); } else cmd_len = sizeof(APCMDBUF_SYS_CONFIGURE) + sizeof(TLVBUF_PROTOCOL) + sizeof(TLVBUF_AKMP); /* Initialize the command buffer */ buffer = (u8 *) malloc(cmd_len); if (!buffer) { printf("ERR:Cannot allocate buffer for command!\n"); return; } bzero((char *) buffer, cmd_len); /* Locate headers */ cmd_buf = (APCMDBUF_SYS_CONFIGURE *) buffer; tlv = (TLVBUF_PROTOCOL *) (buffer + sizeof(APCMDBUF_SYS_CONFIGURE)); akmp_tlv = (TLVBUF_AKMP *) (buffer + sizeof(APCMDBUF_SYS_CONFIGURE) + sizeof(TLVBUF_PROTOCOL)); /* Fill the command buffer */ cmd_buf->CmdCode = APCMD_SYS_CONFIGURE; cmd_buf->Size = cmd_len; cmd_buf->SeqNum = 0; cmd_buf->Result = 0; tlv->Tag = MRVL_PROTOCOL_TLV_ID; tlv->Length = 2; if (argc == 0) { cmd_buf->Action = ACTION_GET; akmp_tlv->Tag = MRVL_AKMP_TLV_ID; akmp_tlv->Length = 2; } else { cmd_buf->Action = ACTION_SET; tlv->Protocol = (u16) atoi(argv[0]); if (tlv->Protocol & (PROTOCOL_WPA | PROTOCOL_WPA2)) { akmp_tlv->Tag = MRVL_AKMP_TLV_ID; akmp_tlv->Length = 2; akmp_tlv->KeyMgmt = KEY_MGMT_PSK; akmp_tlv->KeyMgmt = uap_cpu_to_le16(akmp_tlv->KeyMgmt); } } endian_convert_tlv_header_out(tlv); if (cmd_len == (sizeof(APCMDBUF_SYS_CONFIGURE) + sizeof(TLVBUF_PROTOCOL) + sizeof(TLVBUF_AKMP))) endian_convert_tlv_header_out(akmp_tlv); tlv->Protocol = uap_cpu_to_le16(tlv->Protocol); /* Send the command */ ret = uap_ioctl((u8 *) cmd_buf, &cmd_len, cmd_len); endian_convert_tlv_header_in(tlv); if (cmd_len == (sizeof(APCMDBUF_SYS_CONFIGURE) + sizeof(TLVBUF_PROTOCOL) + sizeof(TLVBUF_AKMP))) endian_convert_tlv_header_in(akmp_tlv); /* Process response */ if (ret == UAP_SUCCESS) { /* Verify response */ if (cmd_buf->CmdCode != (APCMD_SYS_CONFIGURE | APCMD_RESP_CHECK)) { printf("ERR:Corrupted response! CmdCode=%x, Tlv->Tag=%x\n", cmd_buf->CmdCode, tlv->Tag); free(buffer); return; } /* Print response */ if (cmd_buf->Result == CMD_SUCCESS) { if (argc == 0) { print_tlv((u8 *) tlv, cmd_buf->Size - sizeof(APCMDBUF_SYS_CONFIGURE) + BUF_HEADER_SIZE); } else { printf("protocol setting successful\n"); } } else { if (argc == 0) { printf("ERR:Could not get protocol!\n"); } else { printf("ERR:Could not set protocol!\n"); } } } else { printf("ERR:Command sending failed!\n"); } if (buffer) free(buffer); return; } /** * @brief Creates a sys_cfg request for WEP keys settings * and sends to the driver * * Usage: "sys_cfg_wep_key [INDEX_0 IS_DEFAULT KEY_0] [INDEX_1 IS_DEFAULT KEY_1] [INDEX_2 IS_DEFAULT KEY_2] [INDEX_3 IS_DEFAULT KEY_3]" * * Options: INDEX_* : 0 - KeyIndex is 0 * 1 - KeyIndex is 1 * 2 - KeyIndex is 2 * 3 - KeyIndex is 3 * IS_DEFAUL : 0 - KeyIndex is not the default * 1 - KeyIndex is the default transmit key * KEY_* : Key value * empty - Get current WEP key settings * * @param argc Number of arguments * @param argv Pointer to the arguments * @return N/A */ void apcmd_sys_cfg_wep_key(int argc, char *argv[]) { APCMDBUF_SYS_CONFIGURE *cmd_buf = NULL; TLVBUF_WEP_KEY *tlv = NULL; u8 *buffer = NULL; u16 cmd_len; u16 buf_len; int ret = UAP_FAILURE; int key_len = -1; int length = 0; int number_of_keys = 0; int i = 0; int keyindex = -1; int is_default = -1; char *key = NULL; int opt; while ((opt = getopt_long(argc, argv, "+", cmd_options, NULL)) != -1) { switch (opt) { default: print_sys_cfg_wep_key_usage(); return; } } argc -= optind; argv += optind; /* Check arguments */ if (argc > 12) { printf("ERR:Too many arguments.\n"); print_sys_cfg_wep_key_usage(); return; } else if ((argc > 1) && ((argc % 3) != 0)) { printf("ERR:Illegal number of parameters.\n"); print_sys_cfg_wep_key_usage(); return; } else if (argc > 1) { /* Find number of keys provided */ number_of_keys = argc / 3; for (i = 0; i < number_of_keys; i++) { if ((ISDIGIT(argv[(3 * i)]) == 0) || (atoi(argv[(3 * i)]) < 0) || (atoi(argv[(3 * i)]) > 3)) { printf ("ERR:Illegal INDEX %s. Must be either '0', '1', '2', or '3'.\n", argv[(3 * i)]); print_sys_cfg_wep_key_usage(); return; } if ((ISDIGIT(argv[(3 * i) + 1]) == 0) || (atoi(argv[(3 * i) + 1]) < 0) || (atoi(argv[(3 * i) + 1]) > 1)) { printf ("ERR:Illegal IS_DEFAULT %s. Must be either '0', or '1'.\n", argv[(3 * i) + 1]); print_sys_cfg_wep_key_usage(); return; } if ((strlen(argv[(3 * i) + 2]) != 5) && (strlen(argv[(3 * i) + 2]) != 10) && (strlen(argv[(3 * i) + 2]) != 13) && (strlen(argv[(3 * i) + 2]) != 26)) { printf("ERR:Incorrect KEY_%d length %d\n", atoi(argv[(3 * i)]), strlen(argv[(3 * i) + 2])); print_sys_cfg_wep_key_usage(); return; } if ((strlen(argv[(3 * i) + 2]) == 10) || (strlen(argv[(3 * i) + 2]) == 26)) { if (UAP_FAILURE == ishexstring(argv[(3 * i) + 2])) { printf ("ERR:Only hex digits are allowed when key length is 10 or 26\n"); print_sys_cfg_wep_key_usage(); return; } } } } else if (argc == 1) { if ((ISDIGIT(argv[0]) == 0) || (atoi(argv[0]) < 0) || (atoi(argv[0]) > 3)) { printf ("ERR:Illegal INDEX %s. Must be either '0', '1', '2', or '3'.\n", argv[0]); print_sys_cfg_wep_key_usage(); return; } } /* Initialize the command length */ if (argc == 0 || argc == 1) { if (argc == 0) cmd_len = sizeof(APCMDBUF_SYS_CONFIGURE) + sizeof(TLVBUF_HEADER); else cmd_len = sizeof(APCMDBUF_SYS_CONFIGURE) + sizeof(TLVBUF_WEP_KEY) - 1; buf_len = MRVDRV_SIZE_OF_CMD_BUFFER; } else { buf_len = cmd_len = sizeof(APCMDBUF_SYS_CONFIGURE); } /* Initialize the command buffer */ buffer = (u8 *) malloc(buf_len); if (!buffer) { printf("ERR:Cannot allocate buffer for command!\n"); return; } bzero((char *) buffer, buf_len); /* Locate headers */ cmd_buf = (APCMDBUF_SYS_CONFIGURE *) buffer; /* Fill the command buffer */ cmd_buf->CmdCode = APCMD_SYS_CONFIGURE; cmd_buf->SeqNum = 0; cmd_buf->Result = 0; if (argc == 0 || argc == 1) { cmd_buf->Action = ACTION_GET; tlv = (TLVBUF_WEP_KEY *) (buffer + sizeof(APCMDBUF_SYS_CONFIGURE)); tlv->Tag = MRVL_WEP_KEY_TLV_ID; if (argc == 0) tlv->Length = 0; else { tlv->Length = 1; tlv->KeyIndex = atoi(argv[0]); } endian_convert_tlv_header_out(tlv); } else { cmd_buf->Action = ACTION_SET; } /* Add key TLVs */ for (i = 0; i < number_of_keys; i++) { keyindex = atoi(argv[(3 * i)]); is_default = atoi(argv[(3 * i) + 1]); key = argv[(3 * i) + 2]; length = strlen(key); switch (length) { case 5: case 10: key_len = 5; break; case 13: case 26: key_len = 13; break; default: key_len = 0; break; } /* Adjust command buffer */ buffer = realloc(buffer, (cmd_len + sizeof(TLVBUF_WEP_KEY) + key_len)); if (!buffer) { printf("ERR:Cannot append WEP key configurations TLV!\n"); return; } /* Locate headers */ cmd_buf = (APCMDBUF_SYS_CONFIGURE *) buffer; tlv = (TLVBUF_WEP_KEY *) (buffer + cmd_len); /* Adjust command length */ cmd_len += (sizeof(TLVBUF_WEP_KEY) + key_len); /* Set TLV fields */ tlv->Tag = MRVL_WEP_KEY_TLV_ID; tlv->Length = 2 + key_len; tlv->KeyIndex = (u8) keyindex; tlv->IsDefault = (u8) is_default; /* Check if string or raw */ switch (length) { case 5: case 13: memcpy(tlv->Key, key, length); break; case 10: case 26: string2raw(key, tlv->Key); break; default: break; } endian_convert_tlv_header_out(tlv); } /* Update command length */ cmd_buf->Size = cmd_len; if ((argc != 0) && (argc != 1)) buf_len = cmd_len; /* Send the command */ ret = uap_ioctl((u8 *) cmd_buf, &cmd_len, buf_len); /* Process response */ if (ret == UAP_SUCCESS) { /* Verify response */ if (cmd_buf->CmdCode != (APCMD_SYS_CONFIGURE | APCMD_RESP_CHECK)) { printf("ERR:Corrupted response!\n"); free(buffer); return; } /* Print response */ if (cmd_buf->Result == CMD_SUCCESS) { if ((argc != 0) && (argc != 1)) { printf("WEP key setting successful\n"); } else { printf("query WEP key setting successful\n"); tlv = (TLVBUF_WEP_KEY *) (buffer + sizeof(APCMDBUF_SYS_CONFIGURE)); print_tlv((u8 *) tlv, cmd_buf->Size - sizeof(APCMDBUF_SYS_CONFIGURE) + BUF_HEADER_SIZE); } } else { if ((argc != 0) && (argc != 1)) printf("ERR:Could not set WEP keys!\n"); else printf("ERR:Could not get WEP keys!\n"); } } else { printf("ERR:Command sending failed!\n"); } if (buffer) free(buffer); return; } /** * @brief Creates a sys_cfg request for custom IE settings * and sends to the driver * * Usage: "sys_cfg_custom_ie [INDEX] [MASK] [IEBuffer]" * * Options: INDEX : 0 - Get/Set IE index 0 setting * 1 - Get/Set IE index 1 setting * 2 - Get/Set IE index 2 setting * 3 - Get/Set IE index 3 setting * MASK : Management subtype mask value * IEBuffer: IE Buffer in hex * empty - Get all IE settings * * @param argc Number of arguments * @param argv Pointer to the arguments * @return N/A */ void apcmd_sys_cfg_custom_ie(int argc, char *argv[]) { APCMDBUF_SYS_CONFIGURE *cmd_buf = NULL; tlvbuf_custom_ie *tlv = NULL; custom_ie *ie_ptr = NULL; u8 *buffer = NULL; u16 cmd_len; u16 buf_len; u16 mgmt_subtype_mask = 0; int ret = UAP_FAILURE; int ie_buf_len = 0, ie_len = 0; int opt; while ((opt = getopt_long(argc, argv, "+", cmd_options, NULL)) != -1) { switch (opt) { default: print_sys_cfg_custom_ie_usage(); return; } } argc -= optind; argv += optind; /* Check arguments */ if (argc > 3) { printf("ERR:Too many arguments.\n"); print_sys_cfg_custom_ie_usage(); return; } /* Error checks and initialize the command length */ if (argc >= 1) { if ((ISDIGIT(argv[0]) == 0) || (atoi(argv[0]) < 0) || (atoi(argv[0]) > 3)) { printf ("ERR:Illegal index %s. Must be either '0', '1', '2', or '3'.\n", argv[0]); print_sys_cfg_custom_ie_usage(); return; } } switch (argc) { case 0: cmd_len = sizeof(APCMDBUF_SYS_CONFIGURE) + sizeof(tlvbuf_custom_ie); buf_len = MRVDRV_SIZE_OF_CMD_BUFFER; break; case 1: /* TLV header + ie_index */ cmd_len = sizeof(APCMDBUF_SYS_CONFIGURE) + sizeof(tlvbuf_custom_ie) + sizeof(u16); buf_len = MRVDRV_SIZE_OF_CMD_BUFFER; break; case 2: if (UAP_FAILURE == ishexstring(argv[1]) || A2HEXDECIMAL(argv[1]) != 0) { printf("ERR: Mask value should be 0 to clear IEBuffers.\n"); print_sys_cfg_custom_ie_usage(); return; } buf_len = cmd_len = sizeof(APCMDBUF_SYS_CONFIGURE) + sizeof(tlvbuf_custom_ie) + sizeof(custom_ie); break; case 3: if (UAP_FAILURE == ishexstring(argv[1]) || A2HEXDECIMAL(argv[1]) == 0) { printf("ERR: Mask value should not be 0 to set IEBuffers.\n"); print_sys_cfg_custom_ie_usage(); return; } if (UAP_FAILURE == ishexstring(argv[2])) { printf("ERR:Only hex digits are allowed\n"); print_sys_cfg_custom_ie_usage(); return; } ie_buf_len = strlen(argv[2]); if (!strncasecmp("0x", argv[2], 2)) { ie_len = (ie_buf_len - 2 + 1) / 2; argv[2] += 2; } else ie_len = (ie_buf_len + 1) / 2; if (ie_len > MAX_IE_BUFFER_LEN) { printf("ERR:Incorrect IE length %d\n", ie_buf_len); print_sys_cfg_custom_ie_usage(); return; } mgmt_subtype_mask = (u16) A2HEXDECIMAL(argv[1]); buf_len = cmd_len = sizeof(APCMDBUF_SYS_CONFIGURE) + sizeof(tlvbuf_custom_ie) + sizeof(custom_ie) + ie_len; break; } /* Initialize the command buffer */ buffer = (u8 *) malloc(buf_len); if (!buffer) { printf("ERR:Cannot allocate buffer for command!\n"); return; } memset(buffer, 0, buf_len); /* Locate headers */ cmd_buf = (APCMDBUF_SYS_CONFIGURE *) buffer; /* Fill the command buffer */ cmd_buf->CmdCode = APCMD_SYS_CONFIGURE; cmd_buf->SeqNum = 0; cmd_buf->Result = 0; cmd_buf->Size = cmd_len; if (argc == 0 || argc == 1) { cmd_buf->Action = ACTION_GET; tlv = (tlvbuf_custom_ie *) (buffer + sizeof(APCMDBUF_SYS_CONFIGURE)); tlv->Tag = MRVL_MGMT_IE_LIST_TLV_ID; if (argc == 0) tlv->Length = 0; else { tlv->Length = sizeof(u16); ie_ptr = (custom_ie *) (tlv->ie_data); ie_ptr->ie_index = (u16) uap_cpu_to_le16(atoi(argv[0])); } endian_convert_tlv_header_out(tlv); } else { cmd_buf->Action = ACTION_SET; /* Locate headers */ tlv = (tlvbuf_custom_ie *) (buffer + sizeof(APCMDBUF_SYS_CONFIGURE)); ie_ptr = (custom_ie *) (tlv->ie_data); /* Set TLV fields */ tlv->Tag = MRVL_MGMT_IE_LIST_TLV_ID; tlv->Length = sizeof(custom_ie) + ie_len; ie_ptr->ie_index = uap_cpu_to_le16(atoi(argv[0])); ie_ptr->mgmt_subtype_mask = uap_cpu_to_le16(mgmt_subtype_mask); ie_ptr->ie_length = uap_cpu_to_le16(ie_len); if (argc == 3) string2raw(argv[2], ie_ptr->ie_buffer); endian_convert_tlv_header_out(tlv); } /* Send the command */ ret = uap_ioctl((u8 *) cmd_buf, &cmd_len, buf_len); /* Process response */ if (ret == UAP_SUCCESS) { /* Verify response */ if (cmd_buf->CmdCode != (APCMD_SYS_CONFIGURE | APCMD_RESP_CHECK)) { printf("ERR:Corrupted response!\n"); free(buffer); return; } /* Print response */ if (cmd_buf->Result == CMD_SUCCESS) { if ((argc != 0) && (argc != 1)) { printf("custom IE setting successful\n"); } else { printf("Querying custom IE successful\n"); tlv = (tlvbuf_custom_ie *) (buffer + sizeof(APCMDBUF_SYS_CONFIGURE)); print_tlv((u8 *) tlv, cmd_buf->Size - sizeof(APCMDBUF_SYS_CONFIGURE) + BUF_HEADER_SIZE); } } else { if ((argc != 0) && (argc != 1)) printf("ERR:Could not set custom IE elements!\n"); else printf("ERR:Could not get custom IE elements!\n"); } } else { printf("ERR:Command sending failed!\n"); } if (buffer) free(buffer); return; } /** * @brief Creates a sys_cfg request for cipher configurations * and sends to the driver * * Usage: "sys_cfg_security_cfg [PAIRWISE_CIPHER GROUP_CIPHER]" * * Options: PAIRWISE_CIPHER : Bit 2 - TKIP * Bit 3 - AES CCMP * GROUP_CIPHER : Bit 2 - TKIP * Bit 3 - AES CCMP * empty - Get current cipher settings * * @param argc Number of arguments * @param argv Pointer to the arguments * @return N/A */ void apcmd_sys_cfg_cipher(int argc, char *argv[]) { APCMDBUF_SYS_CONFIGURE *cmd_buf = NULL; TLVBUF_CIPHER *tlv = NULL; u8 *buffer = NULL; u16 cmd_len; int ret = UAP_FAILURE; int opt; while ((opt = getopt_long(argc, argv, "+", cmd_options, NULL)) != -1) { switch (opt) { default: print_sys_cfg_cipher_usage(); return; } } argc -= optind; argv += optind; /* Check arguments */ if ((argc != 0) && (argc != 2)) { printf("ERR:wrong arguments.\n"); print_sys_cfg_cipher_usage(); return; } if (argc == 2) { if ((ISDIGIT(argv[0]) == 0) || (ISDIGIT(argv[1]) == 0)) { print_sys_cfg_cipher_usage(); return; } if (atoi(argv[0]) & ~CIPHER_BITMAP) { printf("ERR:Illegal PAIRWISE_CIPHER parameter %s.\n", argv[0]); print_sys_cfg_cipher_usage(); return; } if (atoi(argv[1]) & ~CIPHER_BITMAP) { printf("ERR:Illegal GROUP_CIPHER parameter %s.\n", argv[1]); print_sys_cfg_cipher_usage(); return; } if (is_cipher_valid(atoi(argv[0]), atoi(argv[1])) != UAP_SUCCESS) { printf("ERR:Wrong group and pair cipher combination!\n"); print_sys_cfg_cipher_usage(); return; } } /* Initialize the command length */ cmd_len = sizeof(APCMDBUF_SYS_CONFIGURE) + sizeof(TLVBUF_CIPHER); /* Initialize the command buffer */ buffer = (u8 *) malloc(cmd_len); if (!buffer) { printf("ERR:Cannot allocate buffer for command!\n"); return; } bzero((char *) buffer, cmd_len); /* Locate headers */ cmd_buf = (APCMDBUF_SYS_CONFIGURE *) buffer; tlv = (TLVBUF_CIPHER *) (buffer + sizeof(APCMDBUF_SYS_CONFIGURE)); /* Fill the command buffer */ cmd_buf->CmdCode = APCMD_SYS_CONFIGURE; cmd_buf->Size = cmd_len; cmd_buf->SeqNum = 0; cmd_buf->Result = 0; tlv->Tag = MRVL_CIPHER_TLV_ID; tlv->Length = 2; if (argc == 0) { cmd_buf->Action = ACTION_GET; } else { cmd_buf->Action = ACTION_SET; tlv->PairwiseCipher = (u8) atoi(argv[0]); tlv->GroupCipher = (u8) atoi(argv[1]); } endian_convert_tlv_header_out(tlv); /* Send the command */ ret = uap_ioctl((u8 *) cmd_buf, &cmd_len, cmd_len); endian_convert_tlv_header_in(tlv); /* Process response */ if (ret == UAP_SUCCESS) { /* Verify response */ if ((cmd_buf->CmdCode != (APCMD_SYS_CONFIGURE | APCMD_RESP_CHECK)) || (tlv->Tag != MRVL_CIPHER_TLV_ID)) { printf("ERR:Corrupted response! CmdCode=%x, Tlv->Tag=%x\n", cmd_buf->CmdCode, tlv->Tag); free(buffer); return; } /* Print response */ if (cmd_buf->Result == CMD_SUCCESS) { if (argc == 0) { print_cipher(tlv); } else { printf("cipher setting successful\n"); } } else { if (argc == 0) { printf("ERR:Could not get cipher parameters!\n"); } else { printf("ERR:Could not set cipher parameters!\n"); } } } else { printf("ERR:Command sending failed!\n"); } if (buffer) free(buffer); return; } /** * @brief Creates a sys_cfg request for group re-key timer * and sends to the driver * * Usage: "Usage : sys_cfg_group_rekey_timer [GROUP_REKEY_TIMER]" * * Options: GROUP_REKEY_TIME is represented in seconds * Get current group re-key timer * * @param argc Number of arguments * @param argv Pointer to the arguments * @return N/A */ void apcmd_sys_cfg_group_rekey_timer(int argc, char *argv[]) { APCMDBUF_SYS_CONFIGURE *cmd_buf = NULL; TLVBUF_GROUP_REKEY_TIMER *tlv = NULL; u8 *buffer = NULL; u16 cmd_len; int ret = UAP_FAILURE; int opt; while ((opt = getopt_long(argc, argv, "+", cmd_options, NULL)) != -1) { switch (opt) { default: print_sys_cfg_group_rekey_timer_usage(); return; } } argc -= optind; argv += optind; /* Check arguments */ if (argc && (is_input_valid(GROUPREKEYTIMER, argc, argv) != UAP_SUCCESS)) { print_sys_cfg_group_rekey_timer_usage(); } /* Initialize the command length */ cmd_len = sizeof(APCMDBUF_SYS_CONFIGURE) + sizeof(TLVBUF_GROUP_REKEY_TIMER); /* Initialize the command buffer */ buffer = (u8 *) malloc(cmd_len); if (!buffer) { printf("ERR:Cannot allocate buffer for command!\n"); return; } bzero((char *) buffer, cmd_len); /* Locate headers */ cmd_buf = (APCMDBUF_SYS_CONFIGURE *) buffer; tlv = (TLVBUF_GROUP_REKEY_TIMER *) (buffer + sizeof(APCMDBUF_SYS_CONFIGURE)); /* Fill the command buffer */ cmd_buf->CmdCode = APCMD_SYS_CONFIGURE; cmd_buf->Size = cmd_len; cmd_buf->SeqNum = 0; cmd_buf->Result = 0; tlv->Tag = MRVL_GRP_REKEY_TIME_TLV_ID; tlv->Length = 4; if (argc == 0) { cmd_buf->Action = ACTION_GET; } else { cmd_buf->Action = ACTION_SET; tlv->GroupRekeyTime_sec = (u32) atoi(argv[0]); } endian_convert_tlv_header_out(tlv); tlv->GroupRekeyTime_sec = uap_cpu_to_le32(tlv->GroupRekeyTime_sec); /* Send the command */ ret = uap_ioctl((u8 *) cmd_buf, &cmd_len, cmd_len); endian_convert_tlv_header_in(tlv); tlv->GroupRekeyTime_sec = uap_le32_to_cpu(tlv->GroupRekeyTime_sec); /* Process response */ if (ret == UAP_SUCCESS) { /* Verify response */ if ((cmd_buf->CmdCode != (APCMD_SYS_CONFIGURE | APCMD_RESP_CHECK)) || (tlv->Tag != MRVL_GRP_REKEY_TIME_TLV_ID)) { printf("ERR:Corrupted response! CmdCode=%x, Tlv->Tag=%x\n", cmd_buf->CmdCode, tlv->Tag); free(buffer); return; } /* Print response */ if (cmd_buf->Result == CMD_SUCCESS) { if (argc == 0) { if (tlv->GroupRekeyTime_sec > 0) printf("Group rekey time is %ld s\n", tlv->GroupRekeyTime_sec); else printf("Group rekey time is disabled\n"); } else { printf("group re-key time setting successful\n"); } } else { if (argc == 0) { printf("ERR:Could not get group re-key time!\n"); } else { printf("ERR:Could not set group re-key time!\n"); } } } else { printf("ERR:Command sending failed!\n"); } if (buffer) free(buffer); return; } /** * @brief Creates a sys_cfg request for WPA passphrase * and sends to the driver * * Usage: "sys_cfg_wpa_passphrase [PASSPHRASE]" * if PASSPHRASE is provided, a 'set' is performed * else a 'get' is performed * * @param argc Number of arguments * @param argv Pointer to the arguments * @return N/A */ void apcmd_sys_cfg_wpa_passphrase(int argc, char *argv[]) { APCMDBUF_SYS_CONFIGURE *cmd_buf = NULL; TLVBUF_WPA_PASSPHRASE *tlv = NULL; u8 *buffer = NULL; u16 cmd_len; int ret = UAP_FAILURE; int opt; while ((opt = getopt_long(argc, argv, "+", cmd_options, NULL)) != -1) { switch (opt) { default: print_sys_cfg_wpa_passphrase_usage(); return; } } argc -= optind; argv += optind; /* Check arguments */ if (argc > 1) { printf("ERR:Too many arguments.\n"); print_sys_cfg_wpa_passphrase_usage(); return; } if ((argc == 1) && (strlen(argv[0]) > MAX_WPA_PASSPHRASE_LENGTH)) { printf("ERR:PASSPHRASE too long.\n"); return; } if ((argc == 1) && (strlen(argv[0]) < MIN_WPA_PASSPHRASE_LENGTH)) { printf("ERR:PASSPHRASE too short.\n"); return; } if ((argc == 1) && (strlen(argv[0]) == MAX_WPA_PASSPHRASE_LENGTH)) { if (UAP_FAILURE == ishexstring(argv[0])) { printf ("ERR:Only hex digits are allowed when passphrase's length is 64\n"); return; } } /* Initialize the command length */ if (argc == 0) cmd_len = sizeof(APCMDBUF_SYS_CONFIGURE) + sizeof(TLVBUF_WPA_PASSPHRASE) + MAX_WPA_PASSPHRASE_LENGTH; else cmd_len = sizeof(APCMDBUF_SYS_CONFIGURE) + sizeof(TLVBUF_WPA_PASSPHRASE) + strlen(argv[0]); /* Initialize the command buffer */ buffer = (u8 *) malloc(cmd_len); if (!buffer) { printf("ERR:Cannot allocate buffer for command!\n"); return; } bzero((char *) buffer, cmd_len); /* Locate headers */ cmd_buf = (APCMDBUF_SYS_CONFIGURE *) buffer; tlv = (TLVBUF_WPA_PASSPHRASE *) (buffer + sizeof(APCMDBUF_SYS_CONFIGURE)); /* Fill the command buffer */ cmd_buf->CmdCode = APCMD_SYS_CONFIGURE; cmd_buf->Size = cmd_len; cmd_buf->SeqNum = 0; cmd_buf->Result = 0; tlv->Tag = MRVL_WPA_PASSPHRASE_TLV_ID; if (argc == 0) { cmd_buf->Action = ACTION_GET; tlv->Length = MAX_WPA_PASSPHRASE_LENGTH; } else { cmd_buf->Action = ACTION_SET; tlv->Length = strlen(argv[0]); memcpy(tlv->Passphrase, argv[0], tlv->Length); } endian_convert_tlv_header_out(tlv); /* Send the command */ ret = uap_ioctl((u8 *) cmd_buf, &cmd_len, cmd_len); endian_convert_tlv_header_in(tlv); /* Process response */ if (ret == UAP_SUCCESS) { /* Verify response */ if ((cmd_buf->CmdCode != (APCMD_SYS_CONFIGURE | APCMD_RESP_CHECK)) || (tlv->Tag != MRVL_WPA_PASSPHRASE_TLV_ID)) { printf("ERR:Corrupted response! CmdCode=%x, Tlv->Tag=%x\n", cmd_buf->CmdCode, tlv->Tag); free(buffer); return; } /* Print response */ if (cmd_buf->Result == CMD_SUCCESS) { if (argc == 0) { if (tlv->Length > 0) printf("WPA passphrase = %s\n", tlv->Passphrase); else printf("WPA passphrase: None\n"); } else { printf("WPA passphrase setting successful\n"); } } else { if (argc == 0) { printf("ERR:Could not get WPA passphrase!\n"); } else { printf("ERR:Could not set WPA passphrase!\n"); } } } else { printf("ERR:Command sending failed!\n"); } if (buffer) free(buffer); return; } /** * @brief Creates a STA filter request and sends to the driver * * Usage: "sta_filter_table " * * Options: FILTERMODE : 0 - Disable filter table * 1 - Allow mac address specified in the allwed list * 2 - Block MAC addresses specified in the banned list * MACADDRESS_LIST is the list of MAC addresses to be acted upon. Each * MAC address must be separated with a space. Maximum of * 16 MAC addresses are supported. * * @param argc Number of arguments * @param argv Pointer to the arguments * @return N/A */ void apcmd_sta_filter_table(int argc, char *argv[]) { APCMDBUF_SYS_CONFIGURE *cmd_buf = NULL; TLVBUF_STA_MAC_ADDR_FILTER *tlv = NULL; u8 *buffer = NULL; u16 cmd_len; int ret = UAP_FAILURE; int i = 0; int opt; while ((opt = getopt_long(argc, argv, "+", cmd_options, NULL)) != -1) { switch (opt) { default: print_sta_filter_table_usage(); return; } } argc -= optind; argv += optind; /* Check arguments */ if (argc > (MAX_MAC_ONESHOT_FILTER + 1)) { printf("ERR:Too many arguments.\n"); print_sta_filter_table_usage(); return; } if (argc > 0) { if ((ISDIGIT(argv[0]) == 0) || ((atoi(argv[0]) < 0) || (atoi(argv[0]) > 2))) { printf ("ERR:Illegal FILTERMODE parameter %s. Must be either '0', '1', or '2'.\n", argv[1]); print_sta_filter_table_usage(); return; } if ((atoi(argv[0]) != 0) && (argc == 1)) { printf("ERR:At least one mac is required.\n"); print_sta_filter_table_usage(); return; } } /* Initialize the command length */ if (argc == 0) { cmd_len = sizeof(APCMDBUF_SYS_CONFIGURE) + sizeof(TLVBUF_STA_MAC_ADDR_FILTER) + (MAX_MAC_ONESHOT_FILTER * ETH_ALEN); } else { cmd_len = sizeof(APCMDBUF_SYS_CONFIGURE) + sizeof(TLVBUF_STA_MAC_ADDR_FILTER) + (argc - 1) * ETH_ALEN; } /* Initialize the command buffer */ buffer = (u8 *) malloc(cmd_len); if (!buffer) { printf("ERR:Cannot allocate buffer for command!\n"); return; } bzero((char *) buffer, cmd_len); /* Locate headers */ cmd_buf = (APCMDBUF_SYS_CONFIGURE *) buffer; tlv = (TLVBUF_STA_MAC_ADDR_FILTER *) (buffer + sizeof(APCMDBUF_SYS_CONFIGURE)); /* Fill the command buffer */ cmd_buf->CmdCode = APCMD_SYS_CONFIGURE; cmd_buf->Size = cmd_len; cmd_buf->SeqNum = 0; cmd_buf->Result = 0; tlv->Tag = MRVL_STA_MAC_ADDR_FILTER_TLV_ID; if (argc == 0) { cmd_buf->Action = ACTION_GET; tlv->Count = MAX_MAC_ONESHOT_FILTER; } else { cmd_buf->Action = ACTION_SET; tlv->FilterMode = atoi(argv[0]); tlv->Count = argc - 1; for (i = 0; i < tlv->Count; i++) { if ((ret = mac2raw(argv[i + 1], &tlv->MacAddress[i * ETH_ALEN])) != UAP_SUCCESS) { printf("ERR: %s Address\n", ret == UAP_FAILURE ? "Invalid MAC" : ret == UAP_RET_MAC_BROADCAST ? "Broadcast" : "Multicast"); print_sta_filter_table_usage(); free(buffer); return; } } } tlv->Length = tlv->Count * ETH_ALEN + 2; endian_convert_tlv_header_out(tlv); /* Send the command */ ret = uap_ioctl((u8 *) cmd_buf, &cmd_len, cmd_len); endian_convert_tlv_header_in(tlv); /* Process response */ if (ret == UAP_SUCCESS) { /* Verify response */ if ((cmd_buf->CmdCode != (APCMD_SYS_CONFIGURE | APCMD_RESP_CHECK)) || (tlv->Tag != MRVL_STA_MAC_ADDR_FILTER_TLV_ID)) { printf("ERR:Corrupted response! CmdCode=%x, Tlv->Tag=%x\n", cmd_buf->CmdCode, tlv->Tag); free(buffer); return; } /* Print response */ if (cmd_buf->Result == CMD_SUCCESS) { if (argc == 0) { print_mac_filter(tlv); } else { printf("MAC address filter table setting successful!\n"); } } else { if (argc == 0) { printf ("ERR:Could not get MAC address filter table settings!\n"); } else { printf ("ERR:Could not set MAC address filter table settings!\n"); } } } else { printf("ERR:Command sending failed!\n"); } if (buffer) free(buffer); return; } /** * @brief Creates a sys_cfg request for max station number * and sends to the driver * * Usage: "sys_cfg_max_sta_num [STA_NUM]" * if STA_NUM is provided, a 'set' is performed * else a 'get' is performed. * * STA_NUM should not bigger than 8 * * @param argc Number of arguments * @param argv Pointer to the arguments * @return N/A */ void apcmd_sys_cfg_max_sta_num(int argc, char *argv[]) { APCMDBUF_SYS_CONFIGURE *cmd_buf = NULL; TLVBUF_MAX_STA_NUM *tlv = NULL; u8 *buffer = NULL; u16 cmd_len; int ret = UAP_FAILURE; int opt; while ((opt = getopt_long(argc, argv, "+", cmd_options, NULL)) != -1) { switch (opt) { default: print_sys_cfg_max_sta_num_usage(); return; } } argc -= optind; argv += optind; /* Check arguments */ if (argc && (is_input_valid(MAXSTANUM, argc, argv) != UAP_SUCCESS)) { print_sys_cfg_max_sta_num_usage(); return; } /* Initialize the command length */ cmd_len = sizeof(APCMDBUF_SYS_CONFIGURE) + sizeof(TLVBUF_MAX_STA_NUM); /* Initialize the command buffer */ buffer = (u8 *) malloc(cmd_len); if (!buffer) { printf("ERR:Cannot allocate buffer for command!\n"); return; } bzero((char *) buffer, cmd_len); /* Locate headers */ cmd_buf = (APCMDBUF_SYS_CONFIGURE *) buffer; tlv = (TLVBUF_MAX_STA_NUM *) (buffer + sizeof(APCMDBUF_SYS_CONFIGURE)); /* Fill the command buffer */ cmd_buf->CmdCode = APCMD_SYS_CONFIGURE; cmd_buf->Size = cmd_len; cmd_buf->SeqNum = 0; cmd_buf->Result = 0; tlv->Tag = MRVL_MAX_STA_CNT_TLV_ID; tlv->Length = 2; if (argc == 0) { cmd_buf->Action = ACTION_GET; } else { cmd_buf->Action = ACTION_SET; tlv->Max_sta_num = (u16) atoi(argv[0]); } endian_convert_tlv_header_out(tlv); tlv->Max_sta_num = uap_cpu_to_le16(tlv->Max_sta_num); /* Send the command */ ret = uap_ioctl((u8 *) cmd_buf, &cmd_len, cmd_len); endian_convert_tlv_header_in(tlv); tlv->Max_sta_num = uap_le16_to_cpu(tlv->Max_sta_num); /* Process response */ if (ret == UAP_SUCCESS) { /* Verify response */ if ((cmd_buf->CmdCode != (APCMD_SYS_CONFIGURE | APCMD_RESP_CHECK)) || (tlv->Tag != MRVL_MAX_STA_CNT_TLV_ID)) { printf("ERR:Corrupted response! CmdCode=%x, Tlv->Tag=%x\n", cmd_buf->CmdCode, tlv->Tag); free(buffer); return; } /* Print response */ if (cmd_buf->Result == CMD_SUCCESS) { if (argc == 0) { printf("max station number = %d\n", tlv->Max_sta_num); } else { printf("max station number setting successful\n"); } } else { if (argc == 0) { printf("ERR:Could not get max station number!\n"); } else { printf("ERR:Could not set max station number!\n"); } } } else { printf("ERR:Command sending failed!\n"); } if (buffer) free(buffer); return; } /** * @brief Creates a sys_cfg request for retry limit * and sends to the driver * * Usage: "sys_cfg_retry_limit [RETRY_LIMIT]" * if RETRY_LIMIT is provided, a 'set' is performed * else a 'get' is performed. * * RETRY_LIMIT should not bigger than 14 * * @param argc Number of arguments * @param argv Pointer to the arguments * @return N/A */ void apcmd_sys_cfg_retry_limit(int argc, char *argv[]) { APCMDBUF_SYS_CONFIGURE *cmd_buf = NULL; TLVBUF_RETRY_LIMIT *tlv = NULL; u8 *buffer = NULL; u16 cmd_len; int ret = UAP_FAILURE; int opt; while ((opt = getopt_long(argc, argv, "+", cmd_options, NULL)) != -1) { switch (opt) { default: print_sys_cfg_max_sta_num_usage(); return; } } argc -= optind; argv += optind; /* Check arguments */ if (argc && (is_input_valid(RETRYLIMIT, argc, argv) != UAP_SUCCESS)) { print_sys_cfg_retry_limit_usage(); return; } /* Initialize the command length */ cmd_len = sizeof(APCMDBUF_SYS_CONFIGURE) + sizeof(TLVBUF_RETRY_LIMIT); /* Initialize the command buffer */ buffer = (u8 *) malloc(cmd_len); if (!buffer) { printf("ERR:Cannot allocate buffer for command!\n"); return; } bzero((char *) buffer, cmd_len); /* Locate headers */ cmd_buf = (APCMDBUF_SYS_CONFIGURE *) buffer; tlv = (TLVBUF_RETRY_LIMIT *) (buffer + sizeof(APCMDBUF_SYS_CONFIGURE)); /* Fill the command buffer */ cmd_buf->CmdCode = APCMD_SYS_CONFIGURE; cmd_buf->Size = cmd_len; cmd_buf->SeqNum = 0; cmd_buf->Result = 0; tlv->Tag = MRVL_RETRY_LIMIT_TLV_ID; tlv->Length = 1; if (argc == 0) { cmd_buf->Action = ACTION_GET; } else { cmd_buf->Action = ACTION_SET; tlv->retry_limit = (u8) atoi(argv[0]); } endian_convert_tlv_header_out(tlv); /* Send the command */ ret = uap_ioctl((u8 *) cmd_buf, &cmd_len, cmd_len); endian_convert_tlv_header_in(tlv); /* Process response */ if (ret == UAP_SUCCESS) { /* Verify response */ if ((cmd_buf->CmdCode != (APCMD_SYS_CONFIGURE | APCMD_RESP_CHECK)) || (tlv->Tag != MRVL_RETRY_LIMIT_TLV_ID)) { printf("ERR:Corrupted response! CmdCode=%x, Tlv->Tag=%x\n", cmd_buf->CmdCode, tlv->Tag); free(buffer); return; } /* Print response */ if (cmd_buf->Result == CMD_SUCCESS) { if (argc == 0) { printf("retry limit = %d\n", tlv->retry_limit); } else { printf("retry limit setting successful\n"); } } else { if (argc == 0) { printf("ERR:Could not get retry limit!\n"); } else { printf("ERR:Could not set retry limit!\n"); } } } else { printf("ERR:Command sending failed!\n"); } if (buffer) free(buffer); return; } /** * @brief convert string to integer * * @param ptr A pointer to data buffer * @param chr A pointer to return integer * @return A pointer to next data field */ char * convert2hex(char *ptr, u8 * chr) { u8 val; for (val = 0; *ptr && isxdigit(*ptr); ptr++) { val = (val * 16) + hexc2bin(*ptr); } *chr = val; return ptr; } /** * @brief parse hex data * @param fp A pointer to FILE stream * @param dst A pointer to receive hex data * @return length of hex data */ int fparse_for_hex(FILE * fp, u8 * dst) { char *ptr; u8 *dptr; char buf[256]; dptr = dst; while (fgets(buf, sizeof(buf), fp)) { ptr = buf; while (*ptr) { /* skip leading spaces */ while (*ptr && (isspace(*ptr) || *ptr == '\t')) ptr++; /* skip blank lines and lines beginning with '#' */ if (*ptr == '\0' || *ptr == '#') break; if (isxdigit(*ptr)) { ptr = convert2hex(ptr, dptr++); } else { /* Invalid character on data line */ ptr++; } } } return (dptr - dst); } /** * @brief Creates a cfg_data request * and sends to the driver * * Usage: "cfg_data " * * @param argc Number of arguments * @param argv Pointer to the arguments * @return N/A */ void apcmd_cfg_data(int argc, char *argv[]) { APCMDBUF_CFG_DATA *cmd_buf = NULL; u8 *buf = NULL; u16 cmd_len; u16 buf_len; int ret = UAP_FAILURE; int opt; FILE *fp = NULL; while ((opt = getopt_long(argc, argv, "+", cmd_options, NULL)) != -1) { switch (opt) { default: print_cfg_data_usage(); return; } } argc -= optind; argv += optind; /* Check arguments */ if ((argc == 0) || (argc > 2)) { printf("ERR:wrong arguments.\n"); print_cfg_data_usage(); return; } else { if ((ISDIGIT(argv[0]) == 0) || (atoi(argv[0]) != 2)) { printf("ERR:Illegal type parameter %s. Must be '2'.\n", argv[0]); print_cfg_data_usage(); return; } } buf_len = MRVDRV_SIZE_OF_CMD_BUFFER; buf = (u8 *) malloc(buf_len); memset(buf, 0, buf_len); cmd_buf = (APCMDBUF_CFG_DATA *) buf; if (buf == NULL) { printf("Error: allocate memory for hostcmd failed\n"); return; } if (argc == 2) { /* Check if file exists */ fp = fopen(argv[1], "r"); if (fp == NULL) { printf("\nERR:Config file can not open %s.\n", argv[1]); free(buf); return; } cmd_buf->action = ACTION_SET; cmd_buf->data_len = fparse_for_hex(fp, cmd_buf->data); fclose(fp); if (cmd_buf->data_len > MAX_CFG_DATA_SIZE) { printf("ERR: Config file is too big %d\n", cmd_buf->data_len); free(buf); return; } } else { cmd_buf->action = ACTION_GET; cmd_buf->data_len = 0; } cmd_buf->action = uap_cpu_to_le16(cmd_buf->action); cmd_buf->type = atoi(argv[0]); cmd_buf->type = uap_cpu_to_le16(cmd_buf->type); cmd_buf->data_len = uap_cpu_to_le16(cmd_buf->data_len); /* Fill the command buffer */ cmd_len = cmd_buf->data_len + sizeof(APCMDBUF_CFG_DATA); cmd_buf->CmdCode = HostCmd_CMD_CFG_DATA; cmd_buf->Size = cmd_len; cmd_buf->SeqNum = 0; cmd_buf->Result = 0; /* Send the command */ ret = uap_ioctl((u8 *) cmd_buf, &cmd_len, buf_len); /* Process response */ if (ret == UAP_SUCCESS) { cmd_buf->action = uap_le16_to_cpu(cmd_buf->action); cmd_buf->data_len = uap_le16_to_cpu(cmd_buf->data_len); if (cmd_buf->action == ACTION_GET) { hexdump_data("cfg_data", cmd_buf->data, cmd_buf->data_len, ' '); } else printf("download cfg data successful\n"); } if (buf) free(buf); return; }