summaryrefslogtreecommitdiff
path: root/drivers/net/wwan/t7xx/t7xx_pci_rescan.c
blob: 045777d8a843a5b78e11707ce49cb7c2f4e8a174 (plain)
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
// SPDX-License-Identifier: GPL-2.0-only
/*
 * Copyright (c) 2021, MediaTek Inc.
 * Copyright (c) 2021-2022, Intel Corporation.
 */

#define pr_fmt(fmt) KBUILD_MODNAME ":t7xx:%s: " fmt, __func__
#define dev_fmt(fmt) "t7xx: " fmt

#include <linux/delay.h>
#include <linux/pci.h>
#include <linux/spinlock.h>
#include <linux/workqueue.h>

#include "t7xx_pci.h"
#include "t7xx_pci_rescan.h"

static struct remove_rescan_context g_mtk_rescan_context;

void t7xx_pci_dev_rescan(void)
{
	struct pci_bus *b = NULL;

	pci_lock_rescan_remove();
	while ((b = pci_find_next_bus(b)))
		pci_rescan_bus(b);

	pci_unlock_rescan_remove();
}

void t7xx_rescan_done(void)
{
	unsigned long flags;

	spin_lock_irqsave(&g_mtk_rescan_context.dev_lock, flags);
	if (g_mtk_rescan_context.rescan_done == 0) {
		pr_debug("this is a rescan probe\n");
		g_mtk_rescan_context.rescan_done = 1;
	} else {
		pr_debug("this is a init probe\n");
	}
	spin_unlock_irqrestore(&g_mtk_rescan_context.dev_lock, flags);
}

static void t7xx_remove_rescan(struct work_struct *work)
{
	struct pci_dev *pdev;
	int num_retries = RESCAN_RETRIES;
	unsigned long flags;

	spin_lock_irqsave(&g_mtk_rescan_context.dev_lock, flags);
	g_mtk_rescan_context.rescan_done = 0;
	pdev = g_mtk_rescan_context.dev;
	spin_unlock_irqrestore(&g_mtk_rescan_context.dev_lock, flags);

	if (pdev) {
		pci_stop_and_remove_bus_device_locked(pdev);
		pr_debug("start remove and rescan flow\n");
	}

	do {
		t7xx_pci_dev_rescan();
		spin_lock_irqsave(&g_mtk_rescan_context.dev_lock, flags);
		if (g_mtk_rescan_context.rescan_done) {
			spin_unlock_irqrestore(&g_mtk_rescan_context.dev_lock, flags);
			break;
		}

		spin_unlock_irqrestore(&g_mtk_rescan_context.dev_lock, flags);
		msleep(DELAY_RESCAN_MTIME);
	} while (num_retries--);
}

void t7xx_rescan_queue_work(struct pci_dev *pdev)
{
	unsigned long flags;

	dev_info(&pdev->dev, "start queue_mtk_rescan_work\n");
	spin_lock_irqsave(&g_mtk_rescan_context.dev_lock, flags);
	if (!g_mtk_rescan_context.rescan_done) {
		dev_err(&pdev->dev, "rescan failed because last rescan undone\n");
		spin_unlock_irqrestore(&g_mtk_rescan_context.dev_lock, flags);
		return;
	}

	g_mtk_rescan_context.dev = pdev;
	spin_unlock_irqrestore(&g_mtk_rescan_context.dev_lock, flags);
	queue_work(g_mtk_rescan_context.pcie_rescan_wq, &g_mtk_rescan_context.service_task);
}

int t7xx_rescan_init(void)
{
	spin_lock_init(&g_mtk_rescan_context.dev_lock);
	g_mtk_rescan_context.rescan_done = 1;
	g_mtk_rescan_context.dev = NULL;
	g_mtk_rescan_context.pcie_rescan_wq = create_singlethread_workqueue(MTK_RESCAN_WQ);
	if (!g_mtk_rescan_context.pcie_rescan_wq) {
		pr_err("Failed to create workqueue: %s\n", MTK_RESCAN_WQ);
		return -ENOMEM;
	}

	INIT_WORK(&g_mtk_rescan_context.service_task, t7xx_remove_rescan);

	return 0;
}

void t7xx_rescan_deinit(void)
{
	unsigned long flags;

	spin_lock_irqsave(&g_mtk_rescan_context.dev_lock, flags);
	g_mtk_rescan_context.rescan_done = 0;
	g_mtk_rescan_context.dev = NULL;
	spin_unlock_irqrestore(&g_mtk_rescan_context.dev_lock, flags);
	cancel_work_sync(&g_mtk_rescan_context.service_task);
	destroy_workqueue(g_mtk_rescan_context.pcie_rescan_wq);
}