aspm.c 27.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_DOWNSTREAM) {
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 551 552 553 554 555
/*
 * pcie_aspm_init_link_state: Initiate PCI express link state.
 * It is called after the pcie and its children devices are scaned.
 * @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 (!pci_is_pcie(pdev) || pdev->link_state)
S
Shaohua Li 已提交
563
		return;
564 565
	if (pci_pcie_type(pdev) != PCI_EXP_TYPE_ROOT_PORT &&
	    pci_pcie_type(pdev) != PCI_EXP_TYPE_DOWNSTREAM)
S
Shaohua Li 已提交
566
		return;
567

568
	/* VIA has a strange chipset, root port is under a bridge */
569
	if (pci_pcie_type(pdev) == PCI_EXP_TYPE_ROOT_PORT &&
570
	    pdev->bus->self)
571
		return;
572

S
Shaohua Li 已提交
573 574 575 576 577
	down_read(&pci_bus_sem);
	if (list_empty(&pdev->subordinate->devices))
		goto out;

	mutex_lock(&aspm_lock);
578
	link = alloc_pcie_link_state(pdev);
579 580 581
	if (!link)
		goto unlock;
	/*
582 583 584
	 * 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().
585
	 */
586
	pcie_aspm_cap_init(link, blacklist);
S
Shaohua Li 已提交
587

588
	/* Setup initial Clock PM state */
589
	pcie_clkpm_cap_init(link, blacklist);
590 591 592 593 594 595 596 597 598

	/*
	 * 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 已提交
599
	if (aspm_policy != POLICY_POWERSAVE) {
600 601 602 603
		pcie_config_aspm_path(link);
		pcie_set_clkpm(link, policy_to_clkpm_state(link));
	}

604
unlock:
S
Shaohua Li 已提交
605 606 607 608 609
	mutex_unlock(&aspm_lock);
out:
	up_read(&pci_bus_sem);
}

610 611 612 613 614 615 616 617 618 619 620 621 622 623 624 625
/* 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) {
626 627
			if ((pci_pcie_type(child) != PCI_EXP_TYPE_ENDPOINT) &&
			    (pci_pcie_type(child) != PCI_EXP_TYPE_LEG_END))
628 629 630 631 632 633
				continue;
			pcie_aspm_check_latency(child);
		}
	}
}

S
Shaohua Li 已提交
634 635 636 637
/* @pdev: the endpoint device */
void pcie_aspm_exit_link_state(struct pci_dev *pdev)
{
	struct pci_dev *parent = pdev->bus->self;
638
	struct pcie_link_state *link, *root, *parent_link;
S
Shaohua Li 已提交
639

640
	if (!parent || !parent->link_state)
S
Shaohua Li 已提交
641
		return;
642

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

652
	link = parent->link_state;
653
	root = link->root;
654
	parent_link = link->parent;
655

S
Shaohua Li 已提交
656
	/* All functions are removed, so just disable ASPM for the link */
657
	pcie_config_aspm_link(link, 0);
658 659
	list_del(&link->sibling);
	list_del(&link->link);
S
Shaohua Li 已提交
660
	/* Clock PM is for endpoint device */
661
	free_link_state(link);
662 663

	/* Recheck latencies and configure upstream links */
664 665 666 667
	if (parent_link) {
		pcie_update_aspm_capable(root);
		pcie_config_aspm_path(parent_link);
	}
S
Shaohua Li 已提交
668 669 670 671 672 673 674 675
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)
{
676
	struct pcie_link_state *link = pdev->link_state;
S
Shaohua Li 已提交
677

K
Kenji Kaneshige 已提交
678
	if (aspm_disabled || !pci_is_pcie(pdev) || !link)
S
Shaohua Li 已提交
679
		return;
680 681
	if ((pci_pcie_type(pdev) != PCI_EXP_TYPE_ROOT_PORT) &&
	    (pci_pcie_type(pdev) != PCI_EXP_TYPE_DOWNSTREAM))
S
Shaohua Li 已提交
682 683
		return;
	/*
684 685
	 * Devices changed PM state, we should recheck if latency
	 * meets all functions' requirement
S
Shaohua Li 已提交
686
	 */
687 688 689
	down_read(&pci_bus_sem);
	mutex_lock(&aspm_lock);
	pcie_update_aspm_capable(link->root);
690
	pcie_config_aspm_path(link);
691 692
	mutex_unlock(&aspm_lock);
	up_read(&pci_bus_sem);
S
Shaohua Li 已提交
693 694
}

695 696 697 698 699 700 701 702 703 704
void pcie_aspm_powersave_config_link(struct pci_dev *pdev)
{
	struct pcie_link_state *link = pdev->link_state;

	if (aspm_disabled || !pci_is_pcie(pdev) || !link)
		return;

	if (aspm_policy != POLICY_POWERSAVE)
		return;

705 706
	if ((pci_pcie_type(pdev) != PCI_EXP_TYPE_ROOT_PORT) &&
	    (pci_pcie_type(pdev) != PCI_EXP_TYPE_DOWNSTREAM))
707 708 709 710 711 712 713 714 715 716
		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);
}

S
Shaohua Li 已提交
717 718 719 720
/*
 * pci_disable_link_state - disable pci device's link state, so the link will
 * never enter specific states
 */
M
Matthew Garrett 已提交
721 722
static void __pci_disable_link_state(struct pci_dev *pdev, int state, bool sem,
				     bool force)
S
Shaohua Li 已提交
723 724
{
	struct pci_dev *parent = pdev->bus->self;
725
	struct pcie_link_state *link;
S
Shaohua Li 已提交
726

M
Matthew Garrett 已提交
727
	if (!pci_is_pcie(pdev))
S
Shaohua Li 已提交
728
		return;
M
Matthew Garrett 已提交
729

730 731
	if (pci_pcie_type(pdev) == PCI_EXP_TYPE_ROOT_PORT ||
	    pci_pcie_type(pdev) == PCI_EXP_TYPE_DOWNSTREAM)
S
Shaohua Li 已提交
732 733 734 735
		parent = pdev;
	if (!parent || !parent->link_state)
		return;

736 737 738 739 740 741 742 743 744 745 746 747 748
	/*
	 * 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.
	 */
	if (aspm_disabled && !force) {
		dev_warn(&pdev->dev, "can't disable ASPM; OS doesn't have ASPM control\n");
		return;
	}

749 750
	if (sem)
		down_read(&pci_bus_sem);
S
Shaohua Li 已提交
751
	mutex_lock(&aspm_lock);
752
	link = parent->link_state;
753 754 755 756
	if (state & PCIE_LINK_STATE_L0S)
		link->aspm_disable |= ASPM_STATE_L0S;
	if (state & PCIE_LINK_STATE_L1)
		link->aspm_disable |= ASPM_STATE_L1;
757 758
	pcie_config_aspm_link(link, policy_to_aspm_state(link));

K
Kenji Kaneshige 已提交
759
	if (state & PCIE_LINK_STATE_CLKPM) {
760 761
		link->clkpm_capable = 0;
		pcie_set_clkpm(link, 0);
K
Kenji Kaneshige 已提交
762
	}
S
Shaohua Li 已提交
763
	mutex_unlock(&aspm_lock);
764 765 766 767 768 769
	if (sem)
		up_read(&pci_bus_sem);
}

void pci_disable_link_state_locked(struct pci_dev *pdev, int state)
{
M
Matthew Garrett 已提交
770
	__pci_disable_link_state(pdev, state, false, false);
771 772 773 774 775
}
EXPORT_SYMBOL(pci_disable_link_state_locked);

void pci_disable_link_state(struct pci_dev *pdev, int state)
{
M
Matthew Garrett 已提交
776
	__pci_disable_link_state(pdev, state, true, false);
S
Shaohua Li 已提交
777 778 779
}
EXPORT_SYMBOL(pci_disable_link_state);

M
Matthew Garrett 已提交
780 781 782 783
void pcie_clear_aspm(struct pci_bus *bus)
{
	struct pci_dev *child;

784 785 786
	if (aspm_force)
		return;

M
Matthew Garrett 已提交
787 788 789 790 791 792 793 794 795 796 797
	/*
	 * Clear any ASPM setup that the firmware has carried out on this bus
	 */
	list_for_each_entry(child, &bus->devices, bus_list) {
		__pci_disable_link_state(child, PCIE_LINK_STATE_L0S |
					 PCIE_LINK_STATE_L1 |
					 PCIE_LINK_STATE_CLKPM,
					 false, true);
	}
}

S
Shaohua Li 已提交
798 799 800
static int pcie_aspm_set_policy(const char *val, struct kernel_param *kp)
{
	int i;
801
	struct pcie_link_state *link;
S
Shaohua Li 已提交
802

803 804
	if (aspm_disabled)
		return -EPERM;
S
Shaohua Li 已提交
805 806 807 808 809 810 811 812 813 814 815
	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;
816 817 818
	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 已提交
819 820 821 822 823 824 825 826 827 828 829 830 831 832 833 834 835 836 837 838 839 840 841 842 843 844 845 846
	}
	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;

847
	return sprintf(buf, "%d\n", link_state->aspm_enabled);
S
Shaohua Li 已提交
848 849 850 851 852 853 854
}

static ssize_t link_state_store(struct device *dev,
		struct device_attribute *attr,
		const char *buf,
		size_t n)
{
855
	struct pci_dev *pdev = to_pci_dev(dev);
856
	struct pcie_link_state *link, *root = pdev->link_state->root;
857
	u32 val = buf[0] - '0', state = 0;
S
Shaohua Li 已提交
858

859 860
	if (aspm_disabled)
		return -EPERM;
861
	if (n < 1 || val > 3)
S
Shaohua Li 已提交
862 863
		return -EINVAL;

864 865 866 867 868 869
	/* 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;

870 871 872 873 874 875 876 877 878 879
	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 已提交
880 881 882 883 884 885 886 887 888
}

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;

889
	return sprintf(buf, "%d\n", link_state->clkpm_enabled);
S
Shaohua Li 已提交
890 891 892 893 894 895 896
}

static ssize_t clk_ctl_store(struct device *dev,
		struct device_attribute *attr,
		const char *buf,
		size_t n)
{
K
Kenji Kaneshige 已提交
897
	struct pci_dev *pdev = to_pci_dev(dev);
S
Shaohua Li 已提交
898 899 900 901 902 903 904 905
	int state;

	if (n < 1)
		return -EINVAL;
	state = buf[0]-'0';

	down_read(&pci_bus_sem);
	mutex_lock(&aspm_lock);
K
Kenji Kaneshige 已提交
906
	pcie_set_clkpm_nocheck(pdev->link_state, !!state);
S
Shaohua Li 已提交
907 908 909 910 911 912 913 914 915 916 917 918 919 920
	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;

K
Kenji Kaneshige 已提交
921
	if (!pci_is_pcie(pdev) ||
922 923
	    (pci_pcie_type(pdev) != PCI_EXP_TYPE_ROOT_PORT &&
	     pci_pcie_type(pdev) != PCI_EXP_TYPE_DOWNSTREAM) || !link_state)
S
Shaohua Li 已提交
924 925
		return;

926
	if (link_state->aspm_support)
S
Shaohua Li 已提交
927 928
		sysfs_add_file_to_group(&pdev->dev.kobj,
			&dev_attr_link_state.attr, power_group);
929
	if (link_state->clkpm_capable)
S
Shaohua Li 已提交
930 931 932 933 934 935 936 937
		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;

K
Kenji Kaneshige 已提交
938
	if (!pci_is_pcie(pdev) ||
939 940
	    (pci_pcie_type(pdev) != PCI_EXP_TYPE_ROOT_PORT &&
	     pci_pcie_type(pdev) != PCI_EXP_TYPE_DOWNSTREAM) || !link_state)
S
Shaohua Li 已提交
941 942
		return;

943
	if (link_state->aspm_support)
S
Shaohua Li 已提交
944 945
		sysfs_remove_file_from_group(&pdev->dev.kobj,
			&dev_attr_link_state.attr, power_group);
946
	if (link_state->clkpm_capable)
S
Shaohua Li 已提交
947 948 949 950 951 952 953
		sysfs_remove_file_from_group(&pdev->dev.kobj,
			&dev_attr_clk_ctl.attr, power_group);
}
#endif

static int __init pcie_aspm_disable(char *str)
{
954
	if (!strcmp(str, "off")) {
M
Matthew Garrett 已提交
955
		aspm_policy = POLICY_DEFAULT;
956
		aspm_disabled = 1;
957
		aspm_support_enabled = false;
958 959 960
		printk(KERN_INFO "PCIe ASPM is disabled\n");
	} else if (!strcmp(str, "force")) {
		aspm_force = 1;
M
Michael Witten 已提交
961
		printk(KERN_INFO "PCIe ASPM is forcibly enabled\n");
962
	}
S
Shaohua Li 已提交
963 964 965
	return 1;
}

966
__setup("pcie_aspm=", pcie_aspm_disable);
S
Shaohua Li 已提交
967

968 969
void pcie_no_aspm(void)
{
M
Matthew Garrett 已提交
970 971 972 973 974 975 976 977
	/*
	 * 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;
978
		aspm_disabled = 1;
M
Matthew Garrett 已提交
979
	}
980 981
}

982 983 984 985 986 987 988
/**
 * pcie_aspm_enabled - is PCIe ASPM enabled?
 *
 * Returns true if ASPM has not been disabled by the command-line option
 * pcie_aspm=off.
 **/
int pcie_aspm_enabled(void)
S
Shaohua Li 已提交
989
{
990
       return !aspm_disabled;
S
Shaohua Li 已提交
991
}
992
EXPORT_SYMBOL(pcie_aspm_enabled);
S
Shaohua Li 已提交
993

994 995 996 997 998
bool pcie_aspm_support_enabled(void)
{
	return aspm_support_enabled;
}
EXPORT_SYMBOL(pcie_aspm_support_enabled);