Skip to content

Latest commit

 

History

History
223 lines (166 loc) · 8.71 KB

README.md

File metadata and controls

223 lines (166 loc) · 8.71 KB

hakoniwa-sim-csharp

このリポジトリでは、Unityアプリケーションを箱庭アセットとして活用するための機能を提供します。 箱庭(Hakoniwa)は、仮想環境でロボットやIoTデバイスをシミュレートするための強力なプラットフォームです。

箱庭の特徴

箱庭は現実世界と仮想環境を統合し、すべてを箱庭アセットという概念で管理するシミュレーションプラットフォームです。主な特徴は以下の通りです:

  • 箱庭が提供する主要機能
  • 箱庭PDU通信の特徴
    • データ定義の標準化/抽象化: ROS IDLベースでPDUデータを定義。
    • データ通信APIの標準化: 統一された箱庭PDU APIを提供。
    • 通信プロトコルの柔軟性: 使用するプロトコルを選択可能(例:WebSocket、共有メモリなど)。

概要

以下の図は、このリポジトリが提供する箱庭Unityフレームワークのアーキテクチャを示しています。

Hakoniwa Unity Architecture

  • 最上位層には、ユーザーが作成するUnityアプリケーションがあります。
  • Hakoniwa PDU APIを通じて、外部ソフトウェアとPDUデータを交換可能です。
  • 追加機能として、以下のオプションが利用可能です:
    • 共有メモリベースの通信: Linuxサーバーなどでの高速通信に最適。
    • 箱庭時刻同期機能: 分散シミュレーションにおける正確な時間管理を実現。

利用シーン別に選べる通信方式

  • UnityアプリケーションをAndroid/iOSに組み込む場合PDU通信パッケージを利用することで、モバイル環境でも簡単にPDU通信を実現可能です。
  • Linuxサーバーでの高速分散シミュレーション: 共有メモリベースのPDU通信と時刻同期機能を活用することで、大規模なシミュレーションを効率的に実行できます(本リポジトリが該当機能を提供)。

前提条件

以下の環境で動作を確認しています:

  • OS: macOS(AppleSilicon)
  • Unity: 2022.3.31f1
  • 依存ライブラリ:
    • WebSocketサポート
    • Hakoniwa Core Library (libshakoc.so)

事前に箱庭コア機能をインストールしてください:


セットアップ

  • あなたのUnityプロジェクトを開きます。
  • Package ManagerAdd package from git URL... をクリックし、以下を指定します。
  • 成功すると、以下の2つのパッケージがインストールされます。
    • スクリーンショット 2024-12-06 15 30 10

基本的な使い方

箱庭スクリプトのアタッチ

  1. あなたのUnityシーンを開き、ヒエラルキービューで、空のゲームオブジェクトを作成します。
  2. 作成したゲームオブジェクトのインスペクタビューから Add Componentをクリックし、HakoAssetと入力して、該当の C# スクリプトをアタッチします。
  3. AssetName には、重複しないアセット名を設定します
  4. Pdu Config Pathには、custom.json と ros_typesディレクトリが配置されているディレクトリパスを指定します。デフォルトは、プロジェクト直下であり、. です。
2024-12-05.13.34.07.mov

custom.json の配置

利用する箱庭PDUデータを custom.json として作成してください。

作成例: LaserScan の場合

{
  "robots": [
    {
      "name": "LiDAR2D",
      "rpc_pdu_readers": [],
      "rpc_pdu_writers": [],
      "shm_pdu_readers": [],
      "shm_pdu_writers": [
        {
          "type": "sensor_msgs/LaserScan",
          "org_name": "scan",
          "name": "LiDAR2D_scan",
          "channel_id": 0,
          "pdu_size": 6088
        }
      ]
    }
  ]
}

ros_typesの配置

箱庭PDU管理リポジトリから、PDU定義ファイルをダウンロードして、ros_types ディレクトリ配下に配置します。

  • ros_types
    • json
      • pdu/json から必要なものを取得して配置
    • offset
      • pdu/offset から必要なものを取得して配置

箱庭プログラムの作成

あなたのゲームオブジェクトを箱庭シミュレーションとして動作させるには、IHakoObject を継承してコールバック関数を実装する必要があります。

public interface IHakoObject
{
	void EventInitialize (); /* 初期化処理 */
	void EventStart (); /* 開始処理 */ 
	void EventTick (); /* 毎フレームの処理 */
	void EventStop (); /* 停止処理 */
	void EventReset (); /* リセット処理 */
}

EventInitialize

このコールバック関数は、箱庭のシミュレーション初期化時に1度だけ呼びだされます。

このタイミングで、箱庭PDUデータのI/Oに関する宣言をする必要があります。

// PDU I/O 用のオブジェクトを取得
hakoPdu = HakoAsset.GetHakoPdu();
// 書き込みするPDU宣言する場合。第一引数は、ロボット名、第二引数はPDU名です。
hakoPdu.DeclarePduForWrite("myRobot", "sensor1");
// 読み込みするPDU宣言する場合。第一引数は、ロボット名、第二引数はPDU名です。
hakoPdu.DeclarePduForWrite("myRobot", "motor");

EventStart, EventStop, EventReset

これらのコールバック関数は、箱庭のシミュレーション開始/停止/リセット時に呼びだされます。

このタイミングで、処理すべきものがあれば実装してください。

EventTick

このコールバック関数は、箱庭のシミュレーション・ステップ単位で呼び出されます。

このタイミングで、シミュレーションに必要な処理を行い、PDUデータの読み書きします。

以下は、PDUデータの書き込み処理実装例です。

public async void EventTick()
{
	//PDU I/O 用のオブジェクトを取得
	var pduManager = hakoPdu.GetPduManager();
	//PDUデータを新規作成
	IPdu pdu = pduManager.CreatePdu(robotName, pduName);
	//センサデータの情報取得処理
	this.Scan();
	//センサデータのPDU設定処理
	this.SetScanData(pdu);
	//PDUデータ書き込み
	pduManager.WritePdu(robotName, pdu);
	//PDUデータをフラッシュ
	await pduManager.FlushPdu(robotName, pduName);
}

PDUデータは、データ型毎にPDUデータアクセサがあります。例えば、LaserScanの場合は以下のようにアクセスできます。

public void SetScanData(IPdu pdu)
{
    //PDUデータをもとに、PDUデータのアクセサを作成します
    LaserScan scan = new LaserScan(pdu);

    //header
    long t = UtilTime.GetUnixTime();
    scan.header.stamp.sec = (int)((long)(t / 1000000));
    scan.header.stamp.nanosec = (uint)((long)(t % 1000000)) * 1000;
    scan.header.frame_id = this.sensorParameters.frame_id;

    //body
    scan.angle_min = angle_min;
    scan.angle_max = angle_max;
    scan.range_min = range_min;
    scan.range_max = range_max;


    if (sensorParameters.AngleRange.BlindPaddingRange != null)
    {
        var values = new float[max_count];
        for (int i = 0; i < sensorParameters.AngleRange.BlindPaddingRange.Size; i++)
        {
            values[i] = sensorParameters.AngleRange.BlindPaddingRange.Value;
        }
        for (int i = sensorParameters.AngleRange.BlindPaddingRange.Size; i < max_count; i++)
        {
            values[i] = distances[i - sensorParameters.AngleRange.BlindPaddingRange.Size];
        }
        scan.ranges = values;
    }
    else
    {
        scan.ranges = distances;
    }
    scan.angle_increment = angle_increment;
    scan.time_increment = time_increment;
    scan.scan_time = scan_time;
    scan.intensities = intensities;
}

詳細な実装例

詳しい実装例については、このサンプルコードを参考にしてください。