aspm.c 26.9 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
	if (!pci_is_pcie(pdev) || pdev->link_state)
S
Shaohua Li 已提交
560
		return;
561 562
	if (pci_pcie_type(pdev) != PCI_EXP_TYPE_ROOT_PORT &&
	    pci_pcie_type(pdev) != PCI_EXP_TYPE_DOWNSTREAM)
S
Shaohua Li 已提交
563
		return;
564

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

S
Shaohua Li 已提交
570 571 572 573 574
	down_read(&pci_bus_sem);
	if (list_empty(&pdev->subordinate->devices))
		goto out;

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

585
	/* Setup initial Clock PM state */
586
	pcie_clkpm_cap_init(link, blacklist);
587 588 589 590 591 592 593 594 595

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

601
unlock:
S
Shaohua Li 已提交
602 603 604 605 606
	mutex_unlock(&aspm_lock);
out:
	up_read(&pci_bus_sem);
}

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

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

637
	if (!parent || !parent->link_state)
S
Shaohua Li 已提交
638
		return;
639

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

649
	link = parent->link_state;
650
	root = link->root;
651
	parent_link = link->parent;
652

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

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

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

692 693 694 695 696 697 698 699 700 701
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;

702 703
	if ((pci_pcie_type(pdev) != PCI_EXP_TYPE_ROOT_PORT) &&
	    (pci_pcie_type(pdev) != PCI_EXP_TYPE_DOWNSTREAM))
704 705 706 707 708 709 710 711 712 713
		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 已提交
714 715 716 717
/*
 * pci_disable_link_state - disable pci device's link state, so the link will
 * never enter specific states
 */
M
Matthew Garrett 已提交
718 719
static void __pci_disable_link_state(struct pci_dev *pdev, int state, bool sem,
				     bool force)
S
Shaohua Li 已提交
720 721
{
	struct pci_dev *parent = pdev->bus->self;
722
	struct pcie_link_state *link;
S
Shaohua Li 已提交
723

M
Matthew Garrett 已提交
724 725 726 727
	if (aspm_disabled && !force)
		return;

	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
	if (sem)
		down_read(&pci_bus_sem);
S
Shaohua Li 已提交
738
	mutex_lock(&aspm_lock);
739
	link = parent->link_state;
740 741 742 743
	if (state & PCIE_LINK_STATE_L0S)
		link->aspm_disable |= ASPM_STATE_L0S;
	if (state & PCIE_LINK_STATE_L1)
		link->aspm_disable |= ASPM_STATE_L1;
744 745
	pcie_config_aspm_link(link, policy_to_aspm_state(link));

K
Kenji Kaneshige 已提交
746
	if (state & PCIE_LINK_STATE_CLKPM) {
747 748
		link->clkpm_capable = 0;
		pcie_set_clkpm(link, 0);
K
Kenji Kaneshige 已提交
749
	}
S
Shaohua Li 已提交
750
	mutex_unlock(&aspm_lock);
751 752 753 754 755 756
	if (sem)
		up_read(&pci_bus_sem);
}

void pci_disable_link_state_locked(struct pci_dev *pdev, int state)
{
M
Matthew Garrett 已提交
757
	__pci_disable_link_state(pdev, state, false, false);
758 759 760 761 762
}
EXPORT_SYMBOL(pci_disable_link_state_locked);

void pci_disable_link_state(struct pci_dev *pdev, int state)
{
M
Matthew Garrett 已提交
763
	__pci_disable_link_state(pdev, state, true, false);
S
Shaohua Li 已提交
764 765 766
}
EXPORT_SYMBOL(pci_disable_link_state);

M
Matthew Garrett 已提交
767 768 769 770 771 772 773 774 775 776 777 778 779 780 781
void pcie_clear_aspm(struct pci_bus *bus)
{
	struct pci_dev *child;

	/*
	 * 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 已提交
782 783 784
static int pcie_aspm_set_policy(const char *val, struct kernel_param *kp)
{
	int i;
785
	struct pcie_link_state *link;
S
Shaohua Li 已提交
786

787 788
	if (aspm_disabled)
		return -EPERM;
S
Shaohua Li 已提交
789 790 791 792 793 794 795 796 797 798 799
	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;
800 801 802
	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 已提交
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 830
	}
	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;

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

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

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

848 849 850 851 852 853
	/* 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;

854 855 856 857 858 859 860 861 862 863
	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 已提交
864 865 866 867 868 869 870 871 872
}

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;

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

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

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

	down_read(&pci_bus_sem);
	mutex_lock(&aspm_lock);
K
Kenji Kaneshige 已提交
890
	pcie_set_clkpm_nocheck(pdev->link_state, !!state);
S
Shaohua Li 已提交
891 892 893 894 895 896 897 898 899 900 901 902 903 904
	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 已提交
905
	if (!pci_is_pcie(pdev) ||
906 907
	    (pci_pcie_type(pdev) != PCI_EXP_TYPE_ROOT_PORT &&
	     pci_pcie_type(pdev) != PCI_EXP_TYPE_DOWNSTREAM) || !link_state)
S
Shaohua Li 已提交
908 909
		return;

910
	if (link_state->aspm_support)
S
Shaohua Li 已提交
911 912
		sysfs_add_file_to_group(&pdev->dev.kobj,
			&dev_attr_link_state.attr, power_group);
913
	if (link_state->clkpm_capable)
S
Shaohua Li 已提交
914 915 916 917 918 919 920 921
		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 已提交
922
	if (!pci_is_pcie(pdev) ||
923 924
	    (pci_pcie_type(pdev) != PCI_EXP_TYPE_ROOT_PORT &&
	     pci_pcie_type(pdev) != PCI_EXP_TYPE_DOWNSTREAM) || !link_state)
S
Shaohua Li 已提交
925 926
		return;

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

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

950
__setup("pcie_aspm=", pcie_aspm_disable);
S
Shaohua Li 已提交
951

952 953
void pcie_no_aspm(void)
{
M
Matthew Garrett 已提交
954 955 956 957 958 959 960 961
	/*
	 * 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;
962
		aspm_disabled = 1;
M
Matthew Garrett 已提交
963
	}
964 965
}

966 967 968 969 970 971 972
/**
 * 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 已提交
973
{
974
       return !aspm_disabled;
S
Shaohua Li 已提交
975
}
976
EXPORT_SYMBOL(pcie_aspm_enabled);
S
Shaohua Li 已提交
977

978 979 980 981 982
bool pcie_aspm_support_enabled(void)
{
	return aspm_support_enabled;
}
EXPORT_SYMBOL(pcie_aspm_support_enabled);