Skip to content
体验新版
项目
组织
正在加载...
登录
切换导航
打开侧边栏
openeuler
raspberrypi-kernel
提交
e5a87625
R
raspberrypi-kernel
项目概览
openeuler
/
raspberrypi-kernel
通知
13
Star
1
Fork
0
代码
文件
提交
分支
Tags
贡献者
分支图
Diff
Issue
0
列表
看板
标记
里程碑
合并请求
0
Wiki
0
Wiki
分析
仓库
DevOps
项目成员
Pages
R
raspberrypi-kernel
项目概览
项目概览
详情
发布
仓库
仓库
文件
提交
分支
标签
贡献者
分支图
比较
Issue
0
Issue
0
列表
看板
标记
里程碑
合并请求
0
合并请求
0
Pages
分析
分析
仓库分析
DevOps
Wiki
0
Wiki
成员
成员
收起侧边栏
关闭侧边栏
动态
分支图
创建新Issue
提交
Issue看板
提交
e5a87625
编写于
9月 24, 2012
作者:
J
John W. Linville
浏览文件
操作
浏览文件
下载
差异文件
Merge branch 'for-john' of
git://git.kernel.org/pub/scm/linux/kernel/git/iwlwifi/iwlwifi-next
上级
9b4e9e75
83f84d7b
变更
7
显示空白变更内容
内联
并排
Showing
7 changed file
with
117 addition
and
90 deletion
+117
-90
drivers/net/wireless/iwlwifi/dvm/ucode.c
drivers/net/wireless/iwlwifi/dvm/ucode.c
+2
-2
drivers/net/wireless/iwlwifi/iwl-drv.c
drivers/net/wireless/iwlwifi/iwl-drv.c
+14
-12
drivers/net/wireless/iwlwifi/iwl-fw.h
drivers/net/wireless/iwlwifi/iwl-fw.h
+1
-2
drivers/net/wireless/iwlwifi/pcie/drv.c
drivers/net/wireless/iwlwifi/pcie/drv.c
+0
-13
drivers/net/wireless/iwlwifi/pcie/internal.h
drivers/net/wireless/iwlwifi/pcie/internal.h
+1
-1
drivers/net/wireless/iwlwifi/pcie/rx.c
drivers/net/wireless/iwlwifi/pcie/rx.c
+44
-31
drivers/net/wireless/iwlwifi/pcie/trans.c
drivers/net/wireless/iwlwifi/pcie/trans.c
+55
-29
未找到文件。
drivers/net/wireless/iwlwifi/dvm/ucode.c
浏览文件 @
e5a87625
...
...
@@ -295,7 +295,7 @@ static int iwl_alive_notify(struct iwl_priv *priv)
static
int
iwl_verify_sec_sparse
(
struct
iwl_priv
*
priv
,
const
struct
fw_desc
*
fw_desc
)
{
__le32
*
image
=
(
__le32
*
)
fw_desc
->
v_addr
;
__le32
*
image
=
(
__le32
*
)
fw_desc
->
data
;
u32
len
=
fw_desc
->
len
;
u32
val
;
u32
i
;
...
...
@@ -319,7 +319,7 @@ static int iwl_verify_sec_sparse(struct iwl_priv *priv,
static
void
iwl_print_mismatch_sec
(
struct
iwl_priv
*
priv
,
const
struct
fw_desc
*
fw_desc
)
{
__le32
*
image
=
(
__le32
*
)
fw_desc
->
v_addr
;
__le32
*
image
=
(
__le32
*
)
fw_desc
->
data
;
u32
len
=
fw_desc
->
len
;
u32
val
;
u32
offs
;
...
...
drivers/net/wireless/iwlwifi/iwl-drv.c
浏览文件 @
e5a87625
...
...
@@ -64,6 +64,7 @@
#include <linux/dma-mapping.h>
#include <linux/firmware.h>
#include <linux/module.h>
#include <linux/vmalloc.h>
#include "iwl-drv.h"
#include "iwl-debug.h"
...
...
@@ -164,10 +165,8 @@ struct fw_sec {
static
void
iwl_free_fw_desc
(
struct
iwl_drv
*
drv
,
struct
fw_desc
*
desc
)
{
if
(
desc
->
v_addr
)
dma_free_coherent
(
drv
->
trans
->
dev
,
desc
->
len
,
desc
->
v_addr
,
desc
->
p_addr
);
desc
->
v_addr
=
NULL
;
vfree
(
desc
->
data
);
desc
->
data
=
NULL
;
desc
->
len
=
0
;
}
...
...
@@ -188,19 +187,22 @@ static void iwl_dealloc_ucode(struct iwl_drv *drv)
static
int
iwl_alloc_fw_desc
(
struct
iwl_drv
*
drv
,
struct
fw_desc
*
desc
,
struct
fw_sec
*
sec
)
{
if
(
!
sec
||
!
sec
->
size
)
{
desc
->
v_addr
=
NULL
;
void
*
data
;
desc
->
data
=
NULL
;
if
(
!
sec
||
!
sec
->
size
)
return
-
EINVAL
;
}
desc
->
v_addr
=
dma_alloc_coherent
(
drv
->
trans
->
dev
,
sec
->
size
,
&
desc
->
p_addr
,
GFP_KERNEL
);
if
(
!
desc
->
v_addr
)
data
=
vmalloc
(
sec
->
size
);
if
(
!
data
)
return
-
ENOMEM
;
desc
->
len
=
sec
->
size
;
desc
->
offset
=
sec
->
offset
;
memcpy
(
desc
->
v_addr
,
sec
->
data
,
sec
->
size
);
memcpy
(
data
,
sec
->
data
,
desc
->
len
);
desc
->
data
=
data
;
return
0
;
}
...
...
drivers/net/wireless/iwlwifi/iwl-fw.h
浏览文件 @
e5a87625
...
...
@@ -124,8 +124,7 @@ struct iwl_ucode_capabilities {
/* one for each uCode image (inst/data, init/runtime/wowlan) */
struct
fw_desc
{
dma_addr_t
p_addr
;
/* hardware address */
void
*
v_addr
;
/* software address */
const
void
*
data
;
/* vmalloc'ed data */
u32
len
;
/* size in bytes */
u32
offset
;
/* offset in the device */
};
...
...
drivers/net/wireless/iwlwifi/pcie/drv.c
浏览文件 @
e5a87625
...
...
@@ -263,8 +263,6 @@ MODULE_DEVICE_TABLE(pci, iwl_hw_card_ids);
/* PCI registers */
#define PCI_CFG_RETRY_TIMEOUT 0x041
#ifndef CONFIG_IWLWIFI_IDI
static
int
iwl_pci_probe
(
struct
pci_dev
*
pdev
,
const
struct
pci_device_id
*
ent
)
{
const
struct
iwl_cfg
*
cfg
=
(
struct
iwl_cfg
*
)(
ent
->
driver_data
);
...
...
@@ -307,8 +305,6 @@ static void __devexit iwl_pci_remove(struct pci_dev *pdev)
pci_set_drvdata
(
pdev
,
NULL
);
}
#endif
/* CONFIG_IWLWIFI_IDI */
#ifdef CONFIG_PM_SLEEP
static
int
iwl_pci_suspend
(
struct
device
*
device
)
...
...
@@ -353,15 +349,6 @@ static SIMPLE_DEV_PM_OPS(iwl_dev_pm_ops, iwl_pci_suspend, iwl_pci_resume);
#endif
#ifdef CONFIG_IWLWIFI_IDI
/*
* Defined externally in iwl-idi.c
*/
int
iwl_pci_probe
(
struct
pci_dev
*
pdev
,
const
struct
pci_device_id
*
ent
);
void
__devexit
iwl_pci_remove
(
struct
pci_dev
*
pdev
);
#endif
/* CONFIG_IWLWIFI_IDI */
static
struct
pci_driver
iwl_pci_driver
=
{
.
name
=
DRV_NAME
,
.
id_table
=
iwl_hw_card_ids
,
...
...
drivers/net/wireless/iwlwifi/pcie/internal.h
浏览文件 @
e5a87625
...
...
@@ -311,7 +311,7 @@ void iwl_trans_pcie_free(struct iwl_trans *trans);
******************************************************/
void
iwl_bg_rx_replenish
(
struct
work_struct
*
data
);
void
iwl_irq_tasklet
(
struct
iwl_trans
*
trans
);
void
iwl
agn
_rx_replenish
(
struct
iwl_trans
*
trans
);
void
iwl_rx_replenish
(
struct
iwl_trans
*
trans
);
void
iwl_rx_queue_update_write_ptr
(
struct
iwl_trans
*
trans
,
struct
iwl_rx_queue
*
q
);
...
...
drivers/net/wireless/iwlwifi/pcie/rx.c
浏览文件 @
e5a87625
...
...
@@ -35,10 +35,6 @@
#include "internal.h"
#include "iwl-op-mode.h"
#ifdef CONFIG_IWLWIFI_IDI
#include "iwl-amfh.h"
#endif
/******************************************************************************
*
* RX path functions
...
...
@@ -181,15 +177,15 @@ void iwl_rx_queue_update_write_ptr(struct iwl_trans *trans,
}
/**
* iwl
agn
_dma_addr2rbd_ptr - convert a DMA address to a uCode read buffer ptr
* iwl_dma_addr2rbd_ptr - convert a DMA address to a uCode read buffer ptr
*/
static
inline
__le32
iwl
agn
_dma_addr2rbd_ptr
(
dma_addr_t
dma_addr
)
static
inline
__le32
iwl_dma_addr2rbd_ptr
(
dma_addr_t
dma_addr
)
{
return
cpu_to_le32
((
u32
)(
dma_addr
>>
8
));
}
/**
* iwl
agn
_rx_queue_restock - refill RX queue from pre-allocated pool
* iwl_rx_queue_restock - refill RX queue from pre-allocated pool
*
* If there are slots in the RX queue that need to be restocked,
* and we have free pre-allocated buffers, fill the ranks as much
...
...
@@ -199,7 +195,7 @@ static inline __le32 iwlagn_dma_addr2rbd_ptr(dma_addr_t dma_addr)
* also updates the memory address in the firmware to reference the new
* target buffer.
*/
static
void
iwl
agn
_rx_queue_restock
(
struct
iwl_trans
*
trans
)
static
void
iwl_rx_queue_restock
(
struct
iwl_trans
*
trans
)
{
struct
iwl_trans_pcie
*
trans_pcie
=
IWL_TRANS_GET_PCIE_TRANS
(
trans
);
struct
iwl_rx_queue
*
rxq
=
&
trans_pcie
->
rxq
;
...
...
@@ -207,6 +203,17 @@ static void iwlagn_rx_queue_restock(struct iwl_trans *trans)
struct
iwl_rx_mem_buffer
*
rxb
;
unsigned
long
flags
;
/*
* If the device isn't enabled - not need to try to add buffers...
* This can happen when we stop the device and still have an interrupt
* pending. We stop the APM before we sync the interrupts / tasklets
* because we have to (see comment there). On the other hand, since
* the APM is stopped, we cannot access the HW (in particular not prph).
* So don't try to restock if the APM has been already stopped.
*/
if
(
!
test_bit
(
STATUS_DEVICE_ENABLED
,
&
trans_pcie
->
status
))
return
;
spin_lock_irqsave
(
&
rxq
->
lock
,
flags
);
while
((
iwl_rx_queue_space
(
rxq
)
>
0
)
&&
(
rxq
->
free_count
))
{
/* The overwritten rxb must be a used one */
...
...
@@ -219,7 +226,7 @@ static void iwlagn_rx_queue_restock(struct iwl_trans *trans)
list_del
(
element
);
/* Point to Rx buffer via next RBD in circular buffer */
rxq
->
bd
[
rxq
->
write
]
=
iwl
agn
_dma_addr2rbd_ptr
(
rxb
->
page_dma
);
rxq
->
bd
[
rxq
->
write
]
=
iwl_dma_addr2rbd_ptr
(
rxb
->
page_dma
);
rxq
->
queue
[
rxq
->
write
]
=
rxb
;
rxq
->
write
=
(
rxq
->
write
+
1
)
&
RX_QUEUE_MASK
;
rxq
->
free_count
--
;
...
...
@@ -230,7 +237,6 @@ static void iwlagn_rx_queue_restock(struct iwl_trans *trans)
if
(
rxq
->
free_count
<=
RX_LOW_WATERMARK
)
schedule_work
(
&
trans_pcie
->
rx_replenish
);
/* If we've added more space for the firmware to place data, tell it.
* Increment device's write pointer in multiples of 8. */
if
(
rxq
->
write_actual
!=
(
rxq
->
write
&
~
0x7
))
{
...
...
@@ -241,15 +247,16 @@ static void iwlagn_rx_queue_restock(struct iwl_trans *trans)
}
}
/**
* iwlagn_rx_replenish - Move all used packet from rx_used to rx_free
*
* When moving to rx_free an SKB is allocated for the slot.
/*
* iwl_rx_allocate - allocate a page for each used RBD
*
* Also restock the Rx queue via iwl_rx_queue_restock.
* This is called as a scheduled work item (except for during initialization)
* A used RBD is an Rx buffer that has been given to the stack. To use it again
* a page must be allocated and the RBD must point to the page. This function
* doesn't change the HW pointer but handles the list of pages that is used by
* iwl_rx_queue_restock. The latter function will update the HW to use the newly
* allocated buffers.
*/
static
void
iwl
agn
_rx_allocate
(
struct
iwl_trans
*
trans
,
gfp_t
priority
)
static
void
iwl_rx_allocate
(
struct
iwl_trans
*
trans
,
gfp_t
priority
)
{
struct
iwl_trans_pcie
*
trans_pcie
=
IWL_TRANS_GET_PCIE_TRANS
(
trans
);
struct
iwl_rx_queue
*
rxq
=
&
trans_pcie
->
rxq
;
...
...
@@ -328,23 +335,31 @@ static void iwlagn_rx_allocate(struct iwl_trans *trans, gfp_t priority)
}
}
void
iwlagn_rx_replenish
(
struct
iwl_trans
*
trans
)
/*
* iwl_rx_replenish - Move all used buffers from rx_used to rx_free
*
* When moving to rx_free an page is allocated for the slot.
*
* Also restock the Rx queue via iwl_rx_queue_restock.
* This is called as a scheduled work item (except for during initialization)
*/
void
iwl_rx_replenish
(
struct
iwl_trans
*
trans
)
{
struct
iwl_trans_pcie
*
trans_pcie
=
IWL_TRANS_GET_PCIE_TRANS
(
trans
);
unsigned
long
flags
;
iwl
agn
_rx_allocate
(
trans
,
GFP_KERNEL
);
iwl_rx_allocate
(
trans
,
GFP_KERNEL
);
spin_lock_irqsave
(
&
trans_pcie
->
irq_lock
,
flags
);
iwl
agn
_rx_queue_restock
(
trans
);
iwl_rx_queue_restock
(
trans
);
spin_unlock_irqrestore
(
&
trans_pcie
->
irq_lock
,
flags
);
}
static
void
iwl
agn
_rx_replenish_now
(
struct
iwl_trans
*
trans
)
static
void
iwl_rx_replenish_now
(
struct
iwl_trans
*
trans
)
{
iwl
agn
_rx_allocate
(
trans
,
GFP_ATOMIC
);
iwl_rx_allocate
(
trans
,
GFP_ATOMIC
);
iwl
agn
_rx_queue_restock
(
trans
);
iwl_rx_queue_restock
(
trans
);
}
void
iwl_bg_rx_replenish
(
struct
work_struct
*
data
)
...
...
@@ -352,7 +367,7 @@ void iwl_bg_rx_replenish(struct work_struct *data)
struct
iwl_trans_pcie
*
trans_pcie
=
container_of
(
data
,
struct
iwl_trans_pcie
,
rx_replenish
);
iwl
agn
_rx_replenish
(
trans_pcie
->
trans
);
iwl_rx_replenish
(
trans_pcie
->
trans
);
}
static
void
iwl_rx_handle_rxbuf
(
struct
iwl_trans
*
trans
,
...
...
@@ -530,7 +545,7 @@ static void iwl_rx_handle(struct iwl_trans *trans)
count
++
;
if
(
count
>=
8
)
{
rxq
->
read
=
i
;
iwl
agn
_rx_replenish_now
(
trans
);
iwl_rx_replenish_now
(
trans
);
count
=
0
;
}
}
...
...
@@ -539,9 +554,9 @@ static void iwl_rx_handle(struct iwl_trans *trans)
/* Backtrack one entry */
rxq
->
read
=
i
;
if
(
fill_rx
)
iwl
agn
_rx_replenish_now
(
trans
);
iwl_rx_replenish_now
(
trans
);
else
iwl
agn
_rx_queue_restock
(
trans
);
iwl_rx_queue_restock
(
trans
);
}
/**
...
...
@@ -723,11 +738,9 @@ void iwl_irq_tasklet(struct iwl_trans *trans)
/* Disable periodic interrupt; we use it as just a one-shot. */
iwl_write8
(
trans
,
CSR_INT_PERIODIC_REG
,
CSR_INT_PERIODIC_DIS
);
#ifdef CONFIG_IWLWIFI_IDI
iwl_amfh_rx_handler
();
#else
iwl_rx_handle
(
trans
);
#endif
/*
* Enable periodic interrupt in 8 msec only if we received
* real RX interrupt (instead of just periodic int), to catch
...
...
drivers/net/wireless/iwlwifi/pcie/trans.c
浏览文件 @
e5a87625
...
...
@@ -216,7 +216,7 @@ static int iwl_rx_init(struct iwl_trans *trans)
rxq
->
free_count
=
0
;
spin_unlock_irqrestore
(
&
rxq
->
lock
,
flags
);
iwl
agn
_rx_replenish
(
trans
);
iwl_rx_replenish
(
trans
);
iwl_trans_rx_hw_init
(
trans
,
rxq
);
...
...
@@ -855,10 +855,8 @@ static int iwl_nic_init(struct iwl_trans *trans)
iwl_op_mode_nic_config
(
trans
->
op_mode
);
#ifndef CONFIG_IWLWIFI_IDI
/* Allocate the RX queue, or reset if it is already allocated */
iwl_rx_init
(
trans
);
#endif
/* Allocate or reset and init all Tx and Command queues */
if
(
iwl_tx_init
(
trans
))
...
...
@@ -925,13 +923,10 @@ static int iwl_prepare_card_hw(struct iwl_trans *trans)
/*
* ucode
*/
static
int
iwl_load_
section
(
struct
iwl_trans
*
trans
,
u8
section_num
,
const
struct
fw_desc
*
section
)
static
int
iwl_load_
firmware_chunk
(
struct
iwl_trans
*
trans
,
u32
dst_addr
,
dma_addr_t
phy_addr
,
u32
byte_cnt
)
{
struct
iwl_trans_pcie
*
trans_pcie
=
IWL_TRANS_GET_PCIE_TRANS
(
trans
);
dma_addr_t
phy_addr
=
section
->
p_addr
;
u32
byte_cnt
=
section
->
len
;
u32
dst_addr
=
section
->
offset
;
int
ret
;
trans_pcie
->
ucode_write_complete
=
false
;
...
...
@@ -965,27 +960,58 @@ static int iwl_load_section(struct iwl_trans *trans, u8 section_num,
FH_TCSR_TX_CONFIG_REG_VAL_DMA_CREDIT_DISABLE
|
FH_TCSR_TX_CONFIG_REG_VAL_CIRQ_HOST_ENDTFD
);
IWL_DEBUG_FW
(
trans
,
"[%d] uCode section being loaded...
\n
"
,
section_num
);
ret
=
wait_event_timeout
(
trans_pcie
->
ucode_write_waitq
,
trans_pcie
->
ucode_write_complete
,
5
*
HZ
);
if
(
!
ret
)
{
IWL_ERR
(
trans
,
"Could not load the [%d] uCode section
\n
"
,
section_num
);
IWL_ERR
(
trans
,
"Failed to load firmware chunk!
\n
"
);
return
-
ETIMEDOUT
;
}
return
0
;
}
static
int
iwl_load_section
(
struct
iwl_trans
*
trans
,
u8
section_num
,
const
struct
fw_desc
*
section
)
{
u8
*
v_addr
;
dma_addr_t
p_addr
;
u32
offset
;
int
ret
=
0
;
IWL_DEBUG_FW
(
trans
,
"[%d] uCode section being loaded...
\n
"
,
section_num
);
v_addr
=
dma_alloc_coherent
(
trans
->
dev
,
PAGE_SIZE
,
&
p_addr
,
GFP_KERNEL
);
if
(
!
v_addr
)
return
-
ENOMEM
;
for
(
offset
=
0
;
offset
<
section
->
len
;
offset
+=
PAGE_SIZE
)
{
u32
copy_size
;
copy_size
=
min_t
(
u32
,
PAGE_SIZE
,
section
->
len
-
offset
);
memcpy
(
v_addr
,
(
u8
*
)
section
->
data
+
offset
,
copy_size
);
ret
=
iwl_load_firmware_chunk
(
trans
,
section
->
offset
+
offset
,
p_addr
,
copy_size
);
if
(
ret
)
{
IWL_ERR
(
trans
,
"Could not load the [%d] uCode section
\n
"
,
section_num
);
break
;
}
}
dma_free_coherent
(
trans
->
dev
,
PAGE_SIZE
,
v_addr
,
p_addr
);
return
ret
;
}
static
int
iwl_load_given_ucode
(
struct
iwl_trans
*
trans
,
const
struct
fw_img
*
image
)
{
int
ret
=
0
;
int
i
;
int
i
,
ret
=
0
;
for
(
i
=
0
;
i
<
IWL_UCODE_SECTION_MAX
;
i
++
)
{
if
(
!
image
->
sec
[
i
].
p_addr
)
if
(
!
image
->
sec
[
i
].
data
)
break
;
ret
=
iwl_load_section
(
trans
,
i
,
&
image
->
sec
[
i
]);
...
...
@@ -1184,9 +1210,8 @@ static void iwl_trans_pcie_stop_device(struct iwl_trans *trans)
*/
if
(
test_bit
(
STATUS_DEVICE_ENABLED
,
&
trans_pcie
->
status
))
{
iwl_trans_tx_stop
(
trans
);
#ifndef CONFIG_IWLWIFI_IDI
iwl_trans_rx_stop
(
trans
);
#endif
/* Power-down device's busmaster DMA clocks */
iwl_write_prph
(
trans
,
APMG_CLK_DIS_REG
,
APMG_CLK_VAL_DMA_CLK_RQT
);
...
...
@@ -1456,14 +1481,16 @@ static void iwl_trans_pcie_stop_hw(struct iwl_trans *trans,
bool
hw_rfkill
;
unsigned
long
flags
;
spin_lock_irqsave
(
&
trans_pcie
->
irq_lock
,
flags
);
iwl_disable_interrupts
(
trans
);
spin_unlock_irqrestore
(
&
trans_pcie
->
irq_lock
,
flags
);
iwl_apm_stop
(
trans
);
spin_lock_irqsave
(
&
trans_pcie
->
irq_lock
,
flags
);
iwl_disable_interrupts
(
trans
);
spin_unlock_irqrestore
(
&
trans_pcie
->
irq_lock
,
flags
);
iwl_write32
(
trans
,
CSR_INT
,
0xFFFFFFFF
);
if
(
!
op_mode_leaving
)
{
/*
* Even if we stop the HW, we still want the RF kill
...
...
@@ -1551,9 +1578,8 @@ void iwl_trans_pcie_free(struct iwl_trans *trans)
struct
iwl_trans_pcie
*
trans_pcie
=
IWL_TRANS_GET_PCIE_TRANS
(
trans
);
iwl_trans_pcie_tx_free
(
trans
);
#ifndef CONFIG_IWLWIFI_IDI
iwl_trans_pcie_rx_free
(
trans
);
#endif
if
(
trans_pcie
->
irq_requested
==
true
)
{
free_irq
(
trans_pcie
->
irq
,
trans
);
iwl_free_isr_ict
(
trans
);
...
...
编辑
预览
Markdown
is supported
0%
请重试
或
添加新附件
.
添加附件
取消
You are about to add
0
people
to the discussion. Proceed with caution.
先完成此消息的编辑!
取消
想要评论请
注册
或
登录