net: ena: move llq configuration from ena_probe to ena_device_init()
When the ENA device resets to recover from some error state, all LLQ configuration values are reset to their defaults, because LLQ is initialized only once during ena_probe(). Changes in this commit: 1. Move the LLQ configuration process into ena_init_device() which is called from both ena_probe() and ena_restore_device(). This way, LLQ setup configurations that are different from the default values will survive resets. 2. Extract the LLQ bar mapping to ena_map_llq_bar(), and call once in the lifetime of the driver from ena_probe(), since there is no need to unmap and map the LLQ bar again every reset. 3. Map the LLQ bar if it exists, regardless if initialization of LLQ placement policy (ENA_ADMIN_PLACEMENT_POLICY_DEV) succeeded or not. Initialization might fail the first time, falling back to the ENA_ADMIN_PLACEMENT_POLICY_HOST placement policy, but later succeed after device reset, in which case the LLQ bar needs to be mapped already. Signed-off-by: Sameeh Jubran <sameehj@amazon.com> Signed-off-by: Arthur Kiyanovski <akiyano@amazon.com> Signed-off-by: David S. Miller <davem@davemloft.net>
This commit is contained in:
parent
0ee60edf46
commit
c29efeae37
|
@ -3280,10 +3280,71 @@ static int ena_device_validate_params(struct ena_adapter *adapter,
|
||||||
return 0;
|
return 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
static void set_default_llq_configurations(struct ena_llq_configurations *llq_config)
|
||||||
|
{
|
||||||
|
llq_config->llq_header_location = ENA_ADMIN_INLINE_HEADER;
|
||||||
|
llq_config->llq_stride_ctrl = ENA_ADMIN_MULTIPLE_DESCS_PER_ENTRY;
|
||||||
|
llq_config->llq_num_decs_before_header = ENA_ADMIN_LLQ_NUM_DESCS_BEFORE_HEADER_2;
|
||||||
|
llq_config->llq_ring_entry_size = ENA_ADMIN_LIST_ENTRY_SIZE_128B;
|
||||||
|
llq_config->llq_ring_entry_size_value = 128;
|
||||||
|
}
|
||||||
|
|
||||||
|
static int ena_set_queues_placement_policy(struct pci_dev *pdev,
|
||||||
|
struct ena_com_dev *ena_dev,
|
||||||
|
struct ena_admin_feature_llq_desc *llq,
|
||||||
|
struct ena_llq_configurations *llq_default_configurations)
|
||||||
|
{
|
||||||
|
int rc;
|
||||||
|
u32 llq_feature_mask;
|
||||||
|
|
||||||
|
llq_feature_mask = 1 << ENA_ADMIN_LLQ;
|
||||||
|
if (!(ena_dev->supported_features & llq_feature_mask)) {
|
||||||
|
dev_err(&pdev->dev,
|
||||||
|
"LLQ is not supported Fallback to host mode policy.\n");
|
||||||
|
ena_dev->tx_mem_queue_type = ENA_ADMIN_PLACEMENT_POLICY_HOST;
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
rc = ena_com_config_dev_mode(ena_dev, llq, llq_default_configurations);
|
||||||
|
if (unlikely(rc)) {
|
||||||
|
dev_err(&pdev->dev,
|
||||||
|
"Failed to configure the device mode. Fallback to host mode policy.\n");
|
||||||
|
ena_dev->tx_mem_queue_type = ENA_ADMIN_PLACEMENT_POLICY_HOST;
|
||||||
|
}
|
||||||
|
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
static int ena_map_llq_mem_bar(struct pci_dev *pdev, struct ena_com_dev *ena_dev,
|
||||||
|
int bars)
|
||||||
|
{
|
||||||
|
bool has_mem_bar = !!(bars & BIT(ENA_MEM_BAR));
|
||||||
|
|
||||||
|
if (!has_mem_bar) {
|
||||||
|
if (ena_dev->tx_mem_queue_type == ENA_ADMIN_PLACEMENT_POLICY_DEV) {
|
||||||
|
dev_err(&pdev->dev,
|
||||||
|
"ENA device does not expose LLQ bar. Fallback to host mode policy.\n");
|
||||||
|
ena_dev->tx_mem_queue_type = ENA_ADMIN_PLACEMENT_POLICY_HOST;
|
||||||
|
}
|
||||||
|
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
ena_dev->mem_bar = devm_ioremap_wc(&pdev->dev,
|
||||||
|
pci_resource_start(pdev, ENA_MEM_BAR),
|
||||||
|
pci_resource_len(pdev, ENA_MEM_BAR));
|
||||||
|
|
||||||
|
if (!ena_dev->mem_bar)
|
||||||
|
return -EFAULT;
|
||||||
|
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
|
||||||
static int ena_device_init(struct ena_com_dev *ena_dev, struct pci_dev *pdev,
|
static int ena_device_init(struct ena_com_dev *ena_dev, struct pci_dev *pdev,
|
||||||
struct ena_com_dev_get_features_ctx *get_feat_ctx,
|
struct ena_com_dev_get_features_ctx *get_feat_ctx,
|
||||||
bool *wd_state)
|
bool *wd_state)
|
||||||
{
|
{
|
||||||
|
struct ena_llq_configurations llq_config;
|
||||||
struct device *dev = &pdev->dev;
|
struct device *dev = &pdev->dev;
|
||||||
bool readless_supported;
|
bool readless_supported;
|
||||||
u32 aenq_groups;
|
u32 aenq_groups;
|
||||||
|
@ -3374,6 +3435,15 @@ static int ena_device_init(struct ena_com_dev *ena_dev, struct pci_dev *pdev,
|
||||||
|
|
||||||
*wd_state = !!(aenq_groups & BIT(ENA_ADMIN_KEEP_ALIVE));
|
*wd_state = !!(aenq_groups & BIT(ENA_ADMIN_KEEP_ALIVE));
|
||||||
|
|
||||||
|
set_default_llq_configurations(&llq_config);
|
||||||
|
|
||||||
|
rc = ena_set_queues_placement_policy(pdev, ena_dev, &get_feat_ctx->llq,
|
||||||
|
&llq_config);
|
||||||
|
if (rc) {
|
||||||
|
dev_err(&pdev->dev, "ena device init failed\n");
|
||||||
|
goto err_admin_init;
|
||||||
|
}
|
||||||
|
|
||||||
return 0;
|
return 0;
|
||||||
|
|
||||||
err_admin_init:
|
err_admin_init:
|
||||||
|
@ -3880,54 +3950,6 @@ static u32 ena_calc_max_io_queue_num(struct pci_dev *pdev,
|
||||||
return max_num_io_queues;
|
return max_num_io_queues;
|
||||||
}
|
}
|
||||||
|
|
||||||
static int ena_set_queues_placement_policy(struct pci_dev *pdev,
|
|
||||||
struct ena_com_dev *ena_dev,
|
|
||||||
struct ena_admin_feature_llq_desc *llq,
|
|
||||||
struct ena_llq_configurations *llq_default_configurations)
|
|
||||||
{
|
|
||||||
bool has_mem_bar;
|
|
||||||
int rc;
|
|
||||||
u32 llq_feature_mask;
|
|
||||||
|
|
||||||
llq_feature_mask = 1 << ENA_ADMIN_LLQ;
|
|
||||||
if (!(ena_dev->supported_features & llq_feature_mask)) {
|
|
||||||
dev_err(&pdev->dev,
|
|
||||||
"LLQ is not supported Fallback to host mode policy.\n");
|
|
||||||
ena_dev->tx_mem_queue_type = ENA_ADMIN_PLACEMENT_POLICY_HOST;
|
|
||||||
return 0;
|
|
||||||
}
|
|
||||||
|
|
||||||
has_mem_bar = pci_select_bars(pdev, IORESOURCE_MEM) & BIT(ENA_MEM_BAR);
|
|
||||||
|
|
||||||
rc = ena_com_config_dev_mode(ena_dev, llq, llq_default_configurations);
|
|
||||||
if (unlikely(rc)) {
|
|
||||||
dev_err(&pdev->dev,
|
|
||||||
"Failed to configure the device mode. Fallback to host mode policy.\n");
|
|
||||||
ena_dev->tx_mem_queue_type = ENA_ADMIN_PLACEMENT_POLICY_HOST;
|
|
||||||
return 0;
|
|
||||||
}
|
|
||||||
|
|
||||||
/* Nothing to config, exit */
|
|
||||||
if (ena_dev->tx_mem_queue_type == ENA_ADMIN_PLACEMENT_POLICY_HOST)
|
|
||||||
return 0;
|
|
||||||
|
|
||||||
if (!has_mem_bar) {
|
|
||||||
dev_err(&pdev->dev,
|
|
||||||
"ENA device does not expose LLQ bar. Fallback to host mode policy.\n");
|
|
||||||
ena_dev->tx_mem_queue_type = ENA_ADMIN_PLACEMENT_POLICY_HOST;
|
|
||||||
return 0;
|
|
||||||
}
|
|
||||||
|
|
||||||
ena_dev->mem_bar = devm_ioremap_wc(&pdev->dev,
|
|
||||||
pci_resource_start(pdev, ENA_MEM_BAR),
|
|
||||||
pci_resource_len(pdev, ENA_MEM_BAR));
|
|
||||||
|
|
||||||
if (!ena_dev->mem_bar)
|
|
||||||
return -EFAULT;
|
|
||||||
|
|
||||||
return 0;
|
|
||||||
}
|
|
||||||
|
|
||||||
static void ena_set_dev_offloads(struct ena_com_dev_get_features_ctx *feat,
|
static void ena_set_dev_offloads(struct ena_com_dev_get_features_ctx *feat,
|
||||||
struct net_device *netdev)
|
struct net_device *netdev)
|
||||||
{
|
{
|
||||||
|
@ -4043,14 +4065,6 @@ static void ena_release_bars(struct ena_com_dev *ena_dev, struct pci_dev *pdev)
|
||||||
pci_release_selected_regions(pdev, release_bars);
|
pci_release_selected_regions(pdev, release_bars);
|
||||||
}
|
}
|
||||||
|
|
||||||
static void set_default_llq_configurations(struct ena_llq_configurations *llq_config)
|
|
||||||
{
|
|
||||||
llq_config->llq_header_location = ENA_ADMIN_INLINE_HEADER;
|
|
||||||
llq_config->llq_ring_entry_size = ENA_ADMIN_LIST_ENTRY_SIZE_128B;
|
|
||||||
llq_config->llq_stride_ctrl = ENA_ADMIN_MULTIPLE_DESCS_PER_ENTRY;
|
|
||||||
llq_config->llq_num_decs_before_header = ENA_ADMIN_LLQ_NUM_DESCS_BEFORE_HEADER_2;
|
|
||||||
llq_config->llq_ring_entry_size_value = 128;
|
|
||||||
}
|
|
||||||
|
|
||||||
static int ena_calc_io_queue_size(struct ena_calc_queue_size_ctx *ctx)
|
static int ena_calc_io_queue_size(struct ena_calc_queue_size_ctx *ctx)
|
||||||
{
|
{
|
||||||
|
@ -4132,7 +4146,6 @@ static int ena_probe(struct pci_dev *pdev, const struct pci_device_id *ent)
|
||||||
{
|
{
|
||||||
struct ena_calc_queue_size_ctx calc_queue_ctx = { 0 };
|
struct ena_calc_queue_size_ctx calc_queue_ctx = { 0 };
|
||||||
struct ena_com_dev_get_features_ctx get_feat_ctx;
|
struct ena_com_dev_get_features_ctx get_feat_ctx;
|
||||||
struct ena_llq_configurations llq_config;
|
|
||||||
struct ena_com_dev *ena_dev = NULL;
|
struct ena_com_dev *ena_dev = NULL;
|
||||||
struct ena_adapter *adapter;
|
struct ena_adapter *adapter;
|
||||||
struct net_device *netdev;
|
struct net_device *netdev;
|
||||||
|
@ -4187,13 +4200,10 @@ static int ena_probe(struct pci_dev *pdev, const struct pci_device_id *ent)
|
||||||
goto err_free_region;
|
goto err_free_region;
|
||||||
}
|
}
|
||||||
|
|
||||||
set_default_llq_configurations(&llq_config);
|
rc = ena_map_llq_mem_bar(pdev, ena_dev, bars);
|
||||||
|
|
||||||
rc = ena_set_queues_placement_policy(pdev, ena_dev, &get_feat_ctx.llq,
|
|
||||||
&llq_config);
|
|
||||||
if (rc) {
|
if (rc) {
|
||||||
dev_err(&pdev->dev, "ena device init failed\n");
|
dev_err(&pdev->dev, "ena llq bar mapping failed\n");
|
||||||
goto err_device_destroy;
|
goto err_free_ena_dev;
|
||||||
}
|
}
|
||||||
|
|
||||||
calc_queue_ctx.ena_dev = ena_dev;
|
calc_queue_ctx.ena_dev = ena_dev;
|
||||||
|
|
Loading…
Reference in New Issue