1 /****************************************************************************** 2 * 3 * Copyright(c) 2009-2013 Realtek Corporation. 4 * 5 * This program is free software; you can redistribute it and/or modify it 6 * under the terms of version 2 of the GNU General Public License as 7 * published by the Free Software Foundation. 8 * 9 * This program is distributed in the hope that it will be useful, but WITHOUT 10 * ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or 11 * FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License for 12 * more details. 13 * 14 * The full GNU General Public License is included in this distribution in the 15 * file called LICENSE. 16 * 17 * Contact Information: 18 * wlanfae <wlanfae@realtek.com> 19 * Realtek Corporation, No. 2, Innovation Road II, Hsinchu Science Park, 20 * Hsinchu 300, Taiwan. 21 * 22 * Larry Finger <Larry.Finger@lwfinger.net> 23 * 24 *****************************************************************************/ 25 26 #include "../wifi.h" 27 #include "../pci.h" 28 #include "../base.h" 29 #include "../core.h" 30 #include "reg.h" 31 #include "def.h" 32 #include "fw.h" 33 34 static void _rtl88e_enable_fw_download(struct ieee80211_hw *hw, bool enable) 35 { 36 struct rtl_priv *rtlpriv = rtl_priv(hw); 37 u8 tmp; 38 39 if (enable) { 40 tmp = rtl_read_byte(rtlpriv, REG_SYS_FUNC_EN + 1); 41 rtl_write_byte(rtlpriv, REG_SYS_FUNC_EN + 1, tmp | 0x04); 42 43 tmp = rtl_read_byte(rtlpriv, REG_MCUFWDL); 44 rtl_write_byte(rtlpriv, REG_MCUFWDL, tmp | 0x01); 45 46 tmp = rtl_read_byte(rtlpriv, REG_MCUFWDL + 2); 47 rtl_write_byte(rtlpriv, REG_MCUFWDL + 2, tmp & 0xf7); 48 } else { 49 tmp = rtl_read_byte(rtlpriv, REG_MCUFWDL); 50 rtl_write_byte(rtlpriv, REG_MCUFWDL, tmp & 0xfe); 51 52 rtl_write_byte(rtlpriv, REG_MCUFWDL + 1, 0x00); 53 } 54 } 55 56 static void _rtl88e_fw_block_write(struct ieee80211_hw *hw, 57 const u8 *buffer, u32 size) 58 { 59 struct rtl_priv *rtlpriv = rtl_priv(hw); 60 u32 blocksize = sizeof(u32); 61 u8 *bufferptr = (u8 *)buffer; 62 u32 *pu4BytePtr = (u32 *)buffer; 63 u32 i, offset, blockcount, remainsize; 64 65 blockcount = size / blocksize; 66 remainsize = size % blocksize; 67 68 for (i = 0; i < blockcount; i++) { 69 offset = i * blocksize; 70 rtl_write_dword(rtlpriv, (FW_8192C_START_ADDRESS + offset), 71 *(pu4BytePtr + i)); 72 } 73 74 if (remainsize) { 75 offset = blockcount * blocksize; 76 bufferptr += offset; 77 for (i = 0; i < remainsize; i++) { 78 rtl_write_byte(rtlpriv, (FW_8192C_START_ADDRESS + 79 offset + i), *(bufferptr + i)); 80 } 81 } 82 } 83 84 static void _rtl88e_fw_page_write(struct ieee80211_hw *hw, 85 u32 page, const u8 *buffer, u32 size) 86 { 87 struct rtl_priv *rtlpriv = rtl_priv(hw); 88 u8 value8; 89 u8 u8page = (u8) (page & 0x07); 90 91 value8 = (rtl_read_byte(rtlpriv, REG_MCUFWDL + 2) & 0xF8) | u8page; 92 93 rtl_write_byte(rtlpriv, (REG_MCUFWDL + 2), value8); 94 _rtl88e_fw_block_write(hw, buffer, size); 95 } 96 97 static void _rtl88e_fill_dummy(u8 *pfwbuf, u32 *pfwlen) 98 { 99 u32 fwlen = *pfwlen; 100 u8 remain = (u8) (fwlen % 4); 101 102 remain = (remain == 0) ? 0 : (4 - remain); 103 104 while (remain > 0) { 105 pfwbuf[fwlen] = 0; 106 fwlen++; 107 remain--; 108 } 109 110 *pfwlen = fwlen; 111 } 112 113 static void _rtl88e_write_fw(struct ieee80211_hw *hw, 114 enum version_8188e version, u8 *buffer, u32 size) 115 { 116 struct rtl_priv *rtlpriv = rtl_priv(hw); 117 u8 *bufferptr = (u8 *)buffer; 118 u32 pagenums, remainsize; 119 u32 page, offset; 120 121 RT_TRACE(rtlpriv, COMP_FW, DBG_LOUD, "FW size is %d bytes,\n", size); 122 123 _rtl88e_fill_dummy(bufferptr, &size); 124 125 pagenums = size / FW_8192C_PAGE_SIZE; 126 remainsize = size % FW_8192C_PAGE_SIZE; 127 128 if (pagenums > 8) { 129 RT_TRACE(rtlpriv, COMP_ERR, DBG_EMERG, 130 "Page numbers should not greater then 8\n"); 131 } 132 133 for (page = 0; page < pagenums; page++) { 134 offset = page * FW_8192C_PAGE_SIZE; 135 _rtl88e_fw_page_write(hw, page, (bufferptr + offset), 136 FW_8192C_PAGE_SIZE); 137 } 138 139 if (remainsize) { 140 offset = pagenums * FW_8192C_PAGE_SIZE; 141 page = pagenums; 142 _rtl88e_fw_page_write(hw, page, (bufferptr + offset), 143 remainsize); 144 } 145 } 146 147 static int _rtl88e_fw_free_to_go(struct ieee80211_hw *hw) 148 { 149 struct rtl_priv *rtlpriv = rtl_priv(hw); 150 int err = -EIO; 151 u32 counter = 0; 152 u32 value32; 153 154 do { 155 value32 = rtl_read_dword(rtlpriv, REG_MCUFWDL); 156 } while ((counter++ < FW_8192C_POLLING_TIMEOUT_COUNT) && 157 (!(value32 & FWDL_CHKSUM_RPT))); 158 159 if (counter >= FW_8192C_POLLING_TIMEOUT_COUNT) { 160 RT_TRACE(rtlpriv, COMP_ERR, DBG_EMERG, 161 "chksum report faill ! REG_MCUFWDL:0x%08x .\n", 162 value32); 163 goto exit; 164 } 165 166 RT_TRACE(rtlpriv, COMP_FW, DBG_TRACE, 167 "Checksum report OK ! REG_MCUFWDL:0x%08x .\n", value32); 168 169 value32 = rtl_read_dword(rtlpriv, REG_MCUFWDL); 170 value32 |= MCUFWDL_RDY; 171 value32 &= ~WINTINI_RDY; 172 rtl_write_dword(rtlpriv, REG_MCUFWDL, value32); 173 174 rtl88e_firmware_selfreset(hw); 175 counter = 0; 176 177 do { 178 value32 = rtl_read_dword(rtlpriv, REG_MCUFWDL); 179 if (value32 & WINTINI_RDY) { 180 RT_TRACE(rtlpriv, COMP_FW, DBG_TRACE, 181 "Polling FW ready success!! REG_MCUFWDL:0x%08x.\n", 182 value32); 183 err = 0; 184 goto exit; 185 } 186 187 udelay(FW_8192C_POLLING_DELAY); 188 189 } while (counter++ < FW_8192C_POLLING_TIMEOUT_COUNT); 190 191 RT_TRACE(rtlpriv, COMP_ERR, DBG_EMERG, 192 "Polling FW ready fail!! REG_MCUFWDL:0x%08x .\n", value32); 193 194 exit: 195 return err; 196 } 197 198 int rtl88e_download_fw(struct ieee80211_hw *hw, 199 bool buse_wake_on_wlan_fw) 200 { 201 struct rtl_priv *rtlpriv = rtl_priv(hw); 202 struct rtl_hal *rtlhal = rtl_hal(rtl_priv(hw)); 203 struct rtlwifi_firmware_header *pfwheader; 204 u8 *pfwdata; 205 u32 fwsize; 206 int err; 207 enum version_8188e version = rtlhal->version; 208 209 if (!rtlhal->pfirmware) 210 return 1; 211 212 pfwheader = (struct rtlwifi_firmware_header *)rtlhal->pfirmware; 213 pfwdata = rtlhal->pfirmware; 214 fwsize = rtlhal->fwsize; 215 RT_TRACE(rtlpriv, COMP_FW, DBG_DMESG, 216 "normal Firmware SIZE %d\n", fwsize); 217 218 if (IS_FW_HEADER_EXIST(pfwheader)) { 219 RT_TRACE(rtlpriv, COMP_FW, DBG_DMESG, 220 "Firmware Version(%d), Signature(%#x), Size(%d)\n", 221 pfwheader->version, pfwheader->signature, 222 (int)sizeof(struct rtlwifi_firmware_header)); 223 224 pfwdata = pfwdata + sizeof(struct rtlwifi_firmware_header); 225 fwsize = fwsize - sizeof(struct rtlwifi_firmware_header); 226 } 227 228 if (rtl_read_byte(rtlpriv, REG_MCUFWDL) & BIT(7)) { 229 rtl_write_byte(rtlpriv, REG_MCUFWDL, 0); 230 rtl88e_firmware_selfreset(hw); 231 } 232 _rtl88e_enable_fw_download(hw, true); 233 _rtl88e_write_fw(hw, version, pfwdata, fwsize); 234 _rtl88e_enable_fw_download(hw, false); 235 236 err = _rtl88e_fw_free_to_go(hw); 237 if (err) { 238 RT_TRACE(rtlpriv, COMP_ERR, DBG_EMERG, 239 "Firmware is not ready to run!\n"); 240 } else { 241 RT_TRACE(rtlpriv, COMP_FW, DBG_LOUD, 242 "Firmware is ready to run!\n"); 243 } 244 245 return 0; 246 } 247 248 static bool _rtl88e_check_fw_read_last_h2c(struct ieee80211_hw *hw, u8 boxnum) 249 { 250 struct rtl_priv *rtlpriv = rtl_priv(hw); 251 u8 val_hmetfr; 252 253 val_hmetfr = rtl_read_byte(rtlpriv, REG_HMETFR); 254 if (((val_hmetfr >> boxnum) & BIT(0)) == 0) 255 return true; 256 return false; 257 } 258 259 static void _rtl88e_fill_h2c_command(struct ieee80211_hw *hw, 260 u8 element_id, u32 cmd_len, 261 u8 *cmd_b) 262 { 263 struct rtl_priv *rtlpriv = rtl_priv(hw); 264 struct rtl_hal *rtlhal = rtl_hal(rtl_priv(hw)); 265 u8 boxnum; 266 u16 box_reg = 0, box_extreg = 0; 267 u8 u1b_tmp; 268 bool isfw_read = false; 269 u8 buf_index = 0; 270 bool write_sucess = false; 271 u8 wait_h2c_limmit = 100; 272 u8 wait_writeh2c_limit = 100; 273 u8 boxcontent[4], boxextcontent[4]; 274 u32 h2c_waitcounter = 0; 275 unsigned long flag; 276 u8 idx; 277 278 RT_TRACE(rtlpriv, COMP_CMD, DBG_LOUD, "come in\n"); 279 280 while (true) { 281 spin_lock_irqsave(&rtlpriv->locks.h2c_lock, flag); 282 if (rtlhal->h2c_setinprogress) { 283 RT_TRACE(rtlpriv, COMP_CMD, DBG_LOUD, 284 "H2C set in progress! Wait to set..element_id(%d).\n", 285 element_id); 286 287 while (rtlhal->h2c_setinprogress) { 288 spin_unlock_irqrestore(&rtlpriv->locks.h2c_lock, 289 flag); 290 h2c_waitcounter++; 291 RT_TRACE(rtlpriv, COMP_CMD, DBG_LOUD, 292 "Wait 100 us (%d times)...\n", 293 h2c_waitcounter); 294 udelay(100); 295 296 if (h2c_waitcounter > 1000) 297 return; 298 spin_lock_irqsave(&rtlpriv->locks.h2c_lock, 299 flag); 300 } 301 spin_unlock_irqrestore(&rtlpriv->locks.h2c_lock, flag); 302 } else { 303 rtlhal->h2c_setinprogress = true; 304 spin_unlock_irqrestore(&rtlpriv->locks.h2c_lock, flag); 305 break; 306 } 307 } 308 309 while (!write_sucess) { 310 wait_writeh2c_limit--; 311 if (wait_writeh2c_limit == 0) { 312 RT_TRACE(rtlpriv, COMP_ERR, DBG_EMERG, 313 "Write H2C fail because no trigger for FW INT!\n"); 314 break; 315 } 316 317 boxnum = rtlhal->last_hmeboxnum; 318 switch (boxnum) { 319 case 0: 320 box_reg = REG_HMEBOX_0; 321 box_extreg = REG_HMEBOX_EXT_0; 322 break; 323 case 1: 324 box_reg = REG_HMEBOX_1; 325 box_extreg = REG_HMEBOX_EXT_1; 326 break; 327 case 2: 328 box_reg = REG_HMEBOX_2; 329 box_extreg = REG_HMEBOX_EXT_2; 330 break; 331 case 3: 332 box_reg = REG_HMEBOX_3; 333 box_extreg = REG_HMEBOX_EXT_3; 334 break; 335 default: 336 RT_TRACE(rtlpriv, COMP_ERR, DBG_LOUD, 337 "switch case not process\n"); 338 break; 339 } 340 isfw_read = _rtl88e_check_fw_read_last_h2c(hw, boxnum); 341 while (!isfw_read) { 342 wait_h2c_limmit--; 343 if (wait_h2c_limmit == 0) { 344 RT_TRACE(rtlpriv, COMP_CMD, DBG_LOUD, 345 "Waiting too long for FW read clear HMEBox(%d)!\n", 346 boxnum); 347 break; 348 } 349 350 udelay(10); 351 352 isfw_read = _rtl88e_check_fw_read_last_h2c(hw, boxnum); 353 u1b_tmp = rtl_read_byte(rtlpriv, 0x130); 354 RT_TRACE(rtlpriv, COMP_CMD, DBG_LOUD, 355 "Waiting for FW read clear HMEBox(%d)!!! 0x130 = %2x\n", 356 boxnum, u1b_tmp); 357 } 358 359 if (!isfw_read) { 360 RT_TRACE(rtlpriv, COMP_CMD, DBG_LOUD, 361 "Write H2C register BOX[%d] fail!!!!! Fw do not read.\n", 362 boxnum); 363 break; 364 } 365 366 memset(boxcontent, 0, sizeof(boxcontent)); 367 memset(boxextcontent, 0, sizeof(boxextcontent)); 368 boxcontent[0] = element_id; 369 RT_TRACE(rtlpriv, COMP_CMD, DBG_LOUD, 370 "Write element_id box_reg(%4x) = %2x\n", 371 box_reg, element_id); 372 373 switch (cmd_len) { 374 case 1: 375 case 2: 376 case 3: 377 /*boxcontent[0] &= ~(BIT(7));*/ 378 memcpy((u8 *)(boxcontent) + 1, 379 cmd_b + buf_index, cmd_len); 380 381 for (idx = 0; idx < 4; idx++) { 382 rtl_write_byte(rtlpriv, box_reg + idx, 383 boxcontent[idx]); 384 } 385 break; 386 case 4: 387 case 5: 388 case 6: 389 case 7: 390 /*boxcontent[0] |= (BIT(7));*/ 391 memcpy((u8 *)(boxextcontent), 392 cmd_b + buf_index+3, cmd_len-3); 393 memcpy((u8 *)(boxcontent) + 1, 394 cmd_b + buf_index, 3); 395 396 for (idx = 0; idx < 2; idx++) { 397 rtl_write_byte(rtlpriv, box_extreg + idx, 398 boxextcontent[idx]); 399 } 400 401 for (idx = 0; idx < 4; idx++) { 402 rtl_write_byte(rtlpriv, box_reg + idx, 403 boxcontent[idx]); 404 } 405 break; 406 default: 407 RT_TRACE(rtlpriv, COMP_ERR, DBG_LOUD, 408 "switch case not process\n"); 409 break; 410 } 411 412 write_sucess = true; 413 414 rtlhal->last_hmeboxnum = boxnum + 1; 415 if (rtlhal->last_hmeboxnum == 4) 416 rtlhal->last_hmeboxnum = 0; 417 418 RT_TRACE(rtlpriv, COMP_CMD, DBG_LOUD, 419 "pHalData->last_hmeboxnum = %d\n", 420 rtlhal->last_hmeboxnum); 421 } 422 423 spin_lock_irqsave(&rtlpriv->locks.h2c_lock, flag); 424 rtlhal->h2c_setinprogress = false; 425 spin_unlock_irqrestore(&rtlpriv->locks.h2c_lock, flag); 426 427 RT_TRACE(rtlpriv, COMP_CMD, DBG_LOUD, "go out\n"); 428 } 429 430 void rtl88e_fill_h2c_cmd(struct ieee80211_hw *hw, 431 u8 element_id, u32 cmd_len, u8 *cmdbuffer) 432 { 433 struct rtl_hal *rtlhal = rtl_hal(rtl_priv(hw)); 434 u32 tmp_cmdbuf[2]; 435 436 if (!rtlhal->fw_ready) { 437 RT_ASSERT(false, 438 "return H2C cmd because of Fw download fail!!!\n"); 439 return; 440 } 441 442 memset(tmp_cmdbuf, 0, 8); 443 memcpy(tmp_cmdbuf, cmdbuffer, cmd_len); 444 _rtl88e_fill_h2c_command(hw, element_id, cmd_len, (u8 *)&tmp_cmdbuf); 445 446 return; 447 } 448 449 void rtl88e_firmware_selfreset(struct ieee80211_hw *hw) 450 { 451 u8 u1b_tmp; 452 struct rtl_priv *rtlpriv = rtl_priv(hw); 453 454 u1b_tmp = rtl_read_byte(rtlpriv, REG_SYS_FUNC_EN+1); 455 rtl_write_byte(rtlpriv, REG_SYS_FUNC_EN+1, (u1b_tmp & (~BIT(2)))); 456 rtl_write_byte(rtlpriv, REG_SYS_FUNC_EN+1, (u1b_tmp | BIT(2))); 457 RT_TRACE(rtlpriv, COMP_INIT, DBG_LOUD, 458 "8051Reset88E(): 8051 reset success\n"); 459 460 } 461 462 void rtl88e_set_fw_pwrmode_cmd(struct ieee80211_hw *hw, u8 mode) 463 { 464 struct rtl_priv *rtlpriv = rtl_priv(hw); 465 u8 u1_h2c_set_pwrmode[H2C_88E_PWEMODE_LENGTH] = { 0 }; 466 struct rtl_ps_ctl *ppsc = rtl_psc(rtl_priv(hw)); 467 u8 rlbm, power_state = 0; 468 RT_TRACE(rtlpriv, COMP_POWER, DBG_LOUD, "FW LPS mode = %d\n", mode); 469 470 SET_H2CCMD_PWRMODE_PARM_MODE(u1_h2c_set_pwrmode, ((mode) ? 1 : 0)); 471 rlbm = 0;/*YJ, temp, 120316. FW now not support RLBM=2.*/ 472 SET_H2CCMD_PWRMODE_PARM_RLBM(u1_h2c_set_pwrmode, rlbm); 473 SET_H2CCMD_PWRMODE_PARM_SMART_PS(u1_h2c_set_pwrmode, 474 (rtlpriv->mac80211.p2p) ? ppsc->smart_ps : 1); 475 SET_H2CCMD_PWRMODE_PARM_AWAKE_INTERVAL(u1_h2c_set_pwrmode, 476 ppsc->reg_max_lps_awakeintvl); 477 SET_H2CCMD_PWRMODE_PARM_ALL_QUEUE_UAPSD(u1_h2c_set_pwrmode, 0); 478 if (mode == FW_PS_ACTIVE_MODE) 479 power_state |= FW_PWR_STATE_ACTIVE; 480 else 481 power_state |= FW_PWR_STATE_RF_OFF; 482 483 SET_H2CCMD_PWRMODE_PARM_PWR_STATE(u1_h2c_set_pwrmode, power_state); 484 485 RT_PRINT_DATA(rtlpriv, COMP_CMD, DBG_DMESG, 486 "rtl92c_set_fw_pwrmode(): u1_h2c_set_pwrmode\n", 487 u1_h2c_set_pwrmode, H2C_88E_PWEMODE_LENGTH); 488 rtl88e_fill_h2c_cmd(hw, H2C_88E_SETPWRMODE, 489 H2C_88E_PWEMODE_LENGTH, u1_h2c_set_pwrmode); 490 } 491 492 void rtl88e_set_fw_joinbss_report_cmd(struct ieee80211_hw *hw, u8 mstatus) 493 { 494 u8 u1_joinbssrpt_parm[1] = { 0 }; 495 496 SET_H2CCMD_JOINBSSRPT_PARM_OPMODE(u1_joinbssrpt_parm, mstatus); 497 498 rtl88e_fill_h2c_cmd(hw, H2C_88E_JOINBSSRPT, 1, u1_joinbssrpt_parm); 499 } 500 501 void rtl88e_set_fw_ap_off_load_cmd(struct ieee80211_hw *hw, 502 u8 ap_offload_enable) 503 { 504 struct rtl_mac *mac = rtl_mac(rtl_priv(hw)); 505 u8 u1_apoffload_parm[H2C_88E_AP_OFFLOAD_LENGTH] = { 0 }; 506 507 SET_H2CCMD_AP_OFFLOAD_ON(u1_apoffload_parm, ap_offload_enable); 508 SET_H2CCMD_AP_OFFLOAD_HIDDEN(u1_apoffload_parm, mac->hiddenssid); 509 SET_H2CCMD_AP_OFFLOAD_DENYANY(u1_apoffload_parm, 0); 510 511 rtl88e_fill_h2c_cmd(hw, H2C_88E_AP_OFFLOAD, 512 H2C_88E_AP_OFFLOAD_LENGTH, u1_apoffload_parm); 513 514 } 515 516 #define BEACON_PG 0 /* ->1 */ 517 #define PSPOLL_PG 2 518 #define NULL_PG 3 519 #define PROBERSP_PG 4 /* ->5 */ 520 521 #define TOTAL_RESERVED_PKT_LEN 768 522 523 static u8 reserved_page_packet[TOTAL_RESERVED_PKT_LEN] = { 524 /* page 0 beacon */ 525 0x80, 0x00, 0x00, 0x00, 0xFF, 0xFF, 0xFF, 0xFF, 526 0xFF, 0xFF, 0x00, 0xE0, 0x4C, 0x76, 0x00, 0x42, 527 0x00, 0x40, 0x10, 0x10, 0x00, 0x03, 0x50, 0x08, 528 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 529 0x64, 0x00, 0x00, 0x04, 0x00, 0x0C, 0x6C, 0x69, 530 0x6E, 0x6B, 0x73, 0x79, 0x73, 0x5F, 0x77, 0x6C, 531 0x61, 0x6E, 0x01, 0x04, 0x82, 0x84, 0x8B, 0x96, 532 0x03, 0x01, 0x01, 0x06, 0x02, 0x00, 0x00, 0x2A, 533 0x01, 0x00, 0x32, 0x08, 0x24, 0x30, 0x48, 0x6C, 534 0x0C, 0x12, 0x18, 0x60, 0x2D, 0x1A, 0x6C, 0x18, 535 0x03, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 536 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 537 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 538 0x3D, 0x00, 0xDD, 0x06, 0x00, 0xE0, 0x4C, 0x02, 539 0x01, 0x70, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 540 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 541 542 /* page 1 beacon */ 543 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 544 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 545 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 546 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 547 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 548 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 549 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 550 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 551 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 552 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 553 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 554 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 555 0x10, 0x00, 0x20, 0x8C, 0x00, 0x12, 0x10, 0x00, 556 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 557 0x00, 0x01, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 558 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 559 560 /* page 2 ps-poll */ 561 0xA4, 0x10, 0x01, 0xC0, 0x00, 0x40, 0x10, 0x10, 562 0x00, 0x03, 0x00, 0xE0, 0x4C, 0x76, 0x00, 0x42, 563 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 564 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 565 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 566 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 567 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 568 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 569 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 570 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 571 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 572 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 573 0x18, 0x00, 0x20, 0x8C, 0x00, 0x12, 0x00, 0x00, 574 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x80, 575 0x80, 0x01, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 576 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 577 578 /* page 3 null */ 579 0x48, 0x01, 0x00, 0x00, 0x00, 0x40, 0x10, 0x10, 580 0x00, 0x03, 0x00, 0xE0, 0x4C, 0x76, 0x00, 0x42, 581 0x00, 0x40, 0x10, 0x10, 0x00, 0x03, 0x00, 0x00, 582 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 583 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 584 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 585 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 586 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 587 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 588 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 589 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 590 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 591 0x72, 0x00, 0x20, 0x8C, 0x00, 0x12, 0x00, 0x00, 592 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x80, 593 0x80, 0x01, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 594 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 595 596 /* page 4 probe_resp */ 597 0x50, 0x00, 0x00, 0x00, 0x00, 0x40, 0x10, 0x10, 598 0x00, 0x03, 0x00, 0xE0, 0x4C, 0x76, 0x00, 0x42, 599 0x00, 0x40, 0x10, 0x10, 0x00, 0x03, 0x00, 0x00, 600 0x9E, 0x46, 0x15, 0x32, 0x27, 0xF2, 0x2D, 0x00, 601 0x64, 0x00, 0x00, 0x04, 0x00, 0x0C, 0x6C, 0x69, 602 0x6E, 0x6B, 0x73, 0x79, 0x73, 0x5F, 0x77, 0x6C, 603 0x61, 0x6E, 0x01, 0x04, 0x82, 0x84, 0x8B, 0x96, 604 0x03, 0x01, 0x01, 0x06, 0x02, 0x00, 0x00, 0x2A, 605 0x01, 0x00, 0x32, 0x08, 0x24, 0x30, 0x48, 0x6C, 606 0x0C, 0x12, 0x18, 0x60, 0x2D, 0x1A, 0x6C, 0x18, 607 0x03, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 608 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 609 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 610 0x3D, 0x00, 0xDD, 0x06, 0x00, 0xE0, 0x4C, 0x02, 611 0x01, 0x70, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 612 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 613 614 /* page 5 probe_resp */ 615 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 616 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 617 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 618 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 619 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 620 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 621 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 622 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 623 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 624 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 625 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 626 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 627 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 628 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 629 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 630 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 631 }; 632 633 void rtl88e_set_fw_rsvdpagepkt(struct ieee80211_hw *hw, bool b_dl_finished) 634 { 635 struct rtl_priv *rtlpriv = rtl_priv(hw); 636 struct rtl_mac *mac = rtl_mac(rtl_priv(hw)); 637 struct sk_buff *skb = NULL; 638 u32 totalpacketlen; 639 bool rtstatus; 640 u8 u1rsvdpageloc[5] = { 0 }; 641 bool b_dlok = false; 642 u8 *beacon; 643 u8 *p_pspoll; 644 u8 *nullfunc; 645 u8 *p_probersp; 646 647 /*--------------------------------------------------------- 648 * (1) beacon 649 *--------------------------------------------------------- 650 */ 651 beacon = &reserved_page_packet[BEACON_PG * 128]; 652 SET_80211_HDR_ADDRESS2(beacon, mac->mac_addr); 653 SET_80211_HDR_ADDRESS3(beacon, mac->bssid); 654 655 /*------------------------------------------------------- 656 * (2) ps-poll 657 *-------------------------------------------------------- 658 */ 659 p_pspoll = &reserved_page_packet[PSPOLL_PG * 128]; 660 SET_80211_PS_POLL_AID(p_pspoll, (mac->assoc_id | 0xc000)); 661 SET_80211_PS_POLL_BSSID(p_pspoll, mac->bssid); 662 SET_80211_PS_POLL_TA(p_pspoll, mac->mac_addr); 663 664 SET_H2CCMD_RSVDPAGE_LOC_PSPOLL(u1rsvdpageloc, PSPOLL_PG); 665 666 /*-------------------------------------------------------- 667 * (3) null data 668 *--------------------------------------------------------- 669 */ 670 nullfunc = &reserved_page_packet[NULL_PG * 128]; 671 SET_80211_HDR_ADDRESS1(nullfunc, mac->bssid); 672 SET_80211_HDR_ADDRESS2(nullfunc, mac->mac_addr); 673 SET_80211_HDR_ADDRESS3(nullfunc, mac->bssid); 674 675 SET_H2CCMD_RSVDPAGE_LOC_NULL_DATA(u1rsvdpageloc, NULL_PG); 676 677 /*--------------------------------------------------------- 678 * (4) probe response 679 *---------------------------------------------------------- 680 */ 681 p_probersp = &reserved_page_packet[PROBERSP_PG * 128]; 682 SET_80211_HDR_ADDRESS1(p_probersp, mac->bssid); 683 SET_80211_HDR_ADDRESS2(p_probersp, mac->mac_addr); 684 SET_80211_HDR_ADDRESS3(p_probersp, mac->bssid); 685 686 SET_H2CCMD_RSVDPAGE_LOC_PROBE_RSP(u1rsvdpageloc, PROBERSP_PG); 687 688 totalpacketlen = TOTAL_RESERVED_PKT_LEN; 689 690 RT_PRINT_DATA(rtlpriv, COMP_CMD, DBG_LOUD, 691 "rtl88e_set_fw_rsvdpagepkt(): HW_VAR_SET_TX_CMD: ALL\n", 692 &reserved_page_packet[0], totalpacketlen); 693 RT_PRINT_DATA(rtlpriv, COMP_CMD, DBG_DMESG, 694 "rtl88e_set_fw_rsvdpagepkt(): HW_VAR_SET_TX_CMD: ALL\n", 695 u1rsvdpageloc, 3); 696 697 skb = dev_alloc_skb(totalpacketlen); 698 memcpy(skb_put(skb, totalpacketlen), 699 &reserved_page_packet, totalpacketlen); 700 701 rtstatus = rtl_cmd_send_packet(hw, skb); 702 703 if (rtstatus) 704 b_dlok = true; 705 706 if (b_dlok) { 707 RT_TRACE(rtlpriv, COMP_POWER, DBG_LOUD, 708 "Set RSVD page location to Fw.\n"); 709 RT_PRINT_DATA(rtlpriv, COMP_CMD, DBG_DMESG, 710 "H2C_RSVDPAGE:\n", u1rsvdpageloc, 3); 711 rtl88e_fill_h2c_cmd(hw, H2C_88E_RSVDPAGE, 712 sizeof(u1rsvdpageloc), u1rsvdpageloc); 713 } else 714 RT_TRACE(rtlpriv, COMP_ERR, DBG_WARNING, 715 "Set RSVD page location to Fw FAIL!!!!!!.\n"); 716 } 717 718 /*Should check FW support p2p or not.*/ 719 static void rtl88e_set_p2p_ctw_period_cmd(struct ieee80211_hw *hw, u8 ctwindow) 720 { 721 u8 u1_ctwindow_period[1] = { ctwindow}; 722 723 rtl88e_fill_h2c_cmd(hw, H2C_88E_P2P_PS_CTW_CMD, 1, u1_ctwindow_period); 724 725 } 726 727 void rtl88e_set_p2p_ps_offload_cmd(struct ieee80211_hw *hw, u8 p2p_ps_state) 728 { 729 struct rtl_priv *rtlpriv = rtl_priv(hw); 730 struct rtl_ps_ctl *rtlps = rtl_psc(rtl_priv(hw)); 731 struct rtl_hal *rtlhal = rtl_hal(rtl_priv(hw)); 732 struct rtl_p2p_ps_info *p2pinfo = &(rtlps->p2p_ps_info); 733 struct p2p_ps_offload_t *p2p_ps_offload = &rtlhal->p2p_ps_offload; 734 u8 i; 735 u16 ctwindow; 736 u32 start_time, tsf_low; 737 738 switch (p2p_ps_state) { 739 case P2P_PS_DISABLE: 740 RT_TRACE(rtlpriv, COMP_FW, DBG_LOUD, "P2P_PS_DISABLE\n"); 741 memset(p2p_ps_offload, 0, sizeof(*p2p_ps_offload)); 742 break; 743 case P2P_PS_ENABLE: 744 RT_TRACE(rtlpriv, COMP_FW, DBG_LOUD, "P2P_PS_ENABLE\n"); 745 /* update CTWindow value. */ 746 if (p2pinfo->ctwindow > 0) { 747 p2p_ps_offload->ctwindow_en = 1; 748 ctwindow = p2pinfo->ctwindow; 749 rtl88e_set_p2p_ctw_period_cmd(hw, ctwindow); 750 } 751 752 /* hw only support 2 set of NoA */ 753 for (i = 0 ; i < p2pinfo->noa_num; i++) { 754 /* To control the register setting for which NOA*/ 755 rtl_write_byte(rtlpriv, 0x5cf, (i << 4)); 756 if (i == 0) 757 p2p_ps_offload->noa0_en = 1; 758 else 759 p2p_ps_offload->noa1_en = 1; 760 761 /* config P2P NoA Descriptor Register */ 762 rtl_write_dword(rtlpriv, 0x5E0, 763 p2pinfo->noa_duration[i]); 764 rtl_write_dword(rtlpriv, 0x5E4, 765 p2pinfo->noa_interval[i]); 766 767 /*Get Current TSF value */ 768 tsf_low = rtl_read_dword(rtlpriv, REG_TSFTR); 769 770 start_time = p2pinfo->noa_start_time[i]; 771 if (p2pinfo->noa_count_type[i] != 1) { 772 while (start_time <= (tsf_low+(50*1024))) { 773 start_time += p2pinfo->noa_interval[i]; 774 if (p2pinfo->noa_count_type[i] != 255) 775 p2pinfo->noa_count_type[i]--; 776 } 777 } 778 rtl_write_dword(rtlpriv, 0x5E8, start_time); 779 rtl_write_dword(rtlpriv, 0x5EC, 780 p2pinfo->noa_count_type[i]); 781 } 782 783 if ((p2pinfo->opp_ps == 1) || (p2pinfo->noa_num > 0)) { 784 /* rst p2p circuit */ 785 rtl_write_byte(rtlpriv, REG_DUAL_TSF_RST, BIT(4)); 786 787 p2p_ps_offload->offload_en = 1; 788 789 if (P2P_ROLE_GO == rtlpriv->mac80211.p2p) { 790 p2p_ps_offload->role = 1; 791 p2p_ps_offload->allstasleep = -1; 792 } else { 793 p2p_ps_offload->role = 0; 794 } 795 796 p2p_ps_offload->discovery = 0; 797 } 798 break; 799 case P2P_PS_SCAN: 800 RT_TRACE(rtlpriv, COMP_FW, DBG_LOUD, "P2P_PS_SCAN\n"); 801 p2p_ps_offload->discovery = 1; 802 break; 803 case P2P_PS_SCAN_DONE: 804 RT_TRACE(rtlpriv, COMP_FW, DBG_LOUD, "P2P_PS_SCAN_DONE\n"); 805 p2p_ps_offload->discovery = 0; 806 p2pinfo->p2p_ps_state = P2P_PS_ENABLE; 807 break; 808 default: 809 break; 810 } 811 812 rtl88e_fill_h2c_cmd(hw, H2C_88E_P2P_PS_OFFLOAD, 1, 813 (u8 *)p2p_ps_offload); 814 815 } 816