پروتکل HID ردیاب سر

پروتکل دستگاه رابط انسانی ردیاب هد (HID) که برای دستگاه‌های دارای اندروید 13 و بالاتر در دسترس است، به دستگاه ردیاب سر اجازه می‌دهد تا از طریق USB یا بلوتوث به دستگاه اندرویدی متصل شود و از طریق سنسورها در معرض چارچوب و برنامه‌های اندروید قرار گیرد. چارچوب این پروتکل برای کنترل افکت مجازی ساز صوتی (صوت سه بعدی) استفاده می شود. این صفحه از اصطلاحات دستگاه و میزبان به معنای بلوتوث خود استفاده می کند که دستگاه به معنای دستگاه ردیابی سر و میزبان به معنای میزبان اندروید است.

سازندگان دستگاه باید دستگاه‌های اندرویدی خود را به گونه‌ای پیکربندی کنند که پشتیبانی از پروتکل HID ردیاب سر را فعال کنند. برای اطلاعات بیشتر در مورد پیکربندی، به سنسورهای دینامیک README مراجعه کنید.

این صفحه به فرض آشنایی با منابع زیر است:

ساختار سطح بالا

چارچوب Android دستگاه ردیاب سر را به عنوان یک دستگاه HID شناسایی می کند.

برای مثال کامل یک توصیفگر HID معتبر، به پیوست 1 مراجعه کنید: نمونه ای از توصیفگر HID .

در سطح بالا، دستگاه ردیاب سر یک مجموعه برنامه با صفحه Sensors ( 0x20 ) و Other: Custom ( 0xE1 ) است. در داخل این مجموعه چندین فیلد داده ( ورودی ها ) و خواص ( ویژگی ها ) وجود دارد.

خواص و فیلدهای داده

این بخش ویژگی ها و فیلدهای داده را در مجموعه برنامه های یک دستگاه ردیاب سر توضیح می دهد.

ویژگی: توضیحات سنسور ( 0x0308 )

ویژگی Sensor Description ( 0x0308 ) یک ویژگی رشته ASCII (8 بیتی) فقط خواندنی است که باید حاوی مقادیر زیر باشد:

ردیاب سر نسخه 1.0:

#AndroidHeadTracker#1.0

Head Tracker نسخه 2.0 (موجود در Android 15 یا بالاتر)، که شامل پشتیبانی از صدای LE است:

#AndroidHeadTracker#2.0#x

x یک عدد صحیح ( 1 ، 2 ، 3 ) است که انتقال پشتیبانی را نشان می دهد:

  • 1: ACL
  • 2: ISO
  • 3: ACL + ISO

هیچ پایان دهنده تهی انتظار نمی رود، به این معنی که اندازه کل این ویژگی 23 کاراکتر 8 بیتی برای نسخه 1.0 است.

این ویژگی به عنوان یک تمایز برای جلوگیری از برخورد با سایر سنسورهای سفارشی عمل می کند.

ویژگی: شناسه منحصر به فرد پایدار ( 0x0302 )

ویژگی Persistent Unique ID ( 0x0302 ) یک آرایه فقط خواندنی از 16 عنصر است که هر کدام 8 بیت (مجموع 128 بیت) است. هیچ پایان دهنده تهی انتظار نمی رود. این ویژگی اختیاری است.

این ویژگی به دستگاه های ردیابی سر که در دستگاه های صوتی ادغام شده اند اجازه می دهد تا به دستگاه صوتی که به آن متصل هستند ارجاع دهند. طرح های زیر پشتیبانی می شوند.

ردیاب سر مستقل

اگر ویژگی Persistent Unique ID ( 0x0302 ) وجود نداشته باشد یا روی تمام صفرها تنظیم شده باشد، به این معنی است که دستگاه ردیاب سر به طور دائم به یک دستگاه صوتی متصل نیست و می تواند به طور جداگانه استفاده شود، به عنوان مثال، با اجازه دادن به کاربر به صورت دستی دستگاه ردیاب سر را با یک دستگاه صوتی جداگانه مرتبط کنید.

ارجاع با استفاده از آدرس مک بلوتوث

اکتت 0 1 2 3 4 5 6 7 8 9 10 11 12 13 14 15
ارزش 0 0 0 0 0 0 0 0 ب تی مک بلوتوث

در این طرح، 8 اکتت اول باید 0 باشند، 8 و 9 باید به ترتیب حاوی مقادیر ASCII B و T باشند، و 6 اکتت زیر به عنوان یک آدرس مک بلوتوث تفسیر می شوند، با فرض اینکه دستگاه ردیاب هد برای هر دستگاه صوتی دارای اعمال می شود. این آدرس مک این آدرس باید آدرس هویت باشد، حتی اگر دستگاه از یک آدرس MAC تصادفی برای برقراری اتصال استفاده کند. دستگاه‌های دو حالته که از طریق بلوتوث کلاسیک (فرمت v1.0 HID) و بلوتوث LE (فرمت v2.0 HID) متصل می‌شوند باید دو توصیفگر HID با آدرس هویت یکسان را نشان دهند. دستگاه‌های دو حالته با دستگاه‌های چپ و راست مجزا باید بلوتوث LE HID را با استفاده از دستگاه حالت دوگانه اولیه به جای دستگاه ثانویه فقط LE در معرض دید قرار دهند.

ارجاع با استفاده از UUID

هر زمان که مهم‌ترین بیت (MSB) از octet 8 تنظیم شود ( ≥0x80 )، فیلد به عنوان UUID تفسیر می‌شود، همانطور که در RFC-4122 مشخص شده است. دستگاه صوتی مربوطه همان UUID را که در چارچوب اندروید ثبت شده است، از طریق مکانیسم نامشخصی که مخصوص نوع حمل و نقل مورد استفاده است، ارائه می‌کند.

ویژگی: وضعیت گزارش ( 0x0316 )

ویژگی State Reporting ( 0x0316 ) یک ویژگی خواندن/نوشتن است که دارای معنای استانداردی است که در مشخصات HID تعریف شده است. میزبان از این ویژگی برای نشان دادن رویدادهایی که باید به دستگاه گزارش دهد استفاده می کند. فقط از مقادیر No Events ( 0x0840 ) و All Events ( 0x0841 ) استفاده می شود.

مقدار اولیه این فیلد باید بدون رویداد باشد و هرگز نباید توسط دستگاه، فقط توسط میزبان تغییر داده شود.

ویژگی: Power State ( 0x0319 )

ویژگی Power State ( 0x0319 ) یک ویژگی خواندن/نوشتن است که دارای معنای استانداردی است که در مشخصات HID تعریف شده است. میزبان از این ویژگی استفاده می کند تا به دستگاه نشان دهد که در کدام حالت برق باید باشد. فقط از مقادیر Full Power ( 0x0851 ) و Power Off ( 0x0855 ) استفاده می شود.

مقدار اولیه برای این فیلد توسط دستگاه تعیین می شود و هرگز نباید توسط دستگاه تغییر داده شود، فقط توسط میزبان.

ویژگی: فاصله گزارش ( 0x030E )

ویژگی Report Interval ( 0x030E ) یک ویژگی خواندن/نوشتن است که دارای معنای استانداردی است که در مشخصات HID تعریف شده است. میزبان از این ویژگی برای نشان دادن تعداد دفعات گزارش خواندن داده های خود به دستگاه استفاده می کند. واحدها ثانیه هستند. محدوده معتبر برای این مقدار توسط دستگاه تعیین می شود و با استفاده از مکانیسم حداقل/حداکثر فیزیکی توصیف می شود. حداقل نرخ گزارش 50 هرتز باید پشتیبانی شود و حداکثر نرخ گزارش توصیه شده 100 هرتز است. بنابراین، حداقل فاصله گزارش باید کمتر یا مساوی 20 میلی ثانیه باشد و توصیه می شود بیشتر یا مساوی 10 میلی ثانیه باشد.

دارایی: LE Transport رزرو شده توسط فروشنده ( 0xF410 )

ویژگی LE Transport رزرو شده توسط فروشنده ( 0xF410 ) یک ویژگی خواندن/نوشتن است که دارای معنای استانداردی است که در مشخصات HID تعریف شده است. میزبان از این ویژگی برای نشان دادن حمل و نقل انتخاب شده (ACL یا ISO) استفاده می کند. فقط از مقادیر ACL ( 0xF800 ) و ISO ( 0xF801 ) استفاده می شود و هر دو باید در مجموعه منطقی گنجانده شوند.

این ویژگی قبل از وضعیت قدرت یا گزارش پیکربندی شده است.

فیلد داده: مقدار سفارشی 1 ( 0x0544 )

فیلد Custom Value 1 ( 0x0544 ) یک فیلد ورودی است که برای گزارش اطلاعات واقعی ردیابی سر استفاده می شود. این یک آرایه 3 عنصری است که بر اساس قوانین HID معمولی برای مقادیر فیزیکی همانطور که در بخش 6.2.2.7 از مشخصات HID مشخص شده است، تفسیر شده است. محدوده معتبر برای هر عنصر [-π, π] راد است. واحدها همیشه رادیان هستند.

عناصر به این صورت تفسیر می شوند: [rx, ry, rz] ، به طوری که [rx, ry, rz] یک بردار چرخشی است که تبدیل از چارچوب مرجع به قاب سر را نشان می دهد. قدر باید در محدوده [0..π] باشد.

چارچوب مرجع دلخواه است، اما به طور کلی ثابت است و باید دست راست باشد. مقدار کمی دریفت قابل قبول است. محورهای سر عبارتند از:

  • X از گوش چپ به راست
  • Y از پشت سر تا بینی (پشت به جلو)
  • Z از گردن تا بالای سر

فیلد داده: مقدار سفارشی 2 ( 0x0545 )

فیلد Custom Value 2 ( 0x0545 ) یک فیلد ورودی است که برای گزارش اطلاعات واقعی ردیابی سر استفاده می شود. این یک آرایه 3 عنصری با نقطه ثابت است که بر اساس قوانین HID معمولی برای مقادیر فیزیکی تفسیر شده است. واحدها همیشه رادیان/ثانیه هستند.

عناصر به این صورت تفسیر می شوند: [vx, vy, vz] ، به طوری که [vx, vy, vz] یک بردار چرخشی است که نشان دهنده سرعت زاویه ای قاب سر (نسبت به خودش) است.

فیلد داده: مقدار سفارشی 3 ( 0x0546 )

فیلد Custom Value 3 ( 0x0546 ) یک فیلد ورودی است که برای ردیابی ناپیوستگی ها در قاب مرجع استفاده می شود. این یک عدد صحیح اسکالر با اندازه 8 بیت است. برای مثال، اگر یک الگوریتم فیلتر جهت که برای تعیین جهت استفاده می‌شود، حالت خود را بازنشانی کرده باشد، هر بار که چارچوب مرجع تغییر می‌کند، باید توسط دستگاه افزایش یابد (با پوشش). این مقدار بر اساس قوانین HID معمولی برای مقادیر فیزیکی تفسیر می شود. با این حال، ارزش فیزیکی و واحدها مهم نیست. تنها اطلاعات مربوط به میزبان یک مقدار تغییر یافته است. برای جلوگیری از مسائل عددی مربوط به از دست دادن دقت هنگام تبدیل از واحدهای منطقی به فیزیکی، توصیه می‌شود مقادیر حداقل فیزیکی، حداکثر فیزیکی و توان واحد را برای این فیلد روی صفر قرار دهید.

ساختار گزارش

گروه بندی ویژگی ها به گزارش ها (با تخصیص شناسه های گزارش) انعطاف پذیر است. برای کارایی، توصیه می‌کنیم ویژگی‌های فقط خواندنی را از ویژگی‌های خواندن/نوشتن جدا کنید.

برای فیلدهای داده، فیلدهای مقدار سفارشی 1، 2، و 3 باید در همان گزارش باشند و فقط در یک گزارش برای دستگاه معین (مجموعه برنامه) باشند.

ارسال گزارش ورودی

دستگاه باید به صورت دوره ای و ناهمزمان (از طریق پیام های HID INPUT) گزارش های ورودی را هنگامی که همه این شرایط برآورده می شود ارسال کند:

  • ویژگی Power State روی Full Power تنظیم شده است.
  • ویژگی دولت گزارشگر روی همه رویدادها تنظیم شده است.
  • ویژگی Reporting Interval غیر صفر است.

ویژگی Reporting Interval تعیین می کند که هر چند وقت یکبار گزارش ها ارسال شود. هنگامی که هر یک از شرایط بالا برآورده نمی شود، دستگاه نباید هیچ گزارشی ارسال کند.

سازگاری رو به جلو و عقب

پروتکل HID ردیاب سر از یک طرح نسخه‌سازی استفاده می‌کند که امکان به‌روزرسانی را فراهم می‌کند، در حالی که امکان همکاری بین یک میزبان و دستگاهی را که از نسخه‌های مختلف پروتکل استفاده می‌کند، می‌دهد. نسخه‌های پروتکل با دو عدد اصلی و کوچک شناسایی می‌شوند که دارای معنایی متمایز هستند که در بخش‌های زیر توضیح داده شده است.

نسخه های پشتیبانی شده توسط یک دستگاه را می توان با بررسی ویژگی Sensor Description ( 0x0308 ) آن تعیین کرد.

سازگاری نسخه جزئی

تغییرات نسخه جزئی با نسخه‌های کوچک قبلی که بر اساس همان نسخه اصلی هستند سازگار هستند. در به‌روزرسانی‌های نسخه کوچک، میزبان فیلدها و ویژگی‌های داده اضافی را نادیده می‌گیرد. به عنوان مثال، دستگاهی که از پروتکل نسخه 1.6 استفاده می کند، با میزبانی سازگار است که از نسخه 1.x، از جمله نسخه 1.5، از پروتکل پشتیبانی می کند.

سازگاری نسخه اصلی

تغییرات غیر سازگار با عقب برای تغییرات در نسخه های اصلی مجاز است. برای پشتیبانی از چندین نسخه اصلی برای قابلیت همکاری با میزبان‌های قدیمی و جدید، دستگاه‌ها می‌توانند مجموعه‌های برنامه‌های متعددی را در توصیفگرهای گزارش خود مشخص کنند. به عنوان مثال:

const unsigned char ReportDescriptor[] = {
    HID_USAGE_PAGE_SENSOR,
    HID_USAGE_SENSOR_TYPE_OTHER_CUSTOM,

    HID_COLLECTION(HID_APPLICATION),
        // Feature report 2 (read-only).
        HID_REPORT_ID(2),

        // Magic value: "#AndroidHeadTracker#1.5"
        HID_USAGE_SENSOR_PROPERTY_SENSOR_DESCRIPTION,
        HID_LOGICAL_MIN_8(0),
        HID_LOGICAL_MAX_8(0xFF),
        HID_REPORT_SIZE(8),
        HID_REPORT_COUNT(23),
        HID_FEATURE(HID_CONST_VAR_ABS),

      ...

    HID_END_COLLECTION,

    HID_COLLECTION(HID_APPLICATION),
        // Feature report 12 (read-only).
        HID_REPORT_ID(12),

        // Magic value: "#AndroidHeadTracker#2.4"
        HID_USAGE_SENSOR_PROPERTY_SENSOR_DESCRIPTION,
        HID_LOGICAL_MIN_8(0),
        HID_LOGICAL_MAX_8(0xFF),
        HID_REPORT_SIZE(8),
        HID_REPORT_COUNT(23),
        HID_FEATURE(HID_CONST_VAR_ABS),

      ...

    HID_END_COLLECTION,
};

در این مورد، میزبان می‌تواند تمام مجموعه‌های برنامه‌های مختلف تبلیغ‌شده توسط دستگاه را برشمرد، ویژگی Sensor Description آن‌ها را برای تعیین نسخه‌های پروتکلی که هر کدام پیاده‌سازی می‌کنند، بررسی کند، سپس آخرین نسخه پروتکلی را که میزبان پشتیبانی می‌کند انتخاب کند. در صورت انتخاب، میزبان با پروتکل واحدی کار می کند که برای طول عمر اتصال دستگاه انتخاب شده است.

پیوست: نمونه ای از توصیفگر HID

مثال زیر یک توصیفگر معتبر HID معمولی را نشان می دهد. از ماکروهای رایج C استفاده می‌کند که در موارد استفاده از حسگر HID (بخش 4.1) ارائه شده است.

const unsigned char ReportDescriptor[] = {
    HID_USAGE_PAGE_SENSOR,
    HID_USAGE_SENSOR_TYPE_OTHER_CUSTOM,
    HID_COLLECTION(HID_APPLICATION),
        // Feature report 2 (read-only).
        HID_REPORT_ID(2),

        // Magic value: "#AndroidHeadTracker#1.0"
        HID_USAGE_SENSOR_PROPERTY_SENSOR_DESCRIPTION,
        HID_LOGICAL_MIN_8(0),
        HID_LOGICAL_MAX_8(0xFF),
        HID_REPORT_SIZE(8),
        HID_REPORT_COUNT(23),
        HID_FEATURE(HID_CONST_VAR_ABS),

        // UUID.
        HID_USAGE_SENSOR_PROPERTY_PERSISTENT_UNIQUE_ID,
        HID_LOGICAL_MIN_8(0),
        HID_LOGICAL_MAX_8(0xFF),
        HID_REPORT_SIZE(8),
        HID_REPORT_COUNT(16),
        HID_FEATURE(HID_CONST_VAR_ABS),

        // Feature report 1 (read/write).
        HID_REPORT_ID(1),

        // 1-bit on/off reporting state.
        HID_USAGE_SENSOR_PROPERTY_REPORTING_STATE,
        HID_LOGICAL_MIN_8(0),
        HID_LOGICAL_MAX_8(1),
        HID_REPORT_SIZE(1),
        HID_REPORT_COUNT(1),
        HID_COLLECTION(HID_LOGICAL),
            HID_USAGE_SENSOR_PROPERTY_REPORTING_STATE_NO_EVENTS,
            HID_USAGE_SENSOR_PROPERTY_REPORTING_STATE_ALL_EVENTS,
            HID_FEATURE(HID_DATA_ARR_ABS),
        HID_END_COLLECTION,

        // 1-bit on/off power state.
        HID_USAGE_SENSOR_PROPERTY_POWER_STATE,
        HID_LOGICAL_MIN_8(0),
        HID_LOGICAL_MAX_8(1),
        HID_REPORT_SIZE(1),
        HID_REPORT_COUNT(1),
        HID_COLLECTION(HID_LOGICAL),
            HID_USAGE_SENSOR_PROPERTY_POWER_STATE_D4_POWER_OFF,
            HID_USAGE_SENSOR_PROPERTY_POWER_STATE_D0_FULL_POWER,
            HID_FEATURE(HID_DATA_ARR_ABS),
        HID_END_COLLECTION,

        // 6-bit reporting interval, with values [0x00..0x3F] corresponding to [10ms..100ms].
        HID_USAGE_SENSOR_PROPERTY_REPORT_INTERVAL,
        HID_LOGICAL_MIN_8(0x00),
        HID_LOGICAL_MAX_8(0x3F),
        HID_PHYSICAL_MIN_8(10),
        HID_PHYSICAL_MAX_8(100),
        HID_REPORT_SIZE(6),
        HID_REPORT_COUNT(1),
        HID_USAGE_SENSOR_UNITS_SECOND,
        HID_UNIT_EXPONENT(0xD),  // 10^-3
        HID_FEATURE(HID_DATA_VAR_ABS),

        // Input report 1

        // Orientation as rotation vector (scaled to [-pi..pi] rad).
        HID_USAGE_SENSOR_DATA_CUSTOM_VALUE_1,
        HID_LOGICAL_MIN_16(0x01, 0x80), // LOGICAL_MINIMUM (-32767)
        HID_LOGICAL_MAX_16(0xFF, 0x7F), // LOGICAL_MAXIMUM (32767)
        HID_PHYSICAL_MIN_32(0x60, 0x4F, 0x46, 0xED),  // -314159265
        HID_PHYSICAL_MAX_32(0xA1, 0xB0, 0xB9, 0x12),  // 314159265
        HID_UNIT_EXPONENT(0x08),  // 10^-8
        HID_REPORT_SIZE(16),
        HID_REPORT_COUNT(3),
        HID_INPUT(HID_DATA_VAR_ABS),

        // Angular velocity as rotation vector (scaled to [-32..32] rad/sec).
        HID_USAGE_SENSOR_DATA_CUSTOM_VALUE_2,
        HID_LOGICAL_MIN_16(0x01, 0x80), // LOGICAL_MINIMUM (-32767)
        HID_LOGICAL_MAX_16(0xFF, 0x7F), // LOGICAL_MAXIMUM (32767)
        HID_PHYSICAL_MIN_8(0xE0),
        HID_PHYSICAL_MAX_8(0x20),
        HID_UNIT_EXPONENT(0x00),  // 10^0
        HID_REPORT_SIZE(16),
        HID_REPORT_COUNT(3),
        HID_INPUT(HID_DATA_VAR_ABS),

        // Reference frame reset counter.
        HID_USAGE_SENSOR_DATA_CUSTOM_VALUE_3,
        HID_LOGICAL_MIN_16(0x00, 0x00), // LOGICAL_MINIMUM (0)
        HID_LOGICAL_MAX_16(0xFF, 0x00), // LOGICAL_MAXIMUM (255)
        HID_PHYSICAL_MIN_8(0x00),
        HID_PHYSICAL_MAX_8(0x00),
        HID_UNIT_EXPONENT(0x00),  // 10^0
        HID_REPORT_SIZE(8),
        HID_REPORT_COUNT(1),
        HID_INPUT(HID_DATA_VAR_ABS),

    HID_END_COLLECTION,
};

پیوست 2: نمونه ای از توصیفگر HID v2.0

مثال زیر یک توصیفگر HID v2.0 را برای دستگاهی نشان می دهد که فقط از انتقال بلوتوث LE ACL پشتیبانی می کند.

const unsigned char ReportDescriptor[] = {
    HID_USAGE_PAGE_SENSOR,
    HID_USAGE_SENSOR_TYPE_OTHER_CUSTOM,
    HID_COLLECTION(HID_APPLICATION),
        // Feature report 2 (read-only).
        HID_REPORT_ID(2),

        // Magic value: "#AndroidHeadTracker#2.0#1"
        HID_USAGE_SENSOR_PROPERTY_SENSOR_DESCRIPTION,
        HID_LOGICAL_MIN_8(0),
        HID_LOGICAL_MAX_8(0xFF),
        HID_REPORT_SIZE(8),
        HID_REPORT_COUNT(25),
        HID_FEATURE(HID_CONST_VAR_ABS),

        // UUID.
        HID_USAGE_SENSOR_PROPERTY_PERSISTENT_UNIQUE_ID,
        HID_LOGICAL_MIN_8(0),
        HID_LOGICAL_MAX_8(0xFF),
        HID_REPORT_SIZE(8),
        HID_REPORT_COUNT(16),
        HID_FEATURE(HID_CONST_VAR_ABS),

        // Feature report 1 (read/write).
        HID_REPORT_ID(1),

        // 1-bit on/off reporting state.
        HID_USAGE_SENSOR_PROPERTY_REPORTING_STATE,
        HID_LOGICAL_MIN_8(0),
        HID_LOGICAL_MAX_8(1),
        HID_REPORT_SIZE(1),
        HID_REPORT_COUNT(1),
        HID_COLLECTION(HID_LOGICAL),
            HID_USAGE_SENSOR_PROPERTY_REPORTING_STATE_NO_EVENTS,
            HID_USAGE_SENSOR_PROPERTY_REPORTING_STATE_ALL_EVENTS,
            HID_FEATURE(HID_DATA_ARR_ABS),
        HID_END_COLLECTION,

        // 1-bit on/off power state.
        HID_USAGE_SENSOR_PROPERTY_POWER_STATE,
        HID_LOGICAL_MIN_8(0),
        HID_LOGICAL_MAX_8(1),
        HID_REPORT_SIZE(1),
        HID_REPORT_COUNT(1),
        HID_COLLECTION(HID_LOGICAL),
            HID_USAGE_SENSOR_PROPERTY_POWER_STATE_D4_POWER_OFF,
            HID_USAGE_SENSOR_PROPERTY_POWER_STATE_D0_FULL_POWER,
            HID_FEATURE(HID_DATA_ARR_ABS),
        HID_END_COLLECTION,

        // 6-bit reporting interval, with values [0x00..0x3F] corresponding to [10ms..100ms].
        HID_USAGE_SENSOR_PROPERTY_REPORT_INTERVAL,
        HID_LOGICAL_MIN_8(0x00),
        HID_LOGICAL_MAX_8(0x3F),
        HID_PHYSICAL_MIN_8(10),
        HID_PHYSICAL_MAX_8(100),
        HID_REPORT_SIZE(6),
        HID_REPORT_COUNT(1),
        HID_USAGE_SENSOR_UNITS_SECOND,
        HID_UNIT_EXPONENT(0xD),  // 10^-3
        HID_FEATURE(HID_DATA_VAR_ABS),

        // 1-bit transport selection
        HID_USAGE_SENSOR_PROPERTY_VENDOR_LE_TRANSPORT,
        HID_LOGICAL_MIN_8(0),
        HID_LOGICAL_MAX_8(1),
        HID_REPORT_SIZE(1),
        HID_REPORT_COUNT(1),
        HID_COLLECTION(HID_LOGICAL),
            HID_USAGE_SENSOR_PROPERTY_VENDOR_LE_TRANSPORT_ACL,
            HID_USAGE_SENSOR_PROPERTY_VENDOR_LE_TRANSPORT_ISO,
            HID_FEATURE(HID_DATA_ARR_ABS),
        HID_END_COLLECTION,

        // Input report 1

        // Orientation as rotation vector (scaled to [-pi..pi] rad).
        HID_USAGE_SENSOR_DATA_CUSTOM_VALUE_1,
        HID_LOGICAL_MIN_16(0x01, 0x80), // LOGICAL_MINIMUM (-32767)
        HID_LOGICAL_MAX_16(0xFF, 0x7F), // LOGICAL_MAXIMUM (32767)
        HID_PHYSICAL_MIN_32(0x60, 0x4F, 0x46, 0xED),  // -314159265
        HID_PHYSICAL_MAX_32(0xA1, 0xB0, 0xB9, 0x12),  // 314159265
        HID_UNIT_EXPONENT(0x08),  // 10^-8
        HID_REPORT_SIZE(16),
        HID_REPORT_COUNT(3),
        HID_INPUT(HID_DATA_VAR_ABS),

        // Angular velocity as rotation vector (scaled to [-32..32] rad/sec).
        HID_USAGE_SENSOR_DATA_CUSTOM_VALUE_2,
        HID_LOGICAL_MIN_16(0x01, 0x80), // LOGICAL_MINIMUM (-32767)
        HID_LOGICAL_MAX_16(0xFF, 0x7F), // LOGICAL_MAXIMUM (32767)
        HID_PHYSICAL_MIN_8(0xE0),
        HID_PHYSICAL_MAX_8(0x20),
        HID_UNIT_EXPONENT(0x00),  // 10^0
        HID_REPORT_SIZE(16),
        HID_REPORT_COUNT(3),
        HID_INPUT(HID_DATA_VAR_ABS),

        // Reference frame reset counter.
        HID_USAGE_SENSOR_DATA_CUSTOM_VALUE_3,
        HID_LOGICAL_MIN_16(0x00, 0x00), // LOGICAL_MINIMUM (0)
        HID_LOGICAL_MAX_16(0xFF, 0x00), // LOGICAL_MAXIMUM (255)
        HID_PHYSICAL_MIN_8(0x00),
        HID_PHYSICAL_MAX_8(0x00),
        HID_UNIT_EXPONENT(0x00),  // 10^0
        HID_REPORT_SIZE(8),
        HID_REPORT_COUNT(1),
        HID_INPUT(HID_DATA_VAR_ABS),

    HID_END_COLLECTION,
};