2021/04/03

WSL2で仮想GPIO

   WSL2でLinuxカーネルのgpio-mockup機能を使って仮想GPIOを使えるようにします。


●今回の環境

 Windows10


 ディストリビューション

 カーネルバージョン

 今回の環境はまっさらなWSLで説明しますが、前回の仮想CANのカスタムカーネルの環境でもできます。

 Linuxカーネルをビルドできる環境が必要です。下記を参考にしてビルド環境を用意してください。

●方法

 GPIO機能、GPIO-MOCKUP機能を有効にしたカスタムカーネルをビルドします。
 カスタムカーネルができれば、あとはgpio-mockupモジュールをロードし、仮想GPIOデバイスを作成すれば、GPIOが無いPC上でGPIO操作ができるようになります。

●カスタムカーネルの設定手順

 menuconfigを立ち上げます。
$ make menuconfig

手順1:Device Drivers に入ります



手順2:GPIO Support を有効にする

 Device Drivers 内へ入り(リターンで入る)、GPIO Support を有効にします。


手順3:GPIO Testing Driver を有効にする

 GPIO Support 内へ入り、GPIO Testing Driver を有効にする。


 GPIO Testing Drivber が仮想GPIOドライバです。
 また、GPIOのsysfsインターフェースが必要なら、「/sys/class/gpio/... (sysfs interface)」も選択します。
 利用できるGPIOの数は「Maximum nuber of GPIOs for fast path」で設定します。

手順4:設定を保存して終了する

ここまでできたら、忘れずに設定を保存して menuconfig を終了します。
また、Linuxカーネル名を変えておくとよいでしょう。


手順5:カーネルをビルドし、仮想GPIOのLKMをインストールする

$ make
$ sudo make modules_install
 → /lib/modules の下に、カーネル名でディレクトリができます。
  この下の kernel ディレクトリ以下に仮想GPIOの gpio-mockup.ko ファイルが格納されます。

手順6:カスタムカーネルでWSL2を起動する

 今作った vmlinux ファイルでWSL2を再起動します。
 やり方は、以下の手順6~を参考にしてください。

手順7:GPIOを有効にしたカーネルで起動できていることを確認する

$ uname -a
Linux earth 5.4.72-microsoft-standard-WSL2-gpio #1 SMP Wed Mar 2 20:32:16 JST 2021 x86_64 x86_64 x86_64 GNU/Linux

 ここまでが、GPIOを有効にしたカスタムカーネルの作り方でした。


●仮想GPIOデバイスを作成する

gpio-mockupドライバをmodprobeコマンドで読み込みます。

書式:
    # modprobe gpio-mockup gpio_mockup_ranges=[x,y](,…,[x,y]) (gpio_mockup_named_lines)

    x:lineの開始番号、または、-1
    y:lineの終了番号、または xが-1のときlineの個数
 ※x = -1 のとき、割り当ては後ろからになるので注意

例1:gpio0~gpio7 の8つのGPIOを割り当てる
$ sudo modprobe gpio-mockup gpio_mockup_ranges=0,8
 → /dev の下に、gpiochip0 ができ、8line が使用できる
  また、sysfsインターフェースがある場合は、gpio0~7 までexportできる

例2:gpiochip0に4つ、gpiochip1に8つのGPIOを割り当てる
$ sudo modprobe gpio-mockup gpio_mockup_ranges=0,4,8,16
 → /dev の下に、gpiochip0 と gpiochip1 ができ、それぞれ 4line と 8line が使用できる
  また、sysfsインターフェースでは、gpio0~3 と gpio8~15 までexportできる

例3:gpiochip0とgpiochip1に後ろから8つずつGPIOを割り当てる
$ sudo modprobe gpio-mockup gpio_mockup_ranges=-1,8,-1,8
 → /dev の下に、gpiochip0 と gpiochip1 ができ、それぞれ 8line が使用できる
  また、sysfsインターフェースでは、gpio504~511 と gpio496~503 までexportできる

例4:各ラインにラベルを付ける
$ sudo modprobe gpio-mockup gpio_mockup_ranges=0,8,8,16 gpio_mockup_named_lines
 → 各ラインにラベル名が gpio-mockup-A-*、gpio-mockup-B-* という名前が付く
  また、sysfsインターフェースでは、gpio-mockup-A-*、gpio-mockup-B-*というような名前でexportされる


以上で仮想GPIOが使えるようになります。
libgpiod tools も普通に使えます。


●sysfsインターフェースのアクセス権について

 WSLでは、initプロセスが動いていないので、udevが動作していないため、/sys/class/gpio 以下のアクセス権がすべて root:root になってしまいます。
これらの対処法は次回まとめます。

2021/03/27

WSL2で仮想CAN通信

  WSL2でLinuxカーネルのSocketCAN機能を使って仮想CAN通信をできるようにします。

●今回の環境

 Windows10


 ディストリビューション

 カーネルバージョン

 環境は前回と同じです。

 Linuxカーネルをビルドできる環境が必要です。下記を参考にしてビルド環境を用意してください。

●方法

 まず、SocketCAN機能、VCAN機能を有効にしたカスタムカーネルをビルドします。
 カスタムカーネルができれば、あとはCAN通信のモジュールをロードし、仮想CANデバイスを作成すれば、CANデバイスが無くてもPC内でCAN通信ができるようになります。

●カスタムカーネルの設定手順

 menuconfigを立ち上げます。
$ make menuconfig

手順1:Networking suportを有効にする


 ※選択は、項目まで↑↓キーで移動して、スペースキーを押すとできます。

手順2:CAN bus subsystem support を有効にする

 Networking support 内へ入り(リターンで入る)、CAN bus subsystem support を有効にします。


手順3:CAN通信機能の3項目を有効にする

 CAN bus subsystem support 内へ入り、さらに下記の3項目を有効にする。


 デフォルトで既にこれらの項目が有効になっていると思いますが、念のために確認しておきます。

手順4:CAN Device Drivers 内の項目を有効にする

 CAN Device Driversの中へ入り、さらに以下の設定を行います。

 Virtual Local CAN Interface (vcan) が仮想CANドライバです。
 そのほか2つがデフォルトで有効ですが、そのままにします。

手順5:設定を保存して終了する

 ここまでできたら、忘れずに設定を保存して menuconfig を終了します。
 また、Linuxカーネル名を変えておくとよいでしょう。

手順6:カーネルをビルドし、CAN通信のLKMをインストールする

$ make
$ sudo make modules_install
 → /lib/modules の下に、カーネル名でディレクトリができます。
  この下の kernel ディレクトリ以下にCAN通信に必要な*.ko ファイルが格納されます。

手順7:カスタムカーネルでWSL2を起動する

 今作った vmlinux ファイルでWSL2を再起動します。
 やり方は、以下の手順6~を参考にしてください。

手順8:CAN通信を有効にしたカーネルで起動できていることを確認する

$ uname -a
Linux earth 5.4.72-microsoft-standard-WSL2-can #1 SMP Wed Mar 27 14:20:57 JST 2021 x86_64 x86_64 x86_64 GNU/Linux

 ここまでが、CAN通信を有効にしたカスタムカーネルの作り方でした。


●仮想CAN通信デバイスを作成する

以下のコマンドを実行します。
$ sudo modprobe can
$ sudo modprobe can_raw
$ sudo modprobe vcan
$ sudo ip link add dev vcan0 type vcan
$ sudo ip link set up vcan0
 → /sys/class/net の下に、vcan0 の名前で仮想CANデバイスができます。


あとは、ふつうに仮想CAN通信をすることができます。
can-utilsも普通に使えます。

※仮想CANデバイスは、PCを再起動すると消えてしまうので、また modprobe からコマンド実行してください。

  


2021/03/21

WSL2のカーネルをカスタムビルドする

 WSL2のLinuxカーネルを、コンフィグ設定してビルド・使用します。

●今回の環境

 Windows10


 ディストリビューション

 カーネルバージョン

 環境は前回と同じです。
 前回のビルド環境をそのまま使う場合は、下記の手順3から始めてください。

●ビルド手順

手順1:カーネルソースをダウンロード

 カーネルのソースコードはGitHubのMicrosoft公式のものを使用します。
$ curl -OL https://github.com/microsoft/WSL2-Linux-Kernel/archive/linux-msft-5.4.72.tar.gz
 → linux-msft-5.4.72.tar.gz ファイルができる

 解凍します。
$ tar -zxf linux-msft-5.4.72.tar.gz
 → WSL2-Linux-Kernel-linux-msft-5.4.72 ディレクトリができる

手順2:カーネルのビルドに必要なソフトをインストール

 カーネルソース内の README-Microsoft.WSL2 ファイルの2に書かれているツールをインストールします。
 ただし、これaptでインストールするのに、sudo とinstall の間に apt が抜けているので、aptを追記して実行すること(昔から抜けてる。いいかげん直しなよ。。。)
$ sudo apt install build-essential flex bison libssl-dev libelf-dev

手順3:.configファイルの準備

 ソースのディレクトリへ入り、Microsoft/config-wsl ファイルをコピーする
$ cd WSL2-Linux-Kernel-linux-msft-5.4.72
$ cp Microsoft/config-wsl .config


手順4:.configファイルを編集

 方法はいくつかありますが、今回はmenuconfigを使ってGUIで編集します。

 事前にmenuconfig用ツールのインストールが必要
$ sudo apt install libncurses-dev

 上記のツールが入っていれば、以下のコマンドを実行
$ make menuconfig


 ここで、カーネルに組込む、またはカーネルモジュールの有効・無効を切り替えます。

 今回は、カーネルの名前を変えることとします。
 以下の項目を設定します。

 General setup   --->
   Local version - append to kernel release

 WSL2のデフォルトは「-microsoft-standard-WSL2」なので、この名前を適当に変えます。
 今回は例として「-microsoft-standard-WSL2-custom」とします。
 この設定は、「uname -r」 で表示される名前が変ります。

 書き換え出来たら、セーブして終了します。

手順5:カーネルをビルド

$ make
 → vmlinuxファイルができる

 前回はコンフィグファイルを指定しましたが、今回は先ほど作った .config を使います。

手順6:vmlinuxファイルをWindows側へコピー

 コピー先は任意の場所を使用できるが、Windowsの領域へコピーする。
 今回の例では、Windowsの%USERPROFILE%ディレクトリの下にwsl2/5.4.72-microsoft-standard-WSL2-customディレクトリを作って、そこへコピーすることにします。
 ※(Windowsのユーザー名)は各環境で変えること
 ※%USERPROFILE%ディレクトリ = c:\Users\(Windowsのユーザー名) のディレクトリ
$ mkdir -p /mnt/c/Users/(Windowsのユーザー名)/wsl2/5.4.72-microsoft-standard-WSL2-custom
$ cp vmlinux /mnt/c/Users/(Windowsのユーザー名)/wsl2/5.4.72-microsoft-standard-WSL2-custom

手順7:.wslconfigファイルで使用するlinuxカーネルを指定する

 %USERPROFILE%ディレクトリの直下に .wslconfigファイルを、以下の内容で作る。
 ※"\"はエスケープシーケンスになるので2つ続けること
[wsl2]
kernel = c:\\Users\\(Windowsのユーザー名)\\wsl2\\5.4.72-microsoft-standard-WSL2-custom\\vmlinux

手順8:wslをシャットダウンする

 PowerShellで、wslをシャットダウンする。
PS > wsl --shutdown

手順9:wslを起動し、unameでバージョンを確認する

$ uname -a
Linux earth 5.4.72-microsoft-standard-WSL2-custom #1 SMP Wed Mar 21 15:20:57 JST 2021 x86_64 x86_64 x86_64 GNU/Linux

 ビルド日時などから、今作ったカーネルで起動できていることを確認する。


以上で、カーネルのカスタムビルドから起動するまででした。

次回は、Linuxカーネルの仮想CAN通信を有効にします。

2021/03/18

WSL2のカーネルを自分でビルドして使う

 WSL2でubuntuをインストールすると、/lib/modules以下がカラなので、PC上で仮想CAN、仮想GPIOなどのドライバが使えなかったり、ローダブルカーネルモジュール(LKM)のビルドができません。
 このため、Linuxカーネルのコンフィグからこれらのオプションを有効にしたり、LKMをビルドできる環境を用意する必要があります。
 まずは、WSL2のLinuxカーネルを自分でビルドして使えることを確認します。

●今回の環境

 Windows10


 ディストリビューション

 カーネルバージョン

今回はカーネルのコンフィグはWSL2デフォルトのままビルドします。
今のカーネルバージョンが5.4.72なので、同じ5.4.72を使います。
仮想CANを有効にしたカスタムビルドは次回行います。

●ビルド手順

手順1:カーネルソースをダウンロード

 カーネルのソースコードはGitHubのMicrosoft公式のものを使用します。
$ curl -OL https://github.com/microsoft/WSL2-Linux-Kernel/archive/linux-msft-5.4.72.tar.gz
 → linux-msft-5.4.72.tar.gz ファイルができる

 解凍します。
$ tar -zxf linux-msft-5.4.72.tar.gz
 → WSL2-Linux-Kernel-linux-msft-5.4.72 ディレクトリができる

手順2:カーネルのビルドに必要なソフトをインストール

 カーネルソース内の README-Microsoft.WSL2 ファイルの2に書かれているツールをインストールします。
ただし、これaptでインストールするのに、sudo とinstall の間に apt が抜けているので、aptを追記して実行すること(昔から抜けてる。いいかげん直しなよ。。。)
$ sudo apt install build-essential flex bison libssl-dev libelf-dev

手順3:カーネルをビルド

$ cd WSL2-Linux-Kernel-linux-msft-5.4.72
$ make KCONFIG_CONFIG=Microsoft/config-wsl
 → vmlinuxファイルができる

手順4:vmlinuxファイルをWindows側へコピー

 コピー先は任意の場所を使用できるが、Windowsの領域へコピーする。
 今回の例では、Windowsの%USERPROFILE%ディレクトリの下にwsl2/5.4.72-microsoft-standard-WSL2ディレクトリを作って、そこへコピーすることにします。
 ※(Windowsのユーザー名)は各環境で変えること
 ※%USERPROFILE%ディレクトリ = c:\Users\(Windowsのユーザー名) のディレクトリ
$ mkdir -p /mnt/c/Users/(Windowsのユーザー名)/wsl2/5.4.72-microsoft-standard-WSL2
$ cp vmlinux /mnt/c/Users/(Windowsのユーザー名)/wsl2/5.4.72-microsoft-standard-WSL2

手順5:.wslconfigファイルで使用するlinuxカーネルを指定する

 %USERPROFILE%ディレクトリの直下に .wslconfigファイルを、以下の内容で作る。
 ※"\"はエスケープシーケンスになるので2つ続けること
[wsl2]
kernel = c:\\Users\\(Windowsのユーザー名)\\wsl2\\5.4.72-microsoft-standard-WSL2\\vmlinux

手順6:wslをシャットダウンする

 PowerShellで、wslをシャットダウンする。
PS > wsl --shutdown

手順7:wslを起動し、unameでバージョンを確認する

$ uname -a
Linux earth 5.4.72-microsoft-standard-WSL2 #1 SMP Wed Mar 17 20:42:16 JST 2021 x86_64 x86_64 x86_64 GNU/Linux

ビルド日時などから、今作ったカーネルで起動できていることを確認する。

以上で終わりです。

次回は、カーネルのカスタムビルドを行います。

2019/05/06

仮想GPIOドライバひとまず完成

仮想GPIOドライバがひとまず完成しました。

GitHubの以下の場所にアップしています。
vgpio

動作確認は、ubuntu 18.04 (2019/05/06時点のupdate済)でしました。

●使い方
0.ビルドする。
cd vgpio
make
→vgpio.koができる。

1.ドライバをinsmodする。
sudo insmod vgpio.ko
→/sys/kernel/vgpioができる。

2.使いたいGPIOポートを作成する。
(例)GPIO23を作成する場合
echo 23 > /sys/kernel/vgpio/export
→GPIO23ができる。

あとは、通常のGPIOと同じ使い方です。

3.direction、value、edge、active_lowを読み書きする。
(例)GPIO23を入力モードにして、1回読み出す
echo "in" > /sys/kernel/vgpio/gpio23/direction
cat /sys/kernel/vgpio/gpio23/value


4.もう使わない場合は、vgpio.koをrmmodする
sudo rmmod vgpio.ko
→/sys/kernel下のvgpioと、作っていたgpio23が消える。
※作成したGPIOポートは、unexport操作しなくても
rmmodすると自動的にunexportします。

使い方は以上です。

●注意点
※1.本物のGPIOは、入力モードに設定しているので
書き込みはできませんが、この仮想GPIOでは、
入力モードでも書き込みができてしまいます。
デバッグで、GPIOの値を変える場合は、別コンソールで
echoなどでvalueの値を書き換えてください。

※2.select/pollのシステムコールが、現在使えません。
すぐできそうですですが、私がやりたいことは
現状でできたので、とりあえずこのままにします。

----------------------------------------------------------

この仮想GPIOドライバは、linuxカーネルのリポジトリの
以下のファイルを参考にしました。

v4.18
/samples/kobject/kobject-example.c
/drivers/gpio/gpiolib-sysfs.c

・アクセス権について
本物のGPIOでは、otherのアクセス権が読出しのみになっています。
これはカーネルレベルで制限かけられていました。
/include/linux/kernel.h VERIFY_OCTAL_PERMISSIONSマクロ
/fs/sysfs/group.c create_files関数

ラズベリーパイなどでは、GPIO操作用のGPIOグループがあり、
piユーザはGPIOグループに属しており、
GPIOの各属性にGPIOグループで読み書きしているようです。

PCでデバッグするとき、GPIOグループ作って、各属性を
GPIOグループに変えて、GPIOグループへの権限を変えて・・・
ということをいちいちするのは面倒なので、
仮想GPIOドライバではちょっと細工して、otherでも読み書き
できるようにしました。

・select/poll対応について
現在はできません。なので、edgeになにか書き込んでも
特に影響ありません。(書き込み自体はできます)
カーネルの以下の機能を使えばできそうであることはわかっています。
sysfs_get_dirent()  edgeがnone以外が書き込まれたら、kernfs_nodeを取得する
sysfs_notify_dirent() valueが変化したら、kernfs_nodeにイベント発生を通知する
sysfs_put() edge、active_lowが変化したら、kernfs_nodeを解放する

2019/05/03

仮想GPIOを/sys/class以下に作ってみたものの

仮想GPIOドライバをとりあえず作ってみました。
設定値をメモリ上にとって、gpiochipへの読み書きを
メモリの値に置き換えるだけなので、とりあえず動くものは
すぐにできました。

gpiolib-sysfs.cを参考にして、/sys/class以下に
vgpioという名前のディレクトリを作って、
その配下にexport等を作るようにしました。
しかし、同時に/sys/devices/virtual/vgpioもできてしまう。

また、動作させてexportへ任意の番号を書き込むとgpioNができるが、
このエイリアスが/sys/devices/virtual/vgpio/gpioNになり、
direction、value、edge、active_low以外にpower、uevent等
よくわからないものがたくさんできていました。

ただ direction/value/edge/active_low だけ読み書きできれば
良いのですが・・他のはsysfsのclassAPIが作ってるものなので、
なにか意図しないことが起きるかもしれない・・

というわけで、別の方法を模索してみます。

linux kernelのリポジトリを見ると、/samples/kobject があります。
これは、/sys/kernel以下にアトリビュートというファイルのようなものを
作って読み書きするサンプルです。
(sysfsはファイルではなくアトリビュートと言う)
これが、ほぼやりたいことをしているので、これを参考にして作り直します。
こんなのがあったんだな。

2019/04/30

GPIO操作のPCテスト環境について考える

GPIO操作のテスト環境は、ほぼ実機でのテストしかできません。
(と思ってるけど、実はあるのだろうか・・)

例えば、GPIO23に出力するとき、コンソールで以下のようにします。
> echo 23 > /sys/class/gpio/export
> echo "out" > /sys/class/gpio/gpio23/direction
> echo 1 > /sys/class/gpio/gpio23/value

この操作を、実機ではなくPC上で行おうとすると、
最初の exportへ23を書き込む ところで、引数エラーになります。
> echo 23 > /sys/class/gpio/export
echo: 書き込みエラー: 無効な引数です

dmesgで見ると、以下のメッセージが出力されています。
> dmesg
    ~省略~
[  955.6530668] export_store: invalid GPIO 23

これを出してるのは、linux v4.18.0 のカーネルソースを見ると、
以下の場所で出力しているようです。
/drivers/gpio/gpiolib-sysfs.c
453 static ssize_t export_store(struct class *class,
454    struct class_attribute *attr,
455    const char *buf, size_t len)
456 {
    ~省略~
465     desc = gpio_to_desc(gpio);
466     /* reject invalid GPIOs */
467     if (!desc) {
468         pr_warn("%s: invalid GPIO %ld\n", __func__, gpio);
469         return -EINVAL;
470     }

エラーになってるのはgpio_to_desc関数で、引数gpioに対応する
ディスクリプタを返すが、このディスクリプタはカーネルソース内で
チップ情報によって作成されています。つまり、この動作を変えるには、
カーネルソースの再ビルドが必要となりそうです。

なので、GPIO操作のテストには実機が必要だが、現場では数人で
実機を共有しているので、気軽にテストできません。
ようやく実機使えてテストしたら、「不等号の向き間違えた」なんかの
つまらないバグが出ると、時間の無駄だし、気が滅入ってきます。
なので、PC上でコーディングミスレベルの不具合は潰しておきたい。

PC上での方法を考えると、デバッグ時は/sys/class/gpio/gpioN/value を
mkfifo で作ったパイプで代替する方法をしている人が、私の職場でいました。
しかし、これは、書き込み側は動作は変わらないが、読み出し側がパイプを
readしたときに書き込みがあるまで関数がブロックしてしまいます。
また、読み出しが遅い場合、書き込みを、例えば2回1を書き込んだ場合、
読み出すと11になるので、一文字ずつ処理するなど、実機の場合には
不要な処理を足さなくてはならないので面倒です。

カーネルソースのgpiolib-sysfs.cを見ていると、実ポートには出力せず、
メモリ上の値を読み書きするだけの、GPIOの動作を模擬できる
仮想GPIOドライバが作れそうです。

というわけで、このゴールデンウィーク中に仮想GPIOドライバを
作ってみることにしました。