Как сбросить USB-устройство из командной строки?

142

Возможно ли сбросить соединение USB-устройства без физического разъединения / подключения с ПК?

В частности, мое устройство представляет собой цифровую камеру. Я использую gphoto2 , но в последнее время я получаю «ошибки чтения устройства», поэтому я хотел бы попытаться выполнить программный сброс соединения.

Из того, что я могу сказать, для камеры не загружаются модули ядра. Единственный, который выглядит связанным, это usbhid .

    
задан cmcginty 01.08.2010 в 19:46
источник

16 ответов

106

Сохраните следующее как usbreset.c

/* usbreset -- send a USB port reset to a USB device */

#include <stdio.h>
#include <unistd.h>
#include <fcntl.h>
#include <errno.h>
#include <sys/ioctl.h>

#include <linux/usbdevice_fs.h>


int main(int argc, char **argv)
{
    const char *filename;
    int fd;
    int rc;

    if (argc != 2) {
        fprintf(stderr, "Usage: usbreset device-filename\n");
        return 1;
    }
    filename = argv[1];

    fd = open(filename, O_WRONLY);
    if (fd < 0) {
        perror("Error opening output file");
        return 1;
    }

    printf("Resetting USB device %s\n", filename);
    rc = ioctl(fd, USBDEVFS_RESET, 0);
    if (rc < 0) {
        perror("Error in ioctl");
        return 1;
    }
    printf("Reset successful\n");

    close(fd);
    return 0;
}

Запустите следующие команды в терминале:

  1. Скомпилируйте программу:

    $ cc usbreset.c -o usbreset
    
  2. Получите идентификатор шины и устройства USB-устройства, которое вы хотите сбросить:

    $ lsusb  
    Bus 002 Device 003: ID 0fe9:9010 DVICO  
    
  3. Сделайте нашу скомпилированную программу исполняемой:

    $ chmod +x usbreset
    
  4. Выполнить программу с привилегиями sudo; сделайте необходимую замену для <Bus> и <Device> ids, найденных при запуске команды lsusb :

    $ sudo ./usbreset /dev/bus/usb/002/003  
    

Источник вышеуказанной программы: Ссылка

    
ответ дан Li Lo 02.08.2010 в 02:27
источник
48

Я раньше не сталкивался с вашими конкретными обстоятельствами, поэтому я не уверен, что это будет сделано достаточно, но самый простой способ, с помощью которого я нашел сброс USB-устройства, - это команда: (Никаких внешних приложений не требуется )

sudo sh -c "echo 0 > /sys/bus/usb/devices/1-4.6/authorized"
sudo sh -c "echo 1 > /sys/bus/usb/devices/1-4.6/authorized"

Это фактический, который я использую для сброса моего Kinect, поскольку у libfreenect, похоже, нет API для повторного сна. Он находится в моем блоке Gentoo, но ядро ​​должно быть достаточно новым, чтобы использовать ту же структуру пути для sysfs.

Очевидно, что это не будет 1-4.6 , но вы можете извлечь этот путь устройства из своего журнала ядра ( dmesg ), или вы можете использовать что-то вроде lsusb , чтобы получить идентификаторы поставщика и продукта, а затем быстро использовать команду, подобную этой, чтобы указать, как пути относятся к различным парам идентификаторов поставщиков / продуктов:

for X in /sys/bus/usb/devices/*; do 
    echo "$X"
    cat "$X/idVendor" 2>/dev/null 
    cat "$X/idProduct" 2>/dev/null
    echo
done
    
ответ дан ssokolow 13.09.2011 в 06:56
39

Это приведет к сбросу всех подключенных портов USB1 / 2/3 [1]:

for i in /sys/bus/pci/drivers/[uoex]hci_hcd/*:*; do
  [ -e "$i" ] || continue
  echo "${i##*/}" > "${i%/*}/unbind"
  echo "${i##*/}" > "${i%/*}/bind"
done

Я считаю, что это решит вашу проблему. Если вы не хотите сбросить все конечные точки USB, вы можете использовать соответствующий идентификатор устройства из /sys/bus/pci/drivers/ehci_hcd

Примечания: [1]: драйверы драйверов *hci_hcd обычно управляют портами USB. ohci_hcd и uhci_hcd для портов USB1.1, ehci_hcd для портов USB2, а xhci_hcd - для портов USB3. (см. Ссылка )

    
ответ дан Tamás Tapsonyi 04.05.2013 в 11:02
9

Мне нужно было автоматизировать это в сценарии python, поэтому я приложил очень полезный ответ LiLo к следующему:

#!/usr/bin/env python
import os
import sys
from subprocess import Popen, PIPE
import fcntl
driver = sys.argv[-1]
print "resetting driver:", driver
USBDEVFS_RESET= 21780

try:
    lsusb_out = Popen("lsusb | grep -i %s"%driver, shell=True, bufsize=64, stdin=PIPE, stdout=PIPE, close_fds=True).stdout.read().strip().split()
    bus = lsusb_out[1]
    device = lsusb_out[3][:-1]
    f = open("/dev/bus/usb/%s/%s"%(bus, device), 'w', os.O_WRONLY)
    fcntl.ioctl(f, USBDEVFS_RESET, 0)
except Exception, msg:
    print "failed to reset device:", msg

В моем случае это был драйвер cp210x (который я мог бы сказать от lsmod | grep usbserial ), поэтому вы можете сохранить приведенный выше фрагмент как reset_usb.py, а затем выполните следующее:

sudo python reset_usb.py cp210x

Это также может быть полезно, если у вас еще нет настройки компилятора c в вашей системе, но у вас есть python.

    
ответ дан Peter 02.03.2015 в 20:38
4

Я использую кувалду, перезаряжая модули. Это мой скрипт usb_reset.sh:

#!/bin/bash

# USB drivers
rmmod xhci_pci
rmmod ehci_pci

# uncomment if you have firewire
#rmmod ohci_pci

modprobe xhci_pci
modprobe ehci_pci

# uncomment if you have firewire
#modprobe ohci_pci

И это мой служебный файл systemd /usr/lib/systemd/system/usbreset.service, который запускает usb_reset.sh после запуска моего диспетчера diplay:

[Unit]
Description=usbreset Service
After=gdm.service
Wants=gdm.service

[Service]
Type=oneshot
ExecStart=/path/to/usb_reset.sh
    
ответ дан Ulrich-Lorenz Schlüter 09.01.2016 в 10:18
4

Поскольку особый случай вопроса - проблема связи gphoto2 с камерой на USB, в gphoto2 есть опция для сброса его USB-соединения:

gphoto2 --reset

Возможно, этот вариант не существовал в 2010 году, когда задавался вопрос.

    
ответ дан mviereck 31.08.2016 в 13:19
3

Самый быстрый способ сброса - сброс самого USB-контроллера. Это приведет к тому, что udev отменит регистрацию устройства при отключении, и регистрация вернется после его включения.

echo -n "0000:00:1a.0" | tee /sys/bus/pci/drivers/ehci_hcd/unbind
echo -n "0000:00:1d.0" | tee /sys/bus/pci/drivers/ehci_hcd/unbind
echo -n "0000:00:1a.0" | tee /sys/bus/pci/drivers/ehci_hcd/bind
echo -n "0000:00:1d.0" | tee /sys/bus/pci/drivers/ehci_hcd/bind

Это должно работать для большинства компьютеров. Однако, если вы используете какое-то специальное оборудование, вы можете просто перебирать имена устройств. С помощью этого метода вам не нужно определять имя устройства lsusb. Вы также можете включить в автоматический сценарий.

    
ответ дан chandank 24.11.2014 в 19:34
2

Я создал скрипт python, который сбросит конкретное USB-устройство на основе номера устройства. Вы можете узнать номер устройства из команды lsusb.

, например:

$ lsusb

Bus 002 Device 004: ID 046d:c312 Logitech, Inc. DeLuxe 250 Keyboard

В этой строке 004 указан номер устройства

import os
import argparse
import subprocess

path='/sys/bus/usb/devices/'

def runbash(cmd):
    p = subprocess.Popen(cmd, shell=True, stdout=subprocess.PIPE)
    out = p.stdout.read().strip()
    return out

def reset_device(dev_num):
    sub_dirs = []
    for root, dirs, files in os.walk(path):
            for name in dirs:
                    sub_dirs.append(os.path.join(root, name))

    dev_found = 0
    for sub_dir in sub_dirs:
            if True == os.path.isfile(sub_dir+'/devnum'):
                    fd = open(sub_dir+'/devnum','r')
                    line = fd.readline()
                    if int(dev_num) == int(line):
                            print ('Your device is at: '+sub_dir)
                            dev_found = 1
                            break

                    fd.close()

    if dev_found == 1:
            reset_file = sub_dir+'/authorized'
            runbash('echo 0 > '+reset_file) 
            runbash('echo 1 > '+reset_file) 
            print ('Device reset successful')

    else:
            print ("No such device")

def main():
    parser = argparse.ArgumentParser()
    parser.add_argument('-d', '--devnum', dest='devnum')
    args = parser.parse_args()

    if args.devnum is None:
            print('Usage:usb_reset.py -d <device_number> \nThe device    number can be obtained from lsusb command result')
            return

    reset_device(args.devnum)

if __name__=='__main__':
    main()
    
ответ дан Raghu 07.09.2016 в 11:42
2

Вот сценарий, который только сбросит соответствующий идентификатор продукта / поставщика.

#!/bin/bash

set -euo pipefail
IFS=$'\n\t'

VENDOR="045e"
PRODUCT="0719"

for DIR in $(find /sys/bus/usb/devices/ -maxdepth 1 -type l); do
  if [[ -f $DIR/idVendor && -f $DIR/idProduct &&
        $(cat $DIR/idVendor) == $VENDOR && $(cat $DIR/idProduct) == $PRODUCT ]]; then
    echo 0 > $DIR/authorized
    sleep 0.5
    echo 1 > $DIR/authorized
  fi
done
    
ответ дан cmcginty 30.04.2017 в 03:50
2

Я создал скрипт Python, который упрощает весь процесс на основе ответов здесь.

Сохраните сценарий ниже как reset_usb.py или клонируйте это репо: Ссылка .

Использование:

python reset_usb.py help: Показать эту справку

Список sudo python reset_usb.py: список всех USB-устройств

sudo python reset_usb.py path / dev / bus / usb / XXX / YYY: Сброс устройства USB с использованием пути / dev / bus / usb / XXX / YYY

sudo python reset_usb.py search "search terms": поиск устройства USB с использованием условий поиска в строке поиска, возвращаемой списком и устройством для подтверждения соответствия

sudo python reset_usb.py listpci: список всех USB-устройств PCI

sudo python reset_usb.py pathpci /sys/bus/pci/drivers/.../XXXX:XX:XX.X: Сброс устройства PCI USB с использованием пути /sys/bus/pci/drivers/.../XXXX : XX: XX.X

sudo python reset_usb.py searchpci «условия поиска»: поиск устройства PCI USB с использованием условий поиска в строке поиска, возвращаемых списком listpci и сбросом соответствия

#!/usr/bin/env python
import os
import sys
from subprocess import Popen, PIPE
import fcntl

instructions = '''
Usage: python reset_usb.py help : Show this help
       sudo python reset_usb.py list : List all USB devices
       sudo python reset_usb.py path /dev/bus/usb/XXX/YYY : Reset USB device using path /dev/bus/usb/XXX/YYY
       sudo python reset_usb.py search "search terms" : Search for USB device using the search terms within the search string returned by list and reset matching device
       sudo python reset_usb.py listpci : List all PCI USB devices
       sudo python reset_usb.py pathpci /sys/bus/pci/drivers/.../XXXX:XX:XX.X : Reset PCI USB device using path
       sudo python reset_usb.py searchpci "search terms" : Search for PCI USB device using the search terms within the search string returned by listpci and reset matching device       
       '''


if len(sys.argv) < 2:
    print(instructions)
    sys.exit(0)

option = sys.argv[1].lower()
if 'help' in option:
    print(instructions)
    sys.exit(0)


def create_pci_list():
    pci_usb_list = list()
    try:
        lspci_out = Popen('lspci -Dvmm', shell=True, bufsize=64, stdin=PIPE, stdout=PIPE, close_fds=True).stdout.read().strip().decode('utf-8')
        pci_devices = lspci_out.split('%s%s' % (os.linesep, os.linesep))
        for pci_device in pci_devices:
            device_dict = dict()
            categories = pci_device.split(os.linesep)
            for category in categories:
                key, value = category.split('\t')
                device_dict[key[:-1]] = value.strip()
            if 'USB' not in device_dict['Class']:
                continue
            for root, dirs, files in os.walk('/sys/bus/pci/drivers/'):
                slot = device_dict['Slot']
                if slot in dirs:
                    device_dict['path'] = os.path.join(root, slot)
                    break
            pci_usb_list.append(device_dict)
    except Exception as ex:
        print('Failed to list pci devices! Error: %s' % ex)
        sys.exit(-1)
    return pci_usb_list


def create_usb_list():
    device_list = list()
    try:
        lsusb_out = Popen('lsusb -v', shell=True, bufsize=64, stdin=PIPE, stdout=PIPE, close_fds=True).stdout.read().strip().decode('utf-8')
        usb_devices = lsusb_out.split('%s%s' % (os.linesep, os.linesep))
        for device_categories in usb_devices:
            if not device_categories:
                continue
            categories = device_categories.split(os.linesep)
            device_stuff = categories[0].strip().split()
            bus = device_stuff[1]
            device = device_stuff[3][:-1]
            device_dict = {'bus': bus, 'device': device}
            device_info = ' '.join(device_stuff[6:])
            device_dict['description'] = device_info
            for category in categories:
                if not category:
                    continue
                categoryinfo = category.strip().split()
                if categoryinfo[0] == 'iManufacturer':
                    manufacturer_info = ' '.join(categoryinfo[2:])
                    device_dict['manufacturer'] = manufacturer_info
                if categoryinfo[0] == 'iProduct':
                    device_info = ' '.join(categoryinfo[2:])
                    device_dict['device'] = device_info
            path = '/dev/bus/usb/%s/%s' % (bus, device)
            device_dict['path'] = path

            device_list.append(device_dict)
    except Exception as ex:
        print('Failed to list usb devices! Error: %s' % ex)
        sys.exit(-1)
    return device_list


if 'listpci' in option:
    pci_usb_list = create_pci_list()
    for device in pci_usb_list:
        print('path=%s' % device['path'])
        print('    manufacturer=%s' % device['SVendor'])
        print('    device=%s' % device['SDevice'])
        print('    search string=%s %s' % (device['SVendor'], device['SDevice']))
    sys.exit(0)

if 'list' in option:
    usb_list = create_usb_list()
    for device in usb_list:
        print('path=%s' % device['path'])
        print('    description=%s' % device['description'])
        print('    manufacturer=%s' % device['manufacturer'])
        print('    device=%s' % device['device'])
        print('    search string=%s %s %s' % (device['description'], device['manufacturer'], device['device']))
    sys.exit(0)

if len(sys.argv) < 3:
    print(instructions)
    sys.exit(0)

option2 = sys.argv[2]

print('Resetting device: %s' % option2)


# echo -n "0000:39:00.0" | tee /sys/bus/pci/drivers/xhci_hcd/unbind;echo -n "0000:39:00.0" | tee /sys/bus/pci/drivers/xhci_hcd/bind
def reset_pci_usb_device(dev_path):
    folder, slot = os.path.split(dev_path)
    try:
        fp = open(os.path.join(folder, 'unbind'), 'wt')
        fp.write(slot)
        fp.close()
        fp = open(os.path.join(folder, 'bind'), 'wt')
        fp.write(slot)
        fp.close()
        print('Successfully reset %s' % dev_path)
        sys.exit(0)
    except Exception as ex:
        print('Failed to reset device! Error: %s' % ex)
        sys.exit(-1)


if 'pathpci' in option:
    reset_pci_usb_device(option2)


if 'searchpci' in option:
    pci_usb_list = create_pci_list()
    for device in pci_usb_list:
        text = '%s %s' % (device['SVendor'], device['SDevice'])
        if option2 in text:
            reset_pci_usb_device(device['path'])
    print('Failed to find device!')
    sys.exit(-1)


def reset_usb_device(dev_path):
    USBDEVFS_RESET = 21780
    try:
        f = open(dev_path, 'w', os.O_WRONLY)
        fcntl.ioctl(f, USBDEVFS_RESET, 0)
        print('Successfully reset %s' % dev_path)
        sys.exit(0)
    except Exception as ex:
        print('Failed to reset device! Error: %s' % ex)
        sys.exit(-1)


if 'path' in option:
    reset_usb_device(option2)


if 'search' in option:
    usb_list = create_usb_list()
    for device in usb_list:
        text = '%s %s %s' % (device['description'], device['manufacturer'], device['device'])
        if option2 in text:
            reset_usb_device(device['path'])
    print('Failed to find device!')
    sys.exit(-1)
    
ответ дан mcarans 21.12.2017 в 10:15
1

Кто-нибудь заказывал кувалду? Здесь собрано множество других ответов.

#!/bin/bash

# Root required
if (( UID )); then
        exec sudo "$0" "$@"
fi

cd /sys/bus/pci/drivers

function reinit {(
        local d="$1"
        test -e "$d" || return

        rmmod "$d"

        cd "$d"

        for i in $(ls | grep :); do
                echo "$i" > unbind
        done

        sleep 1

        for i in $(ls | grep :); do
                echo "$i" > bind
        done

        modprobe "$d"

)}

for d in ?hci_???; do
        echo " - $d"
        reinit "$d"
done
    
ответ дан Mark K Cowan 28.06.2016 в 14:08
1

Иногда я хочу выполнить эту операцию на определенном устройстве, как это определено идентификатором VID (идентификатор продавца) и идентификатором продукта (идентификатор продукта). Это сценарий, который я нашел полезным для этой цели, который использует отличную библиотеку libusb.

Первый запуск:

sudo apt-get install libusb-dev

Затем этот сброс файла c ++DeviceConnection должен выполнить эту задачу, сбросив соединение устройства, идентифицированное с помощью vid и pid.

#include <libusb-1.0/libusb.h>

int resetDeviceConnection(UINT_16 vid, UINT_16 pid){
    /*Open libusb*/
    int resetStatus = 0;
    libusb_context * context;
    libusb_init(&context);

    libusb_device_handle * dev_handle = libusb_open_device_with_vid_pid(context,vid,pid);
    if (dev_handle == NULL){
      printf("usb resetting unsuccessful! No matching device found, or error encountered!\n");
      resetStatus = 1;
    }
    else{
      /*reset the device, if one was found*/
      resetStatus = libusb_reset_device(dev_handle);
    }
    /*exit libusb*/
    libusb_exit(context);
    return resetStatus;
}

(украденный из моего личного каталога TIL: Ссылка )

    
ответ дан Marviel 29.12.2016 в 13:53
0

Возможно, это работает и для камеры:

После возобновления голодающего USB 3.0 HDD на 3.4.42 (kernel.org) Linux на моей стороне. dmesg сказал, что это было тайм-аут команд после 360-х (извините, я не могу скопировать syslog здесь, не подключенные сети), и диск полностью зависел. Процессы, обращающиеся к устройству, были заблокированы в ядре, что было неудобно. NFS висели, ZFS висели, dd висели.

После этого все снова работало. dmesg указала только одну строку о найденном устройстве USB .

Я действительно не знаю, что следует делать в деталях. Но это сработало.

Следующий пример выводится из Debian Squeeze с ядром 2.6.32-5-686 , поэтому я думаю, что он работает для версии 2.6 и выше:

$ ls -al /dev/sdb
brw-rw---T 1 root floppy 8, 16 Jun  3 20:24 /dev/sdb

$ ls -al /sys/dev/block/8:16/device/rescan
--w------- 1 root root 4096 Jun  6 01:46 /sys/dev/block/8:16/device/rescan

$ echo 1 > /sys/dev/block/8:16/device/rescan

Если это не сработает, возможно, кто-то еще сможет выяснить, как отправить реальный сброс на устройство.

    
ответ дан Tino 06.06.2013 в 00:08
0

Попробуйте это, это программное обеспечение отключит (Eject).

Иногда не работает просто отключение устройства для некоторых устройств.

Пример:

Я хочу удалить или изгнать свой «Genius NetScroll 120».

Затем сначала проверьте свое подключенное устройство USB

$ lsusb
Bus 001 Device 001: ID 1d6b:0002 Linux Foundation 2.0 root hub
Bus 002 Device 001: ID 1d6b:0002 Linux Foundation 2.0 root hub
Bus 001 Device 002: ID 8087:0020 Intel Corp. Integrated Rate Matching Hub
Bus 002 Device 002: ID 8087:0020 Intel Corp. Integrated Rate Matching Hub
Bus 001 Device 003: ID 03f0:231d Hewlett-Packard 
Bus 001 Device 004: ID 138a:0007 Validity Sensors, Inc. VFS451 Fingerprint Reader
Bus 001 Device 005: ID 04f2:b163 Chicony Electronics Co., Ltd 
Bus 002 Device 009: ID 0458:003a KYE Systems Corp. (Mouse Systems) NetScroll+ Mini Traveler / Genius NetScroll 120  **<----This my Mouse! XDDD**

Хорошо, я нашел свою мышь, у нее есть Bus 002, Device 009, idVendor 0458 и idProduct 003a, так что это справочная информация об устройстве мыши.

Это важно, номер шины - это путь начала имени к устройству, и я проверю идентификатор продукта и поставщика, чтобы убедиться, что устройство правильно удалено.

$ ls /sys/bus/usb/drivers/usb/
1-1/    1-1.1/  1-1.3/  1-1.5/  2-1/    2-1.3/  bind    uevent  unbind  usb1/   usb2/

Обратите внимание на папки, проверьте начало с папкой номер 2, я проверю это, потому что моя Шина 002, и один за другим я проверил каждую папку, содержащую правильный idVendor и idProduct, о моей информации о мыши.

В этом случае я получаю информацию с помощью этой команды:

cat /sys/bus/usb/drivers/usb/2-1.3/idVendor
0458
cat /sys/bus/usb/drivers/usb/2-1.3/idProduct
003a

Хорошо, путь /sys/bus/usb/drivers/usb/2-1.3/ соответствует моей информационной мыши! XDDD.

Пришло время удалить устройство!

su -c "echo 1 > /sys/bus/usb/drivers/usb/2-1.3/remove"

Подключите USB-устройство снова, и он снова работает!

    
ответ дан user242078 31.01.2014 в 11:15
0

Если вы знаете имя своего устройства, этот скрипт python будет работать:

#!/usr/bin/python
"""
USB Reset

Call as "usbreset.py <device_file_path>"

With device_file_path like "/dev/bus/usb/bus_number/device_number"
"""
import fcntl, sys, os

USBDEVFS_RESET = ord('U') << (4*2) | 20

def main():
    fd = os.open(sys.argv[1], os.O_WRONLY)
    if fd < 0: sys.exit(1)
    fcntl.ioctl(fd, USBDEVFS_RESET, 0)
    os.close(fd)
    sys.exit(0)
# end main

if __name__ == '__main__':
    main()
    
ответ дан Clay 04.08.2017 в 14:35
-2

Возможно, это руководство может помочь вам:

Если вас беспокоит ошибка, которая не позволяет монтировать USB-устройства в Ubuntu Lucid Lynx, проблема связана с гибким модулем. Отключите его:

sudo modprobe -r floppy

После перезагрузки модуль, скорее всего, перезагрузится.

    
ответ дан User 01.08.2010 в 20:16