aspm.c 26.4 KB
Newer Older
S
Shaohua Li 已提交
1 2
/*
 * File:	drivers/pci/pcie/aspm.c
3
 * Enabling PCIe link L0s/L1 state and Clock Power Management
S
Shaohua Li 已提交
4 5 6 7 8 9 10 11 12 13 14 15 16 17 18
 *
 * Copyright (C) 2007 Intel
 * Copyright (C) Zhang Yanmin (yanmin.zhang@intel.com)
 * Copyright (C) Shaohua Li (shaohua.li@intel.com)
 */

#include <linux/kernel.h>
#include <linux/module.h>
#include <linux/moduleparam.h>
#include <linux/pci.h>
#include <linux/pci_regs.h>
#include <linux/errno.h>
#include <linux/pm.h>
#include <linux/init.h>
#include <linux/slab.h>
19
#include <linux/jiffies.h>
20
#include <linux/delay.h>
S
Shaohua Li 已提交
21 22 23 24 25 26 27 28
#include <linux/pci-aspm.h>
#include "../pci.h"

#ifdef MODULE_PARAM_PREFIX
#undef MODULE_PARAM_PREFIX
#endif
#define MODULE_PARAM_PREFIX "pcie_aspm."

29 30 31 32 33 34 35
/* Note: those are not register definitions */
#define ASPM_STATE_L0S_UP	(1)	/* Upstream direction L0s state */
#define ASPM_STATE_L0S_DW	(2)	/* Downstream direction L0s state */
#define ASPM_STATE_L1		(4)	/* L1 state */
#define ASPM_STATE_L0S		(ASPM_STATE_L0S_UP | ASPM_STATE_L0S_DW)
#define ASPM_STATE_ALL		(ASPM_STATE_L0S | ASPM_STATE_L1)

36 37 38
struct aspm_latency {
	u32 l0s;			/* L0s latency (nsec) */
	u32 l1;				/* L1 latency (nsec) */
S
Shaohua Li 已提交
39 40 41
};

struct pcie_link_state {
42
	struct pci_dev *pdev;		/* Upstream component of the Link */
43
	struct pcie_link_state *root;	/* pointer to the root port link */
44 45 46 47
	struct pcie_link_state *parent;	/* pointer to the parent Link state */
	struct list_head sibling;	/* node in link_list */
	struct list_head children;	/* list of child link states */
	struct list_head link;		/* node in parent's children list */
S
Shaohua Li 已提交
48 49

	/* ASPM state */
50 51 52 53 54
	u32 aspm_support:3;		/* Supported ASPM state */
	u32 aspm_enabled:3;		/* Enabled ASPM state */
	u32 aspm_capable:3;		/* Capable ASPM state with latency */
	u32 aspm_default:3;		/* Default ASPM state by BIOS */
	u32 aspm_disable:3;		/* Disabled ASPM state */
55

56 57 58 59 60
	/* Clock PM state */
	u32 clkpm_capable:1;		/* Clock PM capable? */
	u32 clkpm_enabled:1;		/* Current Clock PM state */
	u32 clkpm_default:1;		/* Default Clock PM state by BIOS */

61 62 63
	/* Exit latencies */
	struct aspm_latency latency_up;	/* Upstream direction exit latency */
	struct aspm_latency latency_dw;	/* Downstream direction exit latency */
S
Shaohua Li 已提交
64
	/*
65 66
	 * Endpoint acceptable latencies. A pcie downstream port only
	 * has one slot under it, so at most there are 8 functions.
S
Shaohua Li 已提交
67
	 */
68
	struct aspm_latency acceptable[8];
S
Shaohua Li 已提交
69 70
};

M
Matthew Garrett 已提交
71
static int aspm_disabled, aspm_force;
72
static bool aspm_support_enabled = true;
S
Shaohua Li 已提交
73 74 75 76 77 78
static DEFINE_MUTEX(aspm_lock);
static LIST_HEAD(link_list);

#define POLICY_DEFAULT 0	/* BIOS default setting */
#define POLICY_PERFORMANCE 1	/* high performance */
#define POLICY_POWERSAVE 2	/* high power saving */
79 80 81 82 83 84

#ifdef CONFIG_PCIEASPM_PERFORMANCE
static int aspm_policy = POLICY_PERFORMANCE;
#elif defined CONFIG_PCIEASPM_POWERSAVE
static int aspm_policy = POLICY_POWERSAVE;
#else
S
Shaohua Li 已提交
85
static int aspm_policy;
86 87
#endif

S
Shaohua Li 已提交
88 89 90 91 92 93
static const char *policy_str[] = {
	[POLICY_DEFAULT] = "default",
	[POLICY_PERFORMANCE] = "performance",
	[POLICY_POWERSAVE] = "powersave"
};

94 95
#define LINK_RETRAIN_TIMEOUT HZ

96
static int policy_to_aspm_state(struct pcie_link_state *link)
S
Shaohua Li 已提交
97 98 99 100 101 102 103
{
	switch (aspm_policy) {
	case POLICY_PERFORMANCE:
		/* Disable ASPM and Clock PM */
		return 0;
	case POLICY_POWERSAVE:
		/* Enable ASPM L0s/L1 */
104
		return ASPM_STATE_ALL;
S
Shaohua Li 已提交
105
	case POLICY_DEFAULT:
106
		return link->aspm_default;
S
Shaohua Li 已提交
107 108 109 110
	}
	return 0;
}

111
static int policy_to_clkpm_state(struct pcie_link_state *link)
S
Shaohua Li 已提交
112 113 114 115 116 117 118 119 120
{
	switch (aspm_policy) {
	case POLICY_PERFORMANCE:
		/* Disable ASPM and Clock PM */
		return 0;
	case POLICY_POWERSAVE:
		/* Disable Clock PM */
		return 1;
	case POLICY_DEFAULT:
121
		return link->clkpm_default;
S
Shaohua Li 已提交
122 123 124 125
	}
	return 0;
}

K
Kenji Kaneshige 已提交
126
static void pcie_set_clkpm_nocheck(struct pcie_link_state *link, int enable)
S
Shaohua Li 已提交
127
{
128 129
	struct pci_dev *child;
	struct pci_bus *linkbus = link->pdev->subordinate;
S
Shaohua Li 已提交
130

131
	list_for_each_entry(child, &linkbus->devices, bus_list) {
S
Shaohua Li 已提交
132
		if (enable)
133 134
			pcie_capability_set_word(child, PCI_EXP_LNKCTL,
						 PCI_EXP_LNKCTL_CLKREQ_EN);
S
Shaohua Li 已提交
135
		else
136 137
			pcie_capability_clear_word(child, PCI_EXP_LNKCTL,
						   PCI_EXP_LNKCTL_CLKREQ_EN);
S
Shaohua Li 已提交
138
	}
139
	link->clkpm_enabled = !!enable;
S
Shaohua Li 已提交
140 141
}

K
Kenji Kaneshige 已提交
142 143 144 145
static void pcie_set_clkpm(struct pcie_link_state *link, int enable)
{
	/* Don't enable Clock PM if the link is not Clock PM capable */
	if (!link->clkpm_capable && enable)
146
		enable = 0;
K
Kenji Kaneshige 已提交
147 148 149 150 151 152
	/* Need nothing if the specified equals to current state */
	if (link->clkpm_enabled == enable)
		return;
	pcie_set_clkpm_nocheck(link, enable);
}

153
static void pcie_clkpm_cap_init(struct pcie_link_state *link, int blacklist)
S
Shaohua Li 已提交
154
{
155
	int capable = 1, enabled = 1;
S
Shaohua Li 已提交
156 157
	u32 reg32;
	u16 reg16;
158 159
	struct pci_dev *child;
	struct pci_bus *linkbus = link->pdev->subordinate;
S
Shaohua Li 已提交
160 161

	/* All functions should have the same cap and state, take the worst */
162
	list_for_each_entry(child, &linkbus->devices, bus_list) {
163
		pcie_capability_read_dword(child, PCI_EXP_LNKCAP, &reg32);
S
Shaohua Li 已提交
164 165 166 167 168
		if (!(reg32 & PCI_EXP_LNKCAP_CLKPM)) {
			capable = 0;
			enabled = 0;
			break;
		}
169
		pcie_capability_read_word(child, PCI_EXP_LNKCTL, &reg16);
S
Shaohua Li 已提交
170 171 172
		if (!(reg16 & PCI_EXP_LNKCTL_CLKREQ_EN))
			enabled = 0;
	}
173 174
	link->clkpm_enabled = enabled;
	link->clkpm_default = enabled;
175
	link->clkpm_capable = (blacklist) ? 0 : capable;
176 177
}

S
Shaohua Li 已提交
178 179 180 181 182
/*
 * pcie_aspm_configure_common_clock: check if the 2 ends of a link
 *   could use common clock. If they are, configure them to use the
 *   common clock. That will reduce the ASPM state exit latency.
 */
183
static void pcie_aspm_configure_common_clock(struct pcie_link_state *link)
S
Shaohua Li 已提交
184
{
185
	int same_clock = 1;
186
	u16 reg16, parent_reg, child_reg[8];
187
	unsigned long start_jiffies;
188 189
	struct pci_dev *child, *parent = link->pdev;
	struct pci_bus *linkbus = parent->subordinate;
S
Shaohua Li 已提交
190
	/*
191
	 * All functions of a slot should have the same Slot Clock
S
Shaohua Li 已提交
192
	 * Configuration, so just check one function
193 194
	 */
	child = list_entry(linkbus->devices.next, struct pci_dev, bus_list);
K
Kenji Kaneshige 已提交
195
	BUG_ON(!pci_is_pcie(child));
S
Shaohua Li 已提交
196 197

	/* Check downstream component if bit Slot Clock Configuration is 1 */
198
	pcie_capability_read_word(child, PCI_EXP_LNKSTA, &reg16);
S
Shaohua Li 已提交
199 200 201 202
	if (!(reg16 & PCI_EXP_LNKSTA_SLC))
		same_clock = 0;

	/* Check upstream component if bit Slot Clock Configuration is 1 */
203
	pcie_capability_read_word(parent, PCI_EXP_LNKSTA, &reg16);
S
Shaohua Li 已提交
204 205 206 207
	if (!(reg16 & PCI_EXP_LNKSTA_SLC))
		same_clock = 0;

	/* Configure downstream component, all functions */
208
	list_for_each_entry(child, &linkbus->devices, bus_list) {
209
		pcie_capability_read_word(child, PCI_EXP_LNKCTL, &reg16);
210
		child_reg[PCI_FUNC(child->devfn)] = reg16;
S
Shaohua Li 已提交
211 212 213 214
		if (same_clock)
			reg16 |= PCI_EXP_LNKCTL_CCC;
		else
			reg16 &= ~PCI_EXP_LNKCTL_CCC;
215
		pcie_capability_write_word(child, PCI_EXP_LNKCTL, reg16);
S
Shaohua Li 已提交
216 217 218
	}

	/* Configure upstream component */
219
	pcie_capability_read_word(parent, PCI_EXP_LNKCTL, &reg16);
220
	parent_reg = reg16;
S
Shaohua Li 已提交
221 222 223 224
	if (same_clock)
		reg16 |= PCI_EXP_LNKCTL_CCC;
	else
		reg16 &= ~PCI_EXP_LNKCTL_CCC;
225
	pcie_capability_write_word(parent, PCI_EXP_LNKCTL, reg16);
S
Shaohua Li 已提交
226

227
	/* Retrain link */
S
Shaohua Li 已提交
228
	reg16 |= PCI_EXP_LNKCTL_RL;
229
	pcie_capability_write_word(parent, PCI_EXP_LNKCTL, reg16);
S
Shaohua Li 已提交
230

231
	/* Wait for link training end. Break out after waiting for timeout */
232
	start_jiffies = jiffies;
233
	for (;;) {
234
		pcie_capability_read_word(parent, PCI_EXP_LNKSTA, &reg16);
S
Shaohua Li 已提交
235 236
		if (!(reg16 & PCI_EXP_LNKSTA_LT))
			break;
237 238 239
		if (time_after(jiffies, start_jiffies + LINK_RETRAIN_TIMEOUT))
			break;
		msleep(1);
S
Shaohua Li 已提交
240
	}
241 242 243 244
	if (!(reg16 & PCI_EXP_LNKSTA_LT))
		return;

	/* Training failed. Restore common clock configurations */
245
	dev_err(&parent->dev, "ASPM: Could not configure common clock\n");
246 247 248 249
	list_for_each_entry(child, &linkbus->devices, bus_list)
		pcie_capability_write_word(child, PCI_EXP_LNKCTL,
					   child_reg[PCI_FUNC(child->devfn)]);
	pcie_capability_write_word(parent, PCI_EXP_LNKCTL, parent_reg);
S
Shaohua Li 已提交
250 251
}

252 253
/* Convert L0s latency encoding to ns */
static u32 calc_l0s_latency(u32 encoding)
S
Shaohua Li 已提交
254
{
255 256 257 258
	if (encoding == 0x7)
		return (5 * 1000);	/* > 4us */
	return (64 << encoding);
}
S
Shaohua Li 已提交
259

260 261 262 263 264 265
/* Convert L0s acceptable latency encoding to ns */
static u32 calc_l0s_acceptable(u32 encoding)
{
	if (encoding == 0x7)
		return -1U;
	return (64 << encoding);
S
Shaohua Li 已提交
266 267
}

268 269
/* Convert L1 latency encoding to ns */
static u32 calc_l1_latency(u32 encoding)
S
Shaohua Li 已提交
270
{
271 272 273 274
	if (encoding == 0x7)
		return (65 * 1000);	/* > 64us */
	return (1000 << encoding);
}
S
Shaohua Li 已提交
275

276 277 278 279 280 281
/* Convert L1 acceptable latency encoding to ns */
static u32 calc_l1_acceptable(u32 encoding)
{
	if (encoding == 0x7)
		return -1U;
	return (1000 << encoding);
S
Shaohua Li 已提交
282 283
}

284 285 286 287 288 289 290 291 292
struct aspm_register_info {
	u32 support:2;
	u32 enabled:2;
	u32 latency_encoding_l0s;
	u32 latency_encoding_l1;
};

static void pcie_get_aspm_reg(struct pci_dev *pdev,
			      struct aspm_register_info *info)
S
Shaohua Li 已提交
293 294
{
	u16 reg16;
295
	u32 reg32;
S
Shaohua Li 已提交
296

297
	pcie_capability_read_dword(pdev, PCI_EXP_LNKCAP, &reg32);
298 299 300
	info->support = (reg32 & PCI_EXP_LNKCAP_ASPMS) >> 10;
	info->latency_encoding_l0s = (reg32 & PCI_EXP_LNKCAP_L0SEL) >> 12;
	info->latency_encoding_l1  = (reg32 & PCI_EXP_LNKCAP_L1EL) >> 15;
301
	pcie_capability_read_word(pdev, PCI_EXP_LNKCTL, &reg16);
302
	info->enabled = reg16 & PCI_EXP_LNKCTL_ASPMC;
S
Shaohua Li 已提交
303 304
}

305 306
static void pcie_aspm_check_latency(struct pci_dev *endpoint)
{
307
	u32 latency, l1_switch_latency = 0;
308 309 310 311 312 313 314 315 316 317 318 319
	struct aspm_latency *acceptable;
	struct pcie_link_state *link;

	/* Device not in D0 doesn't need latency check */
	if ((endpoint->current_state != PCI_D0) &&
	    (endpoint->current_state != PCI_UNKNOWN))
		return;

	link = endpoint->bus->self->link_state;
	acceptable = &link->acceptable[PCI_FUNC(endpoint->devfn)];

	while (link) {
320 321 322 323 324 325 326 327 328
		/* Check upstream direction L0s latency */
		if ((link->aspm_capable & ASPM_STATE_L0S_UP) &&
		    (link->latency_up.l0s > acceptable->l0s))
			link->aspm_capable &= ~ASPM_STATE_L0S_UP;

		/* Check downstream direction L0s latency */
		if ((link->aspm_capable & ASPM_STATE_L0S_DW) &&
		    (link->latency_dw.l0s > acceptable->l0s))
			link->aspm_capable &= ~ASPM_STATE_L0S_DW;
329 330 331 332 333
		/*
		 * Check L1 latency.
		 * Every switch on the path to root complex need 1
		 * more microsecond for L1. Spec doesn't mention L0s.
		 */
334 335 336 337
		latency = max_t(u32, link->latency_up.l1, link->latency_dw.l1);
		if ((link->aspm_capable & ASPM_STATE_L1) &&
		    (latency + l1_switch_latency > acceptable->l1))
			link->aspm_capable &= ~ASPM_STATE_L1;
338 339 340 341 342 343
		l1_switch_latency += 1000;

		link = link->parent;
	}
}

344
static void pcie_aspm_cap_init(struct pcie_link_state *link, int blacklist)
S
Shaohua Li 已提交
345
{
346 347
	struct pci_dev *child, *parent = link->pdev;
	struct pci_bus *linkbus = parent->subordinate;
348
	struct aspm_register_info upreg, dwreg;
S
Shaohua Li 已提交
349

350
	if (blacklist) {
351
		/* Set enabled/disable so that we will disable ASPM later */
352 353
		link->aspm_enabled = ASPM_STATE_ALL;
		link->aspm_disable = ASPM_STATE_ALL;
354 355 356 357 358 359
		return;
	}

	/* Configure common clock before checking latencies */
	pcie_aspm_configure_common_clock(link);

360 361
	/* Get upstream/downstream components' register state */
	pcie_get_aspm_reg(parent, &upreg);
362
	child = list_entry(linkbus->devices.next, struct pci_dev, bus_list);
363 364 365 366 367 368 369 370 371 372 373 374 375 376 377 378 379 380 381 382 383 384 385 386 387
	pcie_get_aspm_reg(child, &dwreg);

	/*
	 * Setup L0s state
	 *
	 * Note that we must not enable L0s in either direction on a
	 * given link unless components on both sides of the link each
	 * support L0s.
	 */
	if (dwreg.support & upreg.support & PCIE_LINK_STATE_L0S)
		link->aspm_support |= ASPM_STATE_L0S;
	if (dwreg.enabled & PCIE_LINK_STATE_L0S)
		link->aspm_enabled |= ASPM_STATE_L0S_UP;
	if (upreg.enabled & PCIE_LINK_STATE_L0S)
		link->aspm_enabled |= ASPM_STATE_L0S_DW;
	link->latency_up.l0s = calc_l0s_latency(upreg.latency_encoding_l0s);
	link->latency_dw.l0s = calc_l0s_latency(dwreg.latency_encoding_l0s);

	/* Setup L1 state */
	if (upreg.support & dwreg.support & PCIE_LINK_STATE_L1)
		link->aspm_support |= ASPM_STATE_L1;
	if (upreg.enabled & dwreg.enabled & PCIE_LINK_STATE_L1)
		link->aspm_enabled |= ASPM_STATE_L1;
	link->latency_up.l1 = calc_l1_latency(upreg.latency_encoding_l1);
	link->latency_dw.l1 = calc_l1_latency(dwreg.latency_encoding_l1);
388

389 390
	/* Save default state */
	link->aspm_default = link->aspm_enabled;
391 392 393

	/* Setup initial capable state. Will be updated later */
	link->aspm_capable = link->aspm_support;
394 395 396 397 398
	/*
	 * If the downstream component has pci bridge function, don't
	 * do ASPM for now.
	 */
	list_for_each_entry(child, &linkbus->devices, bus_list) {
399
		if (pci_pcie_type(child) == PCI_EXP_TYPE_PCI_BRIDGE) {
400
			link->aspm_disable = ASPM_STATE_ALL;
401 402 403
			break;
		}
	}
404

405
	/* Get and check endpoint acceptable latencies */
406
	list_for_each_entry(child, &linkbus->devices, bus_list) {
407
		u32 reg32, encoding;
408
		struct aspm_latency *acceptable =
409
			&link->acceptable[PCI_FUNC(child->devfn)];
S
Shaohua Li 已提交
410

411 412
		if (pci_pcie_type(child) != PCI_EXP_TYPE_ENDPOINT &&
		    pci_pcie_type(child) != PCI_EXP_TYPE_LEG_END)
S
Shaohua Li 已提交
413 414
			continue;

415
		pcie_capability_read_dword(child, PCI_EXP_DEVCAP, &reg32);
416
		/* Calculate endpoint L0s acceptable latency */
417 418
		encoding = (reg32 & PCI_EXP_DEVCAP_L0S) >> 6;
		acceptable->l0s = calc_l0s_acceptable(encoding);
419 420 421 422 423
		/* Calculate endpoint L1 acceptable latency */
		encoding = (reg32 & PCI_EXP_DEVCAP_L1) >> 9;
		acceptable->l1 = calc_l1_acceptable(encoding);

		pcie_aspm_check_latency(child);
S
Shaohua Li 已提交
424 425 426
	}
}

427
static void pcie_config_aspm_dev(struct pci_dev *pdev, u32 val)
S
Shaohua Li 已提交
428
{
429 430
	pcie_capability_clear_and_set_word(pdev, PCI_EXP_LNKCTL,
					   PCI_EXP_LNKCTL_ASPMC, val);
S
Shaohua Li 已提交
431 432
}

433
static void pcie_config_aspm_link(struct pcie_link_state *link, u32 state)
S
Shaohua Li 已提交
434
{
435
	u32 upstream = 0, dwstream = 0;
436 437
	struct pci_dev *child, *parent = link->pdev;
	struct pci_bus *linkbus = parent->subordinate;
S
Shaohua Li 已提交
438

439
	/* Nothing to do if the link is already in the requested state */
440
	state &= (link->aspm_capable & ~link->aspm_disable);
441 442
	if (link->aspm_enabled == state)
		return;
443 444
	/* Convert ASPM state to upstream/downstream ASPM register state */
	if (state & ASPM_STATE_L0S_UP)
445
		dwstream |= PCI_EXP_LNKCTL_ASPM_L0S;
446
	if (state & ASPM_STATE_L0S_DW)
447
		upstream |= PCI_EXP_LNKCTL_ASPM_L0S;
448
	if (state & ASPM_STATE_L1) {
449 450
		upstream |= PCI_EXP_LNKCTL_ASPM_L1;
		dwstream |= PCI_EXP_LNKCTL_ASPM_L1;
451
	}
S
Shaohua Li 已提交
452
	/*
453 454 455 456
	 * Spec 2.0 suggests all functions should be configured the
	 * same setting for ASPM. Enabling ASPM L1 should be done in
	 * upstream component first and then downstream, and vice
	 * versa for disabling ASPM L1. Spec doesn't mention L0S.
S
Shaohua Li 已提交
457
	 */
458 459
	if (state & ASPM_STATE_L1)
		pcie_config_aspm_dev(parent, upstream);
460
	list_for_each_entry(child, &linkbus->devices, bus_list)
461 462 463
		pcie_config_aspm_dev(child, dwstream);
	if (!(state & ASPM_STATE_L1))
		pcie_config_aspm_dev(parent, upstream);
S
Shaohua Li 已提交
464

465
	link->aspm_enabled = state;
S
Shaohua Li 已提交
466 467
}

468
static void pcie_config_aspm_path(struct pcie_link_state *link)
S
Shaohua Li 已提交
469
{
470 471 472
	while (link) {
		pcie_config_aspm_link(link, policy_to_aspm_state(link));
		link = link->parent;
473
	}
S
Shaohua Li 已提交
474 475
}

476
static void free_link_state(struct pcie_link_state *link)
S
Shaohua Li 已提交
477
{
478 479
	link->pdev->link_state = NULL;
	kfree(link);
S
Shaohua Li 已提交
480 481
}

482 483
static int pcie_aspm_sanity_check(struct pci_dev *pdev)
{
484
	struct pci_dev *child;
485
	u32 reg32;
486

487
	/*
488
	 * Some functions in a slot might not all be PCIe functions,
489
	 * very strange. Disable ASPM for the whole slot
490
	 */
491
	list_for_each_entry(child, &pdev->subordinate->devices, bus_list) {
492
		if (!pci_is_pcie(child))
493
			return -EINVAL;
494 495 496 497 498 499 500 501 502 503

		/*
		 * If ASPM is disabled then we're not going to change
		 * the BIOS state. It's safe to continue even if it's a
		 * pre-1.1 device
		 */

		if (aspm_disabled)
			continue;

504 505 506 507
		/*
		 * Disable ASPM for pre-1.1 PCIe device, we follow MS to use
		 * RBER bit to determine if a function is 1.1 version device
		 */
508
		pcie_capability_read_dword(child, PCI_EXP_DEVCAP, &reg32);
S
Sitsofe Wheeler 已提交
509
		if (!(reg32 & PCI_EXP_DEVCAP_RBER) && !aspm_force) {
510
			dev_info(&child->dev, "disabling ASPM on pre-1.1 PCIe device.  You can enable it with 'pcie_aspm=force'\n");
511 512
			return -EINVAL;
		}
513 514 515 516
	}
	return 0;
}

517
static struct pcie_link_state *alloc_pcie_link_state(struct pci_dev *pdev)
518 519 520 521 522 523 524 525 526 527
{
	struct pcie_link_state *link;

	link = kzalloc(sizeof(*link), GFP_KERNEL);
	if (!link)
		return NULL;
	INIT_LIST_HEAD(&link->sibling);
	INIT_LIST_HEAD(&link->children);
	INIT_LIST_HEAD(&link->link);
	link->pdev = pdev;
528
	if (pci_pcie_type(pdev) != PCI_EXP_TYPE_ROOT_PORT) {
529 530 531 532 533 534 535 536 537
		struct pcie_link_state *parent;
		parent = pdev->bus->parent->self->link_state;
		if (!parent) {
			kfree(link);
			return NULL;
		}
		link->parent = parent;
		list_add(&link->link, &parent->children);
	}
538 539 540 541 542 543
	/* Setup a pointer to the root port link */
	if (!link->parent)
		link->root = link;
	else
		link->root = link->parent->root;

544 545 546 547 548
	list_add(&link->sibling, &link_list);
	pdev->link_state = link;
	return link;
}

S
Shaohua Li 已提交
549 550
/*
 * pcie_aspm_init_link_state: Initiate PCI express link state.
551
 * It is called after the pcie and its children devices are scanned.
S
Shaohua Li 已提交
552 553 554 555
 * @pdev: the root port or switch downstream port
 */
void pcie_aspm_init_link_state(struct pci_dev *pdev)
{
556
	struct pcie_link_state *link;
557
	int blacklist = !!pcie_aspm_sanity_check(pdev);
S
Shaohua Li 已提交
558

559 560 561
	if (!aspm_support_enabled)
		return;

562
	if (pdev->link_state)
S
Shaohua Li 已提交
563
		return;
564 565 566 567 568 569 570

	/*
	 * We allocate pcie_link_state for the component on the upstream
	 * end of a Link, so there's nothing to do unless this device has a
	 * Link on its secondary side.
	 */
	if (!pdev->has_secondary_link)
S
Shaohua Li 已提交
571
		return;
572

573
	/* VIA has a strange chipset, root port is under a bridge */
574
	if (pci_pcie_type(pdev) == PCI_EXP_TYPE_ROOT_PORT &&
575
	    pdev->bus->self)
576
		return;
577

S
Shaohua Li 已提交
578 579 580 581 582
	down_read(&pci_bus_sem);
	if (list_empty(&pdev->subordinate->devices))
		goto out;

	mutex_lock(&aspm_lock);
583
	link = alloc_pcie_link_state(pdev);
584 585 586
	if (!link)
		goto unlock;
	/*
587 588 589
	 * Setup initial ASPM state. Note that we need to configure
	 * upstream links also because capable state of them can be
	 * update through pcie_aspm_cap_init().
590
	 */
591
	pcie_aspm_cap_init(link, blacklist);
S
Shaohua Li 已提交
592

593
	/* Setup initial Clock PM state */
594
	pcie_clkpm_cap_init(link, blacklist);
595 596 597 598 599 600 601 602 603

	/*
	 * At this stage drivers haven't had an opportunity to change the
	 * link policy setting. Enabling ASPM on broken hardware can cripple
	 * it even before the driver has had a chance to disable ASPM, so
	 * default to a safe level right now. If we're enabling ASPM beyond
	 * the BIOS's expectation, we'll do so once pci_enable_device() is
	 * called.
	 */
M
Matthew Garrett 已提交
604
	if (aspm_policy != POLICY_POWERSAVE) {
605 606 607 608
		pcie_config_aspm_path(link);
		pcie_set_clkpm(link, policy_to_clkpm_state(link));
	}

609
unlock:
S
Shaohua Li 已提交
610 611 612 613 614
	mutex_unlock(&aspm_lock);
out:
	up_read(&pci_bus_sem);
}

615 616 617 618 619 620 621 622 623 624 625 626 627 628 629 630
/* Recheck latencies and update aspm_capable for links under the root */
static void pcie_update_aspm_capable(struct pcie_link_state *root)
{
	struct pcie_link_state *link;
	BUG_ON(root->parent);
	list_for_each_entry(link, &link_list, sibling) {
		if (link->root != root)
			continue;
		link->aspm_capable = link->aspm_support;
	}
	list_for_each_entry(link, &link_list, sibling) {
		struct pci_dev *child;
		struct pci_bus *linkbus = link->pdev->subordinate;
		if (link->root != root)
			continue;
		list_for_each_entry(child, &linkbus->devices, bus_list) {
631 632
			if ((pci_pcie_type(child) != PCI_EXP_TYPE_ENDPOINT) &&
			    (pci_pcie_type(child) != PCI_EXP_TYPE_LEG_END))
633 634 635 636 637 638
				continue;
			pcie_aspm_check_latency(child);
		}
	}
}

S
Shaohua Li 已提交
639 640 641 642
/* @pdev: the endpoint device */
void pcie_aspm_exit_link_state(struct pci_dev *pdev)
{
	struct pci_dev *parent = pdev->bus->self;
643
	struct pcie_link_state *link, *root, *parent_link;
S
Shaohua Li 已提交
644

645
	if (!parent || !parent->link_state)
S
Shaohua Li 已提交
646
		return;
647

S
Shaohua Li 已提交
648 649 650 651
	down_read(&pci_bus_sem);
	mutex_lock(&aspm_lock);
	/*
	 * All PCIe functions are in one slot, remove one function will remove
652
	 * the whole slot, so just wait until we are the last function left.
S
Shaohua Li 已提交
653
	 */
654
	if (!list_is_last(&pdev->bus_list, &parent->subordinate->devices))
S
Shaohua Li 已提交
655 656
		goto out;

657
	link = parent->link_state;
658
	root = link->root;
659
	parent_link = link->parent;
660

S
Shaohua Li 已提交
661
	/* All functions are removed, so just disable ASPM for the link */
662
	pcie_config_aspm_link(link, 0);
663 664
	list_del(&link->sibling);
	list_del(&link->link);
S
Shaohua Li 已提交
665
	/* Clock PM is for endpoint device */
666
	free_link_state(link);
667 668

	/* Recheck latencies and configure upstream links */
669 670 671 672
	if (parent_link) {
		pcie_update_aspm_capable(root);
		pcie_config_aspm_path(parent_link);
	}
S
Shaohua Li 已提交
673 674 675 676 677 678 679 680
out:
	mutex_unlock(&aspm_lock);
	up_read(&pci_bus_sem);
}

/* @pdev: the root port or switch downstream port */
void pcie_aspm_pm_state_change(struct pci_dev *pdev)
{
681
	struct pcie_link_state *link = pdev->link_state;
S
Shaohua Li 已提交
682

683
	if (aspm_disabled || !link)
S
Shaohua Li 已提交
684 685
		return;
	/*
686 687
	 * Devices changed PM state, we should recheck if latency
	 * meets all functions' requirement
S
Shaohua Li 已提交
688
	 */
689 690 691
	down_read(&pci_bus_sem);
	mutex_lock(&aspm_lock);
	pcie_update_aspm_capable(link->root);
692
	pcie_config_aspm_path(link);
693 694
	mutex_unlock(&aspm_lock);
	up_read(&pci_bus_sem);
S
Shaohua Li 已提交
695 696
}

697 698 699 700
void pcie_aspm_powersave_config_link(struct pci_dev *pdev)
{
	struct pcie_link_state *link = pdev->link_state;

701
	if (aspm_disabled || !link)
702 703 704 705 706 707 708 709 710 711 712 713 714
		return;

	if (aspm_policy != POLICY_POWERSAVE)
		return;

	down_read(&pci_bus_sem);
	mutex_lock(&aspm_lock);
	pcie_config_aspm_path(link);
	pcie_set_clkpm(link, policy_to_clkpm_state(link));
	mutex_unlock(&aspm_lock);
	up_read(&pci_bus_sem);
}

715
static void __pci_disable_link_state(struct pci_dev *pdev, int state, bool sem)
S
Shaohua Li 已提交
716 717
{
	struct pci_dev *parent = pdev->bus->self;
718
	struct pcie_link_state *link;
S
Shaohua Li 已提交
719

M
Matthew Garrett 已提交
720
	if (!pci_is_pcie(pdev))
S
Shaohua Li 已提交
721
		return;
M
Matthew Garrett 已提交
722

723
	if (pdev->has_secondary_link)
S
Shaohua Li 已提交
724 725 726 727
		parent = pdev;
	if (!parent || !parent->link_state)
		return;

728 729 730 731 732 733 734 735
	/*
	 * A driver requested that ASPM be disabled on this device, but
	 * if we don't have permission to manage ASPM (e.g., on ACPI
	 * systems we have to observe the FADT ACPI_FADT_NO_ASPM bit and
	 * the _OSC method), we can't honor that request.  Windows has
	 * a similar mechanism using "PciASPMOptOut", which is also
	 * ignored in this situation.
	 */
736
	if (aspm_disabled) {
737 738 739 740
		dev_warn(&pdev->dev, "can't disable ASPM; OS doesn't have ASPM control\n");
		return;
	}

741 742
	if (sem)
		down_read(&pci_bus_sem);
S
Shaohua Li 已提交
743
	mutex_lock(&aspm_lock);
744
	link = parent->link_state;
745 746 747 748
	if (state & PCIE_LINK_STATE_L0S)
		link->aspm_disable |= ASPM_STATE_L0S;
	if (state & PCIE_LINK_STATE_L1)
		link->aspm_disable |= ASPM_STATE_L1;
749 750
	pcie_config_aspm_link(link, policy_to_aspm_state(link));

K
Kenji Kaneshige 已提交
751
	if (state & PCIE_LINK_STATE_CLKPM) {
752 753
		link->clkpm_capable = 0;
		pcie_set_clkpm(link, 0);
K
Kenji Kaneshige 已提交
754
	}
S
Shaohua Li 已提交
755
	mutex_unlock(&aspm_lock);
756 757 758 759 760 761
	if (sem)
		up_read(&pci_bus_sem);
}

void pci_disable_link_state_locked(struct pci_dev *pdev, int state)
{
762
	__pci_disable_link_state(pdev, state, false);
763 764 765
}
EXPORT_SYMBOL(pci_disable_link_state_locked);

766 767 768 769 770 771 772 773 774
/**
 * pci_disable_link_state - Disable device's link state, so the link will
 * never enter specific states.  Note that if the BIOS didn't grant ASPM
 * control to the OS, this does nothing because we can't touch the LNKCTL
 * register.
 *
 * @pdev: PCI device
 * @state: ASPM link state to disable
 */
775 776
void pci_disable_link_state(struct pci_dev *pdev, int state)
{
777
	__pci_disable_link_state(pdev, state, true);
S
Shaohua Li 已提交
778 779 780 781 782 783
}
EXPORT_SYMBOL(pci_disable_link_state);

static int pcie_aspm_set_policy(const char *val, struct kernel_param *kp)
{
	int i;
784
	struct pcie_link_state *link;
S
Shaohua Li 已提交
785

786 787
	if (aspm_disabled)
		return -EPERM;
S
Shaohua Li 已提交
788 789 790 791 792 793 794 795 796 797 798
	for (i = 0; i < ARRAY_SIZE(policy_str); i++)
		if (!strncmp(val, policy_str[i], strlen(policy_str[i])))
			break;
	if (i >= ARRAY_SIZE(policy_str))
		return -EINVAL;
	if (i == aspm_policy)
		return 0;

	down_read(&pci_bus_sem);
	mutex_lock(&aspm_lock);
	aspm_policy = i;
799 800 801
	list_for_each_entry(link, &link_list, sibling) {
		pcie_config_aspm_link(link, policy_to_aspm_state(link));
		pcie_set_clkpm(link, policy_to_clkpm_state(link));
S
Shaohua Li 已提交
802 803 804 805 806 807 808 809 810 811 812 813 814 815 816 817 818 819 820 821 822 823 824 825 826 827 828 829
	}
	mutex_unlock(&aspm_lock);
	up_read(&pci_bus_sem);
	return 0;
}

static int pcie_aspm_get_policy(char *buffer, struct kernel_param *kp)
{
	int i, cnt = 0;
	for (i = 0; i < ARRAY_SIZE(policy_str); i++)
		if (i == aspm_policy)
			cnt += sprintf(buffer + cnt, "[%s] ", policy_str[i]);
		else
			cnt += sprintf(buffer + cnt, "%s ", policy_str[i]);
	return cnt;
}

module_param_call(policy, pcie_aspm_set_policy, pcie_aspm_get_policy,
	NULL, 0644);

#ifdef CONFIG_PCIEASPM_DEBUG
static ssize_t link_state_show(struct device *dev,
		struct device_attribute *attr,
		char *buf)
{
	struct pci_dev *pci_device = to_pci_dev(dev);
	struct pcie_link_state *link_state = pci_device->link_state;

830
	return sprintf(buf, "%d\n", link_state->aspm_enabled);
S
Shaohua Li 已提交
831 832 833 834 835 836 837
}

static ssize_t link_state_store(struct device *dev,
		struct device_attribute *attr,
		const char *buf,
		size_t n)
{
838
	struct pci_dev *pdev = to_pci_dev(dev);
839
	struct pcie_link_state *link, *root = pdev->link_state->root;
840 841 842 843
	u32 val, state = 0;

	if (kstrtouint(buf, 10, &val))
		return -EINVAL;
S
Shaohua Li 已提交
844

845 846
	if (aspm_disabled)
		return -EPERM;
847
	if (n < 1 || val > 3)
S
Shaohua Li 已提交
848 849
		return -EINVAL;

850 851 852 853 854 855
	/* Convert requested state to ASPM state */
	if (val & PCIE_LINK_STATE_L0S)
		state |= ASPM_STATE_L0S;
	if (val & PCIE_LINK_STATE_L1)
		state |= ASPM_STATE_L1;

856 857 858 859 860 861 862 863 864 865
	down_read(&pci_bus_sem);
	mutex_lock(&aspm_lock);
	list_for_each_entry(link, &link_list, sibling) {
		if (link->root != root)
			continue;
		pcie_config_aspm_link(link, state);
	}
	mutex_unlock(&aspm_lock);
	up_read(&pci_bus_sem);
	return n;
S
Shaohua Li 已提交
866 867 868 869 870 871 872 873 874
}

static ssize_t clk_ctl_show(struct device *dev,
		struct device_attribute *attr,
		char *buf)
{
	struct pci_dev *pci_device = to_pci_dev(dev);
	struct pcie_link_state *link_state = pci_device->link_state;

875
	return sprintf(buf, "%d\n", link_state->clkpm_enabled);
S
Shaohua Li 已提交
876 877 878 879 880 881 882
}

static ssize_t clk_ctl_store(struct device *dev,
		struct device_attribute *attr,
		const char *buf,
		size_t n)
{
K
Kenji Kaneshige 已提交
883
	struct pci_dev *pdev = to_pci_dev(dev);
884
	bool state;
S
Shaohua Li 已提交
885

886
	if (strtobool(buf, &state))
S
Shaohua Li 已提交
887 888 889 890
		return -EINVAL;

	down_read(&pci_bus_sem);
	mutex_lock(&aspm_lock);
891
	pcie_set_clkpm_nocheck(pdev->link_state, state);
S
Shaohua Li 已提交
892 893 894 895 896 897 898 899 900 901 902 903 904 905
	mutex_unlock(&aspm_lock);
	up_read(&pci_bus_sem);

	return n;
}

static DEVICE_ATTR(link_state, 0644, link_state_show, link_state_store);
static DEVICE_ATTR(clk_ctl, 0644, clk_ctl_show, clk_ctl_store);

static char power_group[] = "power";
void pcie_aspm_create_sysfs_dev_files(struct pci_dev *pdev)
{
	struct pcie_link_state *link_state = pdev->link_state;

906
	if (!link_state)
S
Shaohua Li 已提交
907 908
		return;

909
	if (link_state->aspm_support)
S
Shaohua Li 已提交
910 911
		sysfs_add_file_to_group(&pdev->dev.kobj,
			&dev_attr_link_state.attr, power_group);
912
	if (link_state->clkpm_capable)
S
Shaohua Li 已提交
913 914 915 916 917 918 919 920
		sysfs_add_file_to_group(&pdev->dev.kobj,
			&dev_attr_clk_ctl.attr, power_group);
}

void pcie_aspm_remove_sysfs_dev_files(struct pci_dev *pdev)
{
	struct pcie_link_state *link_state = pdev->link_state;

921
	if (!link_state)
S
Shaohua Li 已提交
922 923
		return;

924
	if (link_state->aspm_support)
S
Shaohua Li 已提交
925 926
		sysfs_remove_file_from_group(&pdev->dev.kobj,
			&dev_attr_link_state.attr, power_group);
927
	if (link_state->clkpm_capable)
S
Shaohua Li 已提交
928 929 930 931 932 933 934
		sysfs_remove_file_from_group(&pdev->dev.kobj,
			&dev_attr_clk_ctl.attr, power_group);
}
#endif

static int __init pcie_aspm_disable(char *str)
{
935
	if (!strcmp(str, "off")) {
M
Matthew Garrett 已提交
936
		aspm_policy = POLICY_DEFAULT;
937
		aspm_disabled = 1;
938
		aspm_support_enabled = false;
939 940 941
		printk(KERN_INFO "PCIe ASPM is disabled\n");
	} else if (!strcmp(str, "force")) {
		aspm_force = 1;
M
Michael Witten 已提交
942
		printk(KERN_INFO "PCIe ASPM is forcibly enabled\n");
943
	}
S
Shaohua Li 已提交
944 945 946
	return 1;
}

947
__setup("pcie_aspm=", pcie_aspm_disable);
S
Shaohua Li 已提交
948

949 950
void pcie_no_aspm(void)
{
M
Matthew Garrett 已提交
951 952 953 954 955 956 957 958
	/*
	 * Disabling ASPM is intended to prevent the kernel from modifying
	 * existing hardware state, not to clear existing state. To that end:
	 * (a) set policy to POLICY_DEFAULT in order to avoid changing state
	 * (b) prevent userspace from changing policy
	 */
	if (!aspm_force) {
		aspm_policy = POLICY_DEFAULT;
959
		aspm_disabled = 1;
M
Matthew Garrett 已提交
960
	}
961 962
}

963 964 965 966 967
bool pcie_aspm_support_enabled(void)
{
	return aspm_support_enabled;
}
EXPORT_SYMBOL(pcie_aspm_support_enabled);