diff options
| author | Gustavo A. R. Silva <gustavo@embeddedor.com> | 2018-05-18 21:13:42 -0400 |
|---|---|---|
| committer | Greg Kroah-Hartman <gregkh@linuxfoundation.org> | 2018-05-24 12:14:28 -0400 |
| commit | a0d6ec88090d7b1b008429c44532a388e29bb1bd (patch) | |
| tree | c89de91b9859f2767d7be6ffd6aa89e6009d0c80 /drivers/usb/usbip | |
| parent | 4a014a7339f441b0851ce012f469c0fadac61c81 (diff) | |
usbip: vhci_sysfs: fix potential Spectre v1
pdev_nr and rhport can be controlled by user-space, hence leading to
a potential exploitation of the Spectre variant 1 vulnerability.
This issue was detected with the help of Smatch:
drivers/usb/usbip/vhci_sysfs.c:238 detach_store() warn: potential spectre issue 'vhcis'
drivers/usb/usbip/vhci_sysfs.c:328 attach_store() warn: potential spectre issue 'vhcis'
drivers/usb/usbip/vhci_sysfs.c:338 attach_store() warn: potential spectre issue 'vhci->vhci_hcd_ss->vdev'
drivers/usb/usbip/vhci_sysfs.c:340 attach_store() warn: potential spectre issue 'vhci->vhci_hcd_hs->vdev'
Fix this by sanitizing pdev_nr and rhport before using them to index
vhcis and vhci->vhci_hcd_ss->vdev respectively.
Notice that given that speculation windows are large, the policy is
to kill the speculation on the first load and not worry if it can be
completed with a dependent load/store [1].
[1] https://marc.info/?l=linux-kernel&m=152449131114778&w=2
Cc: stable@vger.kernel.org
Signed-off-by: Gustavo A. R. Silva <gustavo@embeddedor.com>
Acked-by: Shuah Khan (Samsung OSG) <shuah@kernel.org>
Signed-off-by: Greg Kroah-Hartman <gregkh@linuxfoundation.org>
Diffstat (limited to 'drivers/usb/usbip')
| -rw-r--r-- | drivers/usb/usbip/vhci_sysfs.c | 24 |
1 files changed, 16 insertions, 8 deletions
diff --git a/drivers/usb/usbip/vhci_sysfs.c b/drivers/usb/usbip/vhci_sysfs.c index 48808388ec33..be37aec250c2 100644 --- a/drivers/usb/usbip/vhci_sysfs.c +++ b/drivers/usb/usbip/vhci_sysfs.c | |||
| @@ -10,6 +10,9 @@ | |||
| 10 | #include <linux/platform_device.h> | 10 | #include <linux/platform_device.h> |
| 11 | #include <linux/slab.h> | 11 | #include <linux/slab.h> |
| 12 | 12 | ||
| 13 | /* Hardening for Spectre-v1 */ | ||
| 14 | #include <linux/nospec.h> | ||
| 15 | |||
| 13 | #include "usbip_common.h" | 16 | #include "usbip_common.h" |
| 14 | #include "vhci.h" | 17 | #include "vhci.h" |
| 15 | 18 | ||
| @@ -205,16 +208,20 @@ static int vhci_port_disconnect(struct vhci_hcd *vhci_hcd, __u32 rhport) | |||
| 205 | return 0; | 208 | return 0; |
| 206 | } | 209 | } |
| 207 | 210 | ||
| 208 | static int valid_port(__u32 pdev_nr, __u32 rhport) | 211 | static int valid_port(__u32 *pdev_nr, __u32 *rhport) |
| 209 | { | 212 | { |
| 210 | if (pdev_nr >= vhci_num_controllers) { | 213 | if (*pdev_nr >= vhci_num_controllers) { |
| 211 | pr_err("pdev %u\n", pdev_nr); | 214 | pr_err("pdev %u\n", *pdev_nr); |
| 212 | return 0; | 215 | return 0; |
| 213 | } | 216 | } |
| 214 | if (rhport >= VHCI_HC_PORTS) { | 217 | *pdev_nr = array_index_nospec(*pdev_nr, vhci_num_controllers); |
| 215 | pr_err("rhport %u\n", rhport); | 218 | |
| 219 | if (*rhport >= VHCI_HC_PORTS) { | ||
| 220 | pr_err("rhport %u\n", *rhport); | ||
| 216 | return 0; | 221 | return 0; |
| 217 | } | 222 | } |
| 223 | *rhport = array_index_nospec(*rhport, VHCI_HC_PORTS); | ||
| 224 | |||
| 218 | return 1; | 225 | return 1; |
| 219 | } | 226 | } |
| 220 | 227 | ||
| @@ -232,7 +239,7 @@ static ssize_t detach_store(struct device *dev, struct device_attribute *attr, | |||
| 232 | pdev_nr = port_to_pdev_nr(port); | 239 | pdev_nr = port_to_pdev_nr(port); |
| 233 | rhport = port_to_rhport(port); | 240 | rhport = port_to_rhport(port); |
| 234 | 241 | ||
| 235 | if (!valid_port(pdev_nr, rhport)) | 242 | if (!valid_port(&pdev_nr, &rhport)) |
| 236 | return -EINVAL; | 243 | return -EINVAL; |
| 237 | 244 | ||
| 238 | hcd = platform_get_drvdata(vhcis[pdev_nr].pdev); | 245 | hcd = platform_get_drvdata(vhcis[pdev_nr].pdev); |
| @@ -258,7 +265,8 @@ static ssize_t detach_store(struct device *dev, struct device_attribute *attr, | |||
| 258 | } | 265 | } |
| 259 | static DEVICE_ATTR_WO(detach); | 266 | static DEVICE_ATTR_WO(detach); |
| 260 | 267 | ||
| 261 | static int valid_args(__u32 pdev_nr, __u32 rhport, enum usb_device_speed speed) | 268 | static int valid_args(__u32 *pdev_nr, __u32 *rhport, |
| 269 | enum usb_device_speed speed) | ||
| 262 | { | 270 | { |
| 263 | if (!valid_port(pdev_nr, rhport)) { | 271 | if (!valid_port(pdev_nr, rhport)) { |
| 264 | return 0; | 272 | return 0; |
| @@ -322,7 +330,7 @@ static ssize_t attach_store(struct device *dev, struct device_attribute *attr, | |||
| 322 | sockfd, devid, speed); | 330 | sockfd, devid, speed); |
| 323 | 331 | ||
| 324 | /* check received parameters */ | 332 | /* check received parameters */ |
| 325 | if (!valid_args(pdev_nr, rhport, speed)) | 333 | if (!valid_args(&pdev_nr, &rhport, speed)) |
| 326 | return -EINVAL; | 334 | return -EINVAL; |
| 327 | 335 | ||
| 328 | hcd = platform_get_drvdata(vhcis[pdev_nr].pdev); | 336 | hcd = platform_get_drvdata(vhcis[pdev_nr].pdev); |
