00001
00002
00003
00004
00005
00006
00007
00008
00009
00010
00011
00012
00013 #include "Tools/Math/PIDsmoothedValue.h"
00014 #include "GT2004SimpleBasicBehaviors.h"
00015 #include "Tools/Math/Geometry.h"
00016 #include "Tools/Math/Matrix2x2.h"
00017 #include "Tools/Math/Common.h"
00018
00019 #include "Tools/Debugging/DebugDrawings.h"
00020 #include "Tools/Debugging/Debugging.h"
00021
00022
00023 #include "Tools/FieldDimensions.h"
00024
00025 void GT2004BasicBehaviorGoToBall::execute()
00026 {
00027
00028
00029 double targetAngle;
00030
00031 if (maxSpeed == 0) maxSpeed = 350;
00032 if (maxSpeedY == 0) maxSpeed = 350;
00033 if (targetAngleToBall == 0)
00034 targetAngle = 0;
00035 else
00036 targetAngle = targetAngleToBall;
00037
00038
00039 double maxTurnSpeed;
00040 if (this->maxTurnSpeed == 0)
00041 maxTurnSpeed = fromDegrees(180);
00042 else
00043 maxTurnSpeed = fromDegrees(this->maxTurnSpeed);
00044
00045 Vector2<double> ballInWorldCoord = ballModel.getKnownPosition(
00046 BallModel::behaviorControlTimeAfterWhichCommunicatedBallsAreAccepted);
00047 double angleToBall = Geometry::angleTo(robotPose.getPose(),ballInWorldCoord);
00048 double distanceToBall = Geometry::distanceTo(robotPose.getPose(),ballInWorldCoord);
00049
00050
00051
00052 double factor = 1 - (distanceToBall - 400) / 200;
00053 if (factor > 1) factor = 1;
00054 if (factor < 0) factor = 0;
00055
00056
00057
00058
00059
00060
00061 Vector2<double> destination(
00062 distanceToBall * cos(angleToBall) + 400,
00063 distanceToBall * sin(angleToBall) + targetAngleToBall + yOffset);
00064
00065
00066
00067 factor = (pi-fabs(angleToBall))/pi;
00068 if (factor > 1) factor = 1;
00069 if (factor < 0) factor = 0;
00070
00071 destination.normalize();
00072 destination *= (maxSpeed*factor);
00073
00074
00075
00076
00077
00078
00079
00080
00081
00082 motionRequest.motionType = MotionRequest::walk;
00083 motionRequest.walkRequest.walkType = static_cast<WalkRequest::WalkType>(static_cast<int>(walkType));
00084 motionRequest.walkRequest.walkParams.translation.x = destination.x;
00085 motionRequest.walkRequest.walkParams.translation.y = destination.y;
00086
00087 if (motionRequest.walkRequest.walkParams.translation.y > maxSpeedY)
00088 motionRequest.walkRequest.walkParams.translation.y = maxSpeedY;
00089
00090 factor = 1.5;
00091
00092
00093
00094
00095
00096 motionRequest.walkRequest.walkParams.rotation = angleToBall * factor;
00097
00098
00099 motionRequest.walkRequest.walkParams.rotation = min(motionRequest.walkRequest.walkParams.rotation, maxTurnSpeed);
00100 motionRequest.walkRequest.walkParams.rotation = max(motionRequest.walkRequest.walkParams.rotation, -maxTurnSpeed);
00101 }
00102
00103 void GT2004BasicBehaviorGoToBallWithoutTurning::execute()
00104 {
00105 if (maxSpeed == 0) maxSpeed = 250;
00106
00107 Vector2<double> ballInWorldCoord = ballModel.getKnownPosition(
00108 BallModel::behaviorControlTimeAfterWhichCommunicatedBallsAreAccepted);
00109 double angleToBall = Geometry::angleTo(robotPose.getPose(),ballInWorldCoord);
00110 double distanceToBall = Geometry::distanceTo(robotPose.getPose(),ballInWorldCoord);
00111
00112 Vector2<double> destination(
00113 distanceToBall * cos(angleToBall),
00114 distanceToBall * sin(angleToBall) );
00115
00116 destination.normalize();
00117 destination *= maxSpeed;
00118
00119 motionRequest.motionType = MotionRequest::walk;
00120 motionRequest.walkRequest.walkType = WalkRequest::normal;
00121 motionRequest.walkRequest.walkParams.translation.x = destination.x;
00122 motionRequest.walkRequest.walkParams.translation.y = destination.y;
00123
00124 motionRequest.walkRequest.walkParams.rotation = 0;
00125 }
00126
00127 void GT2004BasicBehaviorGoaliePosition::execute()
00128 {
00129
00130 if (!basicBehaviorWasActiveDuringLastExecutionOfEngine) {
00131 goaliePose = lastRobotPose = robotPose;
00132 lastOdometry = odometryData;
00133 }
00134
00135
00136 if (maxSpeed == 0 )
00137 maxSpeed = 150;
00138
00139
00140 if (cutY == 0)
00141 cutY = 300;
00142 if (guardDirectToGoal == 0)
00143 guardDirectToGoal = 200;
00144
00145 if (guardLine == 0)
00146 guardLine = -150;
00147
00148
00149
00150
00151 Pose2D diffPose;
00152 diffPose.translation = robotPose.translation - lastGoaliePose.translation;
00153 diffPose.rotation = (robotPose.rotation - lastGoaliePose.rotation);
00154
00155 Pose2D diffOdo = odometryData - lastOdometry;
00156 Vector2<double> diffTranslation = (diffPose.translation*weightPose + diffOdo.translation*weightOdo);
00157
00158 goaliePose.translation += diffTranslation;
00159 goaliePose.rotation += (diffPose.rotation*weightPose) + (diffOdo.rotation*weightOdo);
00160
00161 CIRCLE(goaliePositionField, goaliePose.translation.x, goaliePose.translation.y, 200, 6, 0, Drawings::blue);
00162 CIRCLE(goaliePositionField, robotPose.translation.x, robotPose.translation.y, 180, 6, 0, Drawings::yellow);
00163
00164
00165
00166
00167
00168
00169
00170 Vector2<double> ballInWorldCoord = ballModel.getKnownPosition(
00171 BallModel::behaviorControlTimeAfterWhichCommunicatedBallsAreAccepted);
00172 double angleToBall = Geometry::angleTo(goaliePose,ballInWorldCoord);
00173
00174 Vector2<double> guardPosition;
00175
00176
00177 guardPosition.x = min(xPosOwnGroundline + guardDirectToGoal, xPosOwnGroundline - (xPosOwnGroundline-ballInWorldCoord.x)/2);
00178
00179
00180
00181
00182
00183 double deltaX = ballInWorldCoord.x - (xPosOwnGroundline+guardLine);
00184 double m = 0;
00185 if (deltaX != 0)
00186 m = ballInWorldCoord.y / deltaX;
00187 double n = ballInWorldCoord.y - m*ballInWorldCoord.x;
00188
00189 guardPosition.y = m*goaliePose.translation.x + n;
00190 ARROW(goaliePositionField, xPosOwnGroundline+guardLine,0,ballInWorldCoord.x,ballInWorldCoord.y,6,0,Drawings::gray);
00191 Vector2<double> pointToGuard = ballInWorldCoord - Vector2<double>(xPosOwnGroundline,0);
00192
00193 if (fabs(guardPosition.y) > cutY){
00194 guardPosition.y = min(cutY, guardPosition.y);
00195 guardPosition.y = max(-cutY, guardPosition.y);
00196 LINE(goaliePositionField,xPosOwnGroundline, guardPosition.y,-xPosOwnGroundline,guardPosition.y,6,0,Drawings::red);
00197 }
00198
00199 if (xPosOwnGroundline-ballInWorldCoord.x > -guardDirectToGoal){
00200 guardPosition.x = xPosOwnGroundline+67;
00201 LINE (goaliePositionField,xPosOwnGroundline, 1000,xPosOwnGroundline+67, -1000,6,0,Drawings::red);
00202 }
00203
00204
00205
00206 CIRCLE(goaliePositionField, guardPosition.x, guardPosition.y, 150, 3, 0, Drawings::red);
00207
00208
00209
00210 Vector2<double> direction = guardPosition - goaliePose.translation;
00211 Vector2<double> c1(cos(goaliePose.rotation), -sin(goaliePose.rotation)),
00212 c2(sin(goaliePose.rotation), cos(goaliePose.rotation));
00213 Matrix2x2<double> rotate(c1, c2);
00214
00215 direction = rotate*direction;
00216 if (direction.abs() > maxSpeed)
00217 {
00218 direction.normalize();
00219 direction *= maxSpeed;
00220 }
00221
00222 double rotation = angleToBall;
00223
00224
00225
00226
00227 if( (fabs(rotation) < 0.2) && (abs(int(direction.x)) < minXTrans) && (abs(int(direction.y)) < minYTrans)){
00228 motionRequest.motionType = MotionRequest::stand;
00229
00230 }
00231 else
00232 {
00233 motionRequest.motionType = MotionRequest::walk;
00234 motionRequest.walkRequest.walkType = WalkRequest::normal;
00235 if (!((abs(int(direction.x)) < minXTrans) && (abs(int(direction.y)) < minYTrans))){
00236 motionRequest.walkRequest.walkParams.translation.x = direction.x;
00237 motionRequest.walkRequest.walkParams.translation.y = direction.y;
00238 }
00239 if(!(fabs(rotation) < 0.2))
00240 motionRequest.walkRequest.walkParams.rotation = rotation;
00241 }
00242 lastGoaliePose = goaliePose;
00243 lastOdometry = odometryData;
00244 lastRobotPose = robotPose;
00245
00246 DEBUG_DRAWING_FINISHED(goaliePositionField);
00247
00248 }
00249
00250 void GT2004BasicBehaviorGoaliePositionReturn::execute()
00251 {
00252 Vector2<double> destination(x,y);
00253 const Vector2<double>& self = robotPose.translation;
00254
00255 double distanceToDestination = Geometry::distanceTo(self,destination);
00256
00257 double angleDifference = normalize(fromDegrees(destinationAngle) - robotPose.rotation);
00258
00259 if ((fabs(toDegrees(angleDifference))<25)&&(distanceToDestination<90))
00260 {
00261 motionRequest.motionType = MotionRequest::stand;
00262 }
00263 else
00264 {
00265 accelerationRestrictor.saveLastWalkParameters();
00266
00267 if (maxSpeed==0) maxSpeed = 200;
00268
00269 double maxTurnSpeed = fromDegrees(90);
00270
00271 double angleToDestination = Geometry::angleTo(robotPose,destination);
00272
00273 motionRequest.motionType = MotionRequest::walk;
00274 motionRequest.walkRequest.walkType = WalkRequest::normal;
00275
00276
00277
00278 if (distanceToDestination > 200)
00279 {
00280 motionRequest.walkRequest.walkParams.translation.x = cos(angleToDestination) * maxSpeed;
00281 motionRequest.walkRequest.walkParams.translation.y = sin(angleToDestination) * maxSpeed;
00282 }
00283 else
00284 {
00285 motionRequest.walkRequest.walkParams.translation.x =
00286 cos(angleToDestination) * maxSpeed*distanceToDestination/200;
00287 motionRequest.walkRequest.walkParams.translation.y =
00288 sin(angleToDestination) * maxSpeed*distanceToDestination/200;
00289 }
00290
00291
00292 motionRequest.walkRequest.walkParams.rotation = angleDifference * 2;
00293
00294 if (motionRequest.walkRequest.walkParams.rotation < -maxTurnSpeed)
00295 motionRequest.walkRequest.walkParams.rotation = -maxTurnSpeed;
00296
00297 if (motionRequest.walkRequest.walkParams.rotation > maxTurnSpeed)
00298 motionRequest.walkRequest.walkParams.rotation = maxTurnSpeed;
00299
00300
00301
00302 accelerationRestrictor.restrictAccelerations(150,150,100);
00303 }
00304
00305 }
00306 void GT2004BasicBehaviorTurnAroundPoint::execute()
00307 {
00308 double maxTurnSpeed = fromDegrees(240);
00309 if (forwardComponent == 0) forwardComponent = 200.0;
00310
00311 Vector2<double> destinationInWorldCoord(x,y);
00312 double angleToDestination = Geometry::angleTo(robotPose.getPose(), destinationInWorldCoord);
00313 double distance = Geometry::distanceTo(robotPose.getPose(), destinationInWorldCoord);
00314
00315
00316
00317 Vector2<double> destination(
00318 distance * cos(angleToDestination),
00319 distance * sin(angleToDestination));
00320
00321 double factor =
00322 Range<double>::Range(0, 1).
00323 limit((distance - radius) / (3*radius));
00324
00325 destination *= factor;
00326 destination.y += (leftRight > 0 ? 1 : -1)*(200)*(1-factor);
00327
00328 if (destination.x < forwardComponent)
00329 destination.x = forwardComponent;
00330
00331 double slowDownForMuchTurning =
00332 Range<double>::Range(0.1, 1).limit
00333 ((pi-fabs(angleToDestination))/pi);
00334
00335 motionRequest.motionType = MotionRequest::walk;
00336 motionRequest.walkRequest.walkType = WalkRequest::normal;
00337 motionRequest.walkRequest.walkParams.translation.x = destination.x * slowDownForMuchTurning;
00338 motionRequest.walkRequest.walkParams.translation.y = destination.y * slowDownForMuchTurning;
00339
00340
00341 motionRequest.walkRequest.walkParams.rotation =
00342 Range<double>::Range(-maxTurnSpeed, maxTurnSpeed).
00343 limit(angleToDestination + .6*(leftRight > 0 ? -1 : 1)*(1-factor));
00344 }
00345
00346 void GT2004BasicBehaviorGoForwardToPoint::execute()
00347 {
00348 accelerationRestrictor.saveLastWalkParameters();
00349
00350 Vector2<double> destination(x,y);
00351
00352 if (maxSpeed == 0) maxSpeed = 350;
00353
00354 double maxTurnSpeed = fromDegrees(120);
00355
00356 double angleToDestination = Geometry::angleTo(robotPose,destination);
00357
00358
00359 motionRequest.motionType = MotionRequest::walk;
00360 motionRequest.walkRequest.walkType = WalkRequest::normal;
00361
00362 motionRequest.walkRequest.walkParams.translation.x = cos(angleToDestination) * maxSpeed;
00363 motionRequest.walkRequest.walkParams.translation.y = sin(angleToDestination) * maxSpeed;
00364
00365 motionRequest.walkRequest.walkParams.rotation = angleToDestination*2;
00366 if (motionRequest.walkRequest.walkParams.rotation > maxTurnSpeed)
00367 {
00368 motionRequest.walkRequest.walkParams.rotation = maxTurnSpeed;
00369 }
00370 if (motionRequest.walkRequest.walkParams.rotation < -maxTurnSpeed)
00371 {
00372 motionRequest.walkRequest.walkParams.rotation = -maxTurnSpeed;
00373 }
00374
00375 accelerationRestrictor.restrictAccelerations(300,300,200);
00376
00377 }
00378
00379 void GT2004BasicBehaviorGoToPointAndAvoidObstacles::execute()
00380 {
00381 double
00382 obstacleAvoidanceDistance,
00383 distanceToDestination = Geometry::distanceTo(robotPose.getPose(),Vector2<double>(x,y)),
00384 angleToDestination = Geometry::angleTo(robotPose.getPose(),Vector2<double>(x,y)),
00385 targetSpeed, targetAngle,
00386 freeSpaceInFront,
00387 widthOfCorridor,
00388 minSpeed,
00389 distanceBelowWhichObstaclesAreAvoided;
00390
00391
00392 if (SystemCall::getTimeSince(lastTimeExecuted) > 500)
00393 {
00394 speedX.resetTo(100);
00395 speedY.resetTo(0);
00396 turnSpeed.resetTo(0);
00397 }
00398 lastTimeExecuted = SystemCall::getCurrentSystemTime();
00399
00400 if (maxSpeed == 0) maxSpeed = 350;
00401
00402
00403
00404
00405
00406
00407
00408
00409
00410
00411
00412
00413
00414
00415 widthOfCorridor = 250;
00416
00417 distanceBelowWhichObstaclesAreAvoided = 500;
00418 minSpeed = -150;
00419
00420
00421 obstacleAvoidanceDistance = Range<double>::Range(0, distanceBelowWhichObstaclesAreAvoided).limit(distanceToDestination);
00422
00423
00424
00425 freeSpaceInFront = obstaclesModel.getDistanceInCorridor(0.0, widthOfCorridor);
00426 targetSpeed = Range<double>::Range(minSpeed, maxSpeed).limit(freeSpaceInFront - 300);
00427
00428
00429 double leftForward = obstaclesModel.getTotalFreeSpaceInSector(.7, 1.39, obstacleAvoidanceDistance);
00430 double rightForward = obstaclesModel.getTotalFreeSpaceInSector(-.7, 1.39, obstacleAvoidanceDistance);
00431 targetAngle = (leftForward - rightForward) / (leftForward + rightForward);
00432
00433
00434 double nextFree = obstaclesModel.getAngleOfNextFreeSector(pi/4, angleToDestination, (int )obstacleAvoidanceDistance);
00435
00436
00437 if (obstaclesModel.getDistanceInCorridor(nextFree, widthOfCorridor) > obstacleAvoidanceDistance)
00438 {
00439 if (fabs(nextFree) < .8)
00440 targetAngle = nextFree;
00441 }
00442
00443
00444 if (obstaclesModel.getDistanceInCorridor(angleToDestination, widthOfCorridor) > obstacleAvoidanceDistance)
00445 targetAngle = angleToDestination;
00446
00447
00448
00449
00450
00451
00452
00453
00454
00455 if (targetSpeed == maxSpeed)
00456 targetSpeed *= Range<double>::Range(0.1, 1).limit(
00457 (pi-fabs(targetAngle))/pi );
00458
00459
00460 if (distanceToDestination < 150)
00461 {
00462 motionRequest.motionType = MotionRequest::stand;
00463 }
00464 else
00465 {
00466 motionRequest.motionType = MotionRequest::walk;
00467 motionRequest.walkRequest.walkType = WalkRequest::normal;
00468 motionRequest.walkRequest.walkParams.translation.x = speedX.approximateVal(targetSpeed);
00469 motionRequest.walkRequest.walkParams.translation.y = speedY.approximateVal(0);
00470 motionRequest.walkRequest.walkParams.rotation = turnSpeed.approximateVal(2*targetAngle);
00471 }
00472 }
00473
00474
00475
00476 void GT2004BasicBehaviorGoToPoint::execute()
00477 {
00478 accelerationRestrictor.saveLastWalkParameters();
00479
00480 Vector2<double> destination(x,y);
00481 const Vector2<double>& self = robotPose.translation;
00482
00483 if (maxSpeed==0) maxSpeed = 350;
00484 if (maxSpeedY==0) maxSpeedY = 350;
00485
00486 double maxTurnSpeed = fromDegrees(40);
00487
00488 double distanceToDestination = Geometry::distanceTo(self,destination);
00489
00490 double angleDifference = normalize(fromDegrees(destinationAngle) - robotPose.rotation);
00491
00492 double angleToDestination = Geometry::angleTo(robotPose,destination);
00493
00494
00495 motionRequest.motionType = MotionRequest::walk;
00496 motionRequest.walkRequest.walkType = static_cast<WalkRequest::WalkType>(static_cast<int>(walkType));
00497
00498
00499 double estimatedTimeToReachDestination;
00500 if (distanceToDestination > 200)
00501 {
00502 estimatedTimeToReachDestination = (distanceToDestination+200) / maxSpeed;
00503
00504 motionRequest.walkRequest.walkParams.translation.x = cos(angleToDestination) * maxSpeed;
00505 motionRequest.walkRequest.walkParams.translation.y = sin(angleToDestination) * maxSpeed;
00506 }
00507 else
00508 {
00509 estimatedTimeToReachDestination = 2*distanceToDestination / maxSpeed + 2*fabs(angleDifference) / maxTurnSpeed;
00510
00511
00512
00513 if (distanceToDestination > 30)
00514 {
00515 motionRequest.walkRequest.walkParams.translation.x =
00516 cos(angleToDestination) * maxSpeed * distanceToDestination/200;
00517 motionRequest.walkRequest.walkParams.translation.y =
00518 sin(angleToDestination) * maxSpeed * distanceToDestination/200;
00519 }
00520 else
00521 {
00522 motionRequest.walkRequest.walkParams.translation.x = 0;
00523 motionRequest.walkRequest.walkParams.translation.y = 0;
00524 }
00525 }
00526
00527
00528 if (estimatedTimeToReachDestination != 0)
00529 {
00530
00531
00532
00533
00534
00535
00536 if (fabs(toDegrees(angleDifference)) > 20)
00537 {
00538 motionRequest.walkRequest.walkParams.rotation =
00539 angleDifference / estimatedTimeToReachDestination;
00540 motionRequest.walkRequest.walkParams.rotation =
00541 min (maxTurnSpeed, motionRequest.walkRequest.walkParams.rotation);
00542 motionRequest.walkRequest.walkParams.rotation =
00543 max (-maxTurnSpeed, motionRequest.walkRequest.walkParams.rotation);
00544 }
00545 else
00546 {
00547 motionRequest.walkRequest.walkParams.rotation = 2*angleDifference;
00548 }
00549
00550
00551
00552
00553
00554
00555 motionRequest.walkRequest.walkParams.translation.y =
00556 min (maxSpeedY, motionRequest.walkRequest.walkParams.translation.y);
00557
00558 if ((fabs(toDegrees(angleDifference))<angleRemain)&&(distanceToDestination<distanceRemain))
00559 {
00560 motionRequest.walkRequest.walkParams.translation.x = 0;
00561 motionRequest.walkRequest.walkParams.translation.y = 0;
00562 motionRequest.walkRequest.walkParams.rotation = 0;
00563 }
00564
00565 accelerationRestrictor.restrictAccelerations(150,150,100);
00566
00567 }
00568 else
00569 {
00570 motionRequest.walkRequest.walkParams.translation.x = 0;
00571 motionRequest.walkRequest.walkParams.translation.y = 0;
00572 motionRequest.walkRequest.walkParams.rotation = 0;
00573 }
00574 }
00575
00576
00577
00578 void GT2004BasicBehaviorMoveTheBridge::execute()
00579 {
00580 outgoingBehaviorTeamMessage.walkRequest.type = type;
00581 outgoingBehaviorTeamMessage.walkRequest.x = x;
00582 if (outgoingBehaviorTeamMessage.walkRequest.x > 50)
00583 outgoingBehaviorTeamMessage.walkRequest.x = 50;
00584 if (outgoingBehaviorTeamMessage.walkRequest.x < -50)
00585 outgoingBehaviorTeamMessage.walkRequest.x = -50;
00586
00587 outgoingBehaviorTeamMessage.walkRequest.y = y;
00588 if (outgoingBehaviorTeamMessage.walkRequest.y > 50)
00589 outgoingBehaviorTeamMessage.walkRequest.y = 50;
00590 if (outgoingBehaviorTeamMessage.walkRequest.y < -50)
00591 outgoingBehaviorTeamMessage.walkRequest.y = -50;
00592
00593 outgoingBehaviorTeamMessage.walkRequest.rotation = rotation;
00594 if (outgoingBehaviorTeamMessage.walkRequest.rotation > 2)
00595 outgoingBehaviorTeamMessage.walkRequest.rotation = 2;
00596 if (outgoingBehaviorTeamMessage.walkRequest.rotation < -2)
00597 outgoingBehaviorTeamMessage.walkRequest.rotation = -2;
00598 }
00599
00600 void GT2004BasicBehaviorDogAsJoystick::execute()
00601 {
00602
00603
00604
00605
00606
00607
00608 double tmpr1 = toDegrees(fromMicroRad(sensorDataBuffer.lastFrame().data[SensorData::legFR1]));
00609 double tmpr2 = toDegrees(fromMicroRad(sensorDataBuffer.lastFrame().data[SensorData::legFR2]));
00610
00611 double tmpl1 = toDegrees(fromMicroRad(sensorDataBuffer.lastFrame().data[SensorData::legFL1]));
00612 double tmpl2 = toDegrees(fromMicroRad(sensorDataBuffer.lastFrame().data[SensorData::legFL2]));
00613
00614
00615 outgoingBehaviorTeamMessage.walkRequest.type = 0;
00616 outgoingBehaviorTeamMessage.walkRequest.x = (tmpr1+tmpl1)*2.5;
00617 outgoingBehaviorTeamMessage.walkRequest.y = (tmpr2-tmpl2)*2.5;
00618
00619 outgoingBehaviorTeamMessage.walkRequest.rotation = (tmpl1-tmpr1)/20 ;
00620 }
00621
00622
00623
00624
00625
00626
00627
00628
00629
00630
00631
00632
00633
00634
00635
00636
00637
00638
00639
00640
00641
00642
00643
00644
00645
00646
00647
00648
00649
00650
00651
00652
00653
00654
00655
00656
00657
00658
00659
00660
00661
00662
00663
00664
00665
00666
00667
00668
00669
00670
00671
00672
00673
00674
00675
00676
00677
00678
00679
00680
00681
00682
00683
00684
00685
00686
00687
00688
00689
00690
00691
00692
00693
00694
00695
00696
00697
00698
00699
00700
00701
00702
00703
00704
00705
00706
00707
00708
00709
00710
00711
00712
00713
00714
00715
00716
00717
00718
00719
00720
00721
00722
00723
00724
00725
00726
00727
00728
00729
00730
00731
00732
00733
00734
00735
00736
00737
00738
00739
00740
00741
00742
00743
00744
00745
00746
00747