●ロボットを完成させる
■歩行プログラムの改善
前節のプログラムと歩行のプログラムなどを合わせて、リモコンで動作をコントロールするプログラムを作ります。準備として歩行のプログラムを少し変更します。
現状歩行のタイミングをコントロールするためにtime.sleep()を使っていますが、これがあるとリモコン用の割り込みと両立ができません。このため歩行の側もタイマー割込みで30Hzの制御をすることにします。リストを改造します。
リモコンの時には端子の状態がトリガーとなって割り込みがかかりましたが、今回は自分で時間を数えて定期的に割り込みをかけます。30Hzで呼び出される関数tick(timer)ではグローバル変数「servo_flag」をTrueに設定して0.03ms(30Hz)経過したことを知らせます。
servo_flag = False
tim = Timer()
# 30Hzのタイマー割り込み
def tick(timer):
global servo_flag
servo_flag = True
tim.init(freq=30, mode=Timer.PERIODIC, callback=tick)
whileループのなかでは、servo_flagがTrueだった場合にのみデータを更新してサーボを駆動します。Servo_flagは次に備えてFalseに戻しておきます。
while True: # 繰り返し
if servo_flag == True:
servo_flag = False
# キーフレームを更新
List19 walk3
from machine import Pin, PWM, Timer
#import time
SV_FREQ = 50.0 # サーボ信号周波数
MAX_DUTY = 65025.0 # 周期内の分割数
MIN_SV_PULSE = 0.6 # 最小パルス幅 0°
MAX_SV_PULSE = 2.4 # 最大パルス幅 180°
correction = [14,-14,4, 0,0,6, -25,-12,4, 20,-4,-8]
servo = []
temp_angle = [90, 90, 90, 90, 90, 90, 90, 90, 90, 90, 90, 90]
angle = [\
[90,105, 70, 90, 86, 95, 90, 94, 85, 90, 75,110, 3],\
[90,100,110, 90, 82,100, 90, 98, 80, 90, 80, 70, 3],\
[90, 90, 90, 90, 78,105, 90,102, 75, 90, 90, 90, 8],\
[90, 94, 85, 90, 75,110, 90,105, 70, 90, 86, 95, 3],\
[90, 98, 80, 90, 80, 70, 90,100,110, 90, 82,100, 3],\
[90,102, 75, 90, 90, 90, 90, 90, 90, 90, 78,105, 8]\
]
div_counter = 0
key_frame = 0
next_key_frame = 1
rows = len(angle)
servo_flag = False
tim = Timer()
#30Hzのタイマー割り込み
def tick(timer):
global servo_flag
servo_flag = True
tim.init(freq=30, mode=Timer.PERIODIC, callback=tick)
#パルス幅を計算する関数
def get_pulse_width(angle):
pulse_ms = MIN_SV_PULSE + (MAX_SV_PULSE - MIN_SV_PULSE) * angle / 180.0
x = (int)(MAX_DUTY * (pulse_ms * SV_FREQ /1000.0))
return x
#全てのサーボを順番に駆動
for i in range(12):
servo.append(PWM(Pin(11 - i)))
servo[i].freq(50)
servo[i].duty_u16(get_pulse_width(90 + correction[i]))
while True: # 繰り返し
if servo_flag == True:
servo_flag = False
# キーフレームを更新
div_counter += 1
if div_counter >= angle[key_frame][12]:
div_counter = 0
key_frame = next_key_frame
next_key_frame += 1
if next_key_frame > rows-1:
next_key_frame = 0
# 角度計算
for i in range(12):
temp_angle[i] = angle[key_frame][i] +\
(angle[next_key_frame][i] - angle[key_frame][i])\
* div_counter / angle[key_frame][12]
# サーボ駆動
for i in range(12):
servo[i].duty_u16(get_pulse_width(int(temp_angle[i]) + correction[i]))
#time.sleep(0.03) # 0.03秒待ち 削除
コメント