2012-12-21 01:13:19 +04:00
/*
* Copyright ( c ) 2012 Qualcomm Atheros , Inc .
*
* Permission to use , copy , modify , and / or distribute this software for any
* purpose with or without fee is hereby granted , provided that the above
* copyright notice and this permission notice appear in all copies .
*
* THE SOFTWARE IS PROVIDED " AS IS " AND THE AUTHOR DISCLAIMS ALL WARRANTIES
* WITH REGARD TO THIS SOFTWARE INCLUDING ALL IMPLIED WARRANTIES OF
* MERCHANTABILITY AND FITNESS . IN NO EVENT SHALL THE AUTHOR BE LIABLE FOR
* ANY SPECIAL , DIRECT , INDIRECT , OR CONSEQUENTIAL DAMAGES OR ANY DAMAGES
* WHATSOEVER RESULTING FROM LOSS OF USE , DATA OR PROFITS , WHETHER IN AN
* ACTION OF CONTRACT , NEGLIGENCE OR OTHER TORTIOUS ACTION , ARISING OUT OF
* OR IN CONNECTION WITH THE USE OR PERFORMANCE OF THIS SOFTWARE .
*/
# include <linux/moduleparam.h>
# include <linux/if_arp.h>
2014-02-27 18:20:53 +04:00
# include <linux/etherdevice.h>
2012-12-21 01:13:19 +04:00
# include "wil6210.h"
2014-02-27 18:20:44 +04:00
# include "txrx.h"
2012-12-21 01:13:19 +04:00
/*
* Due to a hardware issue ,
* one has to read / write to / from NIC in 32 - bit chunks ;
* regular memcpy_fromio and siblings will
* not work on 64 - bit platform - it uses 64 - bit transactions
*
* Force 32 - bit transactions to enable NIC on 64 - bit platforms
*
* To avoid byte swap on big endian host , __raw_ { read | write } l
* should be used - { read | write } l would swap bytes to provide
* little endian on PCI value in host endianness .
*/
void wil_memcpy_fromio_32 ( void * dst , const volatile void __iomem * src ,
size_t count )
{
u32 * d = dst ;
const volatile u32 __iomem * s = src ;
/* size_t is unsigned, if (count%4 != 0) it will wrap */
for ( count + = 4 ; count > 4 ; count - = 4 )
* d + + = __raw_readl ( s + + ) ;
}
void wil_memcpy_toio_32 ( volatile void __iomem * dst , const void * src ,
size_t count )
{
volatile u32 __iomem * d = dst ;
const u32 * s = src ;
for ( count + = 4 ; count > 4 ; count - = 4 )
__raw_writel ( * s + + , d + + ) ;
}
2014-02-27 18:20:50 +04:00
static void wil_disconnect_cid ( struct wil6210_priv * wil , int cid )
{
uint i ;
struct wil_sta_info * sta = & wil - > sta [ cid ] ;
2014-02-27 18:20:54 +04:00
2014-03-17 17:34:06 +04:00
sta - > data_port_open = false ;
2014-02-27 18:20:54 +04:00
if ( sta - > status ! = wil_sta_unused ) {
wmi_disconnect_sta ( wil , sta - > addr , WLAN_REASON_DEAUTH_LEAVING ) ;
sta - > status = wil_sta_unused ;
}
2014-02-27 18:20:50 +04:00
for ( i = 0 ; i < WIL_STA_TID_NUM ; i + + ) {
struct wil_tid_ampdu_rx * r = sta - > tid_rx [ i ] ;
sta - > tid_rx [ i ] = NULL ;
wil_tid_ampdu_rx_free ( wil , r ) ;
}
for ( i = 0 ; i < ARRAY_SIZE ( wil - > vring_tx ) ; i + + ) {
if ( wil - > vring2cid_tid [ i ] [ 0 ] = = cid )
wil_vring_fini_tx ( wil , i ) ;
}
memset ( & sta - > stats , 0 , sizeof ( sta - > stats ) ) ;
}
2012-12-21 01:13:19 +04:00
static void _wil6210_disconnect ( struct wil6210_priv * wil , void * bssid )
{
2014-02-27 18:20:50 +04:00
int cid = - ENOENT ;
2012-12-21 01:13:19 +04:00
struct net_device * ndev = wil_to_ndev ( wil ) ;
2014-02-27 18:20:50 +04:00
struct wireless_dev * wdev = wil - > wdev ;
might_sleep ( ) ;
if ( bssid ) {
cid = wil_find_cid ( wil , bssid ) ;
wil_dbg_misc ( wil , " %s(%pM, CID %d) \n " , __func__ , bssid , cid ) ;
} else {
wil_dbg_misc ( wil , " %s(all) \n " , __func__ ) ;
2014-02-27 18:20:44 +04:00
}
2014-02-27 18:20:50 +04:00
if ( cid > = 0 ) /* disconnect 1 peer */
wil_disconnect_cid ( wil , cid ) ;
else /* disconnect all */
for ( cid = 0 ; cid < WIL6210_MAX_CID ; cid + + )
wil_disconnect_cid ( wil , cid ) ;
/* link state */
switch ( wdev - > iftype ) {
case NL80211_IFTYPE_STATION :
case NL80211_IFTYPE_P2P_CLIENT :
wil_link_off ( wil ) ;
if ( test_bit ( wil_status_fwconnected , & wil - > status ) ) {
clear_bit ( wil_status_fwconnected , & wil - > status ) ;
cfg80211_disconnected ( ndev ,
WLAN_STATUS_UNSPECIFIED_FAILURE ,
NULL , 0 , GFP_KERNEL ) ;
} else if ( test_bit ( wil_status_fwconnecting , & wil - > status ) ) {
cfg80211_connect_result ( ndev , bssid , NULL , 0 , NULL , 0 ,
WLAN_STATUS_UNSPECIFIED_FAILURE ,
GFP_KERNEL ) ;
}
clear_bit ( wil_status_fwconnecting , & wil - > status ) ;
break ;
default :
/* AP-like interface and monitor:
* never scan , always connected
*/
if ( bssid )
cfg80211_del_sta ( ndev , bssid , GFP_KERNEL ) ;
break ;
2012-12-21 01:13:19 +04:00
}
}
static void wil_disconnect_worker ( struct work_struct * work )
{
struct wil6210_priv * wil = container_of ( work ,
struct wil6210_priv , disconnect_worker ) ;
_wil6210_disconnect ( wil , NULL ) ;
}
static void wil_connect_timer_fn ( ulong x )
{
struct wil6210_priv * wil = ( void * ) x ;
2013-01-28 20:31:06 +04:00
wil_dbg_misc ( wil , " Connect timeout \n " ) ;
2012-12-21 01:13:19 +04:00
/* reschedule to thread context - disconnect won't
* run from atomic context
*/
schedule_work ( & wil - > disconnect_worker ) ;
}
2014-02-27 18:20:45 +04:00
static int wil_find_free_vring ( struct wil6210_priv * wil )
{
int i ;
for ( i = 0 ; i < WIL6210_MAX_TX_RINGS ; i + + ) {
if ( ! wil - > vring_tx [ i ] . va )
return i ;
}
return - EINVAL ;
}
2013-03-13 16:12:43 +04:00
static void wil_connect_worker ( struct work_struct * work )
{
int rc ;
struct wil6210_priv * wil = container_of ( work , struct wil6210_priv ,
connect_worker ) ;
int cid = wil - > pending_connect_cid ;
2014-02-27 18:20:45 +04:00
int ringid = wil_find_free_vring ( wil ) ;
2013-03-13 16:12:43 +04:00
if ( cid < 0 ) {
wil_err ( wil , " No connection pending \n " ) ;
return ;
}
wil_dbg_wmi ( wil , " Configure for connection CID %d \n " , cid ) ;
2014-02-27 18:20:45 +04:00
rc = wil_vring_init_tx ( wil , ringid , WIL6210_TX_RING_SIZE , cid , 0 ) ;
2013-03-13 16:12:43 +04:00
wil - > pending_connect_cid = - 1 ;
2014-02-27 18:20:43 +04:00
if ( rc = = 0 ) {
wil - > sta [ cid ] . status = wil_sta_connected ;
2013-03-13 16:12:43 +04:00
wil_link_on ( wil ) ;
2014-02-27 18:20:43 +04:00
} else {
wil - > sta [ cid ] . status = wil_sta_unused ;
}
2013-03-13 16:12:43 +04:00
}
2012-12-21 01:13:19 +04:00
int wil_priv_init ( struct wil6210_priv * wil )
{
2013-01-28 20:31:06 +04:00
wil_dbg_misc ( wil , " %s() \n " , __func__ ) ;
2012-12-21 01:13:19 +04:00
2014-02-27 18:20:43 +04:00
memset ( wil - > sta , 0 , sizeof ( wil - > sta ) ) ;
2012-12-21 01:13:19 +04:00
mutex_init ( & wil - > mutex ) ;
mutex_init ( & wil - > wmi_mutex ) ;
init_completion ( & wil - > wmi_ready ) ;
wil - > pending_connect_cid = - 1 ;
setup_timer ( & wil - > connect_timer , wil_connect_timer_fn , ( ulong ) wil ) ;
2013-03-13 16:12:43 +04:00
INIT_WORK ( & wil - > connect_worker , wil_connect_worker ) ;
2012-12-21 01:13:19 +04:00
INIT_WORK ( & wil - > disconnect_worker , wil_disconnect_worker ) ;
INIT_WORK ( & wil - > wmi_event_worker , wmi_event_worker ) ;
INIT_LIST_HEAD ( & wil - > pending_wmi_ev ) ;
spin_lock_init ( & wil - > wmi_ev_lock ) ;
wil - > wmi_wq = create_singlethread_workqueue ( WIL_NAME " _wmi " ) ;
if ( ! wil - > wmi_wq )
return - EAGAIN ;
wil - > wmi_wq_conn = create_singlethread_workqueue ( WIL_NAME " _connect " ) ;
if ( ! wil - > wmi_wq_conn ) {
destroy_workqueue ( wil - > wmi_wq ) ;
return - EAGAIN ;
}
return 0 ;
}
void wil6210_disconnect ( struct wil6210_priv * wil , void * bssid )
{
del_timer_sync ( & wil - > connect_timer ) ;
_wil6210_disconnect ( wil , bssid ) ;
}
void wil_priv_deinit ( struct wil6210_priv * wil )
{
cancel_work_sync ( & wil - > disconnect_worker ) ;
wil6210_disconnect ( wil , NULL ) ;
wmi_event_flush ( wil ) ;
destroy_workqueue ( wil - > wmi_wq_conn ) ;
destroy_workqueue ( wil - > wmi_wq ) ;
}
static void wil_target_reset ( struct wil6210_priv * wil )
{
2014-03-17 17:34:12 +04:00
int delay = 0 ;
2014-03-17 17:34:10 +04:00
u32 baud_rate ;
u32 rev_id ;
2013-01-28 20:31:06 +04:00
wil_dbg_misc ( wil , " Resetting... \n " ) ;
2012-12-21 01:13:19 +04:00
2014-03-17 17:34:10 +04:00
/* register read */
# define R(a) ioread32(wil->csr + HOSTADDR(a))
2012-12-21 01:13:19 +04:00
/* register write */
# define W(a, v) iowrite32(v, wil->csr + HOSTADDR(a))
/* register set = read, OR, write */
2014-03-17 17:34:15 +04:00
# define S(a, v) W(a, R(a) | v)
/* register clear = read, AND with inverted, write */
# define C(a, v) W(a, R(a) & ~v)
2012-12-21 01:13:19 +04:00
2014-03-17 17:34:14 +04:00
wil - > hw_version = R ( RGF_USER_FW_REV_ID ) ;
2014-03-17 17:34:10 +04:00
rev_id = wil - > hw_version & 0xff ;
2012-12-21 01:13:19 +04:00
/* hpal_perst_from_pad_src_n_mask */
S ( RGF_USER_CLKS_CTL_SW_RST_MASK_0 , BIT ( 6 ) ) ;
/* car_perst_rst_src_n_mask */
S ( RGF_USER_CLKS_CTL_SW_RST_MASK_0 , BIT ( 7 ) ) ;
W ( RGF_USER_MAC_CPU_0 , BIT ( 1 ) ) ; /* mac_cpu_man_rst */
W ( RGF_USER_USER_CPU_0 , BIT ( 1 ) ) ; /* user_cpu_man_rst */
W ( RGF_USER_CLKS_CTL_SW_RST_VEC_2 , 0xFE000000 ) ;
W ( RGF_USER_CLKS_CTL_SW_RST_VEC_1 , 0x0000003F ) ;
W ( RGF_USER_CLKS_CTL_SW_RST_VEC_3 , 0x00000170 ) ;
W ( RGF_USER_CLKS_CTL_SW_RST_VEC_0 , 0xFFE7FC00 ) ;
W ( RGF_USER_CLKS_CTL_SW_RST_VEC_3 , 0 ) ;
W ( RGF_USER_CLKS_CTL_SW_RST_VEC_2 , 0 ) ;
W ( RGF_USER_CLKS_CTL_SW_RST_VEC_1 , 0 ) ;
W ( RGF_USER_CLKS_CTL_SW_RST_VEC_0 , 0 ) ;
W ( RGF_USER_CLKS_CTL_SW_RST_VEC_3 , 0x00000001 ) ;
2014-03-17 17:34:10 +04:00
if ( rev_id = = 1 ) {
W ( RGF_USER_CLKS_CTL_SW_RST_VEC_2 , 0x00000080 ) ;
} else {
2014-03-17 17:34:14 +04:00
W ( RGF_PCIE_LOS_COUNTER_CTL , BIT ( 6 ) | BIT ( 8 ) ) ;
2014-03-17 17:34:10 +04:00
W ( RGF_USER_CLKS_CTL_SW_RST_VEC_2 , 0x00008000 ) ;
}
2012-12-21 01:13:19 +04:00
W ( RGF_USER_CLKS_CTL_SW_RST_VEC_0 , 0 ) ;
2014-03-17 17:34:10 +04:00
/* wait until device ready. Use baud rate */
do {
msleep ( 1 ) ;
baud_rate = R ( RGF_USER_SERIAL_BAUD_RATE ) ;
2014-03-17 17:34:12 +04:00
if ( delay + + > 100 ) {
2014-03-17 17:34:10 +04:00
wil_err ( wil , " Reset not completed \n " ) ;
return ;
}
} while ( baud_rate ! = 0x15e ) ;
if ( rev_id = = 2 )
2014-03-17 17:34:14 +04:00
W ( RGF_PCIE_LOS_COUNTER_CTL , BIT ( 8 ) ) ;
2014-03-17 17:34:10 +04:00
2014-03-17 17:34:15 +04:00
C ( RGF_USER_CLKS_CTL_0 , BIT_USER_CLKS_RST_PWGD ) ;
2014-03-17 17:34:12 +04:00
wil_dbg_misc ( wil , " Reset completed in %d ms \n " , delay ) ;
2012-12-21 01:13:19 +04:00
2014-03-17 17:34:10 +04:00
# undef R
2012-12-21 01:13:19 +04:00
# undef W
# undef S
2014-03-17 17:34:15 +04:00
# undef C
2012-12-21 01:13:19 +04:00
}
void wil_mbox_ring_le2cpus ( struct wil6210_mbox_ring * r )
{
le32_to_cpus ( & r - > base ) ;
le16_to_cpus ( & r - > entry_size ) ;
le16_to_cpus ( & r - > size ) ;
le32_to_cpus ( & r - > tail ) ;
le32_to_cpus ( & r - > head ) ;
}
static int wil_wait_for_fw_ready ( struct wil6210_priv * wil )
{
ulong to = msecs_to_jiffies ( 1000 ) ;
ulong left = wait_for_completion_timeout ( & wil - > wmi_ready , to ) ;
if ( 0 = = left ) {
wil_err ( wil , " Firmware not ready \n " ) ;
return - ETIME ;
} else {
2013-01-28 20:31:06 +04:00
wil_dbg_misc ( wil , " FW ready after %d ms \n " ,
2013-01-28 20:30:58 +04:00
jiffies_to_msecs ( to - left ) ) ;
2012-12-21 01:13:19 +04:00
}
return 0 ;
}
/*
* We reset all the structures , and we reset the UMAC .
* After calling this routine , you ' re expected to reload
* the firmware .
*/
int wil_reset ( struct wil6210_priv * wil )
{
int rc ;
cancel_work_sync ( & wil - > disconnect_worker ) ;
wil6210_disconnect ( wil , NULL ) ;
2013-01-28 20:31:05 +04:00
wil6210_disable_irq ( wil ) ;
wil - > status = 0 ;
2012-12-21 01:13:19 +04:00
wmi_event_flush ( wil ) ;
flush_workqueue ( wil - > wmi_wq_conn ) ;
2013-01-28 20:31:05 +04:00
flush_workqueue ( wil - > wmi_wq ) ;
2012-12-21 01:13:19 +04:00
/* TODO: put MAC in reset */
wil_target_reset ( wil ) ;
2014-03-17 17:34:17 +04:00
wil_rx_fini ( wil ) ;
2012-12-21 01:13:19 +04:00
/* init after reset */
wil - > pending_connect_cid = - 1 ;
2013-11-15 02:32:02 +04:00
reinit_completion ( & wil - > wmi_ready ) ;
2012-12-21 01:13:19 +04:00
/* TODO: release MAC reset */
wil6210_enable_irq ( wil ) ;
/* we just started MAC, wait for FW ready */
rc = wil_wait_for_fw_ready ( wil ) ;
return rc ;
}
void wil_link_on ( struct wil6210_priv * wil )
{
struct net_device * ndev = wil_to_ndev ( wil ) ;
2013-01-28 20:31:06 +04:00
wil_dbg_misc ( wil , " %s() \n " , __func__ ) ;
2012-12-21 01:13:19 +04:00
netif_carrier_on ( ndev ) ;
netif_tx_wake_all_queues ( ndev ) ;
}
void wil_link_off ( struct wil6210_priv * wil )
{
struct net_device * ndev = wil_to_ndev ( wil ) ;
2013-01-28 20:31:06 +04:00
wil_dbg_misc ( wil , " %s() \n " , __func__ ) ;
2012-12-21 01:13:19 +04:00
netif_tx_stop_all_queues ( ndev ) ;
netif_carrier_off ( ndev ) ;
}
static int __wil_up ( struct wil6210_priv * wil )
{
struct net_device * ndev = wil_to_ndev ( wil ) ;
struct wireless_dev * wdev = wil - > wdev ;
int rc ;
rc = wil_reset ( wil ) ;
if ( rc )
return rc ;
2013-06-09 10:12:55 +04:00
/* Rx VRING. After MAC and beacon */
rc = wil_rx_init ( wil ) ;
if ( rc )
return rc ;
2012-12-21 01:13:19 +04:00
switch ( wdev - > iftype ) {
case NL80211_IFTYPE_STATION :
2013-01-28 20:31:06 +04:00
wil_dbg_misc ( wil , " type: STATION \n " ) ;
2012-12-21 01:13:19 +04:00
ndev - > type = ARPHRD_ETHER ;
break ;
case NL80211_IFTYPE_AP :
2013-01-28 20:31:06 +04:00
wil_dbg_misc ( wil , " type: AP \n " ) ;
2012-12-21 01:13:19 +04:00
ndev - > type = ARPHRD_ETHER ;
break ;
case NL80211_IFTYPE_P2P_CLIENT :
2013-01-28 20:31:06 +04:00
wil_dbg_misc ( wil , " type: P2P_CLIENT \n " ) ;
2012-12-21 01:13:19 +04:00
ndev - > type = ARPHRD_ETHER ;
break ;
case NL80211_IFTYPE_P2P_GO :
2013-01-28 20:31:06 +04:00
wil_dbg_misc ( wil , " type: P2P_GO \n " ) ;
2012-12-21 01:13:19 +04:00
ndev - > type = ARPHRD_ETHER ;
break ;
case NL80211_IFTYPE_MONITOR :
2013-01-28 20:31:06 +04:00
wil_dbg_misc ( wil , " type: Monitor \n " ) ;
2012-12-21 01:13:19 +04:00
ndev - > type = ARPHRD_IEEE80211_RADIOTAP ;
/* ARPHRD_IEEE80211 or ARPHRD_IEEE80211_RADIOTAP ? */
break ;
default :
return - EOPNOTSUPP ;
}
/* MAC address - pre-requisite for other commands */
wmi_set_mac_address ( wil , ndev - > dev_addr ) ;
2013-05-12 15:43:36 +04:00
napi_enable ( & wil - > napi_rx ) ;
napi_enable ( & wil - > napi_tx ) ;
2012-12-21 01:13:19 +04:00
return 0 ;
}
int wil_up ( struct wil6210_priv * wil )
{
int rc ;
mutex_lock ( & wil - > mutex ) ;
rc = __wil_up ( wil ) ;
mutex_unlock ( & wil - > mutex ) ;
return rc ;
}
static int __wil_down ( struct wil6210_priv * wil )
{
2013-05-12 15:43:36 +04:00
napi_disable ( & wil - > napi_rx ) ;
napi_disable ( & wil - > napi_tx ) ;
2012-12-21 01:13:19 +04:00
if ( wil - > scan_request ) {
cfg80211_scan_done ( wil - > scan_request , true ) ;
wil - > scan_request = NULL ;
}
wil6210_disconnect ( wil , NULL ) ;
wil_rx_fini ( wil ) ;
return 0 ;
}
int wil_down ( struct wil6210_priv * wil )
{
int rc ;
mutex_lock ( & wil - > mutex ) ;
rc = __wil_down ( wil ) ;
mutex_unlock ( & wil - > mutex ) ;
return rc ;
}
2014-02-27 18:20:43 +04:00
int wil_find_cid ( struct wil6210_priv * wil , const u8 * mac )
{
int i ;
int rc = - ENOENT ;
for ( i = 0 ; i < ARRAY_SIZE ( wil - > sta ) ; i + + ) {
if ( ( wil - > sta [ i ] . status ! = wil_sta_unused ) & &
2014-02-27 18:20:53 +04:00
ether_addr_equal ( wil - > sta [ i ] . addr , mac ) ) {
2014-02-27 18:20:43 +04:00
rc = i ;
break ;
}
}
return rc ;
}