Loading...
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45 46 47 48 49 50 51 52 53 54 55 56 57 58 59 60 61 62 63 64 65 66 67 68 69 70 71 72 73 74 75 76 77 78 79 80 81 82 83 84 85 86 87 88 89 90 91 92 93 94 95 96 97 98 99 100 101 102 103 104 105 106 107 108 109 110 111 112 113 114 115 116 117 118 119 120 121 122 123 124 125 126 127 128 129 130 131 132 133 134 135 136 137 138 139 140 141 142 143 144 145 146 147 148 149 150 151 152 153 154 155 156 157 158 159 160 161 162 163 164 165 166 167 168 169 170 171 172 173 174 175 176 177 178 179 180 181 182 183 184 185 186 187 188 189 190 191 192 193 194 195 196 197 198 199 200 201 202 203 204 205 206 207 208 209 210 211 212 213 214 215 216 217 218 219 220 221 222 223 224 225 226 227 228 229 230 231 232 233 234 235 236 237 238 239 240 241 242 243 | // SPDX-License-Identifier: GPL-2.0 /* * Thunderbolt driver - capabilities lookup * * Copyright (c) 2014 Andreas Noever <andreas.noever@gmail.com> * Copyright (C) 2018, Intel Corporation */ #include <linux/slab.h> #include <linux/errno.h> #include "tb.h" #define CAP_OFFSET_MAX 0xff #define VSE_CAP_OFFSET_MAX 0xffff #define TMU_ACCESS_EN BIT(20) static int tb_port_enable_tmu(struct tb_port *port, bool enable) { struct tb_switch *sw = port->sw; u32 value, offset; int ret; /* * Legacy devices need to have TMU access enabled before port * space can be fully accessed. */ if (tb_switch_is_light_ridge(sw)) offset = 0x26; else if (tb_switch_is_eagle_ridge(sw)) offset = 0x2a; else return 0; ret = tb_sw_read(sw, &value, TB_CFG_SWITCH, offset, 1); if (ret) return ret; if (enable) value |= TMU_ACCESS_EN; else value &= ~TMU_ACCESS_EN; return tb_sw_write(sw, &value, TB_CFG_SWITCH, offset, 1); } static void tb_port_dummy_read(struct tb_port *port) { /* * When reading from next capability pointer location in port * config space the read data is not cleared on LR. To avoid * reading stale data on next read perform one dummy read after * port capabilities are walked. */ if (tb_switch_is_light_ridge(port->sw)) { u32 dummy; tb_port_read(port, &dummy, TB_CFG_PORT, 0, 1); } } /** * tb_port_next_cap() - Return next capability in the linked list * @port: Port to find the capability for * @offset: Previous capability offset (%0 for start) * * Returns dword offset of the next capability in port config space * capability list and returns it. Passing %0 returns the first entry in * the capability list. If no next capability is found returns %0. In case * of failure returns negative errno. */ int tb_port_next_cap(struct tb_port *port, unsigned int offset) { struct tb_cap_any header; int ret; if (!offset) return port->config.first_cap_offset; ret = tb_port_read(port, &header, TB_CFG_PORT, offset, 1); if (ret) return ret; return header.basic.next; } static int __tb_port_find_cap(struct tb_port *port, enum tb_port_cap cap) { int offset = 0; do { struct tb_cap_any header; int ret; offset = tb_port_next_cap(port, offset); if (offset < 0) return offset; ret = tb_port_read(port, &header, TB_CFG_PORT, offset, 1); if (ret) return ret; if (header.basic.cap == cap) return offset; } while (offset > 0); return -ENOENT; } /** * tb_port_find_cap() - Find port capability * @port: Port to find the capability for * @cap: Capability to look * * Returns offset to start of capability or %-ENOENT if no such * capability was found. Negative errno is returned if there was an * error. */ int tb_port_find_cap(struct tb_port *port, enum tb_port_cap cap) { int ret; ret = tb_port_enable_tmu(port, true); if (ret) return ret; ret = __tb_port_find_cap(port, cap); tb_port_dummy_read(port); tb_port_enable_tmu(port, false); return ret; } /** * tb_switch_next_cap() - Return next capability in the linked list * @sw: Switch to find the capability for * @offset: Previous capability offset (%0 for start) * * Finds dword offset of the next capability in router config space * capability list and returns it. Passing %0 returns the first entry in * the capability list. If no next capability is found returns %0. In case * of failure returns negative errno. */ int tb_switch_next_cap(struct tb_switch *sw, unsigned int offset) { struct tb_cap_any header; int ret; if (!offset) return sw->config.first_cap_offset; ret = tb_sw_read(sw, &header, TB_CFG_SWITCH, offset, 2); if (ret) return ret; switch (header.basic.cap) { case TB_SWITCH_CAP_TMU: ret = header.basic.next; break; case TB_SWITCH_CAP_VSE: if (!header.extended_short.length) ret = header.extended_long.next; else ret = header.extended_short.next; break; default: tb_sw_dbg(sw, "unknown capability %#x at %#x\n", header.basic.cap, offset); ret = -EINVAL; break; } return ret >= VSE_CAP_OFFSET_MAX ? 0 : ret; } /** * tb_switch_find_cap() - Find switch capability * @sw Switch to find the capability for * @cap: Capability to look * * Returns offset to start of capability or %-ENOENT if no such * capability was found. Negative errno is returned if there was an * error. */ int tb_switch_find_cap(struct tb_switch *sw, enum tb_switch_cap cap) { int offset = 0; do { struct tb_cap_any header; int ret; offset = tb_switch_next_cap(sw, offset); if (offset < 0) return offset; ret = tb_sw_read(sw, &header, TB_CFG_SWITCH, offset, 1); if (ret) return ret; if (header.basic.cap == cap) return offset; } while (offset); return -ENOENT; } /** * tb_switch_find_vse_cap() - Find switch vendor specific capability * @sw: Switch to find the capability for * @vsec: Vendor specific capability to look * * Functions enumerates vendor specific capabilities (VSEC) of a switch * and returns offset when capability matching @vsec is found. If no * such capability is found returns %-ENOENT. In case of error returns * negative errno. */ int tb_switch_find_vse_cap(struct tb_switch *sw, enum tb_switch_vse_cap vsec) { int offset = 0; do { struct tb_cap_any header; int ret; offset = tb_switch_next_cap(sw, offset); if (offset < 0) return offset; ret = tb_sw_read(sw, &header, TB_CFG_SWITCH, offset, 1); if (ret) return ret; if (header.extended_short.cap == TB_SWITCH_CAP_VSE && header.extended_short.vsec_id == vsec) return offset; } while (offset); return -ENOENT; } |