Skip to content
Projects
Groups
Snippets
Help
Loading...
Help
Support
Keyboard shortcuts
?
Submit feedback
Contribute to GitLab
Sign in / Register
Toggle navigation
L
linux
Project overview
Project overview
Details
Activity
Releases
Repository
Repository
Files
Commits
Branches
Tags
Contributors
Graph
Compare
Issues
0
Issues
0
List
Boards
Labels
Milestones
Merge Requests
0
Merge Requests
0
Analytics
Analytics
Repository
Value Stream
Wiki
Wiki
Snippets
Snippets
Members
Members
Collapse sidebar
Close sidebar
Activity
Graph
Create a new issue
Commits
Issue Boards
Open sidebar
Kirill Smelkov
linux
Commits
e94d5b7a
Commit
e94d5b7a
authored
Jan 30, 2012
by
David S. Miller
Browse files
Options
Browse Files
Download
Plain Diff
Merge branch 'davem-next.r8169' of
git://violet.fr.zoreil.com/romieu/linux
parents
30088a25
da78dbff
Changes
1
Show whitespace changes
Inline
Side-by-side
Showing
1 changed file
with
288 additions
and
270 deletions
+288
-270
drivers/net/ethernet/realtek/r8169.c
drivers/net/ethernet/realtek/r8169.c
+288
-270
No files found.
drivers/net/ethernet/realtek/r8169.c
View file @
e94d5b7a
...
@@ -667,6 +667,13 @@ struct rtl8169_counters {
...
@@ -667,6 +667,13 @@ struct rtl8169_counters {
__le16
tx_underun
;
__le16
tx_underun
;
};
};
enum
rtl_flag
{
RTL_FLAG_TASK_SLOW_PENDING
,
RTL_FLAG_TASK_RESET_PENDING
,
RTL_FLAG_TASK_PHY_PENDING
,
RTL_FLAG_MAX
};
struct
rtl8169_private
{
struct
rtl8169_private
{
void
__iomem
*
mmio_addr
;
/* memory map physical address */
void
__iomem
*
mmio_addr
;
/* memory map physical address */
struct
pci_dev
*
pci_dev
;
struct
pci_dev
*
pci_dev
;
...
@@ -688,9 +695,8 @@ struct rtl8169_private {
...
@@ -688,9 +695,8 @@ struct rtl8169_private {
struct
ring_info
tx_skb
[
NUM_TX_DESC
];
/* Tx data buffers */
struct
ring_info
tx_skb
[
NUM_TX_DESC
];
/* Tx data buffers */
struct
timer_list
timer
;
struct
timer_list
timer
;
u16
cp_cmd
;
u16
cp_cmd
;
u16
intr_event
;
u16
napi_event
;
u16
event_slow
;
u16
intr_mask
;
struct
mdio_ops
{
struct
mdio_ops
{
void
(
*
write
)(
void
__iomem
*
,
int
,
int
);
void
(
*
write
)(
void
__iomem
*
,
int
,
int
);
...
@@ -714,7 +720,14 @@ struct rtl8169_private {
...
@@ -714,7 +720,14 @@ struct rtl8169_private {
unsigned
int
(
*
phy_reset_pending
)(
struct
rtl8169_private
*
tp
);
unsigned
int
(
*
phy_reset_pending
)(
struct
rtl8169_private
*
tp
);
unsigned
int
(
*
link_ok
)(
void
__iomem
*
);
unsigned
int
(
*
link_ok
)(
void
__iomem
*
);
int
(
*
do_ioctl
)(
struct
rtl8169_private
*
tp
,
struct
mii_ioctl_data
*
data
,
int
cmd
);
int
(
*
do_ioctl
)(
struct
rtl8169_private
*
tp
,
struct
mii_ioctl_data
*
data
,
int
cmd
);
struct
delayed_work
task
;
struct
{
DECLARE_BITMAP
(
flags
,
RTL_FLAG_MAX
);
struct
mutex
mutex
;
struct
work_struct
work
;
bool
enabled
;
}
wk
;
unsigned
features
;
unsigned
features
;
struct
mii_if_info
mii
;
struct
mii_if_info
mii
;
...
@@ -764,13 +777,20 @@ static int rtl8169_close(struct net_device *dev);
...
@@ -764,13 +777,20 @@ static int rtl8169_close(struct net_device *dev);
static
void
rtl_set_rx_mode
(
struct
net_device
*
dev
);
static
void
rtl_set_rx_mode
(
struct
net_device
*
dev
);
static
void
rtl8169_tx_timeout
(
struct
net_device
*
dev
);
static
void
rtl8169_tx_timeout
(
struct
net_device
*
dev
);
static
struct
net_device_stats
*
rtl8169_get_stats
(
struct
net_device
*
dev
);
static
struct
net_device_stats
*
rtl8169_get_stats
(
struct
net_device
*
dev
);
static
int
rtl8169_rx_interrupt
(
struct
net_device
*
,
struct
rtl8169_private
*
,
void
__iomem
*
,
u32
budget
);
static
int
rtl8169_change_mtu
(
struct
net_device
*
dev
,
int
new_mtu
);
static
int
rtl8169_change_mtu
(
struct
net_device
*
dev
,
int
new_mtu
);
static
void
rtl8169_down
(
struct
net_device
*
dev
);
static
void
rtl8169_rx_clear
(
struct
rtl8169_private
*
tp
);
static
void
rtl8169_rx_clear
(
struct
rtl8169_private
*
tp
);
static
int
rtl8169_poll
(
struct
napi_struct
*
napi
,
int
budget
);
static
int
rtl8169_poll
(
struct
napi_struct
*
napi
,
int
budget
);
static
void
rtl_lock_work
(
struct
rtl8169_private
*
tp
)
{
mutex_lock
(
&
tp
->
wk
.
mutex
);
}
static
void
rtl_unlock_work
(
struct
rtl8169_private
*
tp
)
{
mutex_unlock
(
&
tp
->
wk
.
mutex
);
}
static
void
rtl_tx_performance_tweak
(
struct
pci_dev
*
pdev
,
u16
force
)
static
void
rtl_tx_performance_tweak
(
struct
pci_dev
*
pdev
,
u16
force
)
{
{
int
cap
=
pci_pcie_cap
(
pdev
);
int
cap
=
pci_pcie_cap
(
pdev
);
...
@@ -1180,12 +1200,51 @@ static u8 rtl8168d_efuse_read(void __iomem *ioaddr, int reg_addr)
...
@@ -1180,12 +1200,51 @@ static u8 rtl8168d_efuse_read(void __iomem *ioaddr, int reg_addr)
return
value
;
return
value
;
}
}
static
u16
rtl_get_events
(
struct
rtl8169_private
*
tp
)
{
void
__iomem
*
ioaddr
=
tp
->
mmio_addr
;
return
RTL_R16
(
IntrStatus
);
}
static
void
rtl_ack_events
(
struct
rtl8169_private
*
tp
,
u16
bits
)
{
void
__iomem
*
ioaddr
=
tp
->
mmio_addr
;
RTL_W16
(
IntrStatus
,
bits
);
mmiowb
();
}
static
void
rtl_irq_disable
(
struct
rtl8169_private
*
tp
)
{
void
__iomem
*
ioaddr
=
tp
->
mmio_addr
;
RTL_W16
(
IntrMask
,
0
);
mmiowb
();
}
static
void
rtl_irq_enable
(
struct
rtl8169_private
*
tp
,
u16
bits
)
{
void
__iomem
*
ioaddr
=
tp
->
mmio_addr
;
RTL_W16
(
IntrMask
,
bits
);
}
#define RTL_EVENT_NAPI_RX (RxOK | RxErr)
#define RTL_EVENT_NAPI_TX (TxOK | TxErr)
#define RTL_EVENT_NAPI (RTL_EVENT_NAPI_RX | RTL_EVENT_NAPI_TX)
static
void
rtl_irq_enable_all
(
struct
rtl8169_private
*
tp
)
{
rtl_irq_enable
(
tp
,
RTL_EVENT_NAPI
|
tp
->
event_slow
);
}
static
void
rtl8169_irq_mask_and_ack
(
struct
rtl8169_private
*
tp
)
static
void
rtl8169_irq_mask_and_ack
(
struct
rtl8169_private
*
tp
)
{
{
void
__iomem
*
ioaddr
=
tp
->
mmio_addr
;
void
__iomem
*
ioaddr
=
tp
->
mmio_addr
;
RTL_W16
(
IntrMask
,
0x0000
);
rtl_irq_disable
(
tp
);
RTL_W16
(
IntrStatus
,
tp
->
intr_event
);
rtl_ack_events
(
tp
,
RTL_EVENT_NAPI
|
tp
->
event_slow
);
RTL_R8
(
ChipCmd
);
RTL_R8
(
ChipCmd
);
}
}
...
@@ -1276,9 +1335,6 @@ static void __rtl8169_check_link_status(struct net_device *dev,
...
@@ -1276,9 +1335,6 @@ static void __rtl8169_check_link_status(struct net_device *dev,
struct
rtl8169_private
*
tp
,
struct
rtl8169_private
*
tp
,
void
__iomem
*
ioaddr
,
bool
pm
)
void
__iomem
*
ioaddr
,
bool
pm
)
{
{
unsigned
long
flags
;
spin_lock_irqsave
(
&
tp
->
lock
,
flags
);
if
(
tp
->
link_ok
(
ioaddr
))
{
if
(
tp
->
link_ok
(
ioaddr
))
{
rtl_link_chg_patch
(
tp
);
rtl_link_chg_patch
(
tp
);
/* This is to cancel a scheduled suspend if there's one. */
/* This is to cancel a scheduled suspend if there's one. */
...
@@ -1293,7 +1349,6 @@ static void __rtl8169_check_link_status(struct net_device *dev,
...
@@ -1293,7 +1349,6 @@ static void __rtl8169_check_link_status(struct net_device *dev,
if
(
pm
)
if
(
pm
)
pm_schedule_suspend
(
&
tp
->
pci_dev
->
dev
,
5000
);
pm_schedule_suspend
(
&
tp
->
pci_dev
->
dev
,
5000
);
}
}
spin_unlock_irqrestore
(
&
tp
->
lock
,
flags
);
}
}
static
void
rtl8169_check_link_status
(
struct
net_device
*
dev
,
static
void
rtl8169_check_link_status
(
struct
net_device
*
dev
,
...
@@ -1336,12 +1391,12 @@ static void rtl8169_get_wol(struct net_device *dev, struct ethtool_wolinfo *wol)
...
@@ -1336,12 +1391,12 @@ static void rtl8169_get_wol(struct net_device *dev, struct ethtool_wolinfo *wol)
{
{
struct
rtl8169_private
*
tp
=
netdev_priv
(
dev
);
struct
rtl8169_private
*
tp
=
netdev_priv
(
dev
);
spin_lock_irq
(
&
tp
->
lock
);
rtl_lock_work
(
tp
);
wol
->
supported
=
WAKE_ANY
;
wol
->
supported
=
WAKE_ANY
;
wol
->
wolopts
=
__rtl8169_get_wol
(
tp
);
wol
->
wolopts
=
__rtl8169_get_wol
(
tp
);
spin_unlock_irq
(
&
tp
->
lock
);
rtl_unlock_work
(
tp
);
}
}
static
void
__rtl8169_set_wol
(
struct
rtl8169_private
*
tp
,
u32
wolopts
)
static
void
__rtl8169_set_wol
(
struct
rtl8169_private
*
tp
,
u32
wolopts
)
...
@@ -1378,14 +1433,15 @@ static int rtl8169_set_wol(struct net_device *dev, struct ethtool_wolinfo *wol)
...
@@ -1378,14 +1433,15 @@ static int rtl8169_set_wol(struct net_device *dev, struct ethtool_wolinfo *wol)
{
{
struct
rtl8169_private
*
tp
=
netdev_priv
(
dev
);
struct
rtl8169_private
*
tp
=
netdev_priv
(
dev
);
spin_lock_irq
(
&
tp
->
lock
);
rtl_lock_work
(
tp
);
if
(
wol
->
wolopts
)
if
(
wol
->
wolopts
)
tp
->
features
|=
RTL_FEATURE_WOL
;
tp
->
features
|=
RTL_FEATURE_WOL
;
else
else
tp
->
features
&=
~
RTL_FEATURE_WOL
;
tp
->
features
&=
~
RTL_FEATURE_WOL
;
__rtl8169_set_wol
(
tp
,
wol
->
wolopts
);
__rtl8169_set_wol
(
tp
,
wol
->
wolopts
);
spin_unlock_irq
(
&
tp
->
lock
);
rtl_unlock_work
(
tp
);
device_set_wakeup_enable
(
&
tp
->
pci_dev
->
dev
,
wol
->
wolopts
);
device_set_wakeup_enable
(
&
tp
->
pci_dev
->
dev
,
wol
->
wolopts
);
...
@@ -1540,15 +1596,14 @@ static int rtl8169_set_speed(struct net_device *dev,
...
@@ -1540,15 +1596,14 @@ static int rtl8169_set_speed(struct net_device *dev,
static
int
rtl8169_set_settings
(
struct
net_device
*
dev
,
struct
ethtool_cmd
*
cmd
)
static
int
rtl8169_set_settings
(
struct
net_device
*
dev
,
struct
ethtool_cmd
*
cmd
)
{
{
struct
rtl8169_private
*
tp
=
netdev_priv
(
dev
);
struct
rtl8169_private
*
tp
=
netdev_priv
(
dev
);
unsigned
long
flags
;
int
ret
;
int
ret
;
del_timer_sync
(
&
tp
->
timer
);
del_timer_sync
(
&
tp
->
timer
);
spin_lock_irqsave
(
&
tp
->
lock
,
flags
);
rtl_lock_work
(
tp
);
ret
=
rtl8169_set_speed
(
dev
,
cmd
->
autoneg
,
ethtool_cmd_speed
(
cmd
),
ret
=
rtl8169_set_speed
(
dev
,
cmd
->
autoneg
,
ethtool_cmd_speed
(
cmd
),
cmd
->
duplex
,
cmd
->
advertising
);
cmd
->
duplex
,
cmd
->
advertising
);
spin_unlock_irqrestore
(
&
tp
->
lock
,
flags
);
rtl_unlock_work
(
tp
);
return
ret
;
return
ret
;
}
}
...
@@ -1568,14 +1623,12 @@ static netdev_features_t rtl8169_fix_features(struct net_device *dev,
...
@@ -1568,14 +1623,12 @@ static netdev_features_t rtl8169_fix_features(struct net_device *dev,
return
features
;
return
features
;
}
}
static
int
rtl8169_set_features
(
struct
net_device
*
dev
,
static
void
__
rtl8169_set_features
(
struct
net_device
*
dev
,
netdev_features_t
features
)
netdev_features_t
features
)
{
{
struct
rtl8169_private
*
tp
=
netdev_priv
(
dev
);
struct
rtl8169_private
*
tp
=
netdev_priv
(
dev
);
void
__iomem
*
ioaddr
=
tp
->
mmio_addr
;
unsigned
long
flags
;
spin_lock_irqsave
(
&
tp
->
lock
,
flags
)
;
void
__iomem
*
ioaddr
=
tp
->
mmio_addr
;
if
(
features
&
NETIF_F_RXCSUM
)
if
(
features
&
NETIF_F_RXCSUM
)
tp
->
cp_cmd
|=
RxChkSum
;
tp
->
cp_cmd
|=
RxChkSum
;
...
@@ -1589,12 +1642,21 @@ static int rtl8169_set_features(struct net_device *dev,
...
@@ -1589,12 +1642,21 @@ static int rtl8169_set_features(struct net_device *dev,
RTL_W16
(
CPlusCmd
,
tp
->
cp_cmd
);
RTL_W16
(
CPlusCmd
,
tp
->
cp_cmd
);
RTL_R16
(
CPlusCmd
);
RTL_R16
(
CPlusCmd
);
}
static
int
rtl8169_set_features
(
struct
net_device
*
dev
,
netdev_features_t
features
)
{
struct
rtl8169_private
*
tp
=
netdev_priv
(
dev
);
spin_unlock_irqrestore
(
&
tp
->
lock
,
flags
);
rtl_lock_work
(
tp
);
__rtl8169_set_features
(
dev
,
features
);
rtl_unlock_work
(
tp
);
return
0
;
return
0
;
}
}
static
inline
u32
rtl8169_tx_vlan_tag
(
struct
rtl8169_private
*
tp
,
static
inline
u32
rtl8169_tx_vlan_tag
(
struct
rtl8169_private
*
tp
,
struct
sk_buff
*
skb
)
struct
sk_buff
*
skb
)
{
{
...
@@ -1643,14 +1705,12 @@ static int rtl8169_gset_xmii(struct net_device *dev, struct ethtool_cmd *cmd)
...
@@ -1643,14 +1705,12 @@ static int rtl8169_gset_xmii(struct net_device *dev, struct ethtool_cmd *cmd)
static
int
rtl8169_get_settings
(
struct
net_device
*
dev
,
struct
ethtool_cmd
*
cmd
)
static
int
rtl8169_get_settings
(
struct
net_device
*
dev
,
struct
ethtool_cmd
*
cmd
)
{
{
struct
rtl8169_private
*
tp
=
netdev_priv
(
dev
);
struct
rtl8169_private
*
tp
=
netdev_priv
(
dev
);
unsigned
long
flags
;
int
rc
;
int
rc
;
spin_lock_irqsave
(
&
tp
->
lock
,
flags
);
rtl_lock_work
(
tp
);
rc
=
tp
->
get_settings
(
dev
,
cmd
);
rc
=
tp
->
get_settings
(
dev
,
cmd
);
rtl_unlock_work
(
tp
);
spin_unlock_irqrestore
(
&
tp
->
lock
,
flags
);
return
rc
;
return
rc
;
}
}
...
@@ -1658,14 +1718,15 @@ static void rtl8169_get_regs(struct net_device *dev, struct ethtool_regs *regs,
...
@@ -1658,14 +1718,15 @@ static void rtl8169_get_regs(struct net_device *dev, struct ethtool_regs *regs,
void
*
p
)
void
*
p
)
{
{
struct
rtl8169_private
*
tp
=
netdev_priv
(
dev
);
struct
rtl8169_private
*
tp
=
netdev_priv
(
dev
);
unsigned
long
flags
;
if
(
regs
->
len
>
R8169_REGS_SIZE
)
if
(
regs
->
len
>
R8169_REGS_SIZE
)
regs
->
len
=
R8169_REGS_SIZE
;
regs
->
len
=
R8169_REGS_SIZE
;
spin_lock_irqsave
(
&
tp
->
lock
,
flags
);
rtl_lock_work
(
tp
);
spin_lock_bh
(
&
tp
->
lock
);
memcpy_fromio
(
p
,
tp
->
mmio_addr
,
regs
->
len
);
memcpy_fromio
(
p
,
tp
->
mmio_addr
,
regs
->
len
);
spin_unlock_irqrestore
(
&
tp
->
lock
,
flags
);
spin_unlock_bh
(
&
tp
->
lock
);
rtl_unlock_work
(
tp
);
}
}
static
u32
rtl8169_get_msglevel
(
struct
net_device
*
dev
)
static
u32
rtl8169_get_msglevel
(
struct
net_device
*
dev
)
...
@@ -3182,18 +3243,14 @@ static void rtl_hw_phy_config(struct net_device *dev)
...
@@ -3182,18 +3243,14 @@ static void rtl_hw_phy_config(struct net_device *dev)
}
}
}
}
static
void
rtl
8169_phy_timer
(
unsigned
long
__opaque
)
static
void
rtl
_phy_work
(
struct
rtl8169_private
*
tp
)
{
{
struct
net_device
*
dev
=
(
struct
net_device
*
)
__opaque
;
struct
rtl8169_private
*
tp
=
netdev_priv
(
dev
);
struct
timer_list
*
timer
=
&
tp
->
timer
;
struct
timer_list
*
timer
=
&
tp
->
timer
;
void
__iomem
*
ioaddr
=
tp
->
mmio_addr
;
void
__iomem
*
ioaddr
=
tp
->
mmio_addr
;
unsigned
long
timeout
=
RTL8169_PHY_TIMEOUT
;
unsigned
long
timeout
=
RTL8169_PHY_TIMEOUT
;
assert
(
tp
->
mac_version
>
RTL_GIGA_MAC_VER_01
);
assert
(
tp
->
mac_version
>
RTL_GIGA_MAC_VER_01
);
spin_lock_irq
(
&
tp
->
lock
);
if
(
tp
->
phy_reset_pending
(
tp
))
{
if
(
tp
->
phy_reset_pending
(
tp
))
{
/*
/*
* A busy loop could burn quite a few cycles on nowadays CPU.
* A busy loop could burn quite a few cycles on nowadays CPU.
...
@@ -3204,32 +3261,45 @@ static void rtl8169_phy_timer(unsigned long __opaque)
...
@@ -3204,32 +3261,45 @@ static void rtl8169_phy_timer(unsigned long __opaque)
}
}
if
(
tp
->
link_ok
(
ioaddr
))
if
(
tp
->
link_ok
(
ioaddr
))
goto
out_unlock
;
return
;
netif_warn
(
tp
,
link
,
dev
,
"PHY reset until link up
\n
"
);
netif_warn
(
tp
,
link
,
tp
->
dev
,
"PHY reset until link up
\n
"
);
tp
->
phy_reset_enable
(
tp
);
tp
->
phy_reset_enable
(
tp
);
out_mod_timer:
out_mod_timer:
mod_timer
(
timer
,
jiffies
+
timeout
);
mod_timer
(
timer
,
jiffies
+
timeout
);
out_unlock:
}
spin_unlock_irq
(
&
tp
->
lock
);
static
void
rtl_schedule_task
(
struct
rtl8169_private
*
tp
,
enum
rtl_flag
flag
)
{
spin_lock
(
&
tp
->
lock
);
if
(
!
test_and_set_bit
(
flag
,
tp
->
wk
.
flags
))
schedule_work
(
&
tp
->
wk
.
work
);
spin_unlock
(
&
tp
->
lock
);
}
static
void
rtl_schedule_task_bh
(
struct
rtl8169_private
*
tp
,
enum
rtl_flag
flag
)
{
local_bh_disable
();
rtl_schedule_task
(
tp
,
flag
);
local_bh_enable
();
}
static
void
rtl8169_phy_timer
(
unsigned
long
__opaque
)
{
struct
net_device
*
dev
=
(
struct
net_device
*
)
__opaque
;
struct
rtl8169_private
*
tp
=
netdev_priv
(
dev
);
rtl_schedule_task_bh
(
tp
,
RTL_FLAG_TASK_PHY_PENDING
);
}
}
#ifdef CONFIG_NET_POLL_CONTROLLER
#ifdef CONFIG_NET_POLL_CONTROLLER
/*
* Polling 'interrupt' - used by things like netconsole to send skbs
* without having to re-enable interrupts. It's not called while
* the interrupt routine is executing.
*/
static
void
rtl8169_netpoll
(
struct
net_device
*
dev
)
static
void
rtl8169_netpoll
(
struct
net_device
*
dev
)
{
{
struct
rtl8169_private
*
tp
=
netdev_priv
(
dev
);
struct
rtl8169_private
*
tp
=
netdev_priv
(
dev
);
struct
pci_dev
*
pdev
=
tp
->
pci_dev
;
disable_irq
(
pdev
->
irq
);
rtl8169_interrupt
(
tp
->
pci_dev
->
irq
,
dev
);
rtl8169_interrupt
(
pdev
->
irq
,
dev
);
enable_irq
(
pdev
->
irq
);
}
}
#endif
#endif
...
@@ -3310,7 +3380,7 @@ static void rtl_rar_set(struct rtl8169_private *tp, u8 *addr)
...
@@ -3310,7 +3380,7 @@ static void rtl_rar_set(struct rtl8169_private *tp, u8 *addr)
low
=
addr
[
0
]
|
(
addr
[
1
]
<<
8
)
|
(
addr
[
2
]
<<
16
)
|
(
addr
[
3
]
<<
24
);
low
=
addr
[
0
]
|
(
addr
[
1
]
<<
8
)
|
(
addr
[
2
]
<<
16
)
|
(
addr
[
3
]
<<
24
);
high
=
addr
[
4
]
|
(
addr
[
5
]
<<
8
);
high
=
addr
[
4
]
|
(
addr
[
5
]
<<
8
);
spin_lock_irq
(
&
tp
->
lock
);
rtl_lock_work
(
tp
);
RTL_W8
(
Cfg9346
,
Cfg9346_Unlock
);
RTL_W8
(
Cfg9346
,
Cfg9346_Unlock
);
...
@@ -3334,7 +3404,7 @@ static void rtl_rar_set(struct rtl8169_private *tp, u8 *addr)
...
@@ -3334,7 +3404,7 @@ static void rtl_rar_set(struct rtl8169_private *tp, u8 *addr)
RTL_W8
(
Cfg9346
,
Cfg9346_Lock
);
RTL_W8
(
Cfg9346
,
Cfg9346_Lock
);
spin_unlock_irq
(
&
tp
->
lock
);
rtl_unlock_work
(
tp
);
}
}
static
int
rtl_set_mac_address
(
struct
net_device
*
dev
,
void
*
p
)
static
int
rtl_set_mac_address
(
struct
net_device
*
dev
,
void
*
p
)
...
@@ -3388,8 +3458,7 @@ static const struct rtl_cfg_info {
...
@@ -3388,8 +3458,7 @@ static const struct rtl_cfg_info {
void
(
*
hw_start
)(
struct
net_device
*
);
void
(
*
hw_start
)(
struct
net_device
*
);
unsigned
int
region
;
unsigned
int
region
;
unsigned
int
align
;
unsigned
int
align
;
u16
intr_event
;
u16
event_slow
;
u16
napi_event
;
unsigned
features
;
unsigned
features
;
u8
default_ver
;
u8
default_ver
;
}
rtl_cfg_infos
[]
=
{
}
rtl_cfg_infos
[]
=
{
...
@@ -3397,9 +3466,7 @@ static const struct rtl_cfg_info {
...
@@ -3397,9 +3466,7 @@ static const struct rtl_cfg_info {
.
hw_start
=
rtl_hw_start_8169
,
.
hw_start
=
rtl_hw_start_8169
,
.
region
=
1
,
.
region
=
1
,
.
align
=
0
,
.
align
=
0
,
.
intr_event
=
SYSErr
|
LinkChg
|
RxOverflow
|
.
event_slow
=
SYSErr
|
LinkChg
|
RxOverflow
|
RxFIFOOver
,
RxFIFOOver
|
TxErr
|
TxOK
|
RxOK
|
RxErr
,
.
napi_event
=
RxFIFOOver
|
TxErr
|
TxOK
|
RxOK
|
RxOverflow
,
.
features
=
RTL_FEATURE_GMII
,
.
features
=
RTL_FEATURE_GMII
,
.
default_ver
=
RTL_GIGA_MAC_VER_01
,
.
default_ver
=
RTL_GIGA_MAC_VER_01
,
},
},
...
@@ -3407,9 +3474,7 @@ static const struct rtl_cfg_info {
...
@@ -3407,9 +3474,7 @@ static const struct rtl_cfg_info {
.
hw_start
=
rtl_hw_start_8168
,
.
hw_start
=
rtl_hw_start_8168
,
.
region
=
2
,
.
region
=
2
,
.
align
=
8
,
.
align
=
8
,
.
intr_event
=
SYSErr
|
LinkChg
|
RxOverflow
|
.
event_slow
=
SYSErr
|
LinkChg
|
RxOverflow
,
TxErr
|
TxOK
|
RxOK
|
RxErr
,
.
napi_event
=
TxErr
|
TxOK
|
RxOK
|
RxOverflow
,
.
features
=
RTL_FEATURE_GMII
|
RTL_FEATURE_MSI
,
.
features
=
RTL_FEATURE_GMII
|
RTL_FEATURE_MSI
,
.
default_ver
=
RTL_GIGA_MAC_VER_11
,
.
default_ver
=
RTL_GIGA_MAC_VER_11
,
},
},
...
@@ -3417,9 +3482,8 @@ static const struct rtl_cfg_info {
...
@@ -3417,9 +3482,8 @@ static const struct rtl_cfg_info {
.
hw_start
=
rtl_hw_start_8101
,
.
hw_start
=
rtl_hw_start_8101
,
.
region
=
2
,
.
region
=
2
,
.
align
=
8
,
.
align
=
8
,
.
intr_event
=
SYSErr
|
LinkChg
|
RxOverflow
|
PCSTimeout
|
.
event_slow
=
SYSErr
|
LinkChg
|
RxOverflow
|
RxFIFOOver
|
RxFIFOOver
|
TxErr
|
TxOK
|
RxOK
|
RxErr
,
PCSTimeout
,
.
napi_event
=
RxFIFOOver
|
TxErr
|
TxOK
|
RxOK
|
RxOverflow
,
.
features
=
RTL_FEATURE_MSI
,
.
features
=
RTL_FEATURE_MSI
,
.
default_ver
=
RTL_GIGA_MAC_VER_13
,
.
default_ver
=
RTL_GIGA_MAC_VER_13
,
}
}
...
@@ -3824,23 +3888,21 @@ static void r8168dp_hw_jumbo_disable(struct rtl8169_private *tp)
...
@@ -3824,23 +3888,21 @@ static void r8168dp_hw_jumbo_disable(struct rtl8169_private *tp)
static
void
r8168e_hw_jumbo_enable
(
struct
rtl8169_private
*
tp
)
static
void
r8168e_hw_jumbo_enable
(
struct
rtl8169_private
*
tp
)
{
{
void
__iomem
*
ioaddr
=
tp
->
mmio_addr
;
void
__iomem
*
ioaddr
=
tp
->
mmio_addr
;
struct
pci_dev
*
pdev
=
tp
->
pci_dev
;
RTL_W8
(
MaxTxPacketSize
,
0x3f
);
RTL_W8
(
MaxTxPacketSize
,
0x3f
);
RTL_W8
(
Config3
,
RTL_R8
(
Config3
)
|
Jumbo_En0
);
RTL_W8
(
Config3
,
RTL_R8
(
Config3
)
|
Jumbo_En0
);
RTL_W8
(
Config4
,
RTL_R8
(
Config4
)
|
0x01
);
RTL_W8
(
Config4
,
RTL_R8
(
Config4
)
|
0x01
);
pci_write_config_byte
(
pdev
,
0x79
,
0x20
);
rtl_tx_performance_tweak
(
tp
->
pci_dev
,
0x2
<<
MAX_READ_REQUEST_SHIFT
);
}
}
static
void
r8168e_hw_jumbo_disable
(
struct
rtl8169_private
*
tp
)
static
void
r8168e_hw_jumbo_disable
(
struct
rtl8169_private
*
tp
)
{
{
void
__iomem
*
ioaddr
=
tp
->
mmio_addr
;
void
__iomem
*
ioaddr
=
tp
->
mmio_addr
;
struct
pci_dev
*
pdev
=
tp
->
pci_dev
;
RTL_W8
(
MaxTxPacketSize
,
0x0c
);
RTL_W8
(
MaxTxPacketSize
,
0x0c
);
RTL_W8
(
Config3
,
RTL_R8
(
Config3
)
&
~
Jumbo_En0
);
RTL_W8
(
Config3
,
RTL_R8
(
Config3
)
&
~
Jumbo_En0
);
RTL_W8
(
Config4
,
RTL_R8
(
Config4
)
&
~
0x01
);
RTL_W8
(
Config4
,
RTL_R8
(
Config4
)
&
~
0x01
);
pci_write_config_byte
(
pdev
,
0x79
,
0x50
);
rtl_tx_performance_tweak
(
tp
->
pci_dev
,
0x5
<<
MAX_READ_REQUEST_SHIFT
);
}
}
static
void
r8168b_0_hw_jumbo_enable
(
struct
rtl8169_private
*
tp
)
static
void
r8168b_0_hw_jumbo_enable
(
struct
rtl8169_private
*
tp
)
...
@@ -4048,11 +4110,11 @@ rtl8169_init_one(struct pci_dev *pdev, const struct pci_device_id *ent)
...
@@ -4048,11 +4110,11 @@ rtl8169_init_one(struct pci_dev *pdev, const struct pci_device_id *ent)
rtl_init_rxcfg
(
tp
);
rtl_init_rxcfg
(
tp
);
RTL_W16
(
IntrMask
,
0x0000
);
rtl_irq_disable
(
tp
);
rtl_hw_reset
(
tp
);
rtl_hw_reset
(
tp
);
RTL_W16
(
IntrStatus
,
0xffff
);
rtl_ack_events
(
tp
,
0xffff
);
pci_set_master
(
pdev
);
pci_set_master
(
pdev
);
...
@@ -4099,6 +4161,7 @@ rtl8169_init_one(struct pci_dev *pdev, const struct pci_device_id *ent)
...
@@ -4099,6 +4161,7 @@ rtl8169_init_one(struct pci_dev *pdev, const struct pci_device_id *ent)
}
}
spin_lock_init
(
&
tp
->
lock
);
spin_lock_init
(
&
tp
->
lock
);
mutex_init
(
&
tp
->
wk
.
mutex
);
/* Get MAC address */
/* Get MAC address */
for
(
i
=
0
;
i
<
ETH_ALEN
;
i
++
)
for
(
i
=
0
;
i
<
ETH_ALEN
;
i
++
)
...
@@ -4126,10 +4189,8 @@ rtl8169_init_one(struct pci_dev *pdev, const struct pci_device_id *ent)
...
@@ -4126,10 +4189,8 @@ rtl8169_init_one(struct pci_dev *pdev, const struct pci_device_id *ent)
/* 8110SCd requires hardware Rx VLAN - disallow toggling */
/* 8110SCd requires hardware Rx VLAN - disallow toggling */
dev
->
hw_features
&=
~
NETIF_F_HW_VLAN_RX
;
dev
->
hw_features
&=
~
NETIF_F_HW_VLAN_RX
;
tp
->
intr_mask
=
0xffff
;
tp
->
hw_start
=
cfg
->
hw_start
;
tp
->
hw_start
=
cfg
->
hw_start
;
tp
->
intr_event
=
cfg
->
intr_event
;
tp
->
event_slow
=
cfg
->
event_slow
;
tp
->
napi_event
=
cfg
->
napi_event
;
tp
->
opts1_mask
=
(
tp
->
mac_version
!=
RTL_GIGA_MAC_VER_01
)
?
tp
->
opts1_mask
=
(
tp
->
mac_version
!=
RTL_GIGA_MAC_VER_01
)
?
~
(
RxBOVF
|
RxFOVF
)
:
~
0
;
~
(
RxBOVF
|
RxFOVF
)
:
~
0
;
...
@@ -4196,7 +4257,7 @@ static void __devexit rtl8169_remove_one(struct pci_dev *pdev)
...
@@ -4196,7 +4257,7 @@ static void __devexit rtl8169_remove_one(struct pci_dev *pdev)
rtl8168_driver_stop
(
tp
);
rtl8168_driver_stop
(
tp
);
}
}
cancel_
delayed_work_sync
(
&
tp
->
tas
k
);
cancel_
work_sync
(
&
tp
->
wk
.
wor
k
);
unregister_netdev
(
dev
);
unregister_netdev
(
dev
);
...
@@ -4257,6 +4318,8 @@ static void rtl_request_firmware(struct rtl8169_private *tp)
...
@@ -4257,6 +4318,8 @@ static void rtl_request_firmware(struct rtl8169_private *tp)
rtl_request_uncached_firmware
(
tp
);
rtl_request_uncached_firmware
(
tp
);
}
}
static
void
rtl_task
(
struct
work_struct
*
);
static
int
rtl8169_open
(
struct
net_device
*
dev
)
static
int
rtl8169_open
(
struct
net_device
*
dev
)
{
{
struct
rtl8169_private
*
tp
=
netdev_priv
(
dev
);
struct
rtl8169_private
*
tp
=
netdev_priv
(
dev
);
...
@@ -4284,7 +4347,7 @@ static int rtl8169_open(struct net_device *dev)
...
@@ -4284,7 +4347,7 @@ static int rtl8169_open(struct net_device *dev)
if
(
retval
<
0
)
if
(
retval
<
0
)
goto
err_free_rx_1
;
goto
err_free_rx_1
;
INIT_
DELAYED_WORK
(
&
tp
->
task
,
NULL
);
INIT_
WORK
(
&
tp
->
wk
.
work
,
rtl_task
);
smp_mb
();
smp_mb
();
...
@@ -4296,16 +4359,24 @@ static int rtl8169_open(struct net_device *dev)
...
@@ -4296,16 +4359,24 @@ static int rtl8169_open(struct net_device *dev)
if
(
retval
<
0
)
if
(
retval
<
0
)
goto
err_release_fw_2
;
goto
err_release_fw_2
;
rtl_lock_work
(
tp
);
tp
->
wk
.
enabled
=
true
;
napi_enable
(
&
tp
->
napi
);
napi_enable
(
&
tp
->
napi
);
rtl8169_init_phy
(
dev
,
tp
);
rtl8169_init_phy
(
dev
,
tp
);
rtl8169_set_features
(
dev
,
dev
->
features
);
__
rtl8169_set_features
(
dev
,
dev
->
features
);
rtl_pll_power_up
(
tp
);
rtl_pll_power_up
(
tp
);
rtl_hw_start
(
dev
);
rtl_hw_start
(
dev
);
netif_start_queue
(
dev
);
rtl_unlock_work
(
tp
);
tp
->
saved_wolopts
=
0
;
tp
->
saved_wolopts
=
0
;
pm_runtime_put_noidle
(
&
pdev
->
dev
);
pm_runtime_put_noidle
(
&
pdev
->
dev
);
...
@@ -4379,7 +4450,7 @@ static void rtl_hw_start(struct net_device *dev)
...
@@ -4379,7 +4450,7 @@ static void rtl_hw_start(struct net_device *dev)
tp
->
hw_start
(
dev
);
tp
->
hw_start
(
dev
);
netif_start_queue
(
dev
);
rtl_irq_enable_all
(
tp
);
}
}
static
void
rtl_set_rx_tx_desc_registers
(
struct
rtl8169_private
*
tp
,
static
void
rtl_set_rx_tx_desc_registers
(
struct
rtl8169_private
*
tp
,
...
@@ -4506,9 +4577,6 @@ static void rtl_hw_start_8169(struct net_device *dev)
...
@@ -4506,9 +4577,6 @@ static void rtl_hw_start_8169(struct net_device *dev)
/* no early-rx interrupts */
/* no early-rx interrupts */
RTL_W16
(
MultiIntr
,
RTL_R16
(
MultiIntr
)
&
0xF000
);
RTL_W16
(
MultiIntr
,
RTL_R16
(
MultiIntr
)
&
0xF000
);
/* Enable all known interrupts by setting the interrupt mask. */
RTL_W16
(
IntrMask
,
tp
->
intr_event
);
}
}
static
void
rtl_csi_access_enable
(
void
__iomem
*
ioaddr
,
u32
bits
)
static
void
rtl_csi_access_enable
(
void
__iomem
*
ioaddr
,
u32
bits
)
...
@@ -4888,8 +4956,8 @@ static void rtl_hw_start_8168(struct net_device *dev)
...
@@ -4888,8 +4956,8 @@ static void rtl_hw_start_8168(struct net_device *dev)
/* Work around for RxFIFO overflow. */
/* Work around for RxFIFO overflow. */
if
(
tp
->
mac_version
==
RTL_GIGA_MAC_VER_11
)
{
if
(
tp
->
mac_version
==
RTL_GIGA_MAC_VER_11
)
{
tp
->
intr_event
|=
RxFIFOOver
|
PCSTimeout
;
tp
->
event_slow
|=
RxFIFOOver
|
PCSTimeout
;
tp
->
intr_event
&=
~
RxOverflow
;
tp
->
event_slow
&=
~
RxOverflow
;
}
}
rtl_set_rx_tx_desc_registers
(
tp
,
ioaddr
);
rtl_set_rx_tx_desc_registers
(
tp
,
ioaddr
);
...
@@ -4977,8 +5045,6 @@ static void rtl_hw_start_8168(struct net_device *dev)
...
@@ -4977,8 +5045,6 @@ static void rtl_hw_start_8168(struct net_device *dev)
RTL_W8
(
Cfg9346
,
Cfg9346_Lock
);
RTL_W8
(
Cfg9346
,
Cfg9346_Lock
);
RTL_W16
(
MultiIntr
,
RTL_R16
(
MultiIntr
)
&
0xF000
);
RTL_W16
(
MultiIntr
,
RTL_R16
(
MultiIntr
)
&
0xF000
);
RTL_W16
(
IntrMask
,
tp
->
intr_event
);
}
}
#define R810X_CPCMD_QUIRK_MASK (\
#define R810X_CPCMD_QUIRK_MASK (\
...
@@ -5077,10 +5143,8 @@ static void rtl_hw_start_8101(struct net_device *dev)
...
@@ -5077,10 +5143,8 @@ static void rtl_hw_start_8101(struct net_device *dev)
void
__iomem
*
ioaddr
=
tp
->
mmio_addr
;
void
__iomem
*
ioaddr
=
tp
->
mmio_addr
;
struct
pci_dev
*
pdev
=
tp
->
pci_dev
;
struct
pci_dev
*
pdev
=
tp
->
pci_dev
;
if
(
tp
->
mac_version
>=
RTL_GIGA_MAC_VER_30
)
{
if
(
tp
->
mac_version
>=
RTL_GIGA_MAC_VER_30
)
tp
->
intr_event
&=
~
RxFIFOOver
;
tp
->
event_slow
&=
~
RxFIFOOver
;
tp
->
napi_event
&=
~
RxFIFOOver
;
}
if
(
tp
->
mac_version
==
RTL_GIGA_MAC_VER_13
||
if
(
tp
->
mac_version
==
RTL_GIGA_MAC_VER_13
||
tp
->
mac_version
==
RTL_GIGA_MAC_VER_16
)
{
tp
->
mac_version
==
RTL_GIGA_MAC_VER_16
)
{
...
@@ -5136,8 +5200,6 @@ static void rtl_hw_start_8101(struct net_device *dev)
...
@@ -5136,8 +5200,6 @@ static void rtl_hw_start_8101(struct net_device *dev)
rtl_set_rx_mode
(
dev
);
rtl_set_rx_mode
(
dev
);
RTL_W16
(
MultiIntr
,
RTL_R16
(
MultiIntr
)
&
0xf000
);
RTL_W16
(
MultiIntr
,
RTL_R16
(
MultiIntr
)
&
0xf000
);
RTL_W16
(
IntrMask
,
tp
->
intr_event
);
}
}
static
int
rtl8169_change_mtu
(
struct
net_device
*
dev
,
int
new_mtu
)
static
int
rtl8169_change_mtu
(
struct
net_device
*
dev
,
int
new_mtu
)
...
@@ -5330,92 +5392,34 @@ static void rtl8169_tx_clear(struct rtl8169_private *tp)
...
@@ -5330,92 +5392,34 @@ static void rtl8169_tx_clear(struct rtl8169_private *tp)
tp
->
cur_tx
=
tp
->
dirty_tx
=
0
;
tp
->
cur_tx
=
tp
->
dirty_tx
=
0
;
}
}
static
void
rtl8169_schedule_work
(
struct
net_device
*
dev
,
work_func_t
task
)
static
void
rtl_reset_work
(
struct
rtl8169_private
*
tp
)
{
struct
rtl8169_private
*
tp
=
netdev_priv
(
dev
);
PREPARE_DELAYED_WORK
(
&
tp
->
task
,
task
);
schedule_delayed_work
(
&
tp
->
task
,
4
);
}
static
void
rtl8169_wait_for_quiescence
(
struct
net_device
*
dev
)
{
struct
rtl8169_private
*
tp
=
netdev_priv
(
dev
);
void
__iomem
*
ioaddr
=
tp
->
mmio_addr
;
synchronize_irq
(
dev
->
irq
);
/* Wait for any pending NAPI task to complete */
napi_disable
(
&
tp
->
napi
);
rtl8169_irq_mask_and_ack
(
tp
);
tp
->
intr_mask
=
0xffff
;
RTL_W16
(
IntrMask
,
tp
->
intr_event
);
napi_enable
(
&
tp
->
napi
);
}
static
void
rtl8169_reinit_task
(
struct
work_struct
*
work
)
{
struct
rtl8169_private
*
tp
=
container_of
(
work
,
struct
rtl8169_private
,
task
.
work
);
struct
net_device
*
dev
=
tp
->
dev
;
int
ret
;
rtnl_lock
();
if
(
!
netif_running
(
dev
))
goto
out_unlock
;
rtl8169_wait_for_quiescence
(
dev
);
rtl8169_close
(
dev
);
ret
=
rtl8169_open
(
dev
);
if
(
unlikely
(
ret
<
0
))
{
if
(
net_ratelimit
())
netif_err
(
tp
,
drv
,
dev
,
"reinit failure (status = %d). Rescheduling
\n
"
,
ret
);
rtl8169_schedule_work
(
dev
,
rtl8169_reinit_task
);
}
out_unlock:
rtnl_unlock
();
}
static
void
rtl8169_reset_task
(
struct
work_struct
*
work
)
{
{
struct
rtl8169_private
*
tp
=
container_of
(
work
,
struct
rtl8169_private
,
task
.
work
);
struct
net_device
*
dev
=
tp
->
dev
;
struct
net_device
*
dev
=
tp
->
dev
;
int
i
;
int
i
;
rtnl_lock
();
napi_disable
(
&
tp
->
napi
);
netif_stop_queue
(
dev
);
if
(
!
netif_running
(
dev
))
synchronize_sched
();
goto
out_unlock
;
rtl8169_hw_reset
(
tp
);
rtl8169_hw_reset
(
tp
);
rtl8169_wait_for_quiescence
(
dev
);
for
(
i
=
0
;
i
<
NUM_RX_DESC
;
i
++
)
for
(
i
=
0
;
i
<
NUM_RX_DESC
;
i
++
)
rtl8169_mark_to_asic
(
tp
->
RxDescArray
+
i
,
rx_buf_sz
);
rtl8169_mark_to_asic
(
tp
->
RxDescArray
+
i
,
rx_buf_sz
);
rtl8169_tx_clear
(
tp
);
rtl8169_tx_clear
(
tp
);
rtl8169_init_ring_indexes
(
tp
);
rtl8169_init_ring_indexes
(
tp
);
napi_enable
(
&
tp
->
napi
);
rtl_hw_start
(
dev
);
rtl_hw_start
(
dev
);
netif_wake_queue
(
dev
);
netif_wake_queue
(
dev
);
rtl8169_check_link_status
(
dev
,
tp
,
tp
->
mmio_addr
);
rtl8169_check_link_status
(
dev
,
tp
,
tp
->
mmio_addr
);
out_unlock:
rtnl_unlock
();
}
}
static
void
rtl8169_tx_timeout
(
struct
net_device
*
dev
)
static
void
rtl8169_tx_timeout
(
struct
net_device
*
dev
)
{
{
rtl8169_schedule_work
(
dev
,
rtl8169_reset_task
);
struct
rtl8169_private
*
tp
=
netdev_priv
(
dev
);
rtl_schedule_task
(
tp
,
RTL_FLAG_TASK_RESET_PENDING
);
}
}
static
int
rtl8169_xmit_frags
(
struct
rtl8169_private
*
tp
,
struct
sk_buff
*
skb
,
static
int
rtl8169_xmit_frags
(
struct
rtl8169_private
*
tp
,
struct
sk_buff
*
skb
,
...
@@ -5552,9 +5556,11 @@ static netdev_tx_t rtl8169_start_xmit(struct sk_buff *skb,
...
@@ -5552,9 +5556,11 @@ static netdev_tx_t rtl8169_start_xmit(struct sk_buff *skb,
RTL_W8
(
TxPoll
,
NPQ
);
RTL_W8
(
TxPoll
,
NPQ
);
mmiowb
();
if
(
TX_BUFFS_AVAIL
(
tp
)
<
MAX_SKB_FRAGS
)
{
if
(
TX_BUFFS_AVAIL
(
tp
)
<
MAX_SKB_FRAGS
)
{
netif_stop_queue
(
dev
);
netif_stop_queue
(
dev
);
smp_
r
mb
();
smp_mb
();
if
(
TX_BUFFS_AVAIL
(
tp
)
>=
MAX_SKB_FRAGS
)
if
(
TX_BUFFS_AVAIL
(
tp
)
>=
MAX_SKB_FRAGS
)
netif_wake_queue
(
dev
);
netif_wake_queue
(
dev
);
}
}
...
@@ -5618,12 +5624,10 @@ static void rtl8169_pcierr_interrupt(struct net_device *dev)
...
@@ -5618,12 +5624,10 @@ static void rtl8169_pcierr_interrupt(struct net_device *dev)
rtl8169_hw_reset
(
tp
);
rtl8169_hw_reset
(
tp
);
rtl
8169_schedule_work
(
dev
,
rtl8169_reinit_task
);
rtl
_schedule_task_bh
(
tp
,
RTL_FLAG_TASK_RESET_PENDING
);
}
}
static
void
rtl8169_tx_interrupt
(
struct
net_device
*
dev
,
static
void
rtl_tx
(
struct
net_device
*
dev
,
struct
rtl8169_private
*
tp
)
struct
rtl8169_private
*
tp
,
void
__iomem
*
ioaddr
)
{
{
unsigned
int
dirty_tx
,
tx_left
;
unsigned
int
dirty_tx
,
tx_left
;
...
@@ -5655,7 +5659,7 @@ static void rtl8169_tx_interrupt(struct net_device *dev,
...
@@ -5655,7 +5659,7 @@ static void rtl8169_tx_interrupt(struct net_device *dev,
if
(
tp
->
dirty_tx
!=
dirty_tx
)
{
if
(
tp
->
dirty_tx
!=
dirty_tx
)
{
tp
->
dirty_tx
=
dirty_tx
;
tp
->
dirty_tx
=
dirty_tx
;
smp_
w
mb
();
smp_mb
();
if
(
netif_queue_stopped
(
dev
)
&&
if
(
netif_queue_stopped
(
dev
)
&&
(
TX_BUFFS_AVAIL
(
tp
)
>=
MAX_SKB_FRAGS
))
{
(
TX_BUFFS_AVAIL
(
tp
)
>=
MAX_SKB_FRAGS
))
{
netif_wake_queue
(
dev
);
netif_wake_queue
(
dev
);
...
@@ -5666,10 +5670,12 @@ static void rtl8169_tx_interrupt(struct net_device *dev,
...
@@ -5666,10 +5670,12 @@ static void rtl8169_tx_interrupt(struct net_device *dev,
* of start_xmit activity is detected (if it is not detected,
* of start_xmit activity is detected (if it is not detected,
* it is slow enough). -- FR
* it is slow enough). -- FR
*/
*/
smp_rmb
();
if
(
tp
->
cur_tx
!=
dirty_tx
)
{
if
(
tp
->
cur_tx
!=
dirty_tx
)
void
__iomem
*
ioaddr
=
tp
->
mmio_addr
;
RTL_W8
(
TxPoll
,
NPQ
);
RTL_W8
(
TxPoll
,
NPQ
);
}
}
}
}
}
static
inline
int
rtl8169_fragmented_frame
(
u32
status
)
static
inline
int
rtl8169_fragmented_frame
(
u32
status
)
...
@@ -5707,9 +5713,7 @@ static struct sk_buff *rtl8169_try_rx_copy(void *data,
...
@@ -5707,9 +5713,7 @@ static struct sk_buff *rtl8169_try_rx_copy(void *data,
return
skb
;
return
skb
;
}
}
static
int
rtl8169_rx_interrupt
(
struct
net_device
*
dev
,
static
int
rtl_rx
(
struct
net_device
*
dev
,
struct
rtl8169_private
*
tp
,
u32
budget
)
struct
rtl8169_private
*
tp
,
void
__iomem
*
ioaddr
,
u32
budget
)
{
{
unsigned
int
cur_rx
,
rx_left
;
unsigned
int
cur_rx
,
rx_left
;
unsigned
int
count
;
unsigned
int
count
;
...
@@ -5737,7 +5741,7 @@ static int rtl8169_rx_interrupt(struct net_device *dev,
...
@@ -5737,7 +5741,7 @@ static int rtl8169_rx_interrupt(struct net_device *dev,
if
(
status
&
RxCRC
)
if
(
status
&
RxCRC
)
dev
->
stats
.
rx_crc_errors
++
;
dev
->
stats
.
rx_crc_errors
++
;
if
(
status
&
RxFOVF
)
{
if
(
status
&
RxFOVF
)
{
rtl
8169_schedule_work
(
dev
,
rtl8169_reset_task
);
rtl
_schedule_task
(
tp
,
RTL_FLAG_TASK_RESET_PENDING
);
dev
->
stats
.
rx_fifo_errors
++
;
dev
->
stats
.
rx_fifo_errors
++
;
}
}
rtl8169_mark_to_asic
(
desc
,
rx_buf_sz
);
rtl8169_mark_to_asic
(
desc
,
rx_buf_sz
);
...
@@ -5798,101 +5802,120 @@ static irqreturn_t rtl8169_interrupt(int irq, void *dev_instance)
...
@@ -5798,101 +5802,120 @@ static irqreturn_t rtl8169_interrupt(int irq, void *dev_instance)
{
{
struct
net_device
*
dev
=
dev_instance
;
struct
net_device
*
dev
=
dev_instance
;
struct
rtl8169_private
*
tp
=
netdev_priv
(
dev
);
struct
rtl8169_private
*
tp
=
netdev_priv
(
dev
);
void
__iomem
*
ioaddr
=
tp
->
mmio_addr
;
int
handled
=
0
;
int
handled
=
0
;
int
status
;
u16
status
;
/* loop handling interrupts until we have no new ones or
* we hit a invalid/hotplug case.
*/
status
=
RTL_R16
(
IntrStatus
);
while
(
status
&&
status
!=
0xffff
)
{
status
&=
tp
->
intr_event
;
if
(
!
status
)
break
;
status
=
rtl_get_events
(
tp
);
if
(
status
&&
status
!=
0xffff
)
{
status
&=
RTL_EVENT_NAPI
|
tp
->
event_slow
;
if
(
status
)
{
handled
=
1
;
handled
=
1
;
/* Handle all of the error cases first. These will reset
rtl_irq_disable
(
tp
);
* the chip, so just exit the loop.
napi_schedule
(
&
tp
->
napi
);
*/
}
if
(
unlikely
(
!
netif_running
(
dev
)))
{
rtl8169_hw_reset
(
tp
);
break
;
}
}
return
IRQ_RETVAL
(
handled
);
}
/*
* Workqueue context.
*/
static
void
rtl_slow_event_work
(
struct
rtl8169_private
*
tp
)
{
struct
net_device
*
dev
=
tp
->
dev
;
u16
status
;
status
=
rtl_get_events
(
tp
)
&
tp
->
event_slow
;
rtl_ack_events
(
tp
,
status
);
if
(
unlikely
(
status
&
RxFIFOOver
))
{
if
(
unlikely
(
status
&
RxFIFOOver
))
{
switch
(
tp
->
mac_version
)
{
switch
(
tp
->
mac_version
)
{
/* Work around for rx fifo overflow */
/* Work around for rx fifo overflow */
case
RTL_GIGA_MAC_VER_11
:
case
RTL_GIGA_MAC_VER_11
:
netif_stop_queue
(
dev
);
netif_stop_queue
(
dev
);
rtl8169_tx_timeout
(
dev
);
rtl_schedule_task_bh
(
tp
,
RTL_FLAG_TASK_RESET_PENDING
);
goto
done
;
default:
default:
break
;
break
;
}
}
}
}
if
(
unlikely
(
status
&
SYSErr
))
{
if
(
unlikely
(
status
&
SYSErr
))
rtl8169_pcierr_interrupt
(
dev
);
rtl8169_pcierr_interrupt
(
dev
);
break
;
}
if
(
status
&
LinkChg
)
if
(
status
&
LinkChg
)
__rtl8169_check_link_status
(
dev
,
tp
,
io
addr
,
true
);
__rtl8169_check_link_status
(
dev
,
tp
,
tp
->
mmio_
addr
,
true
);
/* We need to see the lastest version of tp->intr_mask to
napi_disable
(
&
tp
->
napi
);
* avoid ignoring an MSI interrupt and having to wait for
rtl_irq_disable
(
tp
);
* another event which may never come.
*/
smp_rmb
();
if
(
status
&
tp
->
intr_mask
&
tp
->
napi_event
)
{
RTL_W16
(
IntrMask
,
tp
->
intr_event
&
~
tp
->
napi_event
);
tp
->
intr_mask
=
~
tp
->
napi_event
;
if
(
likely
(
napi_schedule_prep
(
&
tp
->
napi
)))
napi_enable
(
&
tp
->
napi
);
__napi_schedule
(
&
tp
->
napi
);
napi_schedule
(
&
tp
->
napi
);
else
}
netif_info
(
tp
,
intr
,
dev
,
"interrupt %04x in poll
\n
"
,
status
);
}
/* We only get a new MSI interrupt when all active irq
static
void
rtl_task
(
struct
work_struct
*
work
)
* sources on the chip have been acknowledged. So, ack
{
* everything we've seen and check if new sources have become
static
const
struct
{
* active to avoid blocking all interrupts from the chip.
int
bitnr
;
*/
void
(
*
action
)(
struct
rtl8169_private
*
);
RTL_W16
(
IntrStatus
,
}
rtl_work
[]
=
{
(
status
&
RxFIFOOver
)
?
(
status
|
RxOverflow
)
:
status
);
{
RTL_FLAG_TASK_SLOW_PENDING
,
rtl_slow_event_work
},
status
=
RTL_R16
(
IntrStatus
);
{
RTL_FLAG_TASK_RESET_PENDING
,
rtl_reset_work
},
{
RTL_FLAG_TASK_PHY_PENDING
,
rtl_phy_work
}
};
struct
rtl8169_private
*
tp
=
container_of
(
work
,
struct
rtl8169_private
,
wk
.
work
);
struct
net_device
*
dev
=
tp
->
dev
;
int
i
;
rtl_lock_work
(
tp
);
if
(
!
netif_running
(
dev
)
||
!
tp
->
wk
.
enabled
)
goto
out_unlock
;
for
(
i
=
0
;
i
<
ARRAY_SIZE
(
rtl_work
);
i
++
)
{
bool
pending
;
spin_lock_bh
(
&
tp
->
lock
);
pending
=
test_and_clear_bit
(
rtl_work
[
i
].
bitnr
,
tp
->
wk
.
flags
);
spin_unlock_bh
(
&
tp
->
lock
);
if
(
pending
)
rtl_work
[
i
].
action
(
tp
);
}
}
done:
return
IRQ_RETVAL
(
handled
);
out_unlock:
rtl_unlock_work
(
tp
);
}
}
static
int
rtl8169_poll
(
struct
napi_struct
*
napi
,
int
budget
)
static
int
rtl8169_poll
(
struct
napi_struct
*
napi
,
int
budget
)
{
{
struct
rtl8169_private
*
tp
=
container_of
(
napi
,
struct
rtl8169_private
,
napi
);
struct
rtl8169_private
*
tp
=
container_of
(
napi
,
struct
rtl8169_private
,
napi
);
struct
net_device
*
dev
=
tp
->
dev
;
struct
net_device
*
dev
=
tp
->
dev
;
void
__iomem
*
ioaddr
=
tp
->
mmio_addr
;
u16
enable_mask
=
RTL_EVENT_NAPI
|
tp
->
event_slow
;
int
work_done
;
int
work_done
=
0
;
u16
status
;
status
=
rtl_get_events
(
tp
);
rtl_ack_events
(
tp
,
status
&
~
tp
->
event_slow
);
if
(
status
&
RTL_EVENT_NAPI_RX
)
work_done
=
rtl_rx
(
dev
,
tp
,
(
u32
)
budget
);
if
(
status
&
RTL_EVENT_NAPI_TX
)
rtl_tx
(
dev
,
tp
);
work_done
=
rtl8169_rx_interrupt
(
dev
,
tp
,
ioaddr
,
(
u32
)
budget
);
if
(
status
&
tp
->
event_slow
)
{
rtl8169_tx_interrupt
(
dev
,
tp
,
ioaddr
);
enable_mask
&=
~
tp
->
event_slow
;
rtl_schedule_task
(
tp
,
RTL_FLAG_TASK_SLOW_PENDING
);
}
if
(
work_done
<
budget
)
{
if
(
work_done
<
budget
)
{
napi_complete
(
napi
);
napi_complete
(
napi
);
/* We need for force the visibility of tp->intr_mask
rtl_irq_enable
(
tp
,
enable_mask
);
* for other CPUs, as we can loose an MSI interrupt
mmiowb
();
* and potentially wait for a retransmit timeout if we don't.
* The posted write to IntrMask is safe, as it will
* eventually make it to the chip and we won't loose anything
* until it does.
*/
tp
->
intr_mask
=
0xffff
;
wmb
();
RTL_W16
(
IntrMask
,
tp
->
intr_event
);
}
}
return
work_done
;
return
work_done
;
...
@@ -5916,26 +5939,19 @@ static void rtl8169_down(struct net_device *dev)
...
@@ -5916,26 +5939,19 @@ static void rtl8169_down(struct net_device *dev)
del_timer_sync
(
&
tp
->
timer
);
del_timer_sync
(
&
tp
->
timer
);
netif_stop_queue
(
dev
);
napi_disable
(
&
tp
->
napi
);
napi_disable
(
&
tp
->
napi
);
netif_stop_queue
(
dev
);
spin_lock_irq
(
&
tp
->
lock
);
rtl8169_hw_reset
(
tp
);
rtl8169_hw_reset
(
tp
);
/*
/*
* At this point device interrupts can not be enabled in any function,
* At this point device interrupts can not be enabled in any function,
* as netif_running is not true (rtl8169_interrupt, rtl8169_reset_task
,
* as netif_running is not true (rtl8169_interrupt, rtl8169_reset_task
)
*
rtl8169_reinit_task)
and napi is disabled (rtl8169_poll).
* and napi is disabled (rtl8169_poll).
*/
*/
rtl8169_rx_missed
(
dev
,
ioaddr
);
rtl8169_rx_missed
(
dev
,
ioaddr
);
spin_unlock_irq
(
&
tp
->
lock
);
synchronize_irq
(
dev
->
irq
);
/* Give a racing hard_start_xmit a few cycles to complete. */
/* Give a racing hard_start_xmit a few cycles to complete. */
synchronize_sched
();
/* FIXME: should this be synchronize_irq()? */
synchronize_sched
();
rtl8169_tx_clear
(
tp
);
rtl8169_tx_clear
(
tp
);
...
@@ -5954,7 +5970,11 @@ static int rtl8169_close(struct net_device *dev)
...
@@ -5954,7 +5970,11 @@ static int rtl8169_close(struct net_device *dev)
/* Update counters before going down */
/* Update counters before going down */
rtl8169_update_counters
(
dev
);
rtl8169_update_counters
(
dev
);
rtl_lock_work
(
tp
);
tp
->
wk
.
enabled
=
false
;
rtl8169_down
(
dev
);
rtl8169_down
(
dev
);
rtl_unlock_work
(
tp
);
free_irq
(
dev
->
irq
,
dev
);
free_irq
(
dev
->
irq
,
dev
);
...
@@ -5974,7 +5994,6 @@ static void rtl_set_rx_mode(struct net_device *dev)
...
@@ -5974,7 +5994,6 @@ static void rtl_set_rx_mode(struct net_device *dev)
{
{
struct
rtl8169_private
*
tp
=
netdev_priv
(
dev
);
struct
rtl8169_private
*
tp
=
netdev_priv
(
dev
);
void
__iomem
*
ioaddr
=
tp
->
mmio_addr
;
void
__iomem
*
ioaddr
=
tp
->
mmio_addr
;
unsigned
long
flags
;
u32
mc_filter
[
2
];
/* Multicast hash filter */
u32
mc_filter
[
2
];
/* Multicast hash filter */
int
rx_mode
;
int
rx_mode
;
u32
tmp
=
0
;
u32
tmp
=
0
;
...
@@ -6003,7 +6022,7 @@ static void rtl_set_rx_mode(struct net_device *dev)
...
@@ -6003,7 +6022,7 @@ static void rtl_set_rx_mode(struct net_device *dev)
}
}
}
}
spin_lock_
irqsave
(
&
tp
->
lock
,
flags
);
spin_lock_
bh
(
&
tp
->
lock
);
tmp
=
(
RTL_R32
(
RxConfig
)
&
~
RX_CONFIG_ACCEPT_MASK
)
|
rx_mode
;
tmp
=
(
RTL_R32
(
RxConfig
)
&
~
RX_CONFIG_ACCEPT_MASK
)
|
rx_mode
;
...
@@ -6019,7 +6038,7 @@ static void rtl_set_rx_mode(struct net_device *dev)
...
@@ -6019,7 +6038,7 @@ static void rtl_set_rx_mode(struct net_device *dev)
RTL_W32
(
RxConfig
,
tmp
);
RTL_W32
(
RxConfig
,
tmp
);
spin_unlock_
irqrestore
(
&
tp
->
lock
,
flags
);
spin_unlock_
bh
(
&
tp
->
lock
);
}
}
/**
/**
...
@@ -6032,13 +6051,9 @@ static struct net_device_stats *rtl8169_get_stats(struct net_device *dev)
...
@@ -6032,13 +6051,9 @@ static struct net_device_stats *rtl8169_get_stats(struct net_device *dev)
{
{
struct
rtl8169_private
*
tp
=
netdev_priv
(
dev
);
struct
rtl8169_private
*
tp
=
netdev_priv
(
dev
);
void
__iomem
*
ioaddr
=
tp
->
mmio_addr
;
void
__iomem
*
ioaddr
=
tp
->
mmio_addr
;
unsigned
long
flags
;
if
(
netif_running
(
dev
))
{
if
(
netif_running
(
dev
))
spin_lock_irqsave
(
&
tp
->
lock
,
flags
);
rtl8169_rx_missed
(
dev
,
ioaddr
);
rtl8169_rx_missed
(
dev
,
ioaddr
);
spin_unlock_irqrestore
(
&
tp
->
lock
,
flags
);
}
return
&
dev
->
stats
;
return
&
dev
->
stats
;
}
}
...
@@ -6050,10 +6065,15 @@ static void rtl8169_net_suspend(struct net_device *dev)
...
@@ -6050,10 +6065,15 @@ static void rtl8169_net_suspend(struct net_device *dev)
if
(
!
netif_running
(
dev
))
if
(
!
netif_running
(
dev
))
return
;
return
;
rtl_pll_power_down
(
tp
);
netif_device_detach
(
dev
);
netif_device_detach
(
dev
);
netif_stop_queue
(
dev
);
netif_stop_queue
(
dev
);
rtl_lock_work
(
tp
);
napi_disable
(
&
tp
->
napi
);
tp
->
wk
.
enabled
=
false
;
rtl_unlock_work
(
tp
);
rtl_pll_power_down
(
tp
);
}
}
#ifdef CONFIG_PM
#ifdef CONFIG_PM
...
@@ -6076,7 +6096,9 @@ static void __rtl8169_resume(struct net_device *dev)
...
@@ -6076,7 +6096,9 @@ static void __rtl8169_resume(struct net_device *dev)
rtl_pll_power_up
(
tp
);
rtl_pll_power_up
(
tp
);
rtl8169_schedule_work
(
dev
,
rtl8169_reset_task
);
tp
->
wk
.
enabled
=
true
;
rtl_schedule_task_bh
(
tp
,
RTL_FLAG_TASK_RESET_PENDING
);
}
}
static
int
rtl8169_resume
(
struct
device
*
device
)
static
int
rtl8169_resume
(
struct
device
*
device
)
...
@@ -6102,10 +6124,10 @@ static int rtl8169_runtime_suspend(struct device *device)
...
@@ -6102,10 +6124,10 @@ static int rtl8169_runtime_suspend(struct device *device)
if
(
!
tp
->
TxDescArray
)
if
(
!
tp
->
TxDescArray
)
return
0
;
return
0
;
spin_lock_irq
(
&
tp
->
lock
);
rtl_lock_work
(
tp
);
tp
->
saved_wolopts
=
__rtl8169_get_wol
(
tp
);
tp
->
saved_wolopts
=
__rtl8169_get_wol
(
tp
);
__rtl8169_set_wol
(
tp
,
WAKE_ANY
);
__rtl8169_set_wol
(
tp
,
WAKE_ANY
);
spin_unlock_irq
(
&
tp
->
lock
);
rtl_unlock_work
(
tp
);
rtl8169_net_suspend
(
dev
);
rtl8169_net_suspend
(
dev
);
...
@@ -6121,10 +6143,10 @@ static int rtl8169_runtime_resume(struct device *device)
...
@@ -6121,10 +6143,10 @@ static int rtl8169_runtime_resume(struct device *device)
if
(
!
tp
->
TxDescArray
)
if
(
!
tp
->
TxDescArray
)
return
0
;
return
0
;
spin_lock_irq
(
&
tp
->
lock
);
rtl_lock_work
(
tp
);
__rtl8169_set_wol
(
tp
,
tp
->
saved_wolopts
);
__rtl8169_set_wol
(
tp
,
tp
->
saved_wolopts
);
tp
->
saved_wolopts
=
0
;
tp
->
saved_wolopts
=
0
;
spin_unlock_irq
(
&
tp
->
lock
);
rtl_unlock_work
(
tp
);
rtl8169_init_phy
(
dev
,
tp
);
rtl8169_init_phy
(
dev
,
tp
);
...
@@ -6192,12 +6214,8 @@ static void rtl_shutdown(struct pci_dev *pdev)
...
@@ -6192,12 +6214,8 @@ static void rtl_shutdown(struct pci_dev *pdev)
/* Restore original MAC address */
/* Restore original MAC address */
rtl_rar_set
(
tp
,
dev
->
perm_addr
);
rtl_rar_set
(
tp
,
dev
->
perm_addr
);
spin_lock_irq
(
&
tp
->
lock
);
rtl8169_hw_reset
(
tp
);
rtl8169_hw_reset
(
tp
);
spin_unlock_irq
(
&
tp
->
lock
);
if
(
system_state
==
SYSTEM_POWER_OFF
)
{
if
(
system_state
==
SYSTEM_POWER_OFF
)
{
if
(
__rtl8169_get_wol
(
tp
)
&
WAKE_ANY
)
{
if
(
__rtl8169_get_wol
(
tp
)
&
WAKE_ANY
)
{
rtl_wol_suspend_quirk
(
tp
);
rtl_wol_suspend_quirk
(
tp
);
...
...
Write
Preview
Markdown
is supported
0%
Try again
or
attach a new file
Attach a file
Cancel
You are about to add
0
people
to the discussion. Proceed with caution.
Finish editing this message first!
Cancel
Please
register
or
sign in
to comment