Windowsに限らず、多くのOS/プラットフォームで、ビットパターンやフラグと呼ばれるデータ形式が使われている。簡単に言えば、これらは数値の2進数表現で各桁を使って「オン/オフ」「有効/無効」「あり/なし」という情報を扱う。
大抵の言語には、こうしたフラグ/ビットパターンを扱う機能がある。一般に、フラグ/ビットパターンを扱うには、enumeration type(enumerated type)を使う。日本語では、列挙体や列挙型などと呼ぶ。言語中では、略して「emum」と表記されることが少なくない。
.NETにも「列挙型」があるため、PowerShellでもこれを利用できる。PowerShell内でも「enum」キーワードを使う。まずは、列挙値をビットパターンして作ってみる。
[Flags()] enum BitPattern {bit0 = 1;bit1 = 2;bit2 = 4;bit3 = 8}
これで4ビットのフラグができた。構造体の名前を使う(キャストする)ことで、4ビットの数値(0~16)からオンになっているビットを調べることができる。
[BitPattern] 10
「10」を2進数として表示するには、
[Convert]::ToString(10,2)
とする。
実際に使われているシステムのフラグを解析する
さて、これからが本番である。実際にWindowsで使われているenum値を定義して、フラグのオンオフ状態を表示させてみる。コマンドについては、過去記事(「WindowsのPowerShellからBluetoothデバイスを調べる」)で扱っている。
Bluetoothのステレオヘッドホンをペアリングしたのち、デバイスのオンオフで何が変わるかを調べてみる。
$myHeadPhone=(Get-PnpDevice | ? Class -eq "Bluetooth" | ? FriendlyName -eq 'MDR-ZX750BN').InstanceId
# デバイス未接続でプロパティを取得
$DevOff=Get-PnpDeviceProperty -InstanceId $myHeadPhone
# デバイス接続後にプロパティを取得
$DevOn=Get-PnpDeviceProperty -InstanceId $myHeadPhone
# $x1と$x2を比較
Compare-Object $DevOn $DevOff -Property KeyName,data | sort keyname
この比較だと、接続前と後で「{83DA6326-97A6-4088-9453-A1923F573B29} 15」と「DEVPKEY_Device_DevNodeStatus」が変化している。前者はFalseとTrueなので調べる必要もない。問題は「DEVPKEY_Device_DevNodeStatus」である。
DEVPKEY_Device_DevNodeStatusをインターネット検索すると、Microsoftのページ(https://learn.microsoft.com/ja-jp/windows-hardware/drivers/install/devpkey-device-devnodestatus)が見つかる。ここには、「DEVPKEY_Device_DevNodeStatus の値は、Cfg.h で定義されている DN_Xxx ビットフラグのビットごとの OR です。」とある。Cfg.hは、WindowsのSDKに含まれているC/C++インクルードファイルの1つ。ファイルを入手するには、Windows SDKをインストールする必要がある。すでにSDKがインストールされているとすれば、Cfg.hには、以下の画面のような定義がある。
この#define文を前述のenumステートメントに変換すれば、列挙型が定義でき、これを使うことで、対象数値でどのフラグがオンになっているのかを表示させることが可能になる。簡易には、エディタを使った手作業でするか、以下のようなコマンドの出力を使ってスクリプトファイルを作成する。
"[Flags()] enum DN_ {" >C:\temp\mysc.ps1
Get-Content "cfg.h"|sls "^#define DN_[^ ]*[^0]+0x"|%{[Void](($_ -split '/')[0] -match "(DN_[^ ]+)[^0]+(0x[\w]+)");Write-Output "$($matches.1)=$($matches.2)"} >> C:\temp\mysc.ps1
"}" >>C:\temp\mysc.ps1
こうして、「C:\temp\mysc.ps1」ができたので、これをドットソース「. c:\temp\mysc.ps1」コマンドで実行させる。
列挙型DN_Xが定義されたかどうかは、列挙型のメソッド、たとえば、「[DN_X].BaseType」などを実行させてみるとわかる。
では、DEVPKEY_Device_DevNodeStatusの値を見てみることにしよう。
[DN_X]58744842
[Convert]::ToString(58744842,2)
[DN_X]25190410
[Convert]::ToString(25190410,2)
違いはDN_NEEDS_LOCKINGの有無。未接続のときにDN_NEEDS_LOCKINGがオンになっている。cfg.hを見ると、DN_NEEDS_LOCKINGには、別名としてDN_DEVICE_DISCONNECTEDが割り当ててあり、「The function driver for a device reported that the device is not connected.」という注釈があり、デバイスが未接続であることを表しているようだ。
Bluetoothデバイスが接続中かどうかは、DEVPKEY_Device_DevNodeStatusの値にDN_NEEDS_LOCKINGがあるかどうかを判定すればよいことがわかる。
enumを使わないと、数値の中でどのフラグが1になっているのかをPowerShellで調べるにはプログラムを書かねばならない。2進数表示させたとしても、1のビットの桁位置を調べるのは面倒だ。列挙型を定義できるなら、これを使ってどのフラグがオンなのかを簡単に調べることが可能だ。

この連載の記事
-
第508回
PC
Scalable Vector Graphics(SVG)そもそも何なのか? -
第507回
PC
Windows 11の「開発者モード」とは何か? -
第506回
PC
Windows 11は早くも来秋登場の26H2プレビューの準備が始まる -
第505回
PC
結構変化しているWindows 11のエクスプローラーの基本設定を見直す -
第504回
PC
新しいOutlookとOutlook Classic、そろそろ古いOutlookとExchangeの組み合わせは引退の頃合いか -
第503回
PC
機能が増えたこともあり、寄せ集めから統合化に進むWindowsの便利ツール「PowerToys」 -
第502回
PC
Windows 11でBluetoothのオーディオ新規格「Bluetooth LE Audio」を試す -
第501回
PC
Windows 11 Ver.25H2での変更点、新機能を整理する -
第500回
PC
Windows 11 Ver.25H2が完成した -
第499回
PC
Windowsでの致命的だが回復可能なエラーに備える手段を2つ紹介 -
第498回
PC
Windows Terminalの安定版V1.23が公開 設定UIが改良される - この連載の一覧へ
















