Thursday, October 10, 2013

Монтирование Nexus 7

С этой темой я корячился давно и долго, и вот после UP7 в LMDE всё заработало по-человечески.
Использовался вот этот чудесный гайд.

Для работы по протоколу MTP используется библиотека libmtp, и похоже, в последнем релизе Debian её таки починили.
Чтобы примонтировать устройство по протоколу MTP, можно пользоваться кучей способов: mtpfs, jmtpfs, gvfs-mtp, go-mtpfs.
Вот последний и интересен. Разрабатывался (и поддерживается в момент написания статьи) разработчиком Android, быстр и, в последних версиях, лишён багов с отваливанием устройства. Ну, будем думать, что из-за фикса непосредственно в libmtp.

Перво-наперво, нужно разрешить не-root пользователям доступ к файловым системам, смонтированным от лица root. Для этого открываем /etc/fuse.conf и раскомментируем опцию user_allow_other.

Дальше нужно поставить сам go-mtpfs.
Для этого понадобятся git и компилятор Go, а дальше дело занимает пару минут.
Итак:
sudo apt-get install libmtp-dev git golang # готовим build environment
mkdir /tmp/go # здесь будем собирать
export GOPATH=/tmp/go
go get github.com/hanwen/go-mtpfs # забираем проект go-mtpfs с гитхаба; при этом происходит сборка при помощи Go
sudo mv /tmp/go/bin/go-mtpfs ~/bin/go-mtpfs # готово, перемещаем бинарник туда, куда нам удобно
sudo chmod a+x ~/bin/go-mtpfs # не забываем выдать бинарнику права на запуск
Собрали. Можем монтировать:
go-mtpfs /media/mtp
и размонтировать:
fusermount -u /media/mtp
Кроме того, добавим правила udev, чтобы при подключении устройства через USB оно автоматически монтировалось. Создаём файлик /etc/udev/rules.d/99-android.rules с вот таким содержанием:
# /etc/udev/rules.d/

# Google Nexus 7
SUBSYSTEM=="usb", ATTR{idVendor}=="18d1", ATTR{idProduct}=="4e40", MODE="0666" # Bootloader
SUBSYSTEM=="usb", ATTR{idVendor}=="18d1", ATTR{idProduct}=="d001", MODE="0666" # Recovery
SUBSYSTEM=="usb", ATTR{idVendor}=="18d1", ATTR{idProduct}=="4e43", MODE="0666" # PTP media, USB debug off
SUBSYSTEM=="usb", ATTR{idVendor}=="18d1", ATTR{idProduct}=="4e44", MODE="0666" # PTP media, USB debug on
SUBSYSTEM=="usb", ATTR{idVendor}=="18d1", ATTR{idProduct}=="4e41", MODE="0666" # MTP media, USB debug off
SUBSYSTEM=="usb", ATTR{idVendor}=="18d1", ATTR{idProduct}=="4e42", MODE="0666" # MTP media, USB debug on

# Google Nexus 7 MTP mode under JB 4.1 : automatic mount & umount when plugged (USB debug on & off)
ENV{ID_MODEL}=="Nexus", ENV{ID_MODEL_ID}=="4e41", ACTION=="add", RUN+="/usr/bin/sudo -b -u dbz /home/dbz/bin/go-mtpfs -allow-other=true /media/mtp"
ENV{ID_MODEL}=="Nexus", ENV{ID_MODEL_ID}=="4e42", ACTION=="add", RUN+="/usr/bin/sudo -b -u dbz /home/dbz/bin/go-mtpfs -allow-other=true /media/mtp"
ENV{ID_MODEL}=="Nexus", ENV{ID_MODEL_ID}=="4e41", ACTION=="remove", RUN+="/bin/umount /media/mtp"
ENV{ID_MODEL}=="Nexus", ENV{ID_MODEL_ID}=="4e42", ACTION=="remove", RUN+="/bin/umount /media/mtp"

# Google Nexus 7 MTP mode under JB 4.2+ : automatic mount & umount when plugged (USB debug on & off)
ENV{ID_MODEL}=="Nexus_7", ENV{ID_MODEL_ID}=="4e41", ACTION=="add", RUN+="/usr/bin/sudo -b -u dbz /home/dbz/bin/go-mtpfs -allow-other=true /media/mtp"
ENV{ID_MODEL}=="Nexus_7", ENV{ID_MODEL_ID}=="4e42", ACTION=="add", RUN+="/usr/bin/sudo -b -u dbz /home/dbz/bin/go-mtpfs -allow-other=true /media/mtp"
ENV{ID_MODEL}=="Nexus_7", ENV{ID_MODEL_ID}=="4e41", ACTION=="remove", RUN+="/bin/umount /media/mtp"
ENV{ID_MODEL}=="Nexus_7", ENV{ID_MODEL_ID}=="4e42", ACTION=="remove", RUN+="/bin/umount /media/mtp"
Здесь используется монтирование через sudo, т.к. почему-то без него монтировать не хочет. Копаться почему так мне не хотелось. Опция -b используется для запуска монтирования в бекграунде, т.к. udev очень нетерпим к долгим операциям. Пробовал вот такую команду:
ENV{ID_MODEL}=="Nexus_7", ENV{ID_MODEL_ID}=="4e42", ACTION=="add", RUN+="/home/dbz/bin/go-mtpfs -allow-other=true /media/mtp &"
т.е. монтировать без sudo и отправлять процесс в бекграунд при помощи & - не сработало...
В правилах udev нужно прописывать абсолютные пути.
Также видим, что наряду с fusermount -u можно размонтировать обычным umount.

Т.к. после UP7 система, наконец-то, научилась монтировать устройства при помощи gvfs-mtp, при подключении устройства увидим 2 точки монтирования: одна та, что мы только что задали; другая - будет называться Nexus_7 или как-нибудь похоже, и будет представлять собой путь вида mtp://...
Всё бы ничего, но раз уж нам две шары ни к чему (всё прекрасно работает через go-mtpfs), можем удалить правила для gvfs-mtp. Кроме того, они, похоже, мешают перемонтировать устройство вручную при помощи go-mtpfs, если кабель не отключали/включали, а перемонтировать по какой-то причине было нужно.
Идём в /lib/udev/rules.d/69-libmtp.rules и комментируем следующие строки:
# Google Inc (for Asus) Nexus 7 (MTP)
#ATTR{idVendor}=="18d1", ATTR{idProduct}=="4e41", SYMLINK+="libmtp-%k", MODE="660", GROUP="audio", ENV{ID_MTP_DEVICE}="1", ENV{ID_MEDIA_PLAYER}="1"
# Google Inc (for Asus) Nexus 7 (MTP+ADB)
#ATTR{idVendor}=="18d1", ATTR{idProduct}=="4e42", SYMLINK+="libmtp-%k", MODE="660", GROUP="audio", ENV{ID_MTP_DEVICE}="1", ENV{ID_MEDIA_PLAYER}="1"
На этом всё. И чтобы наши правила вступили в силу:
sudo service udev restart
Выключаем/включаем кабель и вуаля - шара в /media/mtp. Разумеется, папку перед выполнением всех описанных операций надо создать :)
Закачка у меня происходит на скорости около 8 МБ/с.
Падений не наблюдается.

Есть ещё один нюанс. Если попытаться в файловом браузере отмонтировать шару, то получим сообщение, что мы не рут и сделать этого не можем.
Чтобы разрешить размонтирование, рекомендуют добавить правило в /etc/fstab:
DeviceFs(Nexus\0407)    /media/mtp   fuse.DeviceFs(Nexus\0407)    allow_other,rw,user,noauto    0    0
Я не смог заставить эту опцию заработать. Кроме того, архитектура MTP такова, что как только вы скопировали файл, можно вытаскивать кабель, не боясь что-либо потерять или испортить файловую систему устройства.
Зато я добавил два удобненьких алиаса в ~/.bashrc:
alias mtpmount='go-mtpfs /media/mtp'
alias mtpumount='fusermount -u /media/mtp'

Wednesday, October 2, 2013

Немного о картинках и бэкграундах


Задача №1: указать папку с картинками для слайдшоу в mate-screensaver.
Решение:
  • открываем /usr/share/applications/screensavers/personal-slideshow.desktop
  • ищем секцию Exec=/usr/lib/mate-screensaver/slideshow
  • дописываем в конец секции: --location=/path/to/your/images

Задача №2: поменять фон экрана блокировки.
Решение:
  • идём в /usr/share/backgrounds/linuxmint
  • видим символьную ссылку default_background.jpg
  • сохраняем её на всякий пожарный: cp default_background.jpg default_background.jpg.old
  • создаём новую символьную ссылку на то, что нам надо: sudo ln -s /path/to/image default_background.jpg
  • надо перелогиниться либо перезапустить mate-screensaver: killall mate-screensaver; mate-screensaver &

Автомонтирование разделов с NTFS


Вернуться к этой теме заставило недавнее обновление моего любимого LMDE до Update Pack 7 (UP7).

Итак, задача: после входа пользователя, разделы с файловой системой NTFS должны быть примонтированы.
Нашёл отличное решение вот здесь.
Заключается оно в следующем:
  • создаётся скрипт, который монтирует разделы по одному, скажем, по их UUID
  • скрипт помещается в автозагрузку
  • profit!

Продублирую это решение:
#!/usr/bin/env bash

# Redsandro 2013-03-01
# Mount (LUKS) drives and such

doMount() {
    # Devices change every now and then. UUID does not.
    DEV=`ls -l --color=never /dev/disk/by-uuid/ | grep $UUID | cut -d\>  -f2 | cut -d/ -f3`
    gvfs-mount -d /dev/$DEV
}

# LUKS-1TB
# /dev/sdb1
UUID='9dd6df52-c901-4b70-8151-27086ba7db16'
doMount

# Projects 1TB
# /dev/sdd1
UUID='6b01a30e-af6e-48b4-810a-f9824558f041'
doMount
Всё. Осталось только подставить сюда UUID своих разделов. В рамках одной системы решение долговечное, т.к. UUID разделов меняются чуть чаще чем никогда.

Ещё советуют запретить автоматически открываемые окна для каждого примонтированного девайса вот такой командой:
gsettings set org.gnome.desktop.media-handling automount-open false
Но у меня эти окна не появляются и без редактирования указанной опции.

Следующий шаг очевиден. После входа в систему мы увидим окошко с запросом пароля для монтирования разделов. Чтобы этого избежать, и монтировать без запроса пароля, рекомендуют сделать следующее:
  • открыть файл /usr/share/polkit-1/actions/org.freedesktop.udisks.policy
  • найти там следующую секцию: <action id="org.freedesktop.udisks.filesystem-mount-system-internal">
  • взглянуть ниже и найти там подсекцию <defaults>
  • для каждой из опций allow_* в этой секции поменять текущие значения на yes
В итоге должно получиться следующее:
    
      yes
      yes
      yes
    
НО.
После UP7 используется новая подсистема udisks2, и значит нужно менять другой конфиг и другие секции.
Конфиг теперь вот такой: /usr/share/polkit-1/actions/org.freedesktop.udisks2.policy
Секции теперь вот такие:
  • <action id="org.freedesktop.udisks2.filesystem-mount">
  • <action id="org.freedesktop.udisks2.filesystem-mount-system">
  • <action id="org.freedesktop.udisks2.filesystem-mount-other-seat">

Ну и напоследок - новая версия udisks преподнесла ещё один сюрприз.
Доселе разделы монтировались в /media, теперь же они стали монтироваться в /media/$USER.
К счастью, решение есть. Как видим, даже два :)
Используем самое простое и очевидное:
  • создаём файл /etc/udev/rules.d/99-udisks2.rules
  • пишем в нём: ENV{ID_FS_USAGE}=="filesystem", ENV{UDISKS_FILESYSTEM_SHARED}="1"
  • сохраняем
  • размонтируем примонтированные в корявое место разделы :)
  • sudo service udev restart
  • profit!