Skip to content

Latest commit

 

History

History
259 lines (179 loc) · 7.92 KB

README.md

File metadata and controls

259 lines (179 loc) · 7.92 KB

SimpleLinkages

Simple Linkages for my robots 自作リンクロボットの簡単な4節リンクのプログラムを検討。 4節リンクの動きをシミュレートして可視化し、設計を検討します。 4節リンク/リンク/ピン の3つのクラスを作成して、順々に進化させていきます。

設計概要

順運動で計算する場合、この構造はモーターの角度から5節リンク4節リンクの計算を順番に解いていくことで計算することができる。 また、以下のように単純化して計算することも可能となる。

単純化

逆運動を計算するとき、下の図のように構造を単純化して計算する。 B1-Yの長さはb、Y-Eの長さはl1+a+e、リンクE-Iの相互関係は固定されているので計算可能、E-I-Jの角度はFのモーターの角度と同じ値となる。B1-Y-Eの角度はB1,B2の角度によって割り出すことができるので、逆算も可能。

逆運動

歩行を単純に考えるため、IとJのどちらか、もしくは両方が接地している条件で逆運動計算をして、B1,B2の角度を確定させてから、順運動で各頂点の位置とリンクの角度を決定する。 例えば、I-Jが接地している条件で、IのX座標の位置を指定する。 Eの位置をIから相対的に割り出すと、逆運動によってB1,B2の角度を計算することができる。


設計

検討資料

4節リンクのプログラムを検討を参照。

構成

以下のようなディレクトリ構成にします。

.
├── Linkages_lib
│   ├── __init__.py
│   ├── cpin.py
│   ├── clink.py
│   └── CFourBarLinkage.py
│
├── Linkages_lib_cli
│   └── call.py
└── setup.py

Python 3.8.10
OpenCV 4.5.4
moviepy 1.0.3

pip install moviepy 

Circle3D

## minicondaにsympyをインストールする
conda install sympy

PyDyで計算できそう https://pydy.readthedocs.io/en/stable/examples/rocket-car.html


2023/7/2 必要な作業

  • PCからの切断後、再接続待ちになるようにArduino側のソフトを改良する...完了

  • Constructを使って、通信データのパースが楽にできるように実装する...完了

  • サーボからの角度データ、電圧を受信して、PC側に表示する

    • Arduinoのwifiテストサンプルをサーボコントロール側にマージする...完了
  • PC側の機構データに反映して、脚の角度が可視化できるようにする

    • ワイヤーフレーム表示に反映する
      • まず、wifi-test.pyの実装をmain.pyにマージして、通信データを受信...完了
      • 他のライブラリとの関係をクラス図にして、受信したデータを self.robot.set_angles(theta1, theta2) self.robot.update_position() の形で値をセットして、アップデートして角度を決める。 サーボのポイントから角度を割り出す関数を作成して、その角度をrobotにセットする。
  • 原点調整ができるようにする。調整モードで立ち上げた後、微調整してファイルに保存し、次回起動時は微調整値を読み込んで起動する

@startuml

class Robot <<Singleton>> {
}

class SharedData {

}

class GraphGui {
    +run()
}

GraphGui -left- SharedData 

class OpenGLManager {
   +run()
}

OpenGLManager -- Robot 

class PygameManager {
    +run()
}

PygameManager -- Robot 

class WifiManager {
    +run()
}

WifiManager -right- SharedData 

SharedData o-u- Robot 

note top of OpenGLManager
  ロボットを3D表示する
end note

note top of PygameManager
  ロボットを2D表示する
end note

note left of Robot
  ロボットの情報を保持
  現在の状態を常時計算する
  <b>スイッチの切り替え</b>で、
  wifi経由で実機を反映するか、
  PCのシミュレーションか選択
end note

note bottom of WifiManager
  実機とWifi接続して、
  ハードのデータを取得する
end note

note bottom of GraphGui
  PCのGUIからユーザの
  入力を取得する
end note

note bottom of SharedData
  スレッドセーフに
  データを共有する
end note

@enduml

SharedData クラス

概要

SharedDataクラスは、複数のスレッド間で共有可能なデータをスレッドセーフに管理するためのクラスです。このクラスでは、特定のデータ(servocmdServoFb) がプロパティとして実装されており、それぞれのアクセスがスレッドセーフとなるようにロックが適用されています。これにより、同時にアクセスしようとする複数のスレッドからのデータ競合や不整合を防いでいます。また、Singletonパターンが適用されており、全体で一つのインスタンスしか生成されません。

クラス図

@startuml

class Singleton {
  + Instance(class_name: str) -> Any
}

Class ServoCmdStruct {
    - cmd: str
    + __init__(self, cmd: str)
    + execute(self): None
    + __eq__(self, other: Any): bool
}

Class ServoFbStruct {
    - fb: str
    + __init__(self, fb: str)
    + get_feedback(self): str
    + __eq__(self, other: Any): bool
}

class SharedData {
    - data: Dict[str, Deque[Tuple[int, Any]]]
    - queue_depth: int
    - lock: threading.Lock
    - _servo_cmd_lock: threading.Lock
    - _servo_fb_lock: threading.Lock
    - _servo_cmd: ServoCmdStruct
    - _servo_fb: ServoFbStruct

    + __init__(self, queue_depth: int = 100): None
    + set_data(self, key: str, timestamp_ms: int, value: Any) -> None
    + get_data(self, key: str) -> List[Tuple[int, Any]]
    + {static} servo_cmd(self)
    + {static} servo_cmd(self, value)
    + {static} servo_fb(self)
    + {static} servo_fb(self, value)
}

Singleton -u-^ SharedData : Inherits >
SharedData o-d- ServoCmdStruct : Contains >
SharedData o-d- ServoFbStruct : Contains >

@enduml

属性

  • data: 各キーの時系列データを保持する辞書。
  • queue_depth: 保持する時系列データの深さ(デフォルト値:100)。
  • lock: スレッドセーフな操作のための総合的なロックオブジェクト。
  • _servo_cmd_lock: ServoCmd オブジェクトへのアクセスをスレッドセーフにするためのロックオブジェクト。
  • _servo_fb_lock: ServoFb オブジェクトへのアクセスをスレッドセーフにするためのロックオブジェクト。
  • _servo_cmd: ServoCmdStruct オブジェクト。
  • _servo_fb: ServoFbStruct オブジェクト。

メソッド

__init__(self, queue_depth: int = 100): None

コンストラクタです。queue_depth は時系列データの最大保存数を指定します。

set_data(self, key: str, timestamp_ms: int, value: Any) -> None

指定されたキーに対応する時系列データを追加します。timestamp_msはUnixエポックからのミリ秒単位のタイムスタンプです。

get_data(self, key: str) -> List[Tuple[int, Any]]

指定されたキーに対応する全時系列データを取得します。

servo_cmd(self)

ServoCmdStruct オブジェクトを取得します。スレッドセーフにアクセスされます。

servo_cmd(self, value)

ServoCmdStruct オブジェクトを設定します。スレッドセーフにアクセスされます。

servo_fb(self)

ServoFbStruct オブジェクトを取得します。スレッドセーフにアクセスされます。

servo_fb(self, value)

ServoFbStruct オブジェクトを設定します。スレッドセーフにアクセスされます。

メモ

  • Singletonメタクラスから継承しています。