-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathphysics.py
146 lines (126 loc) · 5.86 KB
/
physics.py
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
__author__ = "seamonsters"
import math
import wpilib
import inspect, os
import configparser
from pyfrc.physics import drivetrains
from pyfrc.physics.visionsim import VisionSim
import ctre
import robotpy_ext.common_drivers.navx
from networktables import NetworkTables
# make the NavX work with the physics simulator
def createAnalogGyro():
return wpilib.AnalogGyro(0)
robotpy_ext.common_drivers.navx.AHRS.create_spi = createAnalogGyro
class SimulatedTalon:
def __init__(self, name):
if name == '':
self.port = None
else:
self.port = abs(int(name))
self.inv = -1 if name.startswith('-') else 1
self.lastPosition = 0
def getSpeed(self, data, maxVel):
if self.port == None:
return 0.0
try:
talonData = data['CAN'][self.port]
controlMode = talonData['control_mode']
if controlMode == ctre.ControlMode.PercentOutput:
value = talonData['value']
if value < -1:
value = -1.0
if value > 1:
value = 1.0
talonData['quad_position'] += int(value * maxVel / 5)
talonData['quad_velocity'] = int(value * maxVel) # update encoder
return value * self.inv
elif controlMode == ctre.ControlMode.Position:
targetPos = talonData['pid0_target']
diff = targetPos - self.lastPosition
self.lastPosition = targetPos
talonData['quad_position'] = targetPos # update encoder
talonData['quad_velocity'] = int(diff * 5)
return diff / maxVel * 5 * self.inv
elif controlMode == ctre.ControlMode.Velocity:
targetVel = talonData['pid0_target']
talonData['quad_position'] += int(targetVel / 5) # update encoder
talonData['quad_velocity'] = int(targetVel)
return targetVel / maxVel * self.inv
else:
return 0.0
except KeyError:
return 0.0
class PhysicsEngine:
def __init__(self, physicsController):
self.physicsController = physicsController
# NavX simulation
self.physicsController.add_analog_gyro_channel(0)
config = configparser.ConfigParser()
filename = os.path.dirname(os.path.abspath(inspect.getfile(inspect.currentframe()))) + os.sep + "sim" + os.sep + "drivetrain.ini"
print("Reading robot data from", filename)
config.read(filename)
if not 'physics' in config:
print("Please use the [physics] header in drivetrain.ini")
exit()
physics = config['physics']
# get() allows fallback values if the key isn't in the config file
self.xLen = float(physics.get('xlen', '2'))
self.yLen = float(physics.get('ylen', '3'))
self.speed = float(physics.get('speed', '6'))
self.drivetrain = physics.get('drivetrain', 'four')
self.flMotor = SimulatedTalon(physics.get('canfl', '0'))
self.frMotor = SimulatedTalon(physics.get('canfr', '0'))
self.blMotor = SimulatedTalon(physics.get('canbl', '0'))
self.brMotor = SimulatedTalon(physics.get('canbr', '0'))
self.maxVel = int(physics.get('maxvel', '8000'))
ds = config['ds']
location = int(ds.get('location', '1'))
team = 1 if (ds.get('team', 'red').lower() == 'blue') else 0
self.allianceStation = location - 1 + team * 3
field = config['field']
self.visionX = float(field.get('visionx', '0'))
self.visionY = float(field.get('visiony', '0'))
self.visionAngleStart = float(field.get('visionanglestart', '90'))
self.visionAngleEnd = float(field.get('visionangleend', '270'))
def initialize(self, hal_data):
self.visionTable = NetworkTables.getTable('limelight')
self.visionTable.putNumber('tv', 1)
self.visionTable.putNumber('tx', 0)
self.visionTable.putNumber('ty', 0)
self.visionTable.putNumber('ts', 0)
self.visionTable.putNumber('ta', 5)
visionTarget = VisionSim.Target(self.visionX, self.visionY,
self.visionAngleStart,
self.visionAngleEnd)
self.visionSim = VisionSim([visionTarget], 60, 2, 50)
hal_data['alliance_station'] = self.allianceStation
def update_sim(self, data, time, elapsed):
fl = self.flMotor.getSpeed(data, self.maxVel)
fr = self.frMotor.getSpeed(data, self.maxVel)
bl = self.blMotor.getSpeed(data, self.maxVel)
br = self.brMotor.getSpeed(data, self.maxVel)
if self.drivetrain == "mecanum":
vx, vy, vw = drivetrains.mecanum_drivetrain(bl, br, fl, fr,
self.xLen, self.yLen, self.speed)
self.physicsController.vector_drive(vx, vy, vw, elapsed)
elif self.drivetrain == "four":
speed, rot = drivetrains.four_motor_drivetrain(bl, br, fl, fr,
self.xLen, self.speed)
self.physicsController.drive(speed, rot, elapsed)
elif self.drivetrain == "two":
speed, rot = drivetrains.two_motor_drivetrain(fl, fr,
self.xLen, self.speed)
self.physicsController.drive(speed, rot, elapsed)
# https://github.com/robotpy/robotpy-wpilib/issues/291
data['analog_gyro'][0]['angle'] = math.degrees(self.physicsController.angle)
x, y, angle = self.physicsController.get_position()
visionData = self.visionSim.compute(time, x, y, angle)
if visionData is not None:
targetData = visionData[0]
self.visionTable.putNumber('tv', targetData[0])
if targetData[0] != 0:
self.visionTable.putNumber('tx', targetData[2])
else:
# DOESN'T mean no vision. vision just doesn't always update
pass