Linux Audio

Check our new training course

Loading...
Note: File does not exist in v4.17.
  1// SPDX-License-Identifier: GPL-2.0
  2/*
  3 * NHI specific operations
  4 *
  5 * Copyright (C) 2019, Intel Corporation
  6 * Author: Mika Westerberg <mika.westerberg@linux.intel.com>
  7 */
  8
  9#include <linux/delay.h>
 10#include <linux/suspend.h>
 11
 12#include "nhi.h"
 13#include "nhi_regs.h"
 14#include "tb.h"
 15
 16/* Ice Lake specific NHI operations */
 17
 18#define ICL_LC_MAILBOX_TIMEOUT	500 /* ms */
 19
 20static int check_for_device(struct device *dev, void *data)
 21{
 22	return tb_is_switch(dev);
 23}
 24
 25static bool icl_nhi_is_device_connected(struct tb_nhi *nhi)
 26{
 27	struct tb *tb = pci_get_drvdata(nhi->pdev);
 28	int ret;
 29
 30	ret = device_for_each_child(&tb->root_switch->dev, NULL,
 31				    check_for_device);
 32	return ret > 0;
 33}
 34
 35static int icl_nhi_force_power(struct tb_nhi *nhi, bool power)
 36{
 37	u32 vs_cap;
 38
 39	/*
 40	 * The Thunderbolt host controller is present always in Ice Lake
 41	 * but the firmware may not be loaded and running (depending
 42	 * whether there is device connected and so on). Each time the
 43	 * controller is used we need to "Force Power" it first and wait
 44	 * for the firmware to indicate it is up and running. This "Force
 45	 * Power" is really not about actually powering on/off the
 46	 * controller so it is accessible even if "Force Power" is off.
 47	 *
 48	 * The actual power management happens inside shared ACPI power
 49	 * resources using standard ACPI methods.
 50	 */
 51	pci_read_config_dword(nhi->pdev, VS_CAP_22, &vs_cap);
 52	if (power) {
 53		vs_cap &= ~VS_CAP_22_DMA_DELAY_MASK;
 54		vs_cap |= 0x22 << VS_CAP_22_DMA_DELAY_SHIFT;
 55		vs_cap |= VS_CAP_22_FORCE_POWER;
 56	} else {
 57		vs_cap &= ~VS_CAP_22_FORCE_POWER;
 58	}
 59	pci_write_config_dword(nhi->pdev, VS_CAP_22, vs_cap);
 60
 61	if (power) {
 62		unsigned int retries = 350;
 63		u32 val;
 64
 65		/* Wait until the firmware tells it is up and running */
 66		do {
 67			pci_read_config_dword(nhi->pdev, VS_CAP_9, &val);
 68			if (val & VS_CAP_9_FW_READY)
 69				return 0;
 70			usleep_range(3000, 3100);
 71		} while (--retries);
 72
 73		return -ETIMEDOUT;
 74	}
 75
 76	return 0;
 77}
 78
 79static void icl_nhi_lc_mailbox_cmd(struct tb_nhi *nhi, enum icl_lc_mailbox_cmd cmd)
 80{
 81	u32 data;
 82
 83	data = (cmd << VS_CAP_19_CMD_SHIFT) & VS_CAP_19_CMD_MASK;
 84	pci_write_config_dword(nhi->pdev, VS_CAP_19, data | VS_CAP_19_VALID);
 85}
 86
 87static int icl_nhi_lc_mailbox_cmd_complete(struct tb_nhi *nhi, int timeout)
 88{
 89	unsigned long end;
 90	u32 data;
 91
 92	if (!timeout)
 93		goto clear;
 94
 95	end = jiffies + msecs_to_jiffies(timeout);
 96	do {
 97		pci_read_config_dword(nhi->pdev, VS_CAP_18, &data);
 98		if (data & VS_CAP_18_DONE)
 99			goto clear;
100		usleep_range(1000, 1100);
101	} while (time_before(jiffies, end));
102
103	return -ETIMEDOUT;
104
105clear:
106	/* Clear the valid bit */
107	pci_write_config_dword(nhi->pdev, VS_CAP_19, 0);
108	return 0;
109}
110
111static void icl_nhi_set_ltr(struct tb_nhi *nhi)
112{
113	u32 max_ltr, ltr;
114
115	pci_read_config_dword(nhi->pdev, VS_CAP_16, &max_ltr);
116	max_ltr &= 0xffff;
117	/* Program the same value for both snoop and no-snoop */
118	ltr = max_ltr << 16 | max_ltr;
119	pci_write_config_dword(nhi->pdev, VS_CAP_15, ltr);
120}
121
122static int icl_nhi_suspend(struct tb_nhi *nhi)
123{
124	struct tb *tb = pci_get_drvdata(nhi->pdev);
125	int ret;
126
127	if (icl_nhi_is_device_connected(nhi))
128		return 0;
129
130	if (tb_switch_is_icm(tb->root_switch)) {
131		/*
132		 * If there is no device connected we need to perform
133		 * both: a handshake through LC mailbox and force power
134		 * down before entering D3.
135		 */
136		icl_nhi_lc_mailbox_cmd(nhi, ICL_LC_PREPARE_FOR_RESET);
137		ret = icl_nhi_lc_mailbox_cmd_complete(nhi, ICL_LC_MAILBOX_TIMEOUT);
138		if (ret)
139			return ret;
140	}
141
142	return icl_nhi_force_power(nhi, false);
143}
144
145static int icl_nhi_suspend_noirq(struct tb_nhi *nhi, bool wakeup)
146{
147	struct tb *tb = pci_get_drvdata(nhi->pdev);
148	enum icl_lc_mailbox_cmd cmd;
149
150	if (!pm_suspend_via_firmware())
151		return icl_nhi_suspend(nhi);
152
153	if (!tb_switch_is_icm(tb->root_switch))
154		return 0;
155
156	cmd = wakeup ? ICL_LC_GO2SX : ICL_LC_GO2SX_NO_WAKE;
157	icl_nhi_lc_mailbox_cmd(nhi, cmd);
158	return icl_nhi_lc_mailbox_cmd_complete(nhi, ICL_LC_MAILBOX_TIMEOUT);
159}
160
161static int icl_nhi_resume(struct tb_nhi *nhi)
162{
163	int ret;
164
165	ret = icl_nhi_force_power(nhi, true);
166	if (ret)
167		return ret;
168
169	icl_nhi_set_ltr(nhi);
170	return 0;
171}
172
173static void icl_nhi_shutdown(struct tb_nhi *nhi)
174{
175	icl_nhi_force_power(nhi, false);
176}
177
178const struct tb_nhi_ops icl_nhi_ops = {
179	.init = icl_nhi_resume,
180	.suspend_noirq = icl_nhi_suspend_noirq,
181	.resume_noirq = icl_nhi_resume,
182	.runtime_suspend = icl_nhi_suspend,
183	.runtime_resume = icl_nhi_resume,
184	.shutdown = icl_nhi_shutdown,
185};