Eu construí um relógio GPS compatível com PPS no meu Raspberry Pi, e para fazê-lo funcionar, eu precisava usar o udev para criar links simbólicos para a porta serial do hardware e o kernel GPIO PPS em /dev/gps0
e /dev/gpspps0
. Não precisa ser 0, eu só preciso que os dois números sejam iguais para que o NTP possa captar o sinal PPS. Eu criei o seguinte arquivo de regras do udev para fazer isso:
# Symlink /dev/ttyAMA0 to /dev/gps0
KERNEL=="ttyAMA0", SUBSYSTEM=="tty", DRIVER=="", SYMLINK+="gps0"
# Symlink /dev/pps0 to /dev/gpspps0
KERNEL=="pps0", SUBSYSTEM=="pps", DRIVER=="", SYMLINK+="gpspps0"
Por si só, isso se comporta bem. Os links simbólicos são criados e o NTP adquire alegremente um bloqueio PPS no dispositivo. Os seguintes dispositivos são gerados:
pi@srv-GPiS-00:~ $ ls -al /dev/gps*
lrwxrwxrwx 1 root root 7 Aug 12 16:31 /dev/gps0 -> ttyAMA0
lrwxrwxrwx 1 root root 4 Aug 12 16:31 /dev/gpspps0 -> pps0
No entanto, se um GPS USB processado por 60-gpsd.rules
estiver conectado no momento da inicialização, o link simbólico gps0
será sobrescrito para apontar para ttyUSB0
, como mostrado aqui:
pi@srv-GPiS-00:~ $ ls -al /dev/gps*
lrwxrwxrwx 1 root root 7 Aug 12 16:45 /dev/gps0 -> ttyUSB0
lrwxrwxrwx 1 root root 4 Aug 12 16:45 /dev/gpspps0 -> pps0
(antes de perguntar por que tenho dois GPS conectados ao mesmo computador, estou tentando depurar e calibrar o segundo GPS em relação ao meu sistema conhecido)
Desconectar o dispositivo permite que o symlink retorne para ttyAMA0
. Curiosamente, quando o segundo GPS é conectado novamente, o udev tem um comportamento secundário e o coloca em ttyUSB1
e adquire o link simbólico gps1
para ele:
pi@srv-GPiS-00:~ $ ls -al /dev/gps*
lrwxrwxrwx 1 root root 7 Aug 12 16:48 /dev/gps0 -> ttyAMA0
lrwxrwxrwx 1 root root 7 Aug 12 16:48 /dev/gps1 -> ttyUSB1
lrwxrwxrwx 1 root root 4 Aug 12 16:45 /dev/gpspps0 -> pps0
Eu gostaria que o sistema tivesse esse comportamento secundário para que o NTP não pudesse ser interferido se eu reinicializasse o sistema com o segundo GPS conectado.
Eu tentei alterar o nome do arquivo de minhas regras do udev para 50 de 90 para nenhum efeito, e também tentei especificar gps%n
/ gpspps%n
no arquivo de regras em vez de gps0
/ gpspps0
(como em 60-gpsd.rules
), o que resulta na eliminação completa do comportamento secundário. Eu também sei do operador :=
para o udev que define um parâmetro finalmente, mas eu não sei como isso interage com dois dispositivos disputando o mesmo symlink, e eu também não quero estrobrar quaisquer outras regras de links simbólicos de outros arquivos.
Concedido, sim, eu poderia codificar meu dispositivo para ser gps3
/ gpspps3
, que, acredito, é o número máximo de dispositivos suportado pelo NTP. No entanto, quero descobrir por que o udev está fazendo o que está fazendo. Também gostaria de evitar a remoção / desativação do arquivo 60-gpsd.rules, já que isso é fita adesiva e o comportamento não seria estável em muitas atualizações de SO.
Alguma opinião?
Informação adicional:
pi@srv-GPiS-00:~ $ udevadm info --name=/dev/ttyAMA0 --attribute-walk
...
looking at device '/devices/platform/soc/3f201000.serial/tty/ttyAMA0':
KERNEL=="ttyAMA0"
SUBSYSTEM=="tty"
DRIVER==""
looking at parent device '/devices/platform/soc/3f201000.serial':
KERNELS=="3f201000.serial"
SUBSYSTEMS=="amba"
DRIVERS=="uart-pl011"
ATTRS{id}=="00241011"
ATTRS{irq0}=="87"
ATTRS{driver_override}==""
looking at parent device '/devices/platform/soc':
KERNELS=="soc"
SUBSYSTEMS=="platform"
DRIVERS==""
ATTRS{driver_override}=="(null)"
looking at parent device '/devices/platform':
KERNELS=="platform"
SUBSYSTEMS==""
DRIVERS==""
pi@srv-GPiS-00:~ $ udevadm info --name=/dev/ttyUSB1 --attribute-walk
...
looking at device '/devices/platform/soc/3f980000.usb/usb1/1-1/1-1.4/1-1.4:1.0/ttyUSB1/tty/ttyUSB1':
KERNEL=="ttyUSB1"
SUBSYSTEM=="tty"
DRIVER==""
looking at parent device '/devices/platform/soc/3f980000.usb/usb1/1-1/1-1.4/1-1.4:1.0/ttyUSB1':
KERNELS=="ttyUSB1"
SUBSYSTEMS=="usb-serial"
DRIVERS=="garmin_gps"
ATTRS{port_number}=="0"
looking at parent device '/devices/platform/soc/3f980000.usb/usb1/1-1/1-1.4/1-1.4:1.0':
KERNELS=="1-1.4:1.0"
SUBSYSTEMS=="usb"
DRIVERS=="garmin_gps"
ATTRS{bInterfaceProtocol}=="ff"
ATTRS{bInterfaceNumber}=="00"
ATTRS{bInterfaceSubClass}=="ff"
ATTRS{bInterfaceClass}=="ff"
ATTRS{bAlternateSetting}==" 0"
ATTRS{authorized}=="1"
ATTRS{bNumEndpoints}=="03"
ATTRS{supports_autosuspend}=="1"
looking at parent device '/devices/platform/soc/3f980000.usb/usb1/1-1/1-1.4':
KERNELS=="1-1.4"
SUBSYSTEMS=="usb"
DRIVERS=="usb"
ATTRS{bDeviceClass}=="ff"
ATTRS{bmAttributes}=="c0"
ATTRS{bConfigurationValue}=="1"
ATTRS{version}==" 1.10"
ATTRS{devnum}=="7"
ATTRS{bMaxPower}=="0mA"
ATTRS{idProduct}=="0003"
ATTRS{avoid_reset_quirk}=="0"
ATTRS{urbnum}=="3412"
ATTRS{bDeviceSubClass}=="ff"
ATTRS{maxchild}=="0"
ATTRS{bcdDevice}=="0001"
ATTRS{bMaxPacketSize0}=="8"
ATTRS{idVendor}=="091e"
ATTRS{speed}=="12"
ATTRS{removable}=="removable"
ATTRS{ltm_capable}=="no"
ATTRS{bNumConfigurations}=="1"
ATTRS{busnum}=="1"
ATTRS{authorized}=="1"
ATTRS{quirks}=="0x0"
ATTRS{configuration}==""
ATTRS{devpath}=="1.4"
ATTRS{bDeviceProtocol}=="ff"
ATTRS{bNumInterfaces}==" 1"
looking at parent device '/devices/platform/soc/3f980000.usb/usb1/1-1':
KERNELS=="1-1"
SUBSYSTEMS=="usb"
DRIVERS=="usb"
ATTRS{bDeviceClass}=="09"
ATTRS{bmAttributes}=="e0"
ATTRS{bConfigurationValue}=="1"
ATTRS{version}==" 2.00"
ATTRS{devnum}=="2"
ATTRS{bMaxPower}=="2mA"
ATTRS{idProduct}=="9514"
ATTRS{avoid_reset_quirk}=="0"
ATTRS{urbnum}=="81"
ATTRS{bDeviceSubClass}=="00"
ATTRS{maxchild}=="5"
ATTRS{bcdDevice}=="0200"
ATTRS{bMaxPacketSize0}=="64"
ATTRS{idVendor}=="0424"
ATTRS{speed}=="480"
ATTRS{removable}=="unknown"
ATTRS{ltm_capable}=="no"
ATTRS{bNumConfigurations}=="1"
ATTRS{busnum}=="1"
ATTRS{authorized}=="1"
ATTRS{quirks}=="0x0"
ATTRS{configuration}==""
ATTRS{devpath}=="1"
ATTRS{bDeviceProtocol}=="02"
ATTRS{bNumInterfaces}==" 1"
looking at parent device '/devices/platform/soc/3f980000.usb/usb1':
KERNELS=="usb1"
SUBSYSTEMS=="usb"
DRIVERS=="usb"
ATTRS{bDeviceClass}=="09"
ATTRS{manufacturer}=="Linux 4.9.35-v7+ dwc_otg_hcd"
ATTRS{bmAttributes}=="e0"
ATTRS{bConfigurationValue}=="1"
ATTRS{version}==" 2.00"
ATTRS{devnum}=="1"
ATTRS{bMaxPower}=="0mA"
ATTRS{idProduct}=="0002"
ATTRS{avoid_reset_quirk}=="0"
ATTRS{urbnum}=="25"
ATTRS{bDeviceSubClass}=="00"
ATTRS{maxchild}=="1"
ATTRS{bcdDevice}=="0409"
ATTRS{bMaxPacketSize0}=="64"
ATTRS{idVendor}=="1d6b"
ATTRS{product}=="DWC OTG Controller"
ATTRS{speed}=="480"
ATTRS{authorized_default}=="1"
ATTRS{interface_authorized_default}=="1"
ATTRS{removable}=="unknown"
ATTRS{ltm_capable}=="no"
ATTRS{serial}=="3f980000.usb"
ATTRS{bNumConfigurations}=="1"
ATTRS{busnum}=="1"
ATTRS{authorized}=="1"
ATTRS{quirks}=="0x0"
ATTRS{configuration}==""
ATTRS{devpath}=="0"
ATTRS{bDeviceProtocol}=="01"
ATTRS{bNumInterfaces}==" 1"
looking at parent device '/devices/platform/soc/3f980000.usb':
KERNELS=="3f980000.usb"
SUBSYSTEMS=="platform"
DRIVERS=="dwc_otg"
ATTRS{wr_reg_test}=="Time to write GNPTXFSIZ reg 10000000 times: 690 msecs (69 jiffies)"
ATTRS{grxfsiz}=="GRXFSIZ = 0x00000306"
ATTRS{srpcapable}=="SRPCapable = 0x1"
ATTRS{buspower}=="Bus Power = 0x1"
ATTRS{bussuspend}=="Bus Suspend = 0x0"
ATTRS{hptxfsiz}=="HPTXFSIZ = 0x02000406"
ATTRS{hnp}=="HstNegScs = 0x0"
ATTRS{mode}=="Mode = 0x1"
ATTRS{mode_ch_tim_en}=="Mode Change Ready Timer Enable = 0x0"
ATTRS{hsic_connect}=="HSIC Connect = 0x1"
ATTRS{gsnpsid}=="GSNPSID = 0x4f54280a"
ATTRS{driver_override}=="(null)"
ATTRS{hcd_frrem}=="HCD Dump Frame Remaining"
ATTRS{gotgctl}=="GOTGCTL = 0x001c0001"
ATTRS{gpvndctl}=="GPVNDCTL = 0x00000000"
ATTRS{hnpcapable}=="HNPCapable = 0x1"
ATTRS{spramdump}=="SPRAM Dump"
ATTRS{regoffset}=="0xffffffff"
ATTRS{gnptxfsiz}=="GNPTXFSIZ = 0x01000306"
ATTRS{guid}=="GUID = 0x2708a000"
ATTRS{regdump}=="Register Dump"
ATTRS{hprt0}=="HPRT0 = 0x00001005"
ATTRS{hcddump}=="HCD Dump"
ATTRS{rem_wakeup_pwrdn}==""
ATTRS{regvalue}=="invalid offset"
ATTRS{gusbcfg}=="GUSBCFG = 0x20001700"
ATTRS{fr_interval}=="Frame Interval = 0x1d4b"
ATTRS{busconnected}=="Bus Connected = 0x1"
ATTRS{remote_wakeup}=="Remote Wakeup Sig = 0 Enabled = 0 LPM Remote Wakeup = 0"
ATTRS{devspeed}=="Device Speed = 0x0"
ATTRS{rd_reg_test}=="Time to read GNPTXFSIZ reg 10000000 times: 940 msecs (94 jiffies)"
ATTRS{enumspeed}=="Device Enumeration Speed = 0x1"
ATTRS{inv_sel_hsic}=="Invert Select HSIC = 0x0"
ATTRS{ggpio}=="GGPIO = 0x00000000"
ATTRS{srp}=="SesReqScs = 0x1"
looking at parent device '/devices/platform/soc':
KERNELS=="soc"
SUBSYSTEMS=="platform"
DRIVERS==""
ATTRS{driver_override}=="(null)"
looking at parent device '/devices/platform':
KERNELS=="platform"
SUBSYSTEMS==""
DRIVERS==""