Передавать сообщения ядра другому процессу по мере их появления
У меня на ноутбуке OLED-экран, который я настроил для отображения информации о состоянии. Текущий драйвер, который я установил в Linux, может отображать сообщения, отправляя их в сценарий в качестве аргумента, разделенного пробелами.
Пример: команда /opt/asusg50oled/utils/notify.sh Hi Everybody "Hello World"
отображается на экране oled:
Hi
Everybody
Hello World
Если другое сообщение отправляется до того, как старые исчезают, и оно возвращается к информации о состоянии, оно выталкивает верхнее сообщение. Пример: менее чем через 30 секунд после предыдущего примера, /opt/asusg50oled/utils/notify.sh "Bananas have potassium"
выполняется:
Everybody
Hello World
Bananas have potassium
То, что я хочу сделать, это иметь сообщения ядра (то, что вы видите, запустив dmesg
) перенаправлен на этот скрипт. Например, когда я вставляю USB-накопитель, следующая информация будет отображаться на экране OLED по мере их регистрации:
[ 1283.200150] usb 2-4: new high speed USB device using ehci_hcd and address 4
[ 1283.353322] scsi9 : usb-storage 2-4:1.0
[ 1284.351366] scsi 9:0:0:0: Direct-Access SanDisk Cruzer 1.03 PQ: 0 ANSI: 2
[ 1284.352697] sd 9:0:0:0: Attached scsi generic sg4 type 0
[ 1284.355669] sd 9:0:0:0: [sdd] 31266816 512-byte logical blocks: (16.0 GB/14.9 GiB)
[ 1284.357032] sd 9:0:0:0: [sdd] Write Protect is off
[ 1284.357041] sd 9:0:0:0: [sdd] Mode Sense: 03 00 00 00
[ 1284.357047] sd 9:0:0:0: [sdd] Assuming drive cache: write through
[ 1284.364356] sd 9:0:0:0: [sdd] Assuming drive cache: write through
[ 1284.364371] sdd: sdd1
[ 1284.371656] sd 9:0:0:0: [sdd] Assuming drive cache: write through
[ 1284.371666] sd 9:0:0:0: [sdd] Attached SCSI removable disk
Обратите внимание, что с учетом ограниченной ширины экрана извлечение временных меток также будет полезно. Таким образом, в конце записи экран будет отображаться в течение примерно 30 секунд:
sdd: sdd1
sd 9:0:0:0: [sdd] Assuming drive cache: write through
sd 9:0:0:0: [sdd] Attached SCSI removable disk
Итак, чтобы уточнить, я хочу, чтобы сообщения ядра отправлялись в реальном времени путем выполнения /opt/asusg50oled/utils/notify.sh "$MESSAGE"
Я могу добавить фильтры, чтобы отсеять все, что я не хочу видеть позже, я просто хочу знать, как выполнять части, упомянутые выше. Как я могу это сделать?
редактировать
Как предложил Ciclamino, я добавил следующую строку в мой файл /etc/rsyslog.conf:
kern.* ^/opt/asusg50oled/utils/notify.sh
Это почти сработало, но форматирование привело к тому, что OLED просто показывал дату, имя хоста и слово "ядро", прежде чем не хватило места. Немного покопавшись, я понял следующее:
$template OLEDformat,"%msg%0
kern.* ^/opt/asusg50oled/utils/notify.sh;OLEDformat
Это почти там, но опять же, мне не хватает места из-за какой-то метки времени.. пример отображения:
[ 4477.993774] sd 11:0:0:0: [sdb] At
Я хочу избавиться от чисел в скобках, чтобы я остался с
sd 11:0:0:0: [sdb] Attached SCSI rem
Все еще не идеальный, но лучшее, что я получу с размером экрана. Было бы еще лучше, если бы он разделил его на 36 символов, чтобы я мог получить что-то вроде следующего:
sd 11:0:0:0: [sdb] Attached SCSI rem
ovable disk
РЕШИТЬ
Я принимаю ответ Цикламино, так как он привел меня туда, где я должен был быть. Вот что я в итоге сделал:
Я создал и добавил бит выполнения в скрипт /opt/asusg50oled/utils/notify-kern.sh
содержащий
#!/bin/bash
cd `dirname $0`
stringA=$1
stringB=${stringA#\[*\]}
stringC=${stringB:0:36}
stringD=${stringB:36}
./notify.sh "$stringC" "$stringD"
Затем я добавил в /etc/rsyslog.conf
## output kernel messages to OLED
$template OLEDformat,"%msg%"
kern.* ^/opt/asusg50oled/utils/notify-kern.sh;OLEDformat
Наконец, я перезапустил rsyslog с sudo service rsyslog restart
,
3 ответа
Вы можете использовать syslog для перехвата сообщений ядра и передачи их команде. Синтаксис будет немного отличаться для разных реализаций syslogd. Вот пример того, как это сделать с помощью rsyslog (в /etc/rsyslog.conf):
kern.* ^/opt/asusg50oled/utils/notify.sh
Вот быстрое и грязное решение:
tail -n 0 -f /var/log/messages | while read -r MESSAGE; do
/opt/asusg50oled/utils/notify.sh "$MESSAGE"
done
Вы можете повторно прочитать вывод /proc/kmsg
с помощью tail
или любой другой метод, который вы предпочитаете.
Однако я думаю, что для этого вам нужно быть пользователем root.