

Acquisitions and Bibliographic Services Branch

395 Weilington Street Ottawa, Ontario K1A 0N4 Bibliothèque nationale du Canada

Direction des acquisitions et des services bibliographiques

395, rue Wellington Ottawa (Ontario) K1A 0N4

Your file Notice references

Our frei Notic reference

## NOTICE

The quality of this microform is heavily dependent upon the quality of the original thesis submitted for microfilming. Every effort has been made to ensure the highest quality of reproduction possible.

dépend grandement de la qualité de la thèse soumise au microfilmage. Nous avons tout fait pour assurer une qualité supérieure de reproduction.

**AVIS** 

La qualité de cette microforme

If pages are missing, contact the university which granted the degree.

S'il manque des pages, veuillez communiquer avec l'université qui a conféré le grade.

Some pages may have indistinct print especially if the original pages were typed with a poor typewriter ribbon or if the university sent us an inferior photocopy.

La qualité d'impression de certaines pages peut laisser à désirer, surtout si les pages originales ont été dactylographiées à l'aide d'un ruban usé ou si l'université nous a fait parvenir une photocopie de qualité inférieure.

Reproduction in full or in part of this microform is governed by the Canadian Copyright Act, R.S.C. 1970, c. C-30, and subsequent amendments.

La reproduction, même partielle, de cette microforme est soumise à la Loi canadienne sur le droit d'auteur, SRC 1970, c. C-30, et ses amendements subséquents.



# The Design and Implementation of an Advanced Robot Controller

Jonathan Neil Brodkin

A Thesis

in

The Department

of

Electrical and Computer Engineering

Presented in Partial Fulfillment of the Requirements for the Degree of Master of Applied Science at Concordia University Montréal, Québec, Canada

November 1990

© Jonathan Neil Brodkin, 1990



Acquisitions and Bibliographic Services Branch

395 Wellington Street Ottawa, Ontario K1A 0N4 Bibliothèque nationale du Canada

Direction des acquisitions et des services bibliographiques

395, rue Wellington Ottawa (Ontario) K1A 0N4

Your life. Votte référence

Our Ne Notre référence

The author has granted an irrevocable non-exclusive licence allowing the National Library of Canada to reproduce, loan, distribute or sell copies of his/her thesis by any means and in any form or format, making this thesis available to interested persons.

L'auteur a accordé une licence irrévocable et non exclusive Bibliothèque permettant à la nationale Canada du reproduire, prêter, distribuer ou vendre des copies de sa thèse de quelque manière et sous quelque forme que ce soit pour mettre des exemplaires de cette thèse disposition des à la personnes intéressées.

The author retains ownership of the copyright in his/her thesis. Neither the thesis nor substantial extracts from it may be printed or otherwise reproduced without his/her permission. L'auteur conserve la propriété du droit d'auteur qui protège sa thèse. Ni la thèse ni des extraits substantiels de celle-ci ne doivent être imprimés ou autrement reproduits sans son autorisation.

ISBN 0-315-90942-0



# ABSTRACT

# The Design and Implementation of an Advanced Robot Controller

# Jonathan N. Brodkin

This thesis describes an architecture for implementing an Advanced Robot Controller (ARC). The ARC has been designed and built as a replacement for the controllers of conventional industrial robotic systems. The need for such a device arises because most industrial robots are constrained by tileir controllers' hardware and software to performing in a simple, pre-defined manner. The ARC hardware consists of a hierarchically-designed dual-processor operating in a master-slave configuration and communicating via dual-port RAM, a bank of quadrature decoders/counters, and a bank of digital to analog converters. A software library has been established consisting of various control and path planning algorithms. This library also contains a set of functions to facilitate algorithm coding and testing. The ARC was integrated with an IBM 7545 Manufacturing System and proved extremely successful in achieving tight control over the 7545's four joints.

# **ACKNOWLEDGEMENTS**

In submitting this thesis I must acknowledge Dr. R. V. Patel for the guidance and support that he provided to me throughout the course of my research and writing; but above all I must thank Dr. Patel for his unwavering trust in my abilities which contributed to my motivation and confidence in this endeavor.

There are several others whose assistance was most appreciated at particular junctures, including hardware suggestions, circuit construction, programming, debugging and text review: Guy Gosselin, Harvey Brodkin, Alan Robins, Claude-Marie Lafforgue, Henry Polley, Mike Stashin, Vladimir Zeman, Venkatram Pramod, and Claude Tessier.

Finally, I would like to thank my family and friends for their constant and enthusiastic support and encouragement.

# TABLE OF CONTENTS

| LIST       | OF TABLESvii                                                                                                                                                                                            | i           |
|------------|---------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------|-------------|
| LIST       | OF FIGURESi                                                                                                                                                                                             | x           |
|            | PTER 1 RODUCTION                                                                                                                                                                                        | 1           |
| 1.1        | Industrial Robotic Systems                                                                                                                                                                              | 1           |
| 1.2        | Summary of the Research Work                                                                                                                                                                            | 3           |
| 1.3        | Thesis Organization                                                                                                                                                                                     | 4           |
| CHA<br>THE | PTER 2 IBM 7545 MANUFACTURING SYSTEM                                                                                                                                                                    | 5           |
| 2.1        | Introduction                                                                                                                                                                                            | 5           |
| 2.2        | The IBM 7545 Manipulator                                                                                                                                                                                | 7           |
|            | 2.2.1       Motor Location and Transmission System       1         2.2.2       DC Servo Motors and Drivers       1         2.2.3       Incremental Encoders       1         2.2.4       Sensors       1 | 11<br>14    |
| 2.3        | Summary1                                                                                                                                                                                                | 17          |
|            | PTER 3 DWARE DESCRIPTION FOR THE ADVANCED ROBOT CONTROLLER1                                                                                                                                             | 18          |
| 3.1        | Introduction                                                                                                                                                                                            | 8           |
| 3.2        | Master Processor and Dual-Port RAM (DPR)                                                                                                                                                                | <b>X</b> O  |
| 3.3        | Slave Processor                                                                                                                                                                                         | 23          |
|            | 3.3.1 The EV80C196KA Interface                                                                                                                                                                          | 23          |
| 3.4        | Digital To Analog Conversion                                                                                                                                                                            | <b>X</b> 6  |
|            | 3.4.1 Digital Circuit Details                                                                                                                                                                           |             |
| 3.5        | Position Counters                                                                                                                                                                                       |             |
| 3.6        | Home Sensors, Encoder Index and Dual-Port RAM Connection                                                                                                                                                | <b>3</b> 3  |
| 3.7        | Hardware Implementation                                                                                                                                                                                 | <b>3</b> 6i |
| 3.8        | Summary and Future Work                                                                                                                                                                                 | 36          |

|            | TWARI                                                       | 4 E DESCRIPTION FOR THE ADVANCED ROBOT CONTROLLER                                                                                                                                                         | 37                                     |
|------------|-------------------------------------------------------------|-----------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------|----------------------------------------|
| 4.1        | Introd                                                      | duction                                                                                                                                                                                                   | 37                                     |
| 4.2        | Dual-I                                                      | Port RAM Utilization                                                                                                                                                                                      | 38                                     |
|            | 4.2.1<br>4.2.2<br>4.2.3<br>4.2.4<br>4.2.5<br>4.2.6<br>4.2.7 | Sample_Period Register. Timing_A Register. Timing_B Register. Command Register. Error Register Actual_Position Registers Torque Registers                                                                 | 40<br>40<br>41<br>41                   |
| 4.3        | Slave                                                       | Processor Software                                                                                                                                                                                        |                                        |
|            | 4.3.1<br>4.3.2<br>4.3.3<br>4.3.4                            | Main Program  Find-HOME Mode  Control Mode  Additional Procedures  4.3.4.1 Get_Joint_Positions  4.3.4.2 Delay  4.3.4.3 Zero_DACs  4.3.4.4 Joint_Overrun_Check  4.3.4.5 Check_Torques  4.3.4.6 Torques_Out | 44<br>46<br>50<br>51<br>52<br>53<br>54 |
| 4.4        | Maste                                                       | er Processor Software                                                                                                                                                                                     |                                        |
|            | 4.4.1<br>4.4.2<br>4.4.3<br>4.4.4                            | Master Sequence of Operation Servo Control Loop Robot Control Programs 4.4.3.1 PD Control 4.4.3.2 Decentralized Adaptive Control Path Planning Programs 4.4.4.1 The Cycloid Function 4.4.4.2 Cubic Spline | 60<br>61<br>62<br>63<br>63             |
| 4.5        | Summ                                                        | nary and Future Work                                                                                                                                                                                      | 65                                     |
| CHA<br>EXP | FTER S                                                      | 5<br>ENTAL RESULTS                                                                                                                                                                                        | 67                                     |
|            | PTER (                                                      | 6<br>IONS                                                                                                                                                                                                 | 102                                    |
| REF        | ERENC                                                       | CES                                                                                                                                                                                                       | 104                                    |
|            | ENDIX<br><b>754</b> 5                                       | A S WIRING DIAGRAMS                                                                                                                                                                                       | 106                                    |
|            | ENDIX                                                       | B<br>OTOR RATINGS AND SPECIFICATIONS                                                                                                                                                                      | 112                                    |

| APPENDIX C CALIBRATION                             | 114 |
|----------------------------------------------------|-----|
| APPENDIX D<br>PS/2 TIMING DIAGRAMS                 | 116 |
| APPENDIX E DUAL-PORT RAM TIMING WAVEFORMS          | 122 |
| APPENDIX F EV80C196KA SCHEMATIC DIAGRAM            | 126 |
| APPENDIX G<br>80C196KA BUS TIMING                  | 130 |
| APPENDIX H AD667 BLOCK DIAGRAM AND TIMING DIAGRAMS | 132 |
| APPENDIX I<br>HCTL-1000 BLOCK AND TIMING DIAGRAMS  | 133 |
| APPENDIX J<br>SLAVE SOURCE CODE                    | 136 |
| APPENDIX K MASTER SOURCE CODE                      | 143 |

# LIST OF TABLES

| 2.1 | 7545 Joint Position Resolutions        | 14 |
|-----|----------------------------------------|----|
| 3.1 | Digital to Analog Converter Addresses  | 2  |
| 3.2 | Memory Map for the HCTL-1000s          | 3  |
| 4.1 | Dual-Port RAM Register Reference Table | 3  |

# LIST OF FIGURES

| 2.1  | IBM 7545 Manufacturing System                           | . 6          |
|------|---------------------------------------------------------|--------------|
| 2.2  | Block Diagram of the IBM 7545 Manufacturing System      | . 6          |
| 2.3  | IBM 7545 Manipulator                                    | . 8          |
| 2.4  | IBM 7545 Manipulator Work Envelope                      | . 9          |
| 2.5  | Servopack Internal Block Diagram                        | 12           |
| 2.6  | Modified Servopack, Encoder and HOME Sensor Connections | 13           |
| 3.1  | Block Diagram of the ARC Hardware                       | 19           |
| 3.2  | PS/2 - Dual-Port Ram Interface Schematic                | 21           |
| 3.3  | 80C196KA Interface Schematic                            | 25           |
| 3.4  | Digital to Analog Converter Schematic                   | <b>2</b> 8   |
| 3.5  | HCTL-1000 Schematic                                     | 30           |
| 3.6  | Index and HOME Sensor Schematic                         | . 35         |
| 4.1  | Block Diagram of the ARC Software                       | . 33         |
| 4.2  | Flow Chart for the Slave's Main Program                 | . 43         |
| 4.3  | Flow Chart for the Find-HOME Mode                       | . 45         |
| 4.4  | Flow Chart for the Control Mode                         | . 48         |
| 4.5  | Flow Chart for Delay Procedure                          | . 52         |
| 4.6  | Flow Chart for Zero_DACs Procedure                      | . 53         |
| 4.7  | Flow Chart for Joint_Overrun_Check Procedure            | . 54         |
| 4.8  | Flow Chart for Check_Torques Procedure                  | . <b>5</b> 5 |
| 4.9  | Flow Chart for Torques_Out Procedure                    | . 56         |
| 4.10 | Sequence of Operation of the Master Software            | . 58         |
| 5.1  | Joint 1 Tracking Errors, PD Control, Experiment 1       | . 70         |
| 5.2  | Joint 2 Tracking Errors, PD Control, Experiment 1       | . 71         |

| 5.3  | Torques, PD Control, Experiment 1                                            | . <b>7</b> 2 |
|------|------------------------------------------------------------------------------|--------------|
| 5.4  | Joint 1 Tracking Errors, Adaptive Control, Experiment 1                      | . <b>7</b> 3 |
| 5.5  | Joint 2 Tracking Errors, Adaptive Control, Experiment 1                      | , 74         |
| 5.6  | Torques, Adaptive Control, Experiment 1                                      | . 75         |
| 5.7  | Desired Path of Joints 1 and 2 for Experiment 2 Shown in Cartesian Space     | . 76         |
| 5.8  | Joint 1 Tracking Errors, PD Control, Experiment 2                            | . 77         |
| 5.9  | Joint 2 Tracking Errors, PD Control, Experiment 2                            | . 78         |
| 5.10 | Joint Z Tracking Errors, PD Control, Experiment 2                            | . <b>7</b> 9 |
| 5.11 | Joint Roll Tracking Errors, PD Control, Experiment 2                         | . 80         |
| 5.12 | Torques, PD Control, Experiment 2                                            | . 81         |
| 5.13 | Joint 1 Tracking Errors, Adaptive Control, Experiment 2                      | . 82         |
| 5.14 | Joint 2 Tracking Errors, Adaptive Control, Experiment 2                      | . 83         |
| 5.15 | Joint Z Tracking Errors, Adaptive Control, Experiment 2                      | . 84         |
| 5.16 | Joint Roll Tracking Errors, Adaptive Control, Experiment 2                   | . 85         |
| 5.17 | Torques, Adaptive Control, Experiment 2                                      | . 86         |
| 5.18 | Joint 1 Tracking Errors, Adaptive Control, Experiment 3, Three Sec.  Move    | . 87         |
| 5.19 | Joint 2 Tracking Errors, Adaptive Control, Experiment 3, Three Sec.  Move    | . <b>8</b> 8 |
| 5.20 | Joint Z Tracking Errors, Adaptive Control, Experiment 3, Three Sec.  Move    | . 89         |
| 5.21 | Joint Roll Tracking Errors, Adaptive Control, Experiment 3, Three Sec.  Move | . 90         |
| 5.22 | Torques, Adaptive Control, Experiment 3, Three Sec. Move                     | 91           |
| 5.23 | Joint 1 Tracking Errors, Adaptive Control, Experiment 3, Two Sec.  Move      | . 92         |
| 5.24 | Joint 2 Tracking Errors, Adaptive Control, Experiment 3, Two Sec.  Move      | . 93         |
| 5.25 | Joint Z Tracking Errors, Adaptive Control, Experiment 3, Two Sec.  Move      | 94           |

| 5.26 | Joint Roll Tracking Errors, Adaptive Control, Experiment 3, Two Sec.  Move | 95  |
|------|----------------------------------------------------------------------------|-----|
| 5.27 | Torques, Adaptive Control, Experiment 3, Two Sec. Move                     | 96  |
| 5.28 | Joint 1 Tracking Errors, Adaptive Control, Experiment 3, One Sec.  Move    | 97  |
| 5.29 | Joint 2 Tracking Errors, Adaptive Control, Experiment 3, One Sec.  Move    | 98  |
| 5.30 | Joint Z Tracking Errors, Adaptive Control, Experiment 3, One Sec.  Move    | 99  |
| 5.31 | Joint Roll Tracking Errors, Adaptive Control, Experiment 3, One Sec.  Move | 100 |
| 5.32 | Torques, Adaptive Control, Experiment 3, One Sec. Move                     | 101 |

# **CHAPTER 1**

#### INTRODUCTION

#### 1.1 INDUSTRIAL ROBOTIC SYSTEMS

Since the early 1960s, robotic systems have steadily been replacing traditional automation techniques. Successful applications of robot manipulators have included manufacturing and assembly, materials handling and inspection in fixed automation environments. This type of automation involves the repetition of a specific sequence of operations for the continuous high-volume production of identical or nearly identical parts. Industrial robots intended for use in the fixed automation environment have been designed to perform relatively simple tasks in a preconceived manner. To perform these tasks, the robots require few if any sensors and practically no intelligent decision-making ability. Controllers for these systems, therefore, are based on weak processing hardware and unsophisticated control and trajectory generation software. These types of robots are generally referred to as "dumb" robots since they are inflexible to changes in their workspace and product design.

A characteristic of dumb robots is their high-level, application-oriented user-interface which enables the operator to specify a small variety of "move-type" commands. With this type of user-interface, the operator has little control over the desired trajectory and virtually no control over the servoing of the joint motors. In some cases, this limitation may adversely affect the robot's efficiency. For example, there exist dumb robotic systems whose application languages do not permit the specification of via points; to negotiate around an obstacle, the user must specify a piecewise path and the robot stops at the end of each segment.

Although there will always be applications for dumb robots, there is presently a growing demand for industrial robots which can operate in a flexible automation environment. This type of environment is dynamically complex, often containing moving obstacles and/or other manipulators sharing the same workspace. To perform in this environment, a robot must possess a significant level of intelligence as well as an assortment of advanced sensors. Although there will always be applications for which dumb robots are well suited, many may soon prove to be obsolete as the demand for more efficient and intelligent industrial robotic systems increases. This is unfortunate when one considers that relatively expensive robotic manipulators will go to waste simply because of the shortcomings of their controllers.

In addition to the industrial domain, robots are frequently found in research environments. Here, they are being used for testing modern advanced control strategies for the purpose of both basic research and prototype development. Once again, the limitations associated with the controllers of dumb robotic systems render them inadequate to serve in such a domain. The research presented in this thesis was conceived in order to overcome this problem.

The IBM 7545 Manufacturing System that was used in our research is representative of a large class of industrial robotic systems and can be classified as a dumb robot. Although the 7545 system is an older generation IBM system, its kirematic configuration is essentially identical to that of current IBM robotic systems; the difference being the intelligence of the controllers. Because the intelligence of the 7545 controller is markedly lower, it does not possess some of the features that have been incorporated into latest generation systems, e.g., the ability to specify desired trajectories. What these two generations of systems do share, however, is their inability to evolve as advanced requirements arise. This inability can be attributed to the design philosophy of the 7545

controller which like most robotic controllers makes it inherently difficult to modify the hardware and software.

### 1.2 SUMMARY OF THE RESEARCH WORK

One solution to overcoming the shortcomings mentioned above is to bypass the manufacturer-supplied controller with an alternate controller possessing sufficient processing power to execute various advanced robot control strategies and an architecture which would support future enhancement. This controller could then be programmed for use in an industrial or research environment. This was the solution adopted for the IBM 7545 Manufacturing System.

The Advanced Robot Controller (ARC) which has been designed and implemented incorporates a master/slave paradigm that assigns to the slave processor the tedious yet essential elements of implementing servo control (e.g., robot state acquisition and limit checking such as joint overruns and excessive torque demands). The slave takes over a significant percentage of the computations required for servo control and thus allows more computationally complex control strategies to be executed on the master. Although the results presented in this thesis reflect work undertaken on a particular commercial robotic system, the approach taken is easily applicable to virtually any robot. It is worth mentioning that an important constraint in the design and implementation of the ARC is that the resulting hardware/software is inexpensive (a small fraction of the cost of the robot) and relatively easy to develop and maintain.

Other researchers have recently reported developments of specialized robotic controllers for the same purpose [1-4]. The systems in [1] and [2] are based on simple single processor architectures. In [1] a robot controller based on an Intel 310 (6 MHz, 80286 processor) running XENIX and with its servo control algorithm linked to the Kernel was introduced. The controller was implemented and evaluated on a six degrees of freedom

PUMA 560 robot and was able to execute PD control for the PUMA's six joints at a servo rate of 100 Hz [5]. In [2] a specialized computer-robot interface was designed. The interface was used to link an IBM AT to a Mitsubishi RM-501 robot for the evaluation of simple and advanced control strategies. A 6.0 millisecond sampling period was achieved for PD control of the robot's first two joints [6]. The methodology and design philosophies which were adopted by us independently of [1] and [2] are in accordance with the suggestions found in the conclusions of these references. It should be noted that the cost of the ARC is of the same order as those for the architectures proposed in [1] and [2]. The systems described in [3] and [4] consist of host SUN workstations directing the efforts of multiple (3-5) VME-based 68000-series processors. These systems exhibit superior performance but at very significantly higher cost.

#### 1.3 THESIS ORGANIZATION

An introduction to the limitations associated with typical commercial industrial robot systems has been given in this chapter. Chapter 2 describes the IBM 7545 Manufacturing System which was used in this research. The required hardware modifications to the 7545 system are also described. Chapter 3 presents a complete ARC hardware description, justifying the ARC's adequacy for controlling the 7545 manipulator. The ARC software description is given in Chapter 4. This covers the high-level master software consisting of control and path planning algorithms as well as the low-level slave algorithms that were developed to increase the performance of the system. Suggestions for future extensions are included in both Chapters 3 and 4. Chapter 5 presents the results of experiments that were designed to show the ARC's effectiveness in executing various control strategies, including proportional-derivative and decentralized adaptive controllers and various path generation schemes. Concluding remarks are given in Chapter 6.

# **CHAPTER 2**

#### THE IBM 7545 MANUFACTURING SYSTEM

## 2.1 INTRODUCTION

The IBM 7545 Manufacturing System (Figure 2.1) consists of a control unit, an operator control panel, a manipulator which consists of a four-joint, DC servo-actuated, Selective Compliance Assembly Robot Arm (SCARA) [7], and an application programming device (not shown). The contents of these components are shown in the block diagram in Figure 2.2.

The control unit houses a power supply, drivers for the servo motors, a motor control board (MTCB), and an interface and power distribution board (relay board). The MTCB contains a Z80 microprocessor, memory, communications interface circuits, and robot motion control circuits.

The 7545 system is pre-programmed by the user for a particular application using the AML/E (A Manufacturing Language/Editor) programming language which runs on the application programming device (e.g., IBM PC or compatible). The application program is compiled by AML/E and then downloaded to the MTCB and executed. AML/E supports simplistic high-level commands such as "move" and "grasp". All trajectory planning and joint motor control is performed by the MTCB in a predefined fashion. Inherent in the MTCB design are the shortcomings described in Chapter 1; consequently the MTCB is the sole 7545 component which needs to be bypassed. All other 7545 components can remain operationally intact.



Figure 2.1. IBM 7545 Manufacturing System [7].



Figure 2.2. Block Diagram of the IBM 7545 Manufacturing System [7].

This chapter provides the necessary information on the 7545 system to effectively bypass the MTCB with the ARC that has been designed. This includes the 7545 specifications as well as the required modifications to various 7545 system components.

#### 2.2 THE IBM 7545 MANIPULATOR

The manipulator of the 7545 Manufacturing System (Figure 2.3) is a four-degrees-of-freedom mechanism with rigid links. The first two revolute joints of the arm, the shoulder  $(\theta_1)$  and the elbow  $(\theta_2)$ , give two degrees of freedom in the horizontal x-y plane.  $\theta_1$  is measured with respect to the x-axis (see Figure 2.3) and its range of motion is 0-200°.  $\theta_2$  is measured with respect to the radial axis of link 1 and has a range of motion of 0-135°.

Located at the end of the arm is a prismatic joint giving one degree of freedom along the vertical z-axis. Attached to the end of this z-axis shaft is a pneumatic gripper. In its uppermost position, the gripper lies in the x-y plane and the joint variable Z (uppercase denotes joint variable) is zero. The shaft can extend 250 mm in the negative z direction. This is the only joint motion affected by gravity.

The Roll joint consists of the revolution of the z-axis shaft (gripper) providing one degree of orientational freedom. The Roll angle,  $\theta_{roll}$ , is measured with respect to the x-axis and as will be shown in the following section, is independent of  $\theta_1$ ,  $\theta_2$ , and Z. The range of motion for this joint is from -180° to +180°.

The forward kinematic equations for the manipulator are:

$$\mathbf{x} = \mathbf{l}_2 \cos(\theta_1 + \theta_2) + \mathbf{l}_1 \cos \theta_1$$

$$\mathbf{y} = \mathbf{l}_2 \sin(\theta_1 + \theta_2) + \mathbf{l}_1 \sin \theta_1$$

$$\mathbf{z} = -\mathbf{h}$$
(2.1)

where  $l_1 = 400$  mm and  $l_2 = 250$  mm are the lengths of links 1 and 2, respectively.



Figure 2.3. IBM 7545 Manipulator [7].

The inverse kinematic equations are:

$$\theta_{1} = \operatorname{Atan2}(y, x) + \operatorname{Atan2}(\pm \sqrt{x^{2} \cdot y^{2} \cdot \omega^{2}}, \omega)$$

$$\theta_{2} = \operatorname{Atan2}(-x \sin \theta_{1} + y \cos \theta_{1}, x \cos \theta_{1} + y \sin \theta_{1} \cdot l_{1})$$
(2.2)

The dimensions of the manipulator's work envelope are shown in Figure 2.4. The manipulator's HOME (reference) position is located in joint space at  $\theta_1 = \theta_2 = \theta_{roll} = 0^\circ$ , Z = 0 mm, or in Cartesian space at x = 650 mm, y = z = 0 mm.



Figure 2.4. IBM 7545 Manipulator Work Envelope [7].

The manipulator has no positional redundancies, i.e., for any point in the work envelope, there exists a unique set of joint values to position the gripper at that point. Orientationally, there is redundancy in that the gripper can rotate a full 360 degrees, i.e.,  $\theta_{roll} = -180^{\circ}$  is equivalent to  $\theta_{roll} = +180^{\circ}$ .

As indicated in the block diagram in Figure 2.2, each joint of the 7545 manipulator is equipped with a DC servo motor and driver (the latter located in the control unit), a transmission device, an incremental encoder, and various sensors. These components are described in the sections that follow. For reference, the wiring diagrams of the connections between these components and the MTCB are included in Appendix A.

## 2.2.1 Motor Location and Transmission System

Each link of the 7545 manipulator is driven indirectly by a DC servo motor. For joints 1 and 2, the motors are mounted on their respective joint axes. The transmissions for these joints are harmonic drive assemblies with gear ratios of 157:1 and 80:1, respectively.

The joint Z motor is mounted at the distal end of link 2 with its axis parallel to the z-axis. A belt and a ball-screw mechanism transform the rotational motion of the motor into translational motion along the Z-axis shaft. The transmission ratio is 0.2381 revolutions per mm.

The Roll motor and its associated harmonic drive transmission is located in the base of the manipulator. A drive belt transfers motion to the joint axis. Due to this configuration, the orientation of the gripper is independent of arm position, i.e., joint variable  $\theta_{roll}$  is independent of  $\theta_1$  and  $\theta_2$ . An interesting consequence of this configuration is that the drive belt dynamically couples the distal end of the arm to the base

of the manipulator. The harmonic drive together with the drive belt provide a reduction ratio of 51.2:1.

#### 2.2.2 DC Servo Motors and Drivers

The DC servo motors are permanent magnet type motors manufactured by Yaskawa Electric Mfg. Co., Ltd. The motors for joints 1 and 2 belong to the Print Motor series [8] while joint Z and Roll motors belong to the Minertia Motor series [9]. Their ratings and specifications are given in Appendix B.

Each motor is driven by a Yaskawa DC Servomotor Controller or Servopack [10]. Power for these units is obtained from the 7545 power supply via the relay board. Servopack power is controlled by the manipulator power key on the control panel which energizes a relay on the relay board. The Servopacks for the Print motors of joints 1 and 2 provide outputs of up to 200 W and for the Minertia motors of joints Z and Roll, up to 100 W.

Figure 2.5 shows a Servopack's internal block diagram. The output (between terminals A and B) comes from a full bridge transistor switching circuit, the driver of which is controlled by the Pulse Width Modulation (PWM) generator. The PWM generator is fed by the output of the Current Amplifier. This amplifier constitutes an armature current controller. The armature current setpoint is located at the Cref testpoint and the control loop is closed by feedback from the Current Detecting Amplifier. Cref is set by the output of the Speed Amplifier which constitutes an armature speed controller. The setpoint of this amplifier is located at the speed reference input terminals of the Servopack and is driven by a digital to analog converter on the MTCB. The feedback for the speed controller is provided by the MTCB which performs a frequency to voltage conversion on the encoder signal. This feedback signal is fed to the TG (tachogenerator) Feedback input of the Servopack.



Figure 2.5. Servopack Internal Block Diagram [10].

The Servopack, in its manufacturer-supplied form, is incompatible with typical control strategies whose output driving signals are computed torque values. The Servopack does not support a torque input because of its speed control amplifier. The solution is to disable this speed controller. By doing this, the torque output signal from a robot controller such as the ARC would reach testpoint Cref undistorted. At Cref, the driving signal would be recognized as a desired armature current. (Note that armature current is directly proportional to motor torque.) The Speed Amplifier can be modified (by opening the speed control loop) to serve as an adjustable gain amplifier for calibration purposes. The following is a list of the required modifications.

- (a) Transform the Speed Amplifier into a simple inverting amplifier by shortingout its integrating capacitor (refer to Figure 2.5).
- (b) Short the TG feedback input.
- (c) Add an external potentiometer at the speed reference input for gain adjustment.
- (d) Connect a wire from the internal testpoint C<sub>ref</sub> to a point outside the Servopack for easy access during calibration. (The calibration procedure is listed in Appendix C.)

A wiring diagram of the modifications for one joint is shown in Figure 2.6. Note that the modifications are implemented using a SPST for (a) and SPDT selector switches for (b) and (c) so that the Servopack can be easily configured (by throwing the three switches) to serve either the MTCB or the ARC.



Figure 2.6. Modified Servopack, Encoder and HOME Sensor Connections.

#### 2.2.3 Incremental Encoders

An incremental optical shaft encoder is mounted on the motor shaft of each joint to provide relative joint position information. Channel A and B outputs are in quadrature (90° out of phase) and have resolutions of 500 pulses per revolution for joints 1 and 2 and 400 pulses per revolution for joints Z and Roll. Each encoder also has a one pulse per revolution index output.

The MTCB contains a counter for each encoder. On-board logic decodes the incoming pulse streams to determine the count direction. The counters increment for positive and decrement for negative joint motion. The counters are clocked only by a single transition on a single encoder channel (e.g., the rising edge of channel A).

The MTCB sets the counters to zero when the manipulator is initialized to the HOME position. The counters then give joint positions relative to the HOME position. Joint position resolutions in pulses per degree are calculated by multiplying the encoder resolution by  $\frac{1 \text{ rev.}}{360^{\circ}}$  (not necessary for joint Z since it is translational) and the transmission ratios given in section 2.2.1. Table 2.1 lists the joint position resolutions of the four joints of the 7545 manipulator.

Table 2.1. 7545 Joint Position Resolutions.

| Joint | Resolution |                  |
|-------|------------|------------------|
| 1     | 218.05     | pulses<br>degree |
| 2     | 111.1      | pulses<br>degree |
| Z     | 95.24      | pulses<br>m m    |
| Roll  | 56.8       | pulses<br>degree |

Because robot control strategies generally require joint position feedback, the ARC must have access to the outputs of the encoders when it is controlling the manipulator. A potential access point is at a test point on the MTCB (labelled CN16 in [7]) where the two channels and the index of each encoder are conveniently available. Unfortunately, it was discovered that this is not a feasible point for extraction of these signals for the following reason. When manipulator power is switched on, the MTCB's control scheme tries to maintain the manipulator in position. However, while the manipulator is under ARC control, the MTCB's control efforts are futile since its output signals have been prevented (by the SPDT switches mentioned in the previous section) from reaching the Servopacks. The problem arises when the ARC moves a joint to a point where the MTCB's computed tracking error becomes excessive. At this point, the MTCB shuts down manipulator power as a safety feature.

A solution to this problem is to extract the incoder signals before they reach the MTCB and to prevent them from entering the MTCB. Actually, preventing only one channel (channel A for joints 1, 2, and Roll and channel B for joint Z) from entering the MTCB and pulling the MTCB input for that channel to +5 V is sufficient to fool the MTCB into thinking that there is no motion.

To accomplish this, a small circuit board was built to extract the encoder signals at MTCB connector CN7 where the signal levels are TTL compatible. SPDT switches were used for selection of either MTCB or ARC mode. Figure 2.6 illustrates the circuit modification for one joint. (Note that for joint Z, channel B is the channel which is to be interrupted.)

An added benefit to the above approach is that the encoder outputs are available at the extraction point whether or not manipulator power is on while at MTCB testpoint CN7 the signals are provided only when manipulator power is on. The benefit is that the ARC can

continually monitor joint position and that reinitialization of position counters is not necessary if manipulator power is switched off and the manipulator is moved manually.

#### 2.2.4 Sensors

Each joint of the manipulator is equipped with overrun and HOME sensors (see Appendix A for wiring details).

The overrun sensors are limit switches that detect joint hyperextension. When an overrun sensor is tripped, manipulator power is turned off. This safety feature protects the links from hitting their mechanical limits and must remain intact. The ARC does not interfere with this safety feature.

The HOME sensors are proximity switches which are activated when a joint is in its HOME or reference position. Referring to Figure 2.3, the respective sensors indicate a HOME condition when link 1 becomes collinear with the x-axis, link 2 becomes collinear with the radial axis of link 1, the gripper point lies in the x-y plane, and the Roll angle is anywhere in its negative region. The HOME sensors are used in initialization of the joint position counters. The MTCB has a find-HOME routine that consists of moving a joint away from its HOME position if the sensor is already active and then moving it back towards the HOME sensor. When the HOME sensor becomes activated, the position counter for the joint is set to zero and the joint motion is discontinued.

The HOME sensor signals are extracted for use with the ARC from the ribbon cable connecting CN5R to CN12 as illustrated in Figure 2.6. HOME condition is indicated by 0V and not HOME condition by 22.5V.

# 2.3 SUMMARY

This chapter described the IBM 7545 Manufacturing System and identified the MTCB component as the source of some of its limitations. It was indicated that to overcome these limitations, the MTCB must be bypassed with a more powerful processing system such as the ARC which is presented in subsequent chapters. The chapter described important details of the 7545 system and outlined the modifications to be made to the system in order to bypass the MTCB.

# CHAPTER 3

#### HARDWARE DESCRIPTION

# FOR THE ADVANCED ROBOT CONTROLLER

#### 3.1 INTRODUCTION

This chapter discusses the design philosophy of the ARC and describes the hardware that was developed for its implementation.

The ARC was conceived primarily as a replacement for the MTCB in the IBM 7545 Manufacturing System for the purpose of converting the 7545 into a testbed for robotic research. For this purpose the following ARC design objectives were adopted:

- The ability to execute both simple and sophisticated robot control algorithms at respectable servo rates (baseline of 1000 Hz for PD control).
- Basic functions to facilitate the development and implementation of robot control strategies.
- An environment conducive to robotic research.
- The cost of the overall system a fraction of the cost of the 7545 system.

To achieve these objectives, a hierarchical dual-processor architecture was chosen (see Figure 3.1) whereby the master processor is used for developing and executing control strategies and the slave serves as an intelligent interface between the master and the robot. Good performance from this type of architecture is achieved since the slave relieves a significant portion of the computational and I/O burden from the master. The slave handles some of the tedious yet essential functions associated with servo control. For example, the slave can be programmed to keep track of joint positions by continuously reading the joint encoder signals emanating from the robot and test for overrun conditions. It can also verify that the output motor torques as computed by the master are

within the limitations of the robot. All the functions handled by the slave are invisible to the user and thus facilitate the development of the master software.



Figure 3.1. Block Diagram of the ARC Hardware.

The IBM Personal System /2 (PS/2) model 50 computer and the Intel 80C196KA microcontroller were chosen as the master and slave, respectively. The PS/2 model 50 is adequate for initial needs and can easily be substituted by the more powerful model 70 or model 80 for future needs. The Intel 80C196KA has many features which suit the ARC's needs. These will be discussed in Section 3.3.

The block diagram of Figure 3.1 also shows the remaining ARC hardware components: a dual-port RAM (DPR) for communication between the master and slave processors; four digital to analog converters (DACs) to drive the 7545's Servopacks; four

position counters to read the 7545's joint encoders; and an input buffer for reading the 7545's encoder index and HOME sensor lines.

The following sections of this chapter describe each of the components mentioned above and the manner in which they interconnect.

### 3.2 MASTER PROCESSOR AND DUAL-PORT RAM (DPR)

The master processor of the ARC is an 80286-based IBM Personal System /2 (PS/2) model 50 [11] which features the Micro Channel architecture and an 80287 math coprocessor. As indicated in Figure 3.1 the master communicates with the slave via a dual-port RAM (DPR).

The dual-port RAM, as its name implies, is RAM that can be accessed by two processors, one at each port. The DPR was chosen over other types of data communication methods (e.g., FIFO) as it can store control program variables required by both the master and the slave processors and it can hold predefined semaphores for timing purposes thus eliminating the need for hardware interrupts. Variable that are passed between the two processors via the RAM include the robot state from slave to master and computed torques from master to slave.

For implementation, an Integrated Device Technology's IDT7132/IDT7142, master/slave pair [12] was chosen. In parallel, these two 2k x 8 dual-port static RAMs form a 2k x 16 memory that interfaces directly to the 16-bit word width of the PS/2. The PS/2-DPR interface circuit that was designed is shown in Figure 3.2. This circuit was constructed on a Micro Channel prototype adapter card which can be inserted into an expansion slot inside the PS/2. The critical timing parameters of the Micro Channel's memory cycles are shown in Appendix D while the timing waveforms of the DPR are given in Appendix E. A short description of the circuit is given next.



Figure 3.2. PS/2 - Dual-Port Ram Interface Schematic.

Referring to Figure 3.2, the twelve DIP switches enable initialization of the starting address of the DPR. ICs U1 and U2 decode the upper 12 address lines A(12-23) along with the  $M/\overline{I/O}$  and MADE 24 signals to produce the unlatched address decode signal. This signal provides the required feedback to the master on lines  $\overline{CD}$  DS 16 and  $\overline{CD}$  SFDBK. The unlatched address decode signal, along with the lower 12 address lines A(0-11) and the control signals  $\overline{SO}$ ,  $\overline{S1}$ , and  $\overline{SBHE}$ , are latched by U3 and U4 on the leading edge of  $\overline{CMD}$ . The latched address decode signal then enables the left port of the dual-port RAM and, together with  $\overline{CMD}$ , enables the data bus buffers, U8 and U9.  $\overline{SO}$  and  $\overline{S1}$  are the write and  $\overline{CMD}$  read enable signals, respectively.

The IDT7132 has on-chip port arbitration logic to resolve the situation in which both ports simultaneously address the same memory location. When this situation occurs, the IDT7132 determines which processor has access and holds the operation of the other processor through the use of a BUSY flag. For example, the IDT7132 sets BUSYL to indicate right port priority. This signal is then used to delay the PS/2 by setting its CD CHRDY(n) (channel ready) line.

Unfortunately, implementation of the above is complicated by the fact that the propagation delays of the IDT7132 and the other interface circuit components cannot guarantee generation of the BUSYL within the time required by the PS/2. Referring to Figure 3.2, the solution is to automatically set the CD CHRDY(n) signal at the beginning of the memory cycle and reset it on the leading edge. CMD using gates U7A, U5D and U7B. This affords the circuit extra time to set the SYL signal. In this configuration, the hardware extends the Micro Channel's default cycle to the 300 nanosecond synchronous cycle given in Appendix D. It should be noted that the PS/2 model 50 automatically extends the default cycle to the 300 nanosecond synchronous extended cycle. It is performed by the interface hardware, however, to assist portability to other systems.

The starting address of the left port of the DPR is set by the DIP switches to 0C0000H.

An 18 inch twisted pair ribbon cable is connected to the header of the circuit board and links the right port of the DPR to the microcontroller which is described in the next section.

#### 3.3 SLAVE PROCESSOR

The slave processor chosen for the ARC system is the Intel 80C196KA Microcontroller [13]. To simplify the design of the slave system the Intel EV80C196KA Microcontroller Evaluation Board [14] was employed. This evaluation board consists of the 80C196KA 16-bit embedded microcontroller, 16k x 16 static RAM, 16k x 16 EPROM and a UART for communications with a host IBM or compatible. Edge connectors on the board provide easy access to the microcontroller's system (data, address, control) bus. For reference, the EV80C196KA schematic diagram is given in Appendix F.

The EV80C196KA also facilitates software development as it contains a system debug monitor (SDM) for loading, executing and debugging code. The SDM is actually composed of two separate programs. One resides in the EPROM on the evaluation board and executes on the EV80C196KA and the other runs on the host computer, in our case an IBM compatible. The two programs communicate via an RS-232 channel.

#### 3.3.1 The EV80C196KA Interface

As illustrated in the ARC block diagram of Figure 3.1, the microcontroller communicates with the DPR, the digital to analog converters, the position counters, and the robot overrun and HOME sensors. Each one of these components is viewed by the microcontroller as external memory residing somewhere in the upper half of the 80C196KA's 64k address space (8000H-FFFFH). Referring to Appendix F, the EV80C196KA's main interface is located at the JP2 memory expansion connector where the system bus is available. The timing diagram of the bus is given in Appendix G. The

schematic diagram of the first stage of the interface circuit which was designed to connect directly to JP2 is shown in Figure 3.3. A short explanation of the circuit follows.

U5 buffers all output control lines and is always enabled. U1 and U2 latch the 16-bit address from the multiplexed address/data bus (AD0-AD15) on the trailing edge of BALE. U3, U4 and U7 are bidirectional data bus buffers. Their direction is controlled by the BRD signal. D0 has its own buffer so that two position counters can be accessed simultaneously (this will be explained in more detail in the section 3.5). U8, a 3-8 decoder, generates the chip enable signals for the components of the ARC and is enabled for addresses in the range 8000H-FFFFH (A15 high). The outputs CE1, CE2, CE3, and CE4 enable the position counters, the digital to analog converters, the HOME and overrun sensors buffer, and the DPR, respectively. Since the chip enable signals are generated by a latched address decode, they are active for the entire memory cycle of the 80C196KA. In light of this, data transmission timing is controlled by the pulse width of the 80C196KA's read (RD) and write (WRH, WRL) lines. For ARC devices that require longer pulse widths than these lines provide, an antedated read/write signal AR/W is generated. The AR/W pulse begins earlier, on the trailing edge of ALE, and ends on the trailing edge of either RD or WRL.

U21B, the D flip-flop, is set and reset by  $\overline{RD}$  and  $\overline{WRH}$ , respectively. The output of the flip-flop, LR/W (latched read/write), drives the  $R/\overline{W}$  line of the position counters. This will be explained in greater detail in section 3.5. An additional requirement of the position counters is a 2 MHz clock. This clock is generated by dividing the 80C196KA's 6 MHz CLOCKOUT signal by three using the dual J-K flip-flops of U10. The clock signal, as shown in Figure 3.3, is termed DBCLK.



Figure 3.3. 80C196KA Interface Schematic.

TIMER2 is a 16-bit counter on the 80C196KA which is used by the ARC as a sample period timer. TIMER2 increments on any transition at the T2CLK terminal. The desired TIMER2 counting frequency is 1 MHz and the required 500 KHz clock for the T2CLK input is generated by dividing the 2 MHz clock by four using the dual J-K flip-flops of U26.

### 3.4 DIGITAL TO ANALOG CONVERSION

Digital to analog converters (DACs) are the means by which the computed motor torque values are converted into analog voltage signals to drive the Servopacks. In its present configuration, the ARC employs four Analog Devices AD667 12-bit double-buffered DACs [15], one for each joint of the 7545 manipulator. The double-buffered digital inputs enable all DACs to be simultaneously updated without the need for external latches and the 12-bits provides adequate resolution. The AD667's digital and analog circuits are discussed below.

# 3.4.1 Digital Circuit Details

The AD667 functional block diagram and write cycle timing diagrams are given in Appendix H. The AD667 bus interface logic consists of four independently addressable registers in two ranks. The first rank consists of three four-bit registers controlled by address inputs AO-A2. The second rank register holds all 12-bits and is controlled by address input A3.

The AD667-80C196KA interface is shown in Figure 3.4. The first rank register address lines (A0-A2) on each of the four DACs are tied together and connected to 80C196KA address lines A1-A4, respectively. The second rank register address line (A3) on all DACs are tied together and connected to 80C196KA address line A5. In this configuration, the first rank registers of each DAC can be loaded separately with their respective torque data as per Write Cycle #1. The transfer from first to second rank is then performed

simultaneously on all DACs as per Write Cycle #2. This double buffered organization eliminates spurious analog outputs originating from data bus activity while the DACs' chip select inputs (cs) are low.

The cs inputs are driven by CE2 whose base address is A000H. The CE2 signal is gated by the ARVW signal to provide the longer pulse width required by the DACs. The DAC addresses are listed in Table 3.1.

Table 3.1. Digital to Analog Converter Addresses.

| DAC # | RANK<br>REGISTER | 80C196KA<br>ADDRESS |
|-------|------------------|---------------------|
|       | 10001DIC         | 7,22,1255           |
| 1     | 1                | AFFCH               |
| 2     | 1                | AFFAH               |
| 3     | 1                | AFF6H               |
| 4     | 1                | AFEEH               |
| ALL   | 2                | AFDEH               |



Figure 3.4. Digital to Analog Converter Schematic.

# 3.4.2 Analog Circuit Details

The DAC's internal output amplifier supplies up to 40 mA of current to drive the 7545's Servopack. Each AD667 is configured to produce bipolar output voltage ranges of ±10V. Two 100Ω potentiometers are used in calibrating the outputs. The calibration process is described in Appendix C. A 20pF capacitor is connected across the feedback resistor to optimize dynamic performance.

### 3.5 POSITION COUNTERS

As mentioned in the first chapter, the joints of the 7545 manipulator are equipped with position feedback devices in the form of optical incremental encoders (see Section 2.2.3). To recover the encoded position information, the ARC employs four Hewlett Packard HCTL-1000 General Purpose Motion Control ICs [16]. These ICs were chosen as they contain quadrature decoders and 24-bit counters for reading the 7545's joint encoders as well as 3-bit state delay digital filters to remove noise spikes on the encoder lines. Each HCTL-1000 takes the place of a multitude of ICs used in [1] and [2] for the same purpose. Another feature of the HCTL-1000 is that a count is produced for both transitions (low→high and high→low) on Channel A and Channel B. Therefore, the 500-count encoders on the first two 7545 manipulator joints are decoded into 2000 quadrature counts per revolution. As a result, the HCTL-1000s give four times better resolution than the counters on the MTCB as well as the counter described in [1] and [2] which count on one transition of one encoder channel only. In addition, the 24-bit capacity of the HCTL-1000s is sufficient to hold the pulse counts corresponding to the maximum range of motion of the joints. Thus no special software is required to keep track of the total pulse count. The schematic diagram of the HCTL-1000 circuit is shown in Figure 3.5. A description of the circuit is given below.



Figure 3.5. HCTL-1000 Schematic.

The HCTL-1000s are clocked by the 2 MHz DBCLK line as are the D flip-flops in U25. The flip-flops ensure that all encoder pulse transitions occur at the HCTL-1000 encoder inputs coincident with the clock, thus preventing any possibility of erroneous encoder counts as explained in [16].

The 24-bit pulse count is located in three of the HCTL-1000s 64 8-bit internal registers. The registers are accessed over the HCTL-1000s multiplexed 6-bit address/8-bit data bus. The bus timing diagram for the HCTL-1000 is included in Appendix I and a description of the I/O operation follows.

On the leading edge of BALE the HCTL-1000 begins sampling the bus into an internal address latch. This bus information, which represents the 6-bit address of one of the HCTL-1000's 64 8-bit registers, gets latched on the trailing edge of BALE. Next, on the leading edge of  $\overline{CS}$ , the HCTL-1000 begins sampling the bus into an internal data latch. On the trailing edge of  $\overline{CS}$  the HCTL-1000 checks the LRW line and performs one of two operations. In the case of a write operation, the data in the data latch is written into the addressed location. In the case of a read operation, the data in the addressed location is sent to an internal output latch which can then be enabled onto the bus by setting  $\overline{OE}$ . Because the HCTL-1000 takes a relatively long time (minimum 1.8 microseconds) to transfer the data to the output latch, the read operation is performed by two microcontroller read instructions. The first read instruction sets the LRW line high, selects the desired register and begins the transfer process by asserting the appropriate  $\overline{CS}$  line. The second read instruction asserts the  $\overline{OE}$  line to enable the data onto the bus to be read by the microcontroller.

The  $\overline{\text{CS}}$  and  $\overline{\text{OE}}$  signals are generated by U20, a 3-8 decoder, which decodes address lines A6, A14, and A7. U20 is enabled by the  $\overline{\text{CE1}}$  and by the AR/W which provides the longer  $\overline{\text{CS}}$  and  $\overline{\text{OE}}$  pulse widths required by the HCTL-1000s.

To increase the ARC performance, the 8-bit HCTL-1000s are connected in pairs to the 16-bit bus of the EV80C196KA microcontroller. The upper and lower HCTL-1000s that make up a pair are connected to bus lines D8-D15 and D0-D7, respectively. The first pair is controlled by  $\overline{CE1}$  and  $\overline{OE1}$  and the second pair is controlled by  $\overline{CE2}$  and  $\overline{OE2}$ . Two other sets of control lines are generated by U20,  $\overline{CE3}$ ,  $\overline{OE3}$  and  $\overline{CE4}$   $\overline{OE4}$ , for future expansion. The memory map for these control signals is given in Table 3.2.

Table 3.2. Memory Map for the HCTL-1000s.

| CNTRL.     |     |     |     |     |     |     | A  | DDF | RESS | 3   |    |    |     |    |           |           |
|------------|-----|-----|-----|-----|-----|-----|----|-----|------|-----|----|----|-----|----|-----------|-----------|
| SIGNAL     | A15 | A14 | A13 | A12 | A11 | A10 | A9 | A8  | A7   | _A6 | A5 | A4 | A3_ | A2 | <u>A1</u> | <u>A0</u> |
| Œ1         | 1   | 0   | 0   | d   | d   | d   | đ  | d   | 0    | 0   | d  | d  | d   | d  | d         | 0         |
| OE2        | 1   | 0   | 0   | d   | d   | ď   | d  | d   | 0    | 1   | d  | d  | d   | d  | d         | 0         |
| CS1        | 1   | 0   | 0   | X   | X   | X   | X  | X   | 1    | 0   | ×  | X  | X   | X  | x         | 0         |
| CS2        | 1   | 0   | 0   | X   | X   | X   | X  | X   | 1    | 1   | X  | X  | X   | X  | x         | 0         |
| OE3        | 1   | 1   | 0   | d   | d   | d   | d  | d   | 0    | 0   | d  | đ  | d   | d  | d         | 0         |
|            | 1   | 1   | 0   | d   | d   | d   | ď  | d   | 0    | 1   | d  | d  | d   | d  | d         | 0         |
| OE4<br>CS3 | 1   | 1   | 0   | X   | X   | X   | X  | X   | 1    | 0   | X  | X  | X   | X  | X         | 0         |
| CS4        | 1   | 1   | 0   | X   | X   | X   | X  | ×   | 1    | 1   | X  | X  | x   | X  | ×         | 0         |

d-don't care

x-HCTL-1000 register address

Since address line A13 is always 0 only the first 32 registers on the HCTL-1000s can be accessed (i.e. registers R00H - R1FH). This does not pose a problem as the 24-bit actual position count is held in the three 8-bit registers R12H (MSB), R13H, and R14H (LSB) and the only other register of interest is R05H. Writing 01H to R05H commands the HCTL-1000 to enter the Initialization/Idle mode where it simply keeps track of joint position. Upon writing 00H to R05H, the position counter registers (R12H, R13H, R14H) are cleared and the Initialization/Idle mode is entered.

As seen in Table 3.2, address line A0 is always 0. This is due to the fact that the microcontroller requires an even operand in all instructions which refer to word (16-bit) memory locations. This would dictate that only even registers on the lower HCTL-1000 can be addressed. This limitation, however, is overcome by providing bus line D0 with its

own buffer (U7) as shown in the schematic diagram in Figure 3.3. When the buffer is enabled, D0 takes the level of AD0 and when the buffer is disabled, D0 is pulled high by the pull-up resistor R17. It is presumed that the microcontroller addresses the same location in both the upper and lower HCTL-1000s. Therefore, the D0 input of the lower HCTL-1000 should always be the same as D8 on the upper HCTL-1000 during addressing. Hence, during the addressing portion on the multiplexed bus (i.e., if AD8 is high during BALE the buffer will be disabled by U22B and D0 will be high even though AD0 is low. If AD8 is low, the buffer will be enabled and D0 will take on the value of AD0.

As was mentioned previously, the  $\overline{OE}$  line which enables the output of the HCTL-1000s onto the data bus begins with the antedated AR/W line on the rising edge of  $\overline{BALE}$ . Bus contention could occur at the beginning of this period when the microcontroller is asserting an address on AD[0..15]. The contention would last until the  $\overline{BRD}$  line goes low and the direction of the data buffers changes for a read instruction. The contention is avoided by means of U23A, U22D, U6A, U9A, and U9D, which act to disable the data buffers if the HCTL-1000s are being accessed ( $\overline{CE1}$  is low), if it is a read instruction (A7 and  $\overline{OE}$  are low), and if the  $\overline{BRD}$  line is high. This buffer disabling period begins on the address decode and ends with the start of the read pulse.

The final timing requirement of the HCTL-1000 is that its R/W input remain asserted for a short period after the CS pulse. This is accomplished by latching the BRD and BWRH lines in the D flip-flop, U21B. The output of the flip-flop, labelled LR/W (latched read/write), drives the R/W inputs of the HCTL-1000s.

### 3.6 HOME SENSORS, ENCODER INDEX AND DUAL-PORT RAM CONNECTION

As described in section 2.2.4 the extracted HOME sensor signals have a voltage of approximately 22.5 V when the joint is not HOME and 0 V when the joint is HOME. These

voltages are converted to TTL levels by the circuit in Figure 3.6 which consists of a voltage divider, a Darlington transistor and pull-up resistor, and for buffering an opto-isolator. The HOME signals along with the index pulses generated by the optical encoders (TTL compatible) are made available to the microcontroller through buffer U15. The buffer is enabled by CE2 and BRD and its address is B000H.

Also included in Figure 3.6 is the edge connector for the dual-port RAM. Because the dual-port RAM has a very fast access time it can be connected directly to the basic interface circuit in Figure 3.3. The dual-port RAM is chip-enabled by  $\overline{CE4}$  and has a base address of E000H.



Figure 3.6. Index and HOME Sensor Schematic.

#### 3.7 HARDWARE IMPLEMENTATION

The circuits of sections 3.3 - 3.6 were wire-wrapped on a prototype board. The EV80C196KA microcontroller evaluation board was attached along side the prototype board and was electrically connected via a ribbon cable. The assembly was mounted on a Plexiglass base for attachment inside the 7545's control unit housing. As mentioned previously, connection to the dual-port RAM (residing in the PS/2) was via an 18 inch, 64 conductor, twisted-pair ribbon cable.

### 3.8 SUMMARY AND FUTURE WORK

This chapter described the hardware that was designed specifically for the ARC. The dual-processing nature of the ARC provides the processing power needed for advanced control strategies and also enables some of the basic functions executing on the slave to be invisible to the user. The system also serves as a basis for future hardware expansion. For example, supplemental hardware such as force sensor feedback circuits (required for impedance control) and circuitry for additional degrees of freedom can be easily interfaced to the slave system bus. As for the master processor, greater processing power can be easily achieved in two ways. The first is to replace the PS/2 model 50 by the model 70 or the model 80. The second is to add one or more transputer (processor) boards inside the PS/2.

# **CHAPTER 4**

# SOFTWARE DESCRIPTION

# FOR THE ADVANCED ROBOT CONTROLLER

### 4.1 INTRODUCTION

As mentioned in Chapter 1, the ARC was conceived to enhance the operation of the IBM 7545 Manufacturing System and, in the process, transform it into a testbed for robotic research. The previous chapter described the hardware that was designed to achieve this objective. This chapter presents the master and slave software that has been developed for the purpose of creating the research environment.

Figure 4.1 shows a block diagram illustrating the role of each processor in the overall system. Note that the user interfaces only with the master processor; the slave is completely invisible. The slave software performs the function of I/O handler for the master and some basic safety features such as limit checking of variables. The slave also contributes by maintaining the sample period timer. Perhaps the most valuable function of the slave, however, is the reading of the joint position counters (HCTL-1000s). This function, as detailed in Section 3.5, is both complicated and time consuming. The slave, therefore, performs some of the tedious yet essential tasks associated with robotic control and enables the master (and user) to concentrate on higher level issues. The slave software is described in Section 4.3.



Figure 4.1. Block Diagram of the ARC Software.

The master processor is used for developing, storing and executing path planning and robot control algorithms. So far, a library consisting of two path planning algorithms and two control algorithms has been coded in the C language. Each of the four functions in the library resides in its own separate file for developmental purposes. Once debugged, a function is linked to a main program for execution. The main program, also residing in a separate file, is composed of menu, display, and user-interaction functions. These functions help to create a user-friendly environment conducive to robotic research. Details of the master processor software are included in Section 4.4.

The following section describes how the dual-port RAM is set up to handle the flow of data between the master and slave processors.

#### 4.2 DUAL-PORT RAM UTILIZATION

The DPR of the ARC constitutes the ARC's shared memory system and is used for the passing parameters and variables between the master and slave processors. Each parameter and variable is assigned a specific location (register) in the DPR. The address and name of each register in the DPR is listed in Table 4.1.

Table 4.1. Dual-Port RAM Register Reference Table.

| ADDRESS |       | SIZE        | NAME                    | USER A | CCESS |
|---------|-------|-------------|-------------------------|--------|-------|
| Master  | Slave |             |                         | Master | Slave |
| С0000Н  | E000H | word        | sample_period           | w      | r     |
| C0002H  | E002H | word        | timing_A                | r/w    | r/w   |
| C0004H  | E004H | word        | timing_B                | r/w    | r/v/  |
| С0006Н  | E006H | word        | command                 | r/w    | r/w   |
| C0008H  | E008H | word        | error                   | r/w    | r/w   |
| C0010H  | E010H | double word | actual_position_joint_1 | r      | w     |
| C0014H  | E014H | double word | actual_position_joint_2 | r      | w     |
| C0018H  | E018H | double word | actual_position_joint_Z | r      | w     |
| C001CH  | E01CH | double word | actual_position_joint_R | r      | w     |
| C0020H  | E020H | word        | torque_joint_1          | w      | r     |
| C0022H  | E022H | word        | torque_joint_2          | w      | r     |
| C0024H  | E024H | word        | torque_joint_Z          | w      | r     |
| C0026H  | E026H | word        | torque_joint_R          | w      | r     |

The function of each of the registers listed in the table is described below.

# 4.2.1 Sample\_Period Register

The sample\_period register is loaded by the master with a value corresponding to the controller sampling period. As mentioned in Section 3.3.1, sample period timing is handled by TIMER2 on the 80C196KA which counts up at a frequency of 1 MHz. Therefore, multiplying the value in the sample\_period register by 10-6 yields the actual sampling period in seconds. The slave maintains the sample period timer by resetting TIMER2 when it reaches or exceeds the value in the sample\_period register. The reset is accomplished by subtracting the value in the sample\_period register from TIMER2.

### 4.2.2 Timing\_A Register

The timing\_A register contains two semaphores which are set by the slave to initiate the master's sampling period and to indicate to the master that joint positions have been read and stored in the DPR. The semaphore indications are specified in the table below.

| Bit Number | Indication                |
|------------|---------------------------|
| 0          | Begin new servo loop      |
| 1          | Joint positions available |

The master resets both semaphores after completing the computations for the sampling period and then waits for bit 0 to become set indicating the start of the next period.

## 4.2.3 Timing\_B Register

The timing\_B register contains one semaphore (bit 0) which is set by the master to indicate to the slave that the joint motor torques have been computed and stored in the DPR. The slave resets the semaphore after reading the torques.

## 4.2.4 Command Register

As will become clear in Section 4.3, the slave processor can execute in one of two modes of operation: the find-HOME mode, and the control mode. The master indicates the desired mode by setting a semaphore in the command register. The slave reads the command register while in a waiting loop and performs one of the actions specified below.

| Bit Number Set | Indication               |
|----------------|--------------------------|
| 0              | Enter the Control Mode   |
| 7              | Enter the Find Home Mode |
| None           | Continue Waiting         |

# 4.2.5 Error Register

The slave processor informs the master of any error conditions that have been detected by setting flags in the error register. The flag indications are specified below. The errors will be described in their proper context in Section 4.3.

| Bit Number | Indication                      |  |
|------------|---------------------------------|--|
| 0          | Joint 1 in positive overrun     |  |
| 1          | Joint 1 in negative overrun     |  |
| 2          | Joint 2 in positive overrun     |  |
| 3          | Joint 2 in negative overrun     |  |
| 4          | Joint Z in positive overrun     |  |
| 5          | Joint Z in negative overrun     |  |
| 6          | Joint Roll in positive overrun  |  |
| 7          | Joint Roll in negative overrun  |  |
| 8          | Excessive torque for joint 1    |  |
| 9          | Excessive torque for joint 2    |  |
| 10         | Excessive torque for joint Z    |  |
| 11         | Excessive torque for joint Roll |  |
| 12         | Motor torques arriving too late |  |
| 13         | Master processor too slow       |  |

# 4.2.6 Actual\_Position Registers

The slave processor reads the HCTL-1000 position counters and stores the count values in the long integer or double word (32 bit) actual\_position registers.

# 4.2.7 Torque Registers

The master computes the joint motor torques and stores them in the torque registers.

## 4.3 SLAVE PROCESSOR SOFTWARE

The tasks of the slave are shown in Figure 4.1. These tasks are fundamental to virtually all robot control strategies and therefore do not require modification. By assigning these tasks to the slave, they in effect become invisible to the user. As shown in Figure 4.1, the slave software is designed to execute in one of two modes of operation: the find-HOME mode and the control mode. The first mode is used for locating the manipulator's HOME (reference) position, the second mode contains the tasks which assist the master's control program. An operating mode is selected while the processor is looping in a main program. The main program, the two operating modes and the common procedures are described in the following sections. Flow charts are included in the text and source code listings can be found in Appendix J. The slave processor software is written in 80C196KA assembly language and resides in a single module.

### 4.3.1 Main Program

The flow chart for the slave's main program is shown in Figure 4.2. The initialization sequence consists of program variable and pointer initialization, resetting of the HCTL-1000 position counters and putting the 80C196KA's TIMER2 into fast increment mode to enable it to count at the desired 1 MHz frequency. The slave then clears the command register in the DPR and enters the waiting loop.



Figure 4.2. Flow Chart for the Slave's Main Program.

While in the waiting loop, the slave calls procedures to zero the DACs so that no torques are fed to the Servopacks, to acquire the joint positions, and to check joint positions for overruns. If no joint overrun conditions exist, the command register is polled. Depending on the value in this register (see Section 4.2.4), the slave either enters the control mode, enters the find-HOME mode, or restarts the waiting loop. If, however, a joint overrun is detected, command is not polled and the waiting loop is restarted.

### 4.3.2 Find-HOME Mode

The find-HOME mode is entered if, while in the waiting loop, the slave detects a value of 0080H (bit 7 set) in the command register. In the find-HOME mode, the slave autonomously moves the 7545's joints in order to locate the reference position of each joint. As mentioned in Section 2.2.4, the HOME sensor signals are generated by proximity switches mounted on the 7545 manipulator. Because of the imprecise nature of proximity switches, the ARC recognizes a joint's HOME position as being located at the first encoder index pulse following activation of the proximity switch. The sequence of find-HOME is depicted in the flow chart of Figure 4.3 and is described below.



Figure 4.3. Flow Chart for the Find-HOME Mode.

The find-HOME sequence commences with the reading of the joint's HOME sensor. If the HOME sensor is activated (active low) the slave applies a positive constant torque to the motor in order to move the joint away from the HOME sensor (open-loop

control). The torque continues to be applied for a period of approximately 2 seconds (governed by the delay procedure) following the deactivation of the HOME sensor.

Hence or otherwise, the slave applies a negative torque to the motor in order to move the joint toward the HOME sensor. The slave polls the HOME sensor during this motion. When the slave detects activation of the HOME sensor, it begins polling the encoder index line. When the index pulse is detected, the slave processor first instructs the HCTL-1000 to perform an on-chip software reset which among other things resets the 24-bit position counter to zero, and then instructs the joint motor to stop. In the first instruction, the second HCTL-1000 of the pair (recall from Section 3.5 that the HCTL-1000s are accessed in pairs) is instructed to remain in its Idle/Initialization mode.

Joint motion during the find-HOME sequence is the result of a constant torque applied in an open-loop configuration. The torque is of sufficient magnitude to move the joint at a constant slow speed. Because of inertia, motion does not cease instantaneously and there is always slight overshoot of the HOME position. The find-HOME procedure, therefore, does not leave the manipulator in the HOME position; it simply locates HOME and initializes the position counters at that point.

Upon completion of the find-HOME procedure, the slave processor reenters the main program at the point where command is cleared. A clear command register indicates to the master that HOME was found.

## 4.3.3 Control Mode

The control mode is entered if, while in the waiting loop, the slave finds a value of 0001H (bit 0 set) in the command register. In the control mode, the slave processor serves as an intelligent interface between the 7545 manipulator and the master processor which is

executing the robot control algorithm. The control mode procedure is shown in the flow chart in Figure 4.4 and is described below.





Figure 4.4. Flow Chart for the Control Mode.

Upon entering the control mode, the slave resets the sample period timer (80C196KA special purpose register TIMER2) and the timing\_A and timing\_B registers. Following this, the slave begins its servo loop and tells the master to do the same by setting the 'begin servo loop' semaphore (bit 0) in timing\_A. The slave then calls the get\_joint\_positions procedure whereby the position counters are read and stored in the actual\_position registers

and a semaphore set in the timing A register to indicate to the master that the joint positions are available (the procedure get\_joint\_positions is described in detail in Section 4.3.4.1). The slave then calls the joint\_overrun\_check procedure (see Section 4.3.4.4) where the pulse count of each joint is checked for excessive values and the error register updated. The slave returns from this procedure, checks the error register, and exits the control mode if any semaphores are set. This immediately halts manipulator motion by turning off the servo motors (see Figure 4.2).

Continuing in the control mode, the slave tests for the end of the sampling period and for the motor torques to arrive from the master processor. The end of the sampling period is when the sample period timer, TIMER2, reaches the value in the sample\_period register and the arrival of the torques is indicated by a semaphore in timing\_B. If the servo loop ends before the torques arrive, the slave sets the 'torques arriving too late' flag in the error register and exits the control mode. If the master commands the slave to stop during this process, the slave exits the control mode.

The slave proceeds in the *control* mode by calling the *check\_torques* procedure (see Section 4.3.4.5 and Figure 4.4) which checks for excessive torque values and sets the corresponding flag in the *error* register.

Returning from this procedure, the slave checks the error register and exits the control mode if it is not clear. If all torques are within range, the slave continues by clearing timing\_B and outputting the torques to the DACs.

The slave then tests for the end of the sampling period and for the master to indicate completion of its control loop (clearing of timing\_A). If the sampling period ends before the master completes its loop, the slave sets the 'master processor too slow' flag in the error register and exits the control mode. If the master finishes its loop in time, the slave enters another loop where it waits for the end of the sample period.

When the sampling period ends, the sampling period timer value will equal or exceed the value in the sample\_period register. The slave resets the timer by subtracting from it the value in the sample\_period register.

#### 4.3.4 Additional Procedures

This section describes in more detail the procedures mentioned in the previous sections. Flow charts are provided for most of the procedures and the source code is included in Appendix J.

### 4.3.4.1 Get\_Joint\_Positions

This procedure reads the three 8-bit actual position registers of each HCTL-1000, sign-extends the 24-bit position data to 32 bits, and stores them in the DPR actual\_position registers.

As mentioned in section 3.5, reading of an HCTL-1000 8-bit register actually requires two read instructions (the first to generate the chip select signal and the second to generate the output enable signal 1.8 microseconds later). In addition, it was mentioned that the HCTL-1000s are read in pairs. Because of this, the 16-bits of data obtained by the second read instruction must be sorted. In light of these requirements, the two read instructions are interlaced with instructions that sort the data and sign-extend them to 32 bits. The exact instruction sequence can be found in the source code listing in Appendix J.

Sign-extension enables the 24-bit position data to occupy the double word actual\_position registers and still retain their positive or negative sense. To accomplish sign-extension, we recognize that each joint can move a finite distance into its negative region and even in their most negative positions, the most significant byte of their 24-bit

position count will be FFH. Therefore, sign extension is easily accomplished by first clearing the most significant byte of the 32-bit register and then setting it to FFH providing the second most significant byte (i.e., the most significant byte of the 24-bit count) is FFH.

The final step in the procedure is the storage of the 32-bit joint positions in the actual\_position registers and setting the semaphore in timing\_A.

# 4.3.4.2 Delay

This procedure generates a delay of roughly 2 seconds for use in the *find-HOME* mode. The delay is generated by instructing the processor to count down from FFFH to 0 eight times. The flow chart is given in Figure 4.5.



Figure 4.5. Flow Chart for Delay Procedure.

# 4.3.4.3 Zero\_DACs

This procedure simply sets all the DAC outputs to zero. The slave begins by writing 800H (the bipolar offset value) to the first rank register of each DAC. Then the second rank register of each DAC is loaded simultaneously with one write instruction. The flow chart is shown in Figure 4.6.



Figure 4.6. Flow Chart for Zero\_DACs Procedure.

# 4.3.4.4 Joint\_Overrun\_Check

This procedure constitutes software limit switches for the detection of joint overrun conditions. A negative joint position is compared with MIN, a positive joint position is compared with MAX. If a joint position is more negative than MIN or more positive than MAX, a flag is set in the error register. The values for MIN and MAX are chosen so that the software limit switches activate before the 7545's hardware limit switches. This prevents unnecessary shut-down of manipulator power by the 7545's control unit. Figure 4.7 shows the flow chart for this procedure.



Figure 4.7. Flow Chart for Joint\_Overrun\_Check Procedure.

# 4.3.4.5 Check\_Torques

This procedure reads the torques from the DPR and verifies that they are within the 12-bit range (0-FFFH) of the DACs. Each torque is logically ANDed with F000H. If the result is non-zero, the torque is excessive and the corresponding flag is set in the error register. The flow chart is given in Figure 4.8.



Figure 4.8. Flow Chart for Check\_Torques Procedure.

# 4.3.4.6 Torques\_Out

This short procedure sequentially writes the torques to the first rank of each DAC and then simultaneously writes them to the second rank registers. The flow chart is given in Figure 4.9.



Figure 4.9. Flow Chart for Torques\_Out Procedure.

### 4.4 MASTER PROCESSOR SOFTWARE

As indicated in Figure 4.1, the master software consists of path planning algorithms and robot control algorithms. So far, two control and two path planning algorithms have been implemented. These algorithms have been coded as functions in the C language. Each of the functions resides in its own separate file for developmental purposes and once debugged, is linked to a main function for execution. Together, these files form a growing library of robotic control strategies. The library also contains a utility file which consists of the main function and various user-interface functions. These functions were designed to create an environment conducive to the development, integration, execution and evaluation of path planning and control algorithms. The library source code can be found in Appendix K.

The following sections outline the operation of the master processor. The first section describes the sequence of events that lead up to the execution of a move and hence the manner in which the library files are tied together. The second section lists the operations that are performed by the master during the move. This is followed by a description of the various control and path planning algorithms currently in the library.

### 4.4.1 Master Sequence of Operation

The sequence of operation of the master software is shown in Figure 4.10. The events are listed chronologically and are displayed in their respective files.

#### **UTILITY FILE** TYPICAL CONTROL PROGRAM FILE TYPICAL PATH PLANNER FILE maın Display Default Parameters Call display\_current\_position -Call. find-HOME For each parameter Prompt for the following -Call. controller\_menu Call. choose\_new\_gam -preferred space (Cart or joint) -number of via points find-HOME Display sampling period -location of via points and prompt for a change -desired time to reach via points ·Signal slave to enter its end point find-HOME mode and wait -Call: robot\_error\_check -duration of the move for completion -Call path planner menu controller\_menu (points are entered by calling (pass sampling period) either get\_cartesian\_position Display control programs Allocate memory for on-line or gat\_joint\_position -Call: selected program sampled data file Allocate memory for the path choose new gain Signal slave to enter control Compute path mode Execute move as per -Display current gain and prompt for a change the control scheme Return to control program (generate sampled data files) (return address of start of path) path\_planner\_menu Call display\_errors -Display Path Planner programs Save sampled date in Matlab--Call. selected program readable format (pass sampling period) Free allocated memory display\_current\_position Return to Main -Display robot position in both Cart. & joint space get\_cartesian\_position -Prompt for entry of desired Cartesian point -Perform inv. kinematics get\_joint\_position -Prompt for entry of desired joint positions robot\_error\_check -Check error register and alert user to any joint overrun errors Request manual correction

Figure 4.10. Sequence of Operation of the Master Software.

display\_errors
-Check er:or register and display any errors

The sequence begins in the main function with a call to the find-HOME function. This function clears the DPR error register, commands the slave to enter its find-HOME procedure by writing 0080H to the command register, and waits for the slave to clear the register indicating that HOME was found.

Following the find-HOME function the controller\_menu function is called whereby the user is prompted for the selection of a control algorithm. The menu also includes the find-HOME option

The control algorithms typically begin by displaying the parameter (gain) default values and then prompting the user for changes. Parameter changes are handled by the choose\_new\_gain function. The control algorithm proceeds by displaying the sampling period (calculated from the value in the sample\_period register) and prompting the user for a change. Next, the robot\_error\_check function is called whereby the error register is checked and the user alerted to any joint overruns. The function requires that the overrun joints be manually moved to correct the error. The control algorithm then calls the path\_planner\_menu function where the user is prompted for the selection of a path planning algorithm.

The path planning algorithms typically begin by calling the display\_current\_position function which displays the manipulator's current point in both Cartesian and joint space (the term "point" refers to both position and orientation). The user is then prompted for the preferred space (Cartesian or joint) in which to enter desired path points. The user is then prompted for entry of the path end point, the duration of the move, and if the planner has via point specification capability, the number of via points, their values, and the desired time between successive points.

The input of a path point is handled by one of two functions: get\_joint\_position or get\_Cartesian\_position. The latter function solves the inverse kinematic equations. Both functions warn when the desired path point lies outside the manipulator's workspace.

Once all the necessary information has been entered, the entire path (the position of each joint at every sampling period) is computed off-line and stored contiguously in

memory. A flag indicates the path end-point. The arrangement of the path in memory is shown below.

| sample period 1, joint 1 pulse count       |
|--------------------------------------------|
| sample period 1, joint 2 pulse count       |
| sample period 1, joint Z pulse count       |
| sample period 1, joint Roll pulse count    |
| sample period 2, joint 1 pulse count       |
| :                                          |
| ·                                          |
| last sample period, joint Roll pulse count |
| end-of-path flag                           |

Lastly, the path planner algorithm returns to the control algorithm, a pointer to the first path point. In the control algorithm, the master allocates space in its memory to hold the data that will be obtained during the move for future analysis. The control algorithm then checks to make sure there are no joint overruns and prompts the user to begin or abort the move. If the user aborts the move, control passes to the controller\_menu function. If the user specifies to begin the move, the master signals the slave to enter its control mode (sets bit 0 in the command register). The succeeding sequence of operations is described in the following section.

### 4.4.2 Servo Control Loop

The master waits for the slave to signal the start of a new sampling period (0001H in the timing\_A register). When the signal arrives, the master begins by storing certain variables from the previous period for use in the present sampling period (e.g., joint positions from the previous period are employed to compute joint velocities using the difference method). When the actual joint positions become available (indicated by 0003H in timing\_A), the master computes the required tracking errors. The master then

computes the torques using one of the two control algorithms (see Section 4.4.3). It stores them in the torque registers and sets the 'torques available' semaphore (bit 0) in timing\_B. At this point, the slave verifies the desired torques and the master saves the relevant data (e.g., actual and desired joint positions, torques) in data files for future analysis. If the slave finds the torques are acceptable, it clears timing\_B; and the master subsequently clears timing\_A. The master ther waits for the start of the next servo loop. If the slave indicates an error, the master exits the control mode before the start of the next sampling period.

If the entire move is successfully executed (no errors), the control program saves the sampled data files on hard disk in MATLAB readable format for later analysis using PC- or PRO-MATLAB. If the move ends prematurely due to an error, the display\_errors function is called which alerts the user as to the nature of the error.

#### 4.4.3 Robot Control Programs

The library at present contains two robot control algorithms: Proportional-Derivative (PD) control, and decentralized adaptive control.

#### 4.4.3.1 PD Control

The PD control algorithm provides error-driven, independent joint control whereby each joint is controlled separately by a simple position-velocity servo-loop with predefined constant gains. The joint torque at a sampling instant N is given by

$$\tau(N) = k_p e_p(N) + k_v e_v(N)$$
 (4.1)

where  $e_p(N)$  =desired joint position(N)-actual joint position(N) and  $e_v(N) = \frac{e_p(N) - e_p(N-1)}{T_g}$  are the position and velocity tracking errors, and  $k_p > 0$  and  $k_v > 0$  are the position and velocity constant scalar feedback gains.

The PD controller's main attribute is that it requires few mathematical computations and thus can be implemented at high servo rates.

# 4.4.3.2 Decentralized Adaptive Control

The decentralized adaptive controller is based on an algorithm developed at the Jet Propulsion Laboratory by Seraji [17]. The control scheme is based on the independent joint control concept, does not use the complex manipulator dynamic model, and each joint is controlled simply by a PID feedback controller with adjustable gains. The reader should note that the control scheme as presented in [17] features a position-velocity-acceleration feedforward controller (with adjustable gain). However, in our implementation (as in Seraji's practical implementation), the feedforward controller is not included to reduce the on-line computation time.

The control algorithm for each joint is as follows. The motor torque  $\tau$  computed at a sampling instant N is given by

$$\tau(N) = f(N) + k_p e_p(N) + k_v e_v(N)$$
 (4.2)

where f(n) is an auxiliary signal produced by the adaptation scheme to improve tracking performance and partly compensate for disturbances,  $k_p$  is the proportional feedback gain,  $k_v$  is the velocity feedback gain,  $e_p(N) = \text{desired joint position}(N)$  - actual joint position(N) is the position error, and  $e_v(N) = \frac{e_p(N) - e_p(N-1)}{T_s}$  is the velocity error formed by the software. The auxiliary signal and the controller gains are obtained by the following recursive adaptation laws:

$$f(N) = f(N-1) + \delta \frac{T_s}{2} [r(N)+r(N-1)] + \rho [r(N)-r(N-1)]$$
 (4.3)

$$k_p(N) = k_p(N-1) + \alpha_p \frac{T_g}{2} [r(N)e_p(N) + r(N-1)e_p(N-1)] + \beta_p [r(N)e_p(N) - r(N-1)e_p(N-1)]$$

$$k_{v}(N) = k_{v}(N-1) + \alpha_{v} \frac{T_{s}}{2} \left[ r(N)e_{v}(N) + r(N-1)e_{v}(N-1) \right] + \beta_{v}[r(N)e_{v}(N) - r(N-1)e_{v}(N-1)]$$

where  $r(N) = \omega_p e_p(N) + \omega_v e_v(N)$  is a weighted error with  $\{\omega_p, \omega_v\}$  being positive constant scalar weighting factors which reflect the relative significance of the position and velocity errors,  $\{\delta, \alpha_p, \alpha_v\}$  are positive constant scalar integral adaptation gains, and  $\{\rho, \beta_p, \beta_v\}$  are non-negative proportional adaptation gains.

### 4.4.4 Path Planning Programs

The path planning algorithms are called by the control programs and compute the entire path off-line prior to the execution of the move. This affords the master processor more time during the move to execute computationally intensive control algorithms. Two path planning algorithms have been coded for the library: a cycloid function-based generator and a cubic spline generator. The latter features via point specification capability.

### 4.4.4.1 The Cycloid Function

This path generator plans a path of duration T seconds between the initial point  $\theta(0)$  and the goal point according to the function

$$\theta(t) = \theta(0) + \frac{\Delta}{2\pi} [\omega t - \sin \omega t], \qquad 0 \le t \le T$$
 (4.4)

where  $\omega = 2 \pi / T$  and  $\Delta = \theta(T) - \theta(0)$ . The cycloid is a smooth trajectory with a bell-shaped velocity profile and a sinusoidal acceleration profile given by

$$\dot{\theta}(t) = \frac{\Delta}{t} [1 - \cos \omega t] 
\ddot{\theta}(t) = \frac{2\pi\Delta}{T^2} \sin \omega t, \quad 0 \le t \le T$$

$$\dot{\theta}(t) = \ddot{\theta}(t) = 0, \quad \tau \ge T.$$
(4.5)

## 4.4.4.2 Cubic Spline

The algorithm for this path generator enables the specification of via point locations and also provides automatic selection of joint velocities at the via points. The velocities at the via points are computed according to the following simple heuristic scheme. First, the slopes (average joint velocities) between adjacent via points is calculated. The velocity at a via point is then chosen to be zero if the two slopes on either side are of opposite sign. If the slopes are of the same sign, the velocity is computed as the average of the two slopes.

Now that both position and velocities for the joints at the via points are known, the algorithm determines the cubic splines linking adjacent via points. The cubic splines are governed by the following four constraints

$$\theta (0, = \theta_0,$$

$$\theta (t_f) = \theta_f,$$

$$\dot{\theta} (0) = \dot{\theta}_0,$$

$$\dot{\theta} (t_f) = \dot{\theta}_f,$$

$$(4.6)$$

where  $\theta_0$  and  $\theta_f$  are the joint positions at the beginning and end of the spline,  $\theta_0$  and  $\theta_f$  are the joint velocities at the beginning and end of the spline, and  $t_f$  is the time duration of the spline. The four constraints can be satisfied by the third degree (cubic) polynomial

$$\theta(t) = a_0 + a_1 t + a_2 t^2 + a_3 t^3. \tag{4.7}$$

The joint velocity and acceleration are given by the first and second derivatives

$$\dot{\theta}(t) = a_1 + 2a_2 t + 3a_3 t^2$$

$$\ddot{\theta}(t) = 2a_2 + 6a_3 t.$$
(4.8)

Substituting the four constraints into equations (4.7) and (4.8) gives four equations in four unknowns:

$$\theta_{0} = a_{0}$$

$$\theta_{f} = a_{0} + a_{1}t_{f} + a_{2}t_{f}^{2} + a_{3}t_{f}^{3}$$

$$\dot{\theta}_{0} = a_{1},$$

$$\dot{\theta}_{f} = a_{1} + 2a_{2}t_{f} + 3a_{3}t_{f}^{2}.$$
(4.9)

The solution of these equation for the coefficients a; is

$$a_{0} = \theta_{0},$$

$$a_{1} = \dot{\theta}_{0},$$

$$a_{2} = \frac{3}{t_{f}^{2}} (\theta_{f} - \theta_{0}) - \frac{2}{t_{f}} \dot{\theta}_{0} - \frac{1}{t_{f}} \dot{\theta}_{f},$$

$$a_{3} = -\frac{2}{t_{f}^{3}} (\theta_{f} - \theta_{0}) + \frac{1}{t_{f}^{2}} (\dot{\theta}_{f} + \dot{\theta}_{0}).$$

$$(4.10)$$

Expression (4.10) is sequentially evaluated at all adjacent points to derive the cubic splines. The end result is a smooth continuous path linking the initial point, all the via points and the goal point together.

## 4.5 SUMMARY AND FUTURE WORK

This chapter presented the software that was developed to implement a testbed for robotic control strategies. The slave processor software was designed to support a variety of control algorithms and therefore does not require any immediate modifications. Of course, implementation of some of the hardware suggested in Section 3.8 would demand

modification to the slave software. This would also be the case if the ARC were to be used to control a robotic system other than the IBM 7545.

With regard to the master, future work may consist of expanding the library through the implementation of other advanced control strategies. An interesting project here is the provision for on-line path planning. This would allow the researcher to address the problem of collision avoidance (this, of course, would require hardware for workcell operation). Other areas for future work include the enhancement of the user-interface software in the utility file. This may be accomplished through the use of specialized programs such as C-Scape.

With suitable development, the ARC could serve in the industrial domain. This would require a compiler or interpreter for the translation of an existing (eg., AML) or custom-designed high-level application language.

## **CHAPTER 5**

### **EXPERIMENTAL RESULTS**

This chapter presents the results of three experiments which were conducted to verify the operation of the ARC and to judge its effectiveness and performance in robotic control applications. The experiments involved the use of both the cubic spline and cycloid function path generators, as well as the PD and adaptive control schemes. The servo rates for these latter controllers was set to 1 KHz and 250 Hz, respectively. It should be noted that although the controller gains were tuned, no effort was made to optimize the gains for the experiments. Furthermore, comparisons between the control schemes are not given. It is not the purpose of this project to make value judgements regarding the control schemes but simply to show the effectiveness of the ARC for implementing such schemes. A comparison between the performance of the ARC and the MTCB would have been desirable. Unfortunately, the MTCB does not provide the means for acquiring data during execution of a move. The mere fact that the ARC provides this feature makes it a valuable research tool for the evaluation of robotic control strategies.

In first experiment, the ARC was used to evaluate a controller's ability to compensate for unexpected disturbances. A path was planned by the cycloidal function generator for joints 1 and 2 to move from 0° to 90° in 2.5 seconds and for joints Z and Roll to maintain their position. A disturbance was applied at approximately mid-path when the end-effector came into contact with a movable object which was constrained to move only along the x direction. The end-effector was required to push the object as it proceeded toward its goal position. The experiment was performed under both PD and adaptive control. Figures 5.1, 5.2, and 5.3 show the joint 1 tracking errors, the joint 2 tracking errors, and the torque outputs for the PD controller. Figures 5.4, 5.5, and 5.6 show the

corresponding results for the adaptive controller. The effect of the disturbance is clearly shown in each graph. However, despite the disturbance, both controllers are able to compensate and reduce the tracking error within a finite period of time.

In the second experiment, the ARC was used to demonstrate a controller's ability to track a closed path. The cubic spline generator with its via point capability was used to plan a closed path for all four of the manipulator's joints. The duration of the move was 8 seconds and three via points were specified at 2 second intervals. The set of joint variables (q1, q2, Z, Roll) for the start point, the three via points and the end point were (0°, 90°, 0 mm, 0°), (90°, 0°, -70 mm, 135°), (180°, 90°, -210 mm, -30°), (90°, 135°, -140 mm, -150°), (0°, 90°, 0 mm, 0°), respectively. Figure 5.7 shows the desired path in the x-y plane (Cartesian space). Figures 5.8, 5.9, 5.10, 5.11, and 5.12 show the joint tracking errors for joints 1 2, Z, and Roll and the torques for the PD controller, respectively. Corresponding results for the adaptive controller are shown in Figures 5.13, 5.14, 5.15, 5.16 and 5.17. Despite the fact the gains were not optimized, tight control over all four joints was achieved.

The third experiment demonstrated the ARC's effectiveness in handling slow and fast trajectories. In this experiment, the starting point was (90°, 90°, -70 mm, -35°) and the goal point was (10°, 30°, 0 mm, 0°). The cubic spline generator was used to generate a 3 sec., a 2 sec., and a 1 sec. trajectory between these two points for use by the adaptive controller. The results are shown in Figures 5.18 - 5.32. As expected, the joint tracking errors and the torque demands were highest for the 1 sec. trajectory. Attempts at trajectories faster than 1 sec. were met with excessive torque demand errors.

The finite processing power of the ARC imposes a limitation on the maximum servo rate of the controller (e.g., 1000 Hz for PD control, 250 Hz for adaptive control). This limitation reflects on the maximum trajectory that can be executed without changing the controller gains. A faster servo rate will generally produce smaller tracking errors and

thus smaller torque demands for a particular sampling period. This in turn will allow for faster trajectories to be executed. For a fixed servo rate, faster trajectories may be achieved by detuning the controller gains. The detuned gains prevent the demanded torques from reaching the limits for the robot. The faster trajectories, however, come at the expense of accuracy since the detuned gains will degrade performance by increasing the tracking errors.



Figure 5.1. Joint 1 Tracking Errors, PD Control, Experiment 1.



Figure 5.2. Joint 2 Tracking Errors, PD Control, Experiment 1.



Figure 5.3. Torques, PD Control, Experiment 1.



Figure 5.4. Joint 1 Tracking Errors, Adaptive Control, Experiment 1.



Figure 5.5. Joint 2 Tracking Errors, Adaptive Control, Experiment 1.



Figure 5.6. Torques, Adaptive Control, Experiment 1.



Figure 5.7. Desired Path of Joints 1 and 2 for Experiment 2 Shown in Cartesian Space.



Figure 5.8. Joint 1 Tracking Errors, PD Control, Experiment 2.



Figure 5.9. Joint 2 Tracking Errors, PD Control, Experiment 2.

78



Figure 5.10. Joint Z Tracking Errors, PD Control, Experiment 2.



Figure 5.11. Joint Roll Tracking Errors, PD Control, Experiment 2.



Figure 5.12. Torques, PD Control, Experiment 2.



Figure 5.13. Joint 1 Tracking Errors, Adaptive Control, Experiment 2.



Figure 5.14. Joint 2 Tracking Errors, Adaptive Control, Experiment 2.



Figure 5.15. Joint Z Tracking Errors, Adaptive Control, Experiment 2.



Figure 5.16. Joint Roll Tracking Errors, Adaptive Control, Experiment 2.



Figure 5.17. Torques, Adaptive Control, Experiment 2.



Figure 5.18. Joint 1 Tracking Errors, Adaptive Control, Experiment 3, Three Sec. Move.



Figure 5.19. Joint 2 Tracking Errors, Adaptive Control, Experiment 3, Three Sec. Move.



Figure 5.20. Joint Z Tracking Errors, Adaptive Control, Experiment 3, Three Sec. Move.



Figure 5.21. Joint Roll Tracking Errors, Adaptive Control, Experiment 3, Three Sec. Move.



Figure 5.22. Torques, Adaptive Control, Experiment 3, Three Sec. Move.



Figure 5.23. Joint 1 Tracking Errors, Adaptive Control, Experiment 3, Two Sec. Move.



Figure 5.24. Joint 2 Tracking Errors, Adaptive Control, Experiment 3, Two Sec. Move.



Figure 5.25. Joint Z Tracking Errors, Adaptive Control, Experiment 3, Two Sec. Move.



Figure 5.26. Joint Roll Tracking Errors, Adaptive Control, Experiment 3, Two Sec. Move.



Figure 5.27. Torques, Adaptive Control, Experiment 3, Two Sec. Move.



Figure 5.28. Joint 1 Tracking Errors, Adaptive Control, Experiment 3, One Sec. Move.



Figure 5.29. Joint 2 Tracking Errors, Adaptive Control, Experiment 3, One Sec. Move.



Figure 5.30. Joint Z Tracking Errors, Adaptive Control, Experiment 3, One Sec. Move.



Figure 5.31. Joint Roll Tracking Errors, Adaptive Control, Experiment 3, One Sec. Move.



Figure 5.32. Torques, Adaptive Control, Experiment 3, One Sec. Move.

#### CHAPTER 6

#### CONCLUSIONS

An architecture for implementing advanced robot control strategies has been presented. The work that was done is a contribution to the effort that is being made to enhance the operation of conventional industrial robotic systems. The primary goal of the work has been to create an inexpensive (relative to the cost of the robot) environment for implementing robot control systems which is easy-to-develop and maintain and applicable to a variety of robotic systems. The resulting ARC system has been designed as a replacement for manufacturer-supplied controllers to provide greater flexibility. In the design process, we have considered the following aspects of the controller:

- 1) the design of the hardware component;
- 2) the design of the software component; and
- 3) the application of the above-mentioned components to the IBM 7545 Manufacturing System.

From the hardware standpoint, we have created a system possessing sufficient processing power to execute advanced robot control strategies at respectable servo rates. From the software standpoint, we have constructed the building blocks to facilitate the development and implementation of these control algorithms.

In the system design process, we have recognized the fact that there exist common low-level tasks associated with all robot control algorithms, such as robot state acquisition, torque verification, joint overrun checking, and sample period timing. As a result, the controller has been designed to incorporate a slave processor which executes the procedures associated with these tasks. The slave unloads a significant percentage of the

computational burden and thus allows the implementation of more computationally complex control schemes on the master processor.

Through experimentation, we have demonstrated the effectiveness of the ARC to control the 7545 manipulator. We have shown that the ARC is capable of replacing the IBM-supplied controller (MTCB) and is capable of executing both simple and sophisticated control and path planning strategies at respectable servo rates, e.g., 1000 Hz for PD control and 250 Hz for decentralized adaptive control. The experimental results given in Chapter 5 show that the ARC delivers good performance for a variety of control strategies for various types of trajectories.

In its present form, the ARC is well suited for use in research environments for the implementation and evaluation of new control strategies. It was for this purpose that the project was undertaken, and the above conclusions show that this intention has been fully realized using a typical industrial robot.

#### REFERENCES

- [1] D. G. Bihn and T. C. Steve Hsia, "Universal Six-Joint Robot Controller," IEEE Control Systems Magazine, vol. 8, no. 1, pp. 31-36, 1988.
- [2] M. Erlic and W. -S. Lu, "Computer Independent Driver-Sensor Interface (DSI) for Closed Loop Control of a Mechanical Manipulator," in *Proc. of IEEE Canadian Conf. on Elec. & Comp. Eng.*, Montréal, Québec, 1989, pp. 498-501.
- [3] D. B. Stewart, D. E. Schmitz, and P.K. Khosla, "CHIMERA II: A Real-Time Multiprocessing Environment for Sensor-Based Robot Control," in Proc. of IEEE International Symposium on Intelligent Control, Albany, NewYork, 1989, pp. 265-271.
- [4] V. Hayward, L Daneshmend, and S Hayati, "An Overview of KALI: A System to Program and Control Cooperative Manipulators," in *Advanced Robotics*, ed. K. J. Waldron, Springer Verlag, New York, 1989.
- [5] D. G. Bihn, A Universal Six Joint Robot Controller. M.S. Thesis, Department of Electrical and Computer Engineering, University of California, Davis, 1986.
- [6] M. Erlic, A Study in Unconstrained Motion Control of a Robotic Manipulator. Master's Thesis, Dept. of Electrical and Computer Engineering, University of Victoria, 1989.
- [7] IBM 7545 Manufacturing System Hardware Library, IBM Corporation, Boca Raton, Florida, 1984.
- [8] Print Motor, Yaskawa Electric Mfg. Co., Ltd., Fairfield, New Jersey, 1982.
- [9] Minertia Motor for Industrial Robot Drives, Yaskawa Electric Mfg. Co., Ltd., Fairfield, New Jersey, 1989.
- [10] DC Servomotor Controller Servopack, Yaskawa Electric Mfg. Co., Ltd., Fairfield, New Jersey, 1985.
- [11] IBM Personal System/2 Model 50 and 60 Technical Reference, International Business Machines Corporation, 1987.
- [12] High Performance CMOS Data Book, Integrated Device Technology, Santa Clara, California, 1988.
- [13] Embedded Controller Handbook, Intel Corporation, Santa Clara, California, Volume II, 16-bit, 1988.
- [14] EV80C196KA Microcontroller Evaluation Board User's Manual, Intel Corporation, Santa Clara, California, Release 001, March 20, 1988.
- [15] Data Conversion Products Databook, Analog Devices, Norwood, Ma., 1988.

- [16] General Purpose Motion Control IC, HCTL-1000, Hewlett Packard, Palo Alto, California, 1987.
- [17] H. Seraji, "Decentralized Adaptive Control of Manipulators: Theory, simulation and Experimentation," *IEEE Trans. on Robotics and Automation*, vol. RA-5, no. 2, pp. 183-201.

# APPENDIX A

# IBM 7545 WIRING DIAGRAMS

The wiring diagrams in this appendix, taken from [7], provide the necessary details to bypass the MTCB component of the IBM 7545 Manufacturing system.



106

THETA 2 MOTOR AND FEEDBACK

**2-AXIS MOTOR AND FEEDBACK** 









APPENDIX B

JOINT MOTOR RATINGS AND SPECIFICATIONS

The following table contains the ratings and specifications for the Yaskawa Print motors [8] used in joints 1 and 2 and Minertia motors [9] used in joints 2 and Roll.

| Parameter                                     | Joint 1 | Joint 2 | Joints Z and Roll |
|-----------------------------------------------|---------|---------|-------------------|
| Rated Output (W)                              | 200     | 100     | n/a               |
| Rated Torque (kg•cm)                          | 6.5     | 2.43    | <b>57</b> .3      |
| Rated Speed (rpm)                             | 3000    | 4000    | 3000              |
| Rated Armature Voltage (V)                    | 42      | 26      | n/a               |
| Rated Armature Current (A)                    | 6.4     | 5.5     | n/a               |
| Power Rate (kW/sec)                           | 2.4     | 1.3     | 4.45              |
| Torque Inertia Ratio (rads/sec <sup>2</sup> ) | 4300    | 5200    | <b>12</b> 600     |
| Max. Torque / 1 sec (kg • cm)                 | 36.4    | 14.4    | n/a               |
| Max. Armature Current/1 sec (A)               | 33      | 29      | 22.1              |
| Max. Safe Speed/1 sec (rpm)                   | 4950    | 6600    | 4000              |
| Armature Inertia (kg•cm²)                     | 1.5     | 0.46    | n/a               |
| Armature Resistance (Ω)                       | 0.68    | 0.54    | 0.94              |
| Armature Inductance (mH)                      | 0.06    | 0.02    | 0.9               |
| Voltage Constant (mV/rpm)                     | 11.5    | 5.2     | 8.5               |
| Torque Constant (kg•cm/A)                     | 0.23    | 0.11    | 0.83              |
| Mech. Time Constant (msec)                    | 8.5     | 10      | 4.0               |
| Elec. Time Constant (msec)                    | 0.09    | 0.04    | 0.96              |

n/a - not available

#### APPENDIX C

#### **CALIBRATION**

Calibration of the ARC system is a two part process. The first part involves the calibration of the DACs. The second part involves the calibration of the gain of the Speed Amplifier of the Servopacks. The calibration steps listed below should be performed for all four joints.

#### **DAC** Calibration

- 1. Disconnect the DAC from the Servopack input.
- Connect a digital voltmeter (DVM) to the DAC output.
- 3. Write OH to the DAC and adjust the potentiometer at the DAC's REF OUT terminal to obtain a -10.0000 V dc output.
- Write OFFFH to the DAC and adjust the potentiometer at the DAC's REF IN terminal to obtain a +9.9951 V dc output.
- 5. Reconnect the DAC to the Servopack input.

#### Speed Amplifier Calibration

- 1. Disconnect the motor from Servopack. This can be done by unplugging connectors CN3H, CN4H, CN5H, and CN6H for joint motors 1, 2, Z, and Roll, respectively.
- 2. Determine the saturation point of the Servopack's Speed Amplifier as follows:
  - 2.1. Disconnect the ARC from the Servopack.
  - 2.2. Ground the TG Feedback input on the Servopack.
  - 2.3. Connect a DVM to the Servopack's C<sub>ref</sub> testpoint. This point is at the output of the Speed Amplifier.
  - 2.4. Connect a variable DC power supply in series with a 220K (1/4 W) resistor to the Servopack's Speed Reference input.
  - 2.5. Starting at 0 V dc, increase the power supply voltage and note the voltage at C<sub>ref</sub> just before the Speed Amplifier saturates. The voltage

- should be approximately -4.4, -4.0, -3.2, and -3.0 for joints 1, 2, Z, and Roll, respectively.
- 2.6. Disconnect the power supply and resistor and reconnect the ARC. The DAC should now be connected through the potentiometer to the Servopack Speed Reference input and all selector switches should be set for ARC control (see Figure 2.6 for details).
- 3. Write 0FFFH to the DAC and adjust the potentiometer until  $C_{\rm ref}$  reaches the voltage noted in step 2. For better resolution, a resistor can be inserted in series with the potentiometer. The potentiometer can then be substituted by one of a lower value.

### APPENDIX D

# **PS/2 TIMING DIAGRAMS**

This appendix gives the I/O and Memory cycle timing diagrams for the IBM PS/2 model 50 computer as they appear in [11].



|           | Timing Parameter                                         | Min/Max    | Note |
|-----------|----------------------------------------------------------|------------|------|
| T1        | Status active (low) from ADDRESS,M/-IO,-REFRESH valid    | 10 / - ns  |      |
| T2        | -CMD active (low) from Status active (low)               | 55 / - na  | 2    |
| ТЗ        | -ADL active (low) from ADDRESS,M/lÒ,-REFRESH valid       | 45 / - ns  | _    |
| T4        | -ADL active (low) to -CMD active (low)                   | 40 / - ns  |      |
| <b>T5</b> | -ADL active (low) from Status active (low)               | 12 / - na  |      |
| T6        | -ADL pulse width                                         | 40 / - ns  |      |
| 17        | Status hold from -ADL inactive (high)                    | 25 / - ns  | 2    |
| T8        | ADDRESS,M/-IO,-REFRESH,-SBHE hold from -ADL inactive     | 25 / - na  | 2    |
| T9        | ADDRESS,M/-IO,-REFRESH,-SBHE hold from -CMD active (low) | 30 / - ns  | 3    |
| T10       | Status hold from -CMD active                             | 30 / - ns  | 2    |
| T11       | -SBHE setup to -ADL inactive                             | 40 / - ns  | 2    |
| T12       | -SBHE setup to -CMD active                               | 40 / - ns  | 2    |
| T13       | -CD DS 16 active (n) (low) from                          | - / 55 ns  | 3    |
|           | ADDRESS,M/-IO,-REFRESH valid                             |            | -    |
| T14       | -CD SFDBK active (low) from                              | - / 60 ns  | 1    |
|           | ADDRESS,M/-IO,-REFRESH valid                             |            |      |
| T15       | -CMD active (low) from Address valid                     | 85 / - ns  | 2    |
| T16       | -CMD pulse width                                         | 90 / - ns  |      |
| T17       | Write data setup to -CMD active (low)                    | 0 / - ns   |      |
| T18       | Write data hold from -CMD inactive (high)                | 30 / - ns  |      |
| T19       | Status to Read Data valid (Access Time)                  | - / 125 ns |      |
| T20       | Read data valid from -CMD active (low)                   | - / 60 ns  |      |
| T21       | Read data hold from -CMD inactive (high)                 | 0 / - ns   |      |
| T22       | Read data bus tri-state from -CMD inactive (high)        | - / 40 ns  |      |
| T23       | -CMD active to next -CMD active                          | 190 / - ns | 4    |
| T23A      | -CMD inactive to next -CMD active                        | 80 / - ns  | •    |
| T23B      | -CMD inactive to next -ADL active                        | 40 / - ns  |      |
| T24       | Next Status active (low) from Status inactive            | 30 / - ns  |      |
| T25       | Next Status active (low) to -CMD inactive                | - / 20 ns  |      |

Figure 2-34. I/O and Memory Default Cycle (200 nanoseconds minimum)

#### Notes:

- 1. All slaves must drive -CD SFDBK whenever selected either by the system microprocessor or the DMA Controller. The slaves do not drive -CD SFDBK when they are selected by the 'setup' signal.
- It is recommended that slaves use transparent latches to latch information with the leading or trailing edge of -ADL or with the leading edge of -CMD.
- -CD DS 16 and -CD SFDBK must be driven by unlatched address decodes because the next address may come early into the current cycle.
- 4. Any master in any system, including the system microprocessor or DMA controller, can operate at a performance less than the level specified. Designers should not design to a given performance level as this level can be reduced by CD CHRDY, a lower microprocessor rate, a lower DMA controller rate, or system contention.
- Model 50 and Model 60 automatically extend all default cycles to synchronous extended cycles. Adapter designs should support the 200 nanosecond default cycle to assist portability to other systems or drive CD CHRDY regardless of the system synchronous extension cycle.

### **Default Cycle Return Signals**



| Min/Mex   | Note                                |
|-----------|-------------------------------------|
| - / 20 ns | 1                                   |
| - / 20 ns | 1                                   |
| - / 20 ns | 2                                   |
| - / 20 ns | 3                                   |
|           | - / 20 ns<br>- / 20 ns<br>- / 20 ns |

Figure 2-35. Default Cycle Return Signals (200 nanoseconds minimum)

#### Notes:

- 1. This signal is developed from a negative OR of signals received from each channel connector.
- 2. CHRDYRTN becomes active 40 nanoseconds maximum after -ADL becomes active.
- 3. This signal is developed from a positive AND of signals received from each channel connector.

#### 2-64 Micro Channel Architecture, Channel Timing

#### Synchronous Special Case of Extended Cycle

A Synchronous Extended cycle occurs when a slave releases CD CHRDY synchronously within the specified time after the leading edge of -CMD. The slave provides the Read data within a specified time from -CMD. The timing sequence is illustrated by the following figure.



Figure 2-36. Timing Sequence for the Synchronous Special Case of Extended Cycle

# Synchronous Extended Cycle (300 nanoseconds minimum - Special Case)



|             | Timing Parameter                                                                    | Min/Max    | Note             |
|-------------|-------------------------------------------------------------------------------------|------------|------------------|
| T13         | -CD DS 16 (n) active (low) from<br>ADDRESS.M/-IO,-REFRESH valid                     | - / 55 ns  | 2                |
| T14         | -CD SFDBK (n) active (low) ADDRESS,M/-IO,-REFRESH valid                             | - / 60 ns  | 2                |
| T13A        | -CMD pulse width                                                                    | 190 / - ns |                  |
| T26         | CD CHRDY (n) inactive (low) from ADDRESS valid                                      | - / 60 ns  | 3,<br>See<br>T27 |
| T27         | CD CHRDY (n) inactive (low) from Status active                                      | 0 / 30 ns  | 3                |
| T28<br>T28D | CD CHRDY (n) release (high) from -CMD active (low) Read Data valid from -CMD active | 0 / 30 ns  | 1                |
|             | (when used along with T28)                                                          | 0 / 160 ns | 1                |

Figure 2-37. Synchronous Extended Cycle (300 nanoseconds minimum - Special Case)

#### Notes:

- CD CHRDY is released by a slave performing a 300 nanoseconds extended cycle synchronous with the leading edge of -CMD.
   Since CD CHRDY is generally an asynchronous signal, this is referred to as a purely synchronous special case.
- 2. This is the same as default cycle timing (listed here for emphasis).
- 3. T27 is valid only when Status becomes active 30 nanoseconds or more after the address is valid.
- 4. If Status overlaps with previous -CMD, then CD CHRDY state is not valid during the overlapped period.
- 5. Slaves must not hold CD CHRDY inactive (low) in excess of 3.0 microseconds.

### APPENDIX E

#### **DUAL-PORT RAM TIMING WAVEFORMS**

This appendix gives the timing information for the IDT7132/IDT7142 dual-port RAMs as found in [12].

IDT713218A/LA AND IDT714218A/LA CMOS DUAL-PORT RAMS 16K (2Kx8-BIT) WITH INTERRUPTS

MILITARY AND COMMERCIAL TEMPERATURE RANGES

# AC ELECTRICAL CHARACTERISTICS OVER THE OPERATING TEMPERATURE AND SUPPLY VOLTAGE RANGE

| SYMBOL          | PARAMETER                           |     | VLA35 <sup>(7)</sup><br>VLA35 <sup>(7)</sup><br>MAX |    | A/LA45<br>A/LA45<br>MAX. |    | A/LASS<br>A/LASS<br>MAX, |    | A/LA70 <sup>(3)</sup><br>SA/LA70 <sup>(3)</sup><br>MAX. | UNIT |
|-----------------|-------------------------------------|-----|-----------------------------------------------------|----|--------------------------|----|--------------------------|----|---------------------------------------------------------|------|
| READ CY         | CLE                                 |     |                                                     |    |                          |    |                          |    |                                                         |      |
| 1 <sub>RC</sub> | Read Cycle Time                     | 35  | _                                                   | 45 | -                        | 55 |                          | 70 | -                                                       | ns   |
| 144             | Address Access Time                 | -   | 35                                                  |    | 45                       | -  | 55                       | -  | 70                                                      | ns   |
| TACE            | Chip Enable Access Time             |     | 35                                                  | -  | 45                       | -  | 55                       | -  | 70                                                      | ns   |
| TAGE            | Output Enable Access Tirme          | T - | 25                                                  | -  | 30                       | -  | 35                       | _  | 40                                                      | U2   |
| t <sub>oH</sub> | Output Hold From Address Change     | 0   | _                                                   | 0  | -                        | 0  | -                        | 0  | -                                                       | ns   |
| tiz             | Output Low Z Time (1 4)             | 5   |                                                     | 5  |                          | 5  | -                        | 5  | -                                                       | ns   |
| t <sub>HZ</sub> | Output High Z Time (1-4)            | _   | 15                                                  | -  | 20                       | -  | 30                       | _  | 35                                                      | ns   |
| 1 <sub>PU</sub> | Chip Enable to Power Up Time (4)    | 0   |                                                     | 0  |                          | 0  |                          | 0  | -                                                       | ns   |
| 1 <sub>PD</sub> | Chip Disable to Power Down Time (4) | -   | 50                                                  | -  | 50                       | -  | 50                       | _  | 50                                                      | ns   |

#### NOTES

- Transition is measured ±500mV from low or high impedance voltage with load (Figures 1, 2 and 3)
- 2 0°C to +70°C temperature range only
- -55°C to + 125°C temperature range only
- 4. This parameter guaranteed but not tested

#### TIMING WAVEFORM OF READ CYCLE NO. 1, EITHER SIDE (1, 2, 4)



#### TIMING WAVEFORM OF READ CYCLE NO. 2, EITHER SIDE (1, 2)



#### NOTES

- R/W is high for Read Cycles
- Device is continuously enabled  $\overline{CE} = V_{ij}$
- Addresses valid prior to or coincident with  $\overrightarrow{CE}$  transition low  $\overrightarrow{DE} = V_k$

TIMING WAVEFORM OF WRITE CYCLE NO. 1, R/W CONTROLLED TIMING (1,2,2,7)



#### TIMING WAVEFORM OF WRITE CYCLE NO. 2, CE CONTROLLED TIMING (1.2.8.1)



#### NOTES.

1. WE must be high during all address transitions.

can be as short as the specified  $t_{\mbox{\scriptsize WP}}$ 

- 2. A write occurs during the overlap (t<sub>EW</sub> or t<sub>Wh</sub>) of a low CE and a low R/W.

  3. t<sub>WR</sub> is measured from the earlier of CE or R/W going high to the end of write cycle.
- typ is measured north the sellier of CC or HAM going right to the earth of white cycle.
   During this period the I/O pris are in the output state and input signals must not be applied.
   If the CE low transition occurs simultaneously with or after the RAW low transition, the outputs remain in the high impedance state.
- 6 Transition is measured ±500mV from sheady state with a 5pF load (including access and ig). This parameter is earnised and not 100% tested.
  7 If OE is low during a R/W controlled write cycle. the write pulse must be the larger of t<sub>wo</sub> or (t<sub>WZ</sub> + t<sub>QW</sub>) to allow the I/O drivers to turn off data to be placed on the bus for the required t<sub>QW</sub>. If OE is high during an R/W controlled write cycle, this requirement does not apply and the write pulse.

# AC ELECTRICAL CHARACTERISTICS OVER THE OPERATING TEMPERATURE AND SUPPLY VOLTAGE RANGE

| SYMBOL          | PARAMETER                                   | 713218/<br>714218/<br>MIN | VLA35 <sup>(2)</sup><br>VLA35 <sup>(3)</sup><br>MAX |    | BALA45<br>BALA45<br>MAX. |    | A/LASS<br>IA/LASS<br>MAXL | 714215 | A/LA70 <sup>(3)</sup><br>A/LA70 <sup>(3)</sup><br>MAX | UNIT  |
|-----------------|---------------------------------------------|---------------------------|-----------------------------------------------------|----|--------------------------|----|---------------------------|--------|-------------------------------------------------------|-------|
| WRITE C         | CLE                                         |                           |                                                     |    |                          |    |                           |        |                                                       |       |
| 1wc             | Write Cycle Time (5)                        | 35                        | -                                                   | 45 | -                        | 56 | -                         | 70     | -                                                     | ne    |
| 1 <sub>EW</sub> | Chip Enable to End of Write                 | 30                        |                                                     | 35 | -                        | 40 | -                         | 50     | -                                                     | M     |
| 1 <sub>AW</sub> | Address Valid to End of Write               | 30                        | -                                                   | 35 | -                        | 40 | -                         | 50     | -                                                     | ne    |
| 1 <sub>AS</sub> | Address Set-up Time                         | 0                         | -                                                   | 0  |                          | 0  | -                         | 0      |                                                       | m     |
| 1 <sub>WP</sub> | Write Pulse Width                           | 30                        |                                                     | 35 | -                        | 40 |                           | 50     |                                                       | LIS   |
| twa             | Write Recovery Time                         | 0                         | -                                                   | 0  |                          | 0  | -                         | 0      |                                                       | ne    |
| tow             | Data Valid to End of Write                  | 20                        | -                                                   | 20 | -                        | 20 |                           | 30     |                                                       | m     |
| t <sub>HZ</sub> | Output High Z Time (1.4)                    | T -                       | 15                                                  | -  | 20                       | -  | 30                        |        | 35                                                    | ne    |
| t <sub>DH</sub> | Date Hold Time                              | 0                         | -                                                   | 0  |                          | 0  | _                         | 0      | -                                                     | ria . |
| twz             | Write Eriabled to Output in<br>High Z (1.4) | -                         | 15                                                  | -  | 20                       | -  | 30                        | -      | 35                                                    | ns    |
| low             | Output Active From End of Write (* 4)       | 0                         | -                                                   | 0  |                          | 0  | -                         | 0      | -                                                     | กร    |

#### NOTES:

- 1. Transition is measured: ±500mV from low or high voltage with load (Figures 1: 2 and 3).
- 2 0°C to +70°C temperature range only
- 3 -56°C to +125°C temperature range only
- 4. This parameter guaranteed but not tested
- 5 For MASTER/SLAVE combination two = take + twe

#### TIMING WAVEFORM OF READ WITH BUSY



#### TIMING WAVEFORM OF WRITE WITH BUSY



# AC ELECTRICAL CHARACTERISTICS OVER THE OPERATING TEMPERATURE AND SUPPLY VOLTAGE RANGE

| SYMBOL           | PARAMETER                                  |     | BA/LA35 <sup>(1)</sup><br>BA/LA35 <sup>(1)</sup><br>MAX. |    | SA/LA45<br>SA/LA45<br>MAX. |     | SA/LA55<br>SA/LA55<br>MAX. |     | SA/LA70 <sup>(2)</sup><br>SA/LA70 <sup>(2)</sup><br>MAX. | UNIT |
|------------------|--------------------------------------------|-----|----------------------------------------------------------|----|----------------------------|-----|----------------------------|-----|----------------------------------------------------------|------|
| BUSY TIM         | IING                                       |     |                                                          |    |                            |     |                            |     |                                                          |      |
| t <sub>we</sub>  | Write to BUSY (3 6)                        | 0   | -                                                        | 0  | _                          | -10 | _                          | -10 | _                                                        | ns   |
| twn              | Write Hold After BUSY (7)                  | 20  | -                                                        | 20 | _                          | 20  |                            | 20  | _                                                        | ns   |
| 1BAA             | BUSY Access Time to Address                |     | 35                                                       |    | 35                         | _   | 45                         |     | 45                                                       | ns   |
| 1 <sub>BOA</sub> | BUSY Disable Time to Address               |     | 30                                                       | _  | 35                         | -   | 40                         | -   | 40                                                       | ns   |
| 1 <sub>BAC</sub> | BUSY Access Time to Chip Enable            | T - | 30                                                       | _  | 30                         | -   | 35                         | -   | 35                                                       | ns   |
| 1 <sub>BDC</sub> | BUSY Disable Time to Chip Enable           |     | 25                                                       | -  | 25                         | _   | 30                         | -   | 30                                                       | ns   |
| 1 <sub>woo</sub> | Write Pulse to Data Delay (4)              | -   | 60                                                       | -  | 70                         | -   | 80                         | -   | 90                                                       | ns   |
| 1000             | Write Data Valid to<br>Read Data Delay (4) | _   | 35                                                       |    | 45                         | -   | 55                         | -   | 70                                                       | ns   |
| IAPS             | Arbitration Priority Set up Time           | 5   | _                                                        | 5  | -                          | 5   | _                          | 5   | -                                                        | ns   |
| 1800             | BUSY Disable to Valid Data (5)             | -   | Note 5                                                   | _  | Note 5                     | -   | Note 5                     | -   | Note 5                                                   | пş   |

#### NOTES.

- 1 0°C to +70°C temperature range only
- 2 -55°C to + 125°C temperature range only
- 3 For SLAVE part (IDT71421) only
- 4 Port to-port delay through RAM cells from writing port to reading port
- 5 1<sub>800</sub> is a calculated parameter and is the greater of 0 1<sub>WD0</sub> t<sub>WP</sub> (actual) or t<sub>D00</sub> t<sub>DW</sub> (actual)
- 6 To ensure that the write cycle is inhibited during contention
- 7. To ensure that a write cycle is completed after contention.

### TIMING WAVEFORM OF CONTENTION CYCLE NO. 1, CE ARBITRATION

### CE VALID FIRST:



#### CER VALID FIRST:



### TIMING WAVEFORM OF CONTENTION CYCLE NO. 2, ADDRESS VALID ARBITRATION (\*)

#### **LEFT ADDRESS VALID FIRST:**



#### RIGHT ADDRESS VALID FIRST:



# **APPENDIX F**

# **EV80C196KA SCHEMATIC DIAGRAM**

The following schematic diagram for the INTEL EV80C196KA evaluation board is







EVBDC 196KA

ECO Resilications Engineering

Title BECINE Contraction Beared

Sins Decimator Number

B Namery Number

Contract Number

Contract



# APPENDIX G

# 80C196KA BUS TIMING

The following information regarding the timing of the INTEL 80C196KA Microcontroller IC is taken from [13].



A.C. Characteristics (Over specified operating conditions)

These are ADVANCED specifications, the parameters may change before intel releases the product for sale.

Test Conditions Capacitive load on all pins = 100 pF, Rise and fall times = 10 ns, fosc = 12 MHz

#### The system must meet these specifications to work with the 80C196:

| Symbol | Description                       | Min      | Max                    | Unite | Notes    |
|--------|-----------------------------------|----------|------------------------|-------|----------|
| TAVYV  | Address Valid to READY Setup      |          | 2TOSC - 55             | ns    |          |
| TLLYV  | ALE Low to READY Setup            |          | Tosc - 55              | ns    |          |
| TYLYH  | NonREADY Time                     | Nou      | pper limit             | ns    |          |
| TCLYX  | READY Hold after CLKOUT Low       | 0        | T <sub>OSC</sub> - 30  | ns    | (Note 2) |
| TLLYX  | READY Hold after ALE Low          | Tosc + 5 | 2T <sub>QSC</sub> - 40 | ns    | (Note 2) |
| TAVGV  | Address Valid to Buswidth Setup   |          | 2T <sub>OSC</sub> - 55 | ns    |          |
| TLLGV  | ALE Low to Buswidth Setup         |          | T <sub>OSC</sub> 55    | ns    |          |
| TCLGX  | Buswidth Hold after CLKOUT Low    | 0        |                        | ns    |          |
| TAVDV  | Address Valid to Input Data Valid |          | 3T <sub>OSC</sub> - 60 | ns    |          |
| TRLDV  | RD# Active to Input Data Valid    |          | T <sub>OSC</sub> - 25  | ns    |          |
| TCLDV  | CLKOUT Low to Input Data Valid    |          | T <sub>OSC</sub> - 55  | ns    |          |
| TRHDZ  | End of RD ≠ to input Data Float   |          | T <sub>OSC</sub> - 20  | ns    |          |
| TRXDX  | Data Hold after RD ≠ Inactive     | 0        |                        | ns    |          |

- Typical specification not guaranteed
   If max is exceeded additional wait states will occur

#### The 80C196KA will meet these specifications:

| Symbol | Description                            | Min                   | Max                   | Units | Notes    |
|--------|----------------------------------------|-----------------------|-----------------------|-------|----------|
| FXTAL  | Frequency on XTAL1                     | 35                    | 120                   | MHz   |          |
| Tosc   | 1/F <sub>XTAL</sub>                    | 83                    | 286                   | ns    |          |
| TXHCH  | XTAL1 High to CLKOUT High or Low       | 40                    | 110                   | ns    | (Note 1) |
| TCLCL  | CLKOUT Cycle Time                      | 2T <sub>C</sub>       | SC                    | ns    |          |
| TCHCL  | CLKOUT High Period                     | T <sub>OSC</sub> - 10 | T <sub>OSC</sub> + 10 | ns    |          |
| TCLLF  | CLKOUT Falling Edge to ALE Rising      | - 10                  | 10                    | ns    |          |
| TLLCH  | ALE Falling Edge to CLKOUT Rising      | - 10                  | 10                    | ns    |          |
| TLHLH  | ALE Cycle Time                         | 4T <sub>C</sub>       | osc                   | ns    |          |
| TLHLL  | ALE High Period                        | T <sub>OSC</sub> - 10 | Tosc + 10             | ns    |          |
| TAVLL  | Address Setup to ALE Falling Edge      | T <sub>OSC</sub> - 25 |                       | ns    |          |
| TLLAX  | Address Hold after ALE Falling Edge    | T <sub>OSC</sub> - 15 |                       | ns    |          |
| TLLAL  | ALE Falling Edge to RD Falling Edge    | T <sub>OSC</sub> - 25 |                       | ns    |          |
| TALCL  | RD Falling Edge to CLKOUT Falling Edge | 0                     | 20                    | ns    |          |
| TRLAH  | RD Low Period                          | Tosc - 5              |                       | ns    |          |
| TRHLH  | RD Rising Edge to ALE Rising Edge      | T <sub>OSC</sub> - 15 | T <sub>OSC</sub> + 15 | ns    | (Note 2) |
| TLLWL  | ALE Falling Edge to WR Falling Edge    | T <sub>OSC</sub> - 10 |                       | ns    |          |
| TCLWL  | CLKOUT Low to WR Falling Edge          | - 5                   | 15                    | ns    |          |
| TOVWH  | Data Stable to WR Rising Edge          | T <sub>OSC</sub> - 20 |                       | ns    |          |
| Тснун  | CLKOUT Rising Edge to WR Rising Edge   | - 10                  | 10                    | ns    |          |
| TwLwH  | WR Low Period                          | T <sub>OSC</sub> - 20 |                       | กร    |          |
| TWHQX  | Data Hold after WR Rising Edge         | T <sub>OSC</sub> - 20 |                       | ns    |          |
| TWHLH  | WR Rising Edge to ALE Rising Edge      | T <sub>OSC</sub> - 20 | T <sub>OSC</sub> + 20 | ns    | (Note 2) |
| TWHBX  | BHE, INST HOLD after WR Rising Edge    | Tosc - 30             |                       | ns    | }        |

TOSC = 83 3 ns at 12 MHz, TOSC = 125 ns at 8 MHz 1 Typical specification not guaranteed 2 Assuming back to-back bus cycles

#### APPENDIX H

# AD667 BLOCK DIAGRAM AND TIMING DIAGRAMS

The information in this appendix regarding the AD667 has been taken from [15].

### **AD667 FUNCTIONAL BLOCK DIAGRAM**



#### **TIMING DIAGRAMS**

WRITE CYCLE #1

(Load First Rank from Data Bus; A3 = 1)



#### WRITE CYCLE #2

(Load Second Rank from First Rank; A2, A1, A0 = 1)



#### TIMING SPECIFICATIONS

(All Models,  $T_A = 25^{\circ}\text{C}$ ,  $V_{CC} = +12\text{V or } +15\text{V}$ ,  $V_{EE} = -12\text{V or } -15\text{V}$ )

| Symbol            | Parameter                    | Min | Typ | Max |     |
|-------------------|------------------------------|-----|-----|-----|-----|
| tpc               | Data Valid to End of CS      | 50  | _   | _   | DS  |
| t <sub>AC</sub>   | Address Valid to End of CS   | 100 | -   | -   | ns  |
| tor               | CS Pulse Width               | 100 | _   | -   | ns  |
| ton               | Data Hold Time               | 0   | _   | _   | D\$ |
| t <sub>SETT</sub> | Output Voltage Settling Time | -   | 2   | 4   | με  |

### APPENDIX I

## **HCTL-1000 BLOCK AND TIMING DIAGRAMS**

The information in this appendix regarding the HCTL-1000 is taken from [16].



Figure 2. Internal Block Diagram



# A.C. Electrical Characteristics T<sub>A</sub> = 0°C to 70°C V<sub>CC</sub> = 5 V ± 5%, Units = nsec

| ID# | Signal                                                             | Symbol            | Clock Frequency 2 MHz 1 MHz |               |      |              |
|-----|--------------------------------------------------------------------|-------------------|-----------------------------|---------------|------|--------------|
|     |                                                                    |                   | Min.                        | Max.          | Min. | Max.         |
| 1   | Clock Period                                                       | ICPER             | 500                         | · <del></del> | 1000 |              |
| 2   | Pulse Width, Clock High                                            | †CPWH             | 230                         |               | 300  |              |
| 3   | Pulse Width. Clock Low                                             | TCPWL             | 200                         |               | 200  |              |
| 4   | Clock Rise and Fall Time                                           | tca               |                             | 50            |      | 50           |
| 5   | Input Pulse Width Reset                                            | tinst             | 2500                        |               | 5000 |              |
| 6   | Input Pulse Width Stop, Limit                                      | tip               | 600                         |               | 1100 |              |
| 7   | Input Pulse Width Index, Index                                     | t <sub>IX</sub>   | 1600                        |               | 3100 |              |
| 8   | Input Pulse Width CHA, CHB                                         | TIAB              | 1600                        |               | 3100 |              |
| 9   | Delay CHA to CHB Transition                                        | t <sub>AB</sub>   | 600                         |               | 1100 |              |
| 10  | Input Rise/Fall Time CHA, CHB, Index                               | TIABR             |                             | 450           |      | 900          |
| 11  | Input Rise/Fall Time Reset, ALE, CS, OE, Stop, Limit               | tin               |                             | 50            |      | 50           |
| 12  | Input Puise Width ALE, CS                                          | tipw              | 80                          |               | 80   |              |
| 13  | Delay Time, ALE Fall to CS Fall                                    | IAC               | 50                          |               | 50   |              |
| 14  | Delay Time, ALE Rise to CS Rise                                    | tca               | 50                          |               | 50   |              |
| 15  | Address Set Up Time Before ALE Rise                                | IASR1             | 20                          |               | 20   |              |
| 16  | Address Set Up Time Before CS Fall                                 | tasa              | 20                          |               | 20   |              |
| 17  | Write Data Set Up Time Before CS Rise                              | tosa              | 20                          |               | 20   |              |
| 18  | Address/Data Hold Time                                             | t <sub>H</sub>    | 20                          |               | 20   |              |
| 19  | Set Up Time, R/W Before CS Rise                                    | twcs              | 20                          |               | 20   |              |
| 20  | Hold Time, R/W After CS Rise                                       | twH               | 20                          |               | 20   |              |
| 21  | Delay Time, Write Cycle, CS Rise to ALE Fall                       | İCSAL             | 1700                        |               | 3400 |              |
| 22  | Delay Time, Read/Write, CS Rise to CS Fall                         | tcscs             | 1500                        |               | 3000 |              |
| 23  | Write Cycle, ALE Fall to ALE Fall For Next Write                   | twc               | 1830                        |               | 3530 |              |
| 24  | Delay time, CS Rise to OE Fall                                     | †CSOE             | 1700                        |               | 3200 |              |
| 25  | Delay Time, OE Fall to Data Bus Valid                              | TOEDB             | 100                         |               | 100  |              |
| 26  | Delay Time, CS Rise to Data Bus Valid                              | ICSDB             | 1800                        |               | 3300 |              |
| 27  | Input Pulse Width OE                                               | IPWOE             | 100                         |               | 100  |              |
| 28  | Hold Time, Data Held After OE Rise                                 | t <sub>DOEH</sub> | 20                          |               | 20   |              |
| 29  | Delay Time, Read Cycle, CS Rise to ALE Fall                        | ICSALR            | 1820                        |               | 3320 |              |
| 30  | Read Cycle, ALE Fall to ALE Fall For Next Read                     | tRC               | 1950                        |               | 3450 |              |
| 31  | Output Pulse Width, PROF, INIT, Pulse, Sign, PHA-PHD, MC Port      | t <sub>OF</sub>   | 500                         |               | 1000 |              |
| 32  | Output Rise/Fall Time, PROF, INIT, Pulse, Sign<br>PHA-PHD, MC Port | ton               | 20                          | 150           | 20   | 150          |
| 33  | Delay Time, Clock Rise to Output Rise                              | tep               | 20                          | 300           | 20   | 300          |
| 34  | Delay Time, CS Rising to MC Port Valid                             | tosmo             | 1                           | 1600          |      | 320          |
| 35  | Hold Time, ALE High After CS Rise                                  | TALH              | 100                         |               | 100  |              |
| 36  | Pulse Width, ALE High                                              | 1ALPWH            | 100                         |               | 100  | <del> </del> |

#### APPENDIX J

#### SLAVE SOURCE CODE

```
SLAVE.A96 - 80C196KA ASSEMBLY CODE FOR THE ARC SLAVE PROCESSOR.
; Expand all MACROs in slave.lst listing
                      ; Include symbolic definitions from file 8096.inc.
$include(8096.inc)
                       ; (listing of 8096.inc follows this listing)
;; Storage Reservation for Program Variables and Pointers in the 80C196KA's ;; 232-byte Register File. These registers reside on the 80C196KA chip and
;; should not be confused with the so-called dual-port RAM registers.
                       ; Registers are allocated beginning at 40H
       at 40h
rseq
        TEMP1 .
                       dsw
                              1
                                      : Temporary registers.
        TEMP2:
                       dsw
                               1
        N0001:
                       dsw.
                              1
                                      ; Registers to hold frequently used
        N0003:
                                      ; numbers.
        SAMPLE PERIOD: dsw
                                      ; Registers to hold pointers to
                                      ; dual-port RAM registers.
        TIMING A:
                       dsw
                               1
        TIMING B:
                       dsw
        COMMAND:
                       dsw
                               1
        ERROR:
                       dsw
                               1
        POS:
                       dsw
                               1
                               1
        TOR:
                       dsw
                                      ; Register to hold pointer to the buffer
        SENSORS:
                       dsw
                              1
                                      ; for HOME and encoder index signals
                                      ; Registers to hold actual joint
        POS_1:
                       dsl
                               1
        POS_2:
                                       ; position information.
                       dsl
                               1
                       ds]
        POS Z:
                               1
        POS_R:
                       dsl
                               1
        POS 1L:
                                      ; Registers to hold the actual joint
                       dsl
                               1
                                      ; positions from the previous (last)
                       dsl
        POS_2L:
                               1
        POS_ZL:
                       dsl
                                      ; sampling period
        POS RL:
                       dsl
                               1
        TORQUE 1:
                       dsw
                               1
                                      ; Registers to hold joint torques
        TORQUE 2:
                       dsw
                               1
        TORQUE Z:
                       dsw
                               1
        TORQUE R:
                       dsw
                                      : Pointers to HCTL-1000 position counters
        PCZ2 R14 CS:
                       dsw
                               1
                                       ; 22 or 1R => joints Z & 2 or 1 & R
        PC22 R13 CS:
                       dsw
                               1
        PCZ2 R12 CS:
                       dsw
                                      ; R12, R13, R14 => register 12, 13, 14
        PC22_R14_OE:
                                      ; CS => chip select
                       dsw
                               1
        PC22_R13_OE:
PC22_R12_OE:
                       dsw
                               1
                                       ; OE => output enable
                       dsw
                               1
        PC1R R14 CS:
                       dsw
                               1
        PC1R_R13_CS:
                       dsw
                               1
        PC1R R12 CS:
                       dsw
        PC1R R14 OF:
                       dsw
                               1
        PCIR R13 OE:
                       dsw
                               1
        PC1R_R12_OE:
                       dsw
```

```
PC_1_PC:
                       dsw
                              1
                                      ; Registers to hold pointers to the
       PC_2_PC:
                       dsw
                              1
                                      ; Position counters' program counters
       PC Z PC:
                       dsw
                              1
       PC R PC:
                       dsw
                              1
       PC 1 RESET:
                       dsw
                                      ; Registers to hold values to be to
       PC_2_RESET: PC_Z_RESET:
                       dsw
                              1
                                      ; sent to position counters'
                       dsw
                              1
                                      ; program counters
       PC R RESET:
                       dsw
       DAC 1:
                       dsw
                              1
                                      ; Registers to hold pointers to the
       DAC 2:
                       dsw
                              1
                                      ; first rank registers of the DACS
       DAC_Z:
                       dsw
                              1
       DAC R:
                       dsw
                              1
       DACS_OUT:
                       dsw
                               1
                                      ; Register to hold pointer to the
                                       ; second rank register of each DAC
;;;;;
                              INITIALIZATION
cseg
       at 2080h
                       ; Code segment begins at 2080H
;; initialization of Program Variables and Pointers
       LD
               DAC_1, #OAFFCH
                                       ; DAC first rank addresses
       ΤD
               DAC_2, #OAFFAH
               DAC Z, #OAFF6H
DAC R, #OAFEEH
       LD
       LD
       LD
               DACS_OUT, #OAFDEH
                                       ; All DACS second rank address
       LD
               PCZ2_R14_CS, #09494H
                                      : Position counter addresses
               PCZ2_R13_CS, #09392H
       ID
       ID
               PCZ2_R12_CS, #09292H
       LD
               PCZ2_R14_OE, #09414H
               PCZ2_R13_OE, #09312H
PCZ2_R12_OE, #09212H
       LD
       ID
       LD
               PC1R R14 CS, #094D4H
       LD
               PC1R_R13_CS, #093D2H
       TD
               PC1R R12 CS, #092D2H
               PC1R_R14_OE, #09454H
       LD
       ID
               PC1R_R13_OE, #09352H
               PC1R_R12_OE, #09252H
       LD
       LD
               PC_1_PC, #085C4H
                                       ; Address of program counter register
       LD
               PC_2_PC, #08584H
                                          on the position counters.
                                      ;
               PC_Z_PC, #08584H
PC_R_PC, #085C4H
       LD
                                          1 & R, 2 & Z are equal
        LD
                                          since they are accessed in pairs.
       ID
               PC_1_RESET, #0100H
                                       ; 0100 resets 1 but leaves R idling.
                                      ; 0100 resets 2 but leaves Z idling.
       LD
               PC_2_RESET, #0001H
               PC_Z_RESET, #0100H
PC_R_RESET, #0001H
       LD
                                      ; 0100 resets Z but leaves 2 idling.
       LD
                                       ; 0001 resets R but leaves 1 idling.
       LD
               N0001, #0001H
       TD
               N0003, #0003H
       ĽD
               SP, #100H
                                       ; Init stack at top of reg. file
               IOC2, #00000001B
        LDB
                                      ; Put TIMER2 into fast increment mode
       LD
               SAMPLE PERIOD, #0E000H ; Addresses of dual-port RAM registers
               TIMING_A, #OE002H
       LD
       LD
               TIMING_B, #0E004H
               COMMAND, #0E006H
       LD
       ID
               ERROR, #0E008H
               POS, #0E010H
       LD
               TOR, #0E020H
       LD
       LD
               SENSORS, #0B000H
                                       ; Address of buffer for HOME and
                                       ; encoder index signals
       ST
               O, [ERROR]
                              ; Clear error register
       ST
               0, [PC 1 PC]
                              ; Software reset all position counters by
       ST
               0, [PC_2_PC]
                                   writing O to their program counters.
```

```
MAIN PROGRAM
main:
                                 ; Clear Command register
      ST
             O, [COMMAND]
wait for command:
       CALL
                                  ; Turn off all servo motors.
             zero_dacs
       CALL
             get joint positions
                                 ; get state of robot.
       CALL
             joint_overrun_check
                                  ; If there is a joint overrun error
       CMP
             O, [ERROR]
       JNE
             wait for command
                                 ; keep waiting until corrected.
       LD
             TEMP1, [COMMAND]
                                 ; Read Command register
             TEMP1, 0, control
                                 ; If bit 0 set, enter control_mode.
       RRS
             TEMP1, 7, find home
                                 ; If bit 7 set, enter find home mode.
       HKS
       BR
             wait for_command
                                 ; Otherwise keep waiting
;;;;;; FIND HOME MODE ;;;;;;
FH
       MACRO
             joint, positive torque, negative torque, home_bit, index_bit
             nh, fh_1, fh_2, fh_3
       LOCAL
             TEMP1, [SENSORS]
       LD
             TEMP1, home_bit, fh_1
                                        ;; If HOME sensor not set, goto fh_1.
       JRS
             TEMP2, *positive_torque
TEMP2, [DAC_&joint]
                                        ;; else move joint away from HOME.
       ID
       ST
             O, [DACS OUT]
       ST
             TEMP1, [SENSORS]
TEMP1, home_bit, nh
                                         ;; If HOME sensor still set, goto nh
nh:
       LD
                                        ;; else keep moving away from HOME
;; during 'delay' period (~2 secs).
       JPC
       CALL
             delay
                                        ;; Stop the motor.
             zero dacs
       CALL
             TEMP2, #negative_torque
TEMP2, [DAC_&joint]
fh 1:
       ID
                                         ;; Move joint toward home,
       ST
             O, [DACS OUT]
       ST
             TEMP1, [SENSORS]
fh 2:
       LD
                                                ;; Wait for home sensor
             TEMP1, home_bit, fh_2
                                               ;; to set.
       JBS
             TEMP1, [SENSORS]
                                                ;; Wait for index pulse.
fh 3:
       ID
             TEMP1, index bit, fh 3
       JBC
                                                ;;
             PC & joint & RESET, [PC & joint & PC]
                                                ;; Set pos. counter to zero.
       ST
       CALL
             zero dacs
                                                ;; Remove the torque.
       ENDM
find home:
       FH
             z, 900H, 640H, 4, 0
             1, 900H, 700H, 7, 2
       FH
             2, 900H, 700H, 6, 1
       FH
             R, 920H, 6A0H, 5, 3
       FH
CONTROL MODE
                                                              111111
control:
       CLR
             TIMER2
                                  ; Clear the sample period timer
             0, [TIMING A]
0, [TIMING B]
                                  ; Clear the timing registers
       ST
       ST
control loop:
             NOOO1, [TIMING A]
       ST
                                  ; Signal master to begin its loop
       SCALL
             get joint positions
                                  ; get actual joint positions
       SCALL
              joint overrun check
                                  ; Check positions for overruns.
       CMP
              O, [ERROR]
                                  ; End if an overrun error was found
       PINE.
             main
wait_for_torque:
       CMP:
             TIMER2, [SAMPLE PERIOD]
                                        ; Wait for torques but end (with
```

```
JGT
                                              ; error) if they arrive too late.
                torques too late
        CMP
                                               ; End (without error) if master
                O, [COMMAND]
        BE.
                                               ; clears command register.
                main
        AND
                TEMP1, NOOO1, [TIMING B]
                                               ; Check for arrival of torques.
        JE
                wait_for_torques
        SCALL
                check torques
                                                ; Get and check torques.
        CMP
                O, [ERROR]
                                                ; End if excess torque found.
        BNE
                main
                O, [TIMING B]
                                                ; Tell master torques were accepted.
        SCALL
                                                ; Output torques.
                torques_out
wait for master:
                TIMER2, [SAMPLE PERIOD]
        ÖΜΡ
                                              ; Wait for master to finish loop.
        JGT
                master too slow
                                                ; End if master is too slow.
                O, [TIMING A]
        CMP
        JNE
                wait_for_master
wait_next_period:
        CMP
                TIMER2, [SAMPLE PERIOD]
                                                ; Wait for next sample period
        JLT
                wait next period
        SUB
                TIMER2, [SAMPLE_PERIOD]
                                                ; Reset the sample period timer
                control loop
        BR
torques_too_late:
        LD
                TEMP1, [ERROR]
                TEMP1, #1000H
        ST
                TEMP1, [ERROR]
        BR
                main
master_too_slow:
                TEMP1, [ERROR]
TEMP1, #2000H;
        LD
        OR
        ST
                TEMP1, [ERROR]
        BR
                main
GET JOINT POSITIONS PROCEDURE
get_joint positions:
                TEMP1, [PC1R_R14_CS] ; 1st read of R14, pos. counters 14R
TEMP1, [PC22_R14_CS] ; 1st read of R14, pos. counters 262
POS_1, POS_1L ; Store 'last' positions - low word
        ID
        LD
                                       ; Store 'last' positions - low word
; and kill > than 1.8 usec.
        ST
                 POS 2, POS 2L
        ST
                 POS_Z, POS_ZL
        ST
                 POS_R, POS_RL
        ST
                POS_1, [PCIR_R14_OE]
                                       ; 2nd read of R14, pos. counters 1&R
        ID
                                       ; 2nd read of R14, pos. counters 262
        \mathbf{T}
                POS_Z, [PCZ2_R14_OE]
                POS_1+2, POS_1L+2
POS_2+2, POS_2L+2
                                         ; Store 'last' positions - high word
        ST
        ST
                 TEMP1, [PC1R_R13_CS]
        LD
                                        ; 1st read of R13, pos. counters 16R
                 TEMP1, [PCZ2_R13_CS]
                                         ; 1st read of R13, pos. counters 262
        TD.
                POS_1+1, POS_R
POS_Z+1, POS_2
        STB
                                         ; Sort data and kill > 1.8 usec.
        STB
                TEMP1, [PC1R_R13_OE]
TEMP1, POS_1+1
        LD
                                       ; 2nd read of R13, pos. counters 14R
        STB
                                         ; Sort data
        STB
                 TEMP1+1, POS R+1
        ID
                 TEMP1, [PCZ2_R13_OE]
                                         ; 2nd read of R13, pos. counters 262
         ST
                 POS_Z+2, POS_ZL+2
                                         ; Store 'last' positions - high word
                 POS_R+2, POS_RL+2
         ST
                                        ; 1st read of R12, pcs. counters 16P
         ID
                 TEMP2, [PC1R_R12_CS]
                                         ; 1st read of R12, pos. counters 262
                 TEMP2, [PCZ2_R12_CS]
TEMP1, POS_Z+1
         ID
         STB
                                         ; Sort data and kill > 1.8 usec.
         STB
                 TEMP1+1, POS_2+1
                 TEMP1, [PC1R_R12_OE] ; 2nd read of R12, pos. counters 16F TEMP1, POS_1+2 ; Sort the data
         TD
         STB
         STB
                 TEMP1+1, POS R+2
                 TEMP1, [PC22_R12_OE] ; 2nd read of R12, pos. counters %62
         LD
         STB
                 TEMP1, POS Z+2
                                       ; Sort data
```

```
TEMP1+1, POS 2+2
      STB
      CLRB
                               ; Clear the most significant byte
      CLEB
            POS_2+3
      CLRB
            POS Z+3
      CLRB
            POS_R+3
      CMPB
                               ; Set most sig. byte to FF if 2nd most
            POS 1+2, #0FFH
            pos2
                                  significant byte is FF, i.e.,
      JNE
                                  sign-extend to 32 bits.
            POS_1+3, #0FFH
      LDB
            POS 2+2, #0FFH posZ
      СМРВ
pos2:
      JNE
      LDB
            POS_2+3, #0FFH
      CMPB
            POS_2+2, #OFFH
pos2:
            posR
      JNE
            POS_Z+3, #OFFH
      LDB
posR:
      CMPB
            POS_R+2, #OFFH
      JNF
            pos end
            POS_R+3, #OFFH
     LDit
pos end:
      ST
            POS_1, [POS]
                               ; Save positions in dual-port RAM
            POS_1+2, 2[POS]
                               ; actual position registers
      ST
      ST
            POS 2, 4[POS]
            POS_2+2, 6[POS]
      ST
      ST
            POS Z, 8[POS]
            POS Z+2, OAH[POS]
      ST
            POS R, OCH[POS]
      ST
      ST
            POS R+2, OEH[POS]
      ST
            NOOO3, [TIMING A]
                               ; Signal to masterthat joint positions
                               ; are available
      RET
DELAY PROCEDURE
TEMP2, #8H
TEMP1, #0FFFFH
                               ; Generate ~2 sec. delay
delay: LD
dclay0: LD
                                  by counting down from FFFFH
delayl: DEC
            TEMP1
                                  8H times.
            TEMP1, 0
     CMP
      JNE
            delayl
            TEMP2
     DEC
      CMP
            TEMP2, 0
            delay0
      JNE
      RET
ZERO DACS PROCEDURE
                                                         111111
; Reset all DACS to 0 Vdc (offset = 800H)
zero dacs:
     LD
            TEMP1, #800H
            TEMP1, [DAC_1] ; Load first rank registers of all DACS
     ST
      ST
            TEMP1, [DAC_2]
            TEMP1, [DAC_Z]
TEMP1, [DAC_R]
      ST
      ST
      ST
            O, [DACS_OUT]
                        ; load 2nd rank registers of each DAC
      RET
                         ;
                           simultaneously
JOINT OVERRUN CHECK PROCEDURE
OVERRUN
            MACRO JOINT, MIN, MAX, NEG_OVR_FLAG, POS_OVR_FLAG, NEXT_JOINT, MSB
      LOCAL
            or a, or b
      CMP
            POS_6JOINT+2, #OFF/FH ;; If joint is not in negative region
      JNE
            or_a
                                 goto or_a
                               ;;
            POS &JOINT, #MIN
      CMP
                               ;; Else compare its position against
      JC.
                                  the specified MIN.
            or_&NEXT_JOINT
                               ;;
      OR
            TEMP1, #NEG OVR FLAG
                                   If position is < MIN, make note of
                               ;;
            or_&NEXT_JOINT
      BR
                                    neg ovrrun error, then next joint
                               ;;
     CMP
            POS &JOINT+2, #MSB
                               ;; Compare 3rd byte against MSB:
or a:
```

```
JLT
               or_&NEXT_JOINT
                                          IF position < MSB, check next joint
                                     ;;
                                          IF position > MSB, goto or_b
       JGT
               or b
                                     ;;
       QMP
               POS &JOINT, #MAX
                                          Else compare position against MAX.
                                     ;;
       JNH.
               or_ENEXT_JOINT
                                          IF < MAX, check next joint
                                     ;;
or b:
       OR
               TEMP1, #POS OVR FLAG
                                     ;; Make note of positive overrun error.
       ENDM
overrun_check:
               TEMP1
       CLR
       OVERRUN 1, OEEF7H, OBO3CH, 2H, 1H. 2, 2H
OVERRUN 2, OF752H, OEB3EH, 8H, 4H, Z, OH
or_1:
or_2:
       OVERRUN Z, 0FE83H, 07585H, 20H, 10H, R, 1H
OVERRUN R, 05F1DH, 0A0E3H, 80H, 40H, end, 0H
or_Z:
or_R:
               TEMP2, [ERROR]
or_end: LD
                                    ; Update Error Register
       STB
               TEMP1, TEMP2
       ST
               TEMP2, [ERROR]
       RET
CHECK TORQUES PROCEDURE
JOINT, OFFSET, NEXT_JOINT, ERROR_FLAG
TORQUE_&JOINT, OFFSET[TOR] ;; Get torque from DPR register
TEMP1, TORQUE_&JOINT, #0F000H ;; Mask-out all but most sig. byte
CT
       MACRO
       ID
       AND
        JΕ
               ct_&NEXT_JOINT
                                            ;; If result is not zero
       OR
               TEMP2, #ERROR FLAG
                                                  make a note of the error
       ENDM
check_torques:
               TEMP2
       CLR
               1, 0, 2, 100H
ct_1:
       CT
ct_2:
               2, 2, Z, 200H
       CT
               Z, 4, R, 400H
ct Z:
       CT
               R, 6, end, 800H
TEMP2, [ERROR]
ct R:
       CT
ct_end: OR
                                     ; Update Error Register
        ST
               TEMP2, [ERROR]
        RET
TORQUES_OUT PROCEDURE
                                                                    ;;;;;
torques_out:
               TORQUE_1, [DAC_1]
        ST
                                      ; Load first rank registers of DACs.
               TORQUE 2, [DAC 2]
TORQUE Z, [DAC Z]
        ST
        ST
               TORQUE R, [DAC R]
        ST
               O, [DACS_OUT]
                                      ; Load second rank register of DACs.
        ST
        RET
        END
```

```
; 8096.INC - DEFINITION OF SYMBOLIC NAMES FOR THE I/O REGISTERS OF THE
      8096 AND THE 80C196
        (C) INTEL CORPORATION 1983
,
        8096 & 80C196KA SPECIAL FUNCTION REGISTERS
; */
RO
                     OOH:WORD
                                   ; R
                                           CONTAINS THE VALUE 0000H
AD_COMMAND
                    02H:BYTE
              EQU
                                           NOT USED BY ARC
                                 ; W
                                  ; R
AD RESULT LO
               EQU
                    02H:BYTE
                                           **
                                               **
                                                    .
AD RESULT HI EQU
                    O3H:BYTE
                                   ; R
HSI_MODE
                                  ; W
; W
                                            ..
                                                ,,
                                                        **
                                                    **
               EQU
                     O3H:BYTE
HSO_TIME
               EQU
                     O4H:WORD
                                            11
                                                11
                                                        11
                                 ; R
                                            **
                                                **
               EQU
                     04H:WORD
HSO_COMMAND
              EQU
                     O6H:BYTE
                                   ; W
                                            **
                                                ..
                                   ; R
                                            **
                                                **
HSI_STATUS
                     O6H:BYTE
                                                    *1
                                                        10
              EQU
                                  ; R/W
SBUF
               EQU
                     O7H:BYTE
                                            **
                                                **
                                                    ..
INT_MASK
                     O8H:BYTE
                                  ; R/W
; R/W
                                            ..
               EQU
INT PENDING EQU
                                           **
                     O9H:BYTE
                                               **
WATCHDOG
               EQU
                     OAH:BYTE
                                  ; W WATCHDOG TIMER
                                 ; R
; R
TIMER1
               EQU
                     OAH: WORD
                                           NOT USED BY ARC
TIMER2
               EQU
                     OCH:WORD
                                           USED AS SAMPLE PERIOD TIMER
                                 ; W NOT USED BY ARC
BAUD_RATE
              EQU
                     OEH: BYTE
                                  ; R
IOPORTO
               EQU
                     OEH:BYTE
                                            ..
                                               **
                                                    **
IOPORT1
                    OFH:BYTE
                                   ; R/W
                                                    **
               EQU
                                 ; R/W
IOPORT2
                                           **
               EQU
                    10H:BYTE
                                               .
                                                    .,
                                                        ••
SP_CON
SP_STAT
                                           **
                                                **
                                                        "
               EQU
                     11H:BYTE
                                  ; W
                                 ; R
               EQU
                     11H:BYTE
                                            **
                                               **
                                                    .
                                                        **
10<u>C</u>0
               EQU
                     15H:BYTE
                                   ; W
1050
               EQU
                     15H:BYTE
                                            .
                                                11
                                                    **
                                                        **
                                   ; R
IOC1
               EQU
                     16H:BYTE
                                   ; W
                                            •
                                                11
                                                        **
1051
               EQU
                     16H:BYTE
                                            ..
                                                        **
                                   ; R
PWM CONTROL
               EQU
                    17H:BYTE
                                   ; W
                                           **
                                               11
                                                    **
                                                        18
               EQU
SP
                     18H:WORD
                                   ; R/
                                          STACK POINTER
;/*
; *
        SPECIAL FUNCTION REGISTERS FOR 80C196 ONLY
: */
10C2
               EQU
                     OBH:BYTE
                                 ; W PUTS TIMER2 IN FAST INCR. MODE ; R/W NOT USED BY ARC
                     12H:BYTE
IPEND1
               EQU
                                 ; R/W 1101 0022 ____
; R/W 11 11 11
• D/W 11 11 11
                    13H:BYTE
IMASK1
               EQU
                                                        *
                                  ; R/W
WSR
               EQU
                     14H:BYTE
                                                        11
1052
                    17H:BYTE
                                           **
                                               11
               EQU
                                   ; R
```

#### APPENDIX K

#### MASTER SOURCE CODE

```
* UTILITY.ROB - C PROGRAM CONTAINING ARC MAIN AND OTHER FUNCTIONS
#include "stdlib.h"
#include "math.h"
#include "alloc.h"
#include "stdio.h"
/* GLOBAL VARIABLES (begin with a capital letter) */
   double Duration; /* Duration (in sec.) of the move */
int Home found = 0; /* A flag which is set to 1 when home is found */
main()
ŧ
   for(;;) {
       clrscr();
       controller_menu();
   }
}
/***************
int check_home_found_flag(); /* function prototypes */
prop control();
prop_deriv_control();
seraji_decen_adapt_ctrl();
find home();
                               /* Display menu of available control algorithms */
controller_menu()
   char ch;
   printf("Choose controller\n\n");
   printf("[1] FIND HOME\n");
printf("[2] PROPORTIONAL AND DERIVATIVE CONTROL\n");
   printf("[3] SERAJI-DECENTRALIZED ADAPTIVE CONTROL\n");
printf("[0] QUIT\n\n");
   printf("ENTER YOUR CHOICE: ");
   do {
        switch(ch = getch()) {
            case '1': find home();
                      return;
            case '2': check home found flag();
                      prop_deriv_control();
                       return;
            case '3': check_home_found_flag();
                       seraji decen adapt ctrl();
                       return;
            case '0': clrscr();
                      exit(0);
            default : delline();
                       printf("\rINVALID CHOICE. TRY AGAIN: ");
    ) while(ch != '1' && ch != '2' && ch != '3' && ch '= '0');
}
```

```
/>>>>>>> find_home() /* tells slave to enter find-HOME mode */
find_home()
   int far *command = 0xC0000006;
   unsigned int far *error = 0xC0000008;
   *error = 0x0000; /* clear the error register */
*command = 0x0080; /* give 'find-HOME' command */
   clrscr();
   printf("\t\t\tFINDING HOME, PLEASE WAIT");
   while (*command); /* wait for 196 to finish the move */
   Home found = 1;
   dellIne();
   printf("\r\thome was found. Press any key to continue ");
   while ('getch());
/* check home found flag and warn if home was not previously found */
check_home_found_flag()
   if(!Home_found) {
                      Robot will first FIND HOME.\n\n");
(P)roceed or (Q)uit? ");
       printf("\r\t\t
       printf("\t\t
       if(tolower(getch()) == 'p') find home();
       else exit(0);
}
/* Display menu of available path planning algorithms */
long int huge *cubic spline(double); /* function prototypes */
long int huge *cycloid(double);
long int huge *step function(double);
long int huge *choose path_planner(double sample per)
   char ch;
   clrscr();
   printf("\n\nChoose path planner\n\n");
   printf("[1] CUBIC SPLINE\n");
   printf("[2] CYCLOID\n");
printf("[3] STEP FUNCTION\n");
printf("[0] QUIT\n\n");
   printf ("ENTER YOUR CHOICE: ");
   do {
       switch(ch = getche()) (
           case '1': return(cubic_spline(sample_per));
           case '2': return(cycloid(sample_per));
           case '3': return(step_function(sample_per));
           case '0': clrscr();
                    exit(0):
           default : delline();
                    printf("\rINVALID CHOICE. TRY AGAIN: ");
   } while(ch != '1' && ch != '2' && ch != '3' && ch != '0');
/* display current position and orientation of robot */
current_position()
   int out of workspace;
   long int huge *p = 0xc0010000;
   long int i_lp = *p;
   long int i 2p = *(p+1);
long int i zp = - *(p+2);
   long int i rp = *(p+3);
```

```
double i_1 = i_1p / 872.2222;
double i_2 = i_2p / 444.4444;
    double i_z = i_{zp} / 380.96;
   double i r = i rp / 227.5556;
double i lr= i l * M PI / 180;
double i lr= i 2 * M PI / 180;
    printf("Current position:\n\n");
    printf("Cartesian space \tJoint space\n\n");
printf("X = %8.41f mm.\t", 250*cos(i lr + i 2r) + 400*cos(i lr));
    printf("Joint 1 = %06li (%08lxH) pulses or ", i_lp, i_lp);
    printf("%10.6f deg.\n", i_1);
printf("Y = %8.41f mm.\t", 250*sin(i_1r + i_2r) + 400*sin(i_1r));
    printf("Joint 2 = %06li (%08lxH) pulses or ", i_2p, i_2p);
    printf("%10.6f deg.\n", i_2);
printf("Z = %8.4lf mm.\t", i_z);
    printf("Joint z = $061i ($081xH) pulses or ", i zp, i zp);
    printf("%10.6f mm.\n", i_z);
    printf("R = \$8.41f \text{ deg.}\t^{"}, i_r);
    printf("Joint R = %06li (%08lxH) pulses or ", i_rp, i_rp);
    printf("%10.6f dr .\n\n", i r);
/* prompt for the entry of a cartesian point. Verify that points are
 * within robot workspace, perform inverse kinematics to find joint angles */
get_cartesian_position(p)
long int *p;
{
    int out_of_workspace;
    double \overline{w}, \overline{px}, py, pz, pr, j1, j2;
    do {
         do (
             prin.f("\n\tEnter x (in mm.): ");
             scanf("%1f", &px);
if(px<-650 || px>650) printf("\trange of x: -650 to 650 mm.\n");
         ) while ( px < -650 \parallel px > 650 );
         do {
             printf("\n\tEnter y (in mm.): ");
             scanf("%lf", &py);
             if(py<-386 || py>650) printf("\trange of y: -386 to 650 mm.\n");
         } while(py < -386 | | py > 650 |;
         w = (px * px + py * py + 97500.0)/800.0;
         if (px*px+py*py >= w*w) {
              j1 = atan2(py,px)+atan2(-sqrt(px*px+py*py-w*w),w); /* inv. kin. */
              if(j1 < 0) j1 = j1 + 2 * M_PI;
              j2 = atan2(-px*sin(j1)+py*cos(j1), px*cos(j1)+py*sin(j1)-400);
              p = (long int)(j1 * 180 / MPI * 872.2222 + 0.5);
              * (p+1)
                       long int) (j2 * 180 / M PI * 444.4444 + 0.5);
         else (
                           /* out of workspace */
             *p = -1;
             *(p+1) = -1;
         out of workspace = (*p<0 \mid | *p>174445 \mid | *(p+1)<0 \mid | *(p+1) >60001);
         if(out_of_workspace) printf("\n\tOUT OF WORKSPACE IN X-Y PLANE\n");
    } while(out of_workspace);
         do {
             printf("\n\tEnter z (in mm.): ");
             scanf("%lf", &pz);
if(pz < -250 || pz > 0) printf("\trange of z: -250 to 0 mm.\n");
         } while( pz < -250 | | pz > 0 |;
         *(p+2) = (long int)(-pz * 380.96 + 0.5);
         do {
             printf("\n\tEnter r (in deg.): ");
             scanf("%lf", &pr);
              if(pr < -180 || pr > 180) printf("\trange of r: -180 to 180 deg.\n");
         } while( pr < -180 || pr > 180 );
         *(p+3) = (long int)(pr * 227.5556 + 0.5);
         printf("\n\t Equivalent joint-space destination:\n");
         printf("\tjoint l = %li\n", *p);
```

```
printf("\tjoint 2 = li\n", *(p+1));
printf("\tjoint z = li\n", *(p+2));
        printf("\tjoint r = %li\n", *(p+3));
}
/* prompt for the entry of joint positions */
get_joint_position(p)
long int *p;
    double jv;
    do (
        printf("\n\tEnter joint 1 (in degrees): ");
        scanf ("%lf", &jv);
        if(jv<0||jv>200) printf("\t joint 1 range: 0 to 200 degrees\n");
    } while(jv<0||jv>200);
    *p = (long int)(jv * 872.2222 + 0.5);
    do (
       printf("\n\tEnter joint 2 (in degrees): ");
        scanf ("11f", & jv);
        if (jv<0||jv>135) printf("\t joint 2 range: 0 to 135 degrees\n");
    ) while (jv<0[|jv>135);
    *(p+1) = (long int)(jv * 444.4444 + 0.5);
    do {
       printf("\n\tEnter joint z (in mm.): ");
        scanf ("%lf", & jv);
        if(jv<-2501|jv>0) printf("\tjoint z range: -250 to 0 mm.\n");
    ) while (jv<-250||jv>0);
    *(p+2) = (long int) (-jv * 380.96);
    do {
       printf("\n\tEnter joint r (in degrees): ");
        scanf ("%lf", &jv);
        if(jv<-180||jv>180) printf("\tjoint r range: -180 to 180 deg.\n");
    ) while (jv<-180) | jv>180);
    *(p+3) = (long int)(jv * 227.5556);
)
extern double Duration;
get_time(p)
double *p;
    printf("\n\tLength of time (in sec.) to reach this point: ");
       scanf ("%lf",p);
       if(*p <= *(p-1) || *p > Duration) {
    printf("Must be greater than that of the ");
           printf("previous point and less than the ");
           printf("Duration.\nReenter: ");
    } while (*p <= *(p-1) || *p > Duration);
)
/* check for joint overuns */
robot error check()
    int far *error pointer = 0xC0000008;
    int error;
    *error_pointer = (*error_pointer) & (0x00ff);
    if(*error_pointer) {
       do (
           error = *error_pointer;
           clrscr();
           printf("\t\t\tROBOT ERRORS\n\n");
            if(error & 0x0001) printf("Joint 1 in positive overun\n");
            if(error & 0x0002) printf("Joint 1 in negative overun\n");
            if (error & 0x0004) printf ("Joint 2 in positive overun\n");
```

```
if (error & 0x0008) printf ("Joint 2 in negative overun\n");
            if(error & 0x0010) printf("Joint z in positive overun\n");
            if(error & 0x0020) printf("Joint z in negative overun\n");
           if(error & 0x0040) printf("Joint r in positive overun\n");
if(error & 0x0080) printf("Joint r in negative overun\n");
printf("\nMANUALLY MOVE THESE JOINTS BACK INTO THE WORKSPACF");
            while(error == *error pointer);
        ) while (*error pointer);
        printf("\n\nAll overun errors fixed. Press any key to continue ");
        while (!getch());
        return:
    )
}
/* display errors */
display_errors()
    unsigned int far *error = 0xC0000008;
    unsigned int temp;
    temp = *error;
    if(temp & 0x0001) printf("\tjoint 1 entered positive overun\n");
    if(temp & 0x0002) printf("\tjoint 1 entered negative overun\n");
    if(temp & 0x0004) printf("\tjoint 2 entered positive overun\n");
    if(temp & 0x0008) printf("\tjoint 2 entered negative overun\n");
if(temp & 0x0010) printf("\tjoint z entered positive overun\n");
    if(temp & 0x0020) printf("\tjoint z entered negative overun\n");
    if(temp & 0x0040) printf("\tjoint r entered positive overun\n");
    if(temp & 0x0080) printf("\tjoint r entered negative overun\n");
    if(temp & 0x0100) printf("\tToo much torque was applied to joint 1\n");
    if (temp & 0x0200) printf ("\tToo much torque was applied to joint 2\n");
    if (temp & 0 \times 0400) printf("\tToo much torque was applied to joint z\n"); if (temp & 0 \times 0800) printf("\tToo much torque was applied to joint r\n");
    if(temp & 0x1000) printf("\tPS/2 did not provide torques in time\n");
    if (temp & 0x2000) printf("\tPS/2 did not finish its loop in time\n");
/**********************************
/* compare two numbers (double precision).
 * return 0 if their signs are the same.
 * return 1 if their signs are different or if at least one number is zero. */
cmp sign(double num 1, double num 2)
    if(num 1 && num 2) { /* if neither number is zero */
        if ( (num_1 > 0 & num_2 > 0) ) ( (num_1 < 0 & num_2 < 0) ) return 0;
    else return 1;
/* change the value of a gain or parameter */
double choose new gain(char *name, double gain)
        printf("Change %s? [y/n]: ", name);
        if(getch() == 'y') (
                printf("\tFrom %lf to: ", gain);
                scanf("%lf", &gain);
        else printf("\n");
        return (gain);
}
```

```
* Proportional, Derivative (PD) Control Algorithm
 *******************
#include "alloc.h"
#include "stdio.h"
long int huge *choose_path_planner(double); /* function prototypes */
double choose new gain(char *name, double gain);
extern double Duration; /* Global variable declared in utility.rob */
prop_deriv_control()
    static double kp 1 = 5, kp 2 = 7, kp z = 5, kp r = 5; /* controller gains */
    static double kv_1 = .02, kv_2 = .02, kv_z = .02, kv_r = .02;
   int not_enough_memory;
    long int huge *pp; /* pointer to array of desired path points */
    long int huge *s_pp; /* a place to save this pointer */
unsigned int far *sample_timer = 0xC0000000; /* pointer to DPR sample_period Reg. */
    double num samples;
    unsigned int num captures;
    unsigned int capture_counter = 0;
    unsigned int capture intrvl;
    int far *error = 0xC\overline{0}0000008;
                                          /* pointer to DPR error Reg. */
    long int far *act_pos_1 = 0xC0000010; /* pointer to DPR actual pos. Regs. */
    long int far *act_pos_2 = 0xC0000014;
long int far *act_pos_z = 0xC0000018;
    long int far *act pos r = 0xC000001C;
   int far *t 1 = 0xC0000020;
int far *t 2 = 0xC0000022;
                                           /* pointer to DPR torque Regs. */
    int far t_z = 0x00000024;
    int far *t^{-}r = 0xC0000026;
                                       /* pointer to DPR timing_A reg. */
    int far *timing_a = 0xC0000002;
                                      /* pointer to DPR timing B reg. */
/* pointer to DPR command reg. */
    int far *timing b = 0xC0000004;
   int far *command = 0xC0000006; /* pointer to DPR command reg. */
long int *c a_1; /* pointers to arrays of sampled joint positions */
    long int *s c a 1; /* a => actual, d => desired */
    long int *c_d_1;
    long int *s c d 1;
    long int *c_a_2;
    long int *s_c_a_2;
    long int *c_d_2;
    long int *s_c_d_2;
    long int *c_a_z;
    long int 's c_a_z;
    long int *c d z;
    long int *s_c_d_z;
    long int *c a r;
    long int *s_c_a_r;
    long int *cdr;
    long int *s_c_d_r;
    int *c_t_1;
                                /* pointers to arrays of sampled torques */
    int *s_c_t_1;
   int *c_t_2;
    int *s_c_t_2;
    int *c_t_z;
    int *s_c_t_z;
    int *c_t_r;
    int *s_c_t_r;
    int i, counter = 0;
   FILE *fp;
    typedef struct(
                        /* header for MATLAB file */
       long type;
        long mrows;
```

```
long ncols;
        long imagf;
        long namlen;
    } Fmatrix;
    char *data_file_name="xxxx";
    Fmatrix x;
/* DISPLAY THE GAINS */
    clrscr();
    printf("\t\t\PROPORTIONAL + DERIVATIVE CONTROL\n\n");
    printf ("The error gains are: Kp_1 = \frac{1}{2} l \ln (kp_1);
printf (" kp_2 = \frac{1}{2} l \ln (kp_2);
    printf("
                                      Kp_z = \{lf \mid n'', kp_z\};
    printf("
                                      Kp_r = \{lf \mid n \mid n'', kp_r\};
    printf("
                                      Kv_1 = {1f}n^n, kv_1;
                                      Kv_2 = \frac{1}{N} (n^{tt}, kv_2);

Kv_z = \frac{1}{N} (n^{tt}, kv_z);
    printf("
    printf("
                                      Kv_r - \frac{1}{N} \ln n'', kv_r);
    printf("
/* CHANGING THE GAINS PROMPT */
    printf("Change any of the gains? (y/n): ");
    if(getch() == 'y') {
        printf("\n\n");
        kp_1 = choose_new_gain("Kp_1", kp_1);
kp_2 = choose_new_gain("Kp_2", kp_2);
kp_z = choose_new_gain("Kp_z", kp_z);
        kp_r = choose_new_gain("Kp_r", kp_r);
        printf("\n");
        kv_1 = choose_new_gain("Kv_1", kv_1);
        kv 2 = choose new gain("Kv 2", kv 2);
kv z = choose new gain("Kv z", kv z);
        kv_r = choose_new_gain("Kv_r", kv_r);
/* DISPLAY CURRENT SAMPLING PERIOD AND PROMPT FOR A CHANGE */
    sample_per = (double) (*sample_timer)/1000000;
    printf("\n\nThe sampling period is: %lf sec.\tChange? [y/n]: ", sample per);
    if(getch() == 'y') {
        printf("\n\nEnter new sample period (in sec.): ");
        scanf("%lf", &sample per);
         *sample_timer = (unsigned int) (sample_per * 1000000);
        printf("sample timer = %x \n", *sample timer);
        printf("press any key to continue ");
        while (!getch());
/* CHECK ROBOT FOR OVER RUN ERRORS */
    robot error check();
/* GET TRAJECTORY */
    pp = choose_path_planner(sample_per);
    s_pp = pp;
/* ALLOCATE MEMORY FOR DATA CAPTURE
 * Data from at least 1000 equally spaced samples are captured */
    num_samples = Duration / sample_per;
    if (num samples < 1000) capture intrvl = 1;
    else capture intrv1 = (unsigned int) (num_samples/1000);
    num_captures = (unsigned int) (num_samples/capture_intrvl);
    if (!(c_a_1 = s_a_1 = (long int *) malloc(num captures * 4 + 20))) (
         printf("Not enough memory available. Aborting\n\n");
         goto terminate; }
    if(!(c_d_l=s_c_d_l=(long int *)malloc(num_captures * 4 + 20))){
         printf("Not enough memory available. Aborting\n\n");
         goto terminate; }
     if(!(c a 2=s c a 2=(long int *)malloc(num captures * 4 + 20)))(
         printf("Not enough memory available. Aborting\n\n");
         goto terminate;
     if(!(c_d_2=s_c_d_2=(long int *)malloc(num captures * 4 + 20))){
         printf("Not enough memory available. Aborting\n\n");
         qoto terminate: }
     if(!(c_a_z=s_c_a_z=(long int *)malloc(num_captures * 4 + 20)))(
```

```
printf("Not enough memory available. Aborting\n\n");
                        qoto terminate; }
            if(!(c_d z=s_c_d z=(long int *)malloc(num_cap ures * 4 + 20))){
                       printf("Not enough memory available. Aborting\n\n");
                        goto terminate; }
            if(!(c_a_r=s_c_a_r=(long int *)malloc(num_captures * 4 + 20))){
                       printf("Not enough memory available. Aborting\n\n");
                        goto terminate; )
            if(!(c_d_r=s_c_d_r=(long int *)malloc(num_captures * 4 + 20))){
                       printf("Not enough memory available. Aborting\n\n");
                       goto terminate; }
            if(!(c_t_1 = s_c_t_1 = (int *)malloc(num_captures * 2 + 20))){
                       printf("Not enough memory available. Aborting\n\n");
                       goto terminate; )
            if('(c_t_2 = s_c_t_2 = (int *)malloc(num_captures * 2 + 20))){
                       printf("Not enough memory available. Aborting\n\n");
                       goto terminate: )
            if(!(c t z = s c t z = (int *)malloc(num captures * 2 + 20)))(
                       printf("Not enough memory available. Aborting\n\n");
                       goto terminate; }
           if(!(c_t_r = s_c_t_r = (int *)malloc(num_captures * 2 + 20))){
    printf("Not enough memory available. Aborting\n\n");
                       qoto terminate; }
/* PROMPT TO BEGIN EXECUTION */
           delline();
            printf("\r(A) bort or (B) egin the move? \n");
            do (
                       switch(ch = tolower(getch())) {
                                    case 'a': goto terminate;
                                    case 'b': break; /* continue */
default : delline();
                                                                        printf("\rINVALID CHOICE. TRY AGAIN: ");
            } while(ch '= 'a' && ch != 'b');
/* BEGINNING OF THE MOVE */
           clrscr():
            kvl = kv 1 / sample per;
            kv2 = kv_2 / sample_per;
            kv7 = kv_z / sample_per;
            kvr = kv r / sample_per;
            printf("\t\t\tEXECUTING");
            *command = 0 \times 00001;
                                                                                               /* have 196 begin its i o loop */
            while( (*pp != 0xffff0000) && (*error == 0) ) (
            /* wait to for start of sampling period */
                       while('(*timing a & 0x0001));
            /* update |oint position errors */
                       el_l = e l, el 2 = e 2, el z = e z, el r = e r;
            /* wait for joint positions */
                       while(!(*timing_a & 0x0002));
            /* compute joint torques */
                        c_{1} = *t_{1} = (int)(0x0800 + kp_{1}*(e_{1} = ((*c_{d_{1}} = *pp++) - (*c_{d_{1}} = *pp++))
 *act_pos_1))) + kv1*(e 1-el_1));
                        \bar{c}_{c_{1}} = \bar{c}_{i_{1}} = \bar{c}
*act_pos_2))) + kv2*(e_2-e_1_2));
*c_t_z = *t_z = (int)(0x0800 + kp_z*(e_z = ((*c_d_z = *pp++) - (*c_a_z = *pp++))
*act_pos_z))) + kvz*(e_z-el_z));
                        \bar{c}_{c_1} = \bar{c}_{c_2} = \bar{c
*act_pos_r))) + kvr*(e_r-e1_r));
    *timing_b = 0x0001; /* tell 196 that torques are ready */
            /* capture data every capture interval */
                        if (!capture_counter) {
                                    c d 1++;
                                     c d 2++;
                                     c d z++;
                                     c_d_r++;
```

```
c_a_1++;
           c_a_2++;
           c_a_z++;
           c_a_r++;
           c_t_1++;
           c_t_2++;
           c_t_z++;
            c t r++;
            counter++;
            capture_counter = capture_intrvl - 1;
       else {
            capture_counter--;
    /* indicate end of sample period */
        *timing_a = 0x0000;
/* INDICATE END OF COMMAND */
    *command = 0x0000;
    *timing_b = 0x0000;
/* DISPLAY SUCCESS OF EXECUTION */
    delline();
    if (*error)
       printf("\n\n\tCommand terminated prematurely because:\n\n");
        display_errors();
    else {
        printf("\n\n\t\t\tCommand executed normally.\n\n");
        printf("\t\tSaving %i data samples to disk. Please wait.\n\n", counter);
c a 1 = s c a 1; /* reset the pointers */
        c_d_1 = s_c_{d_1};
        ca2 = sca2;
cd2 = scd2;
        caz = scaz;
        c_dz = s_c_dz;
        c_a_r = s_c_a_r;
        c_d_r = s_c_d_r;
        c_t_1 = s_c_t_1;
        c_t^2 = s_c^t_2;
        ctz = sctz;
ctr = sctr;
/* open file for saving to disk */
        if((fp=fopen("robot.mat","wb")) ==NULL) (
    printf("cannot open file. Aborting\n");
            goto terminate;
        x.imagf = 0;
        x.namlen = 5;
        x.mrows = counter;
/* save a vector called 'tc.1' (joint 1 torque) to disk */
        x.type = 30;
                         /* data is signed int type */
        x.ncols = 1;
        data_file_name = "tor1";
         fwrite(&x, sizeof(Fmatrix), 1, fp);
        fwrite(data_file_name, sizeof(char), x.namlen,fp);
         fwrite(c_t_1, 2, counter, fp);
/* save a vector called 'tor2' (joint 2 torque) to disk */
        data_file_name = "tor2";
         fwrite(&x, sizeof(Fmatrix), 1, fp);
         fwrite(data_file_name, sizeof(char), x.namlen,fp);
data_file_name = "torz";
         fwrite(&x, sizeof(Fmatrix), 1, fp);
         fwrite(data_file_name, sizeof(char), x.namlen, (p);
fwrite(c_t_z, 2, counter, fp); /* save a vector called 'torr' (joint r torque) to disk */
         data_file_name = "torr";
         fwrite(&x, sizeof(Fmatrix), 1, fp);
```

```
fwrite(data_file_name, sizeof(char), x.namlen,fp);
         fwrite(c_t_r, 2, counter, fp);
/* save sample period and capture interval to disk */
                          /* data is double precision */
         x.type = 0;
         x.mrows = 1;
         data_file_name = "sper";
         fwrite(&x, sizeof(Fmatrix), 1, fp);
         fwrite(data_file_name, sizeof(char), x.namlen,fp);
         fwrite(&sample_per, 8, 1, fp);
         x.type = 40; /* data is unsigned 16-bit */
         data_file_name = "intv";
         fwrite(&x, sizeof(Fmatrix), 1, fp);
fwrite(data_file_name, sizeof(char), x.namlen,fp);
    fwrite(&capture_intrvl, 2, 1, fp);
/* save a matrix called 'posl' (joint 1 position) to disk
 * column 1 = desired position of joint 1
 * column 2 = actual position of joint 1 */
         x.type = 20;
                          /* data is long int type */
         x.ncols = 2;
         x.mrows = counter;
data file_name = "posl";
         fwrite(&x, sizeof(Fmatrix), 1, fp);
         fwrite(data_file_name, sizeof(char), x.namlen, fp);
fwrite(c d 1, 4, counter, fp);
    fwrite(c a 1, 4, counter, fp);
/* save a matrix called 'pos2' (joint 2 position) to disk
 * column 1 = desire + position of joint 2
 * column 2 = actual position of joint 2 */
         data_file_name = "pos2";
         fwrite(&x, sizeof(Fmatrix), 1, fp);
         fwrite(data_file_name, sizeof(char), x.namlen, fp);
fwrite(c d 2, 4, counter, fp);
  fwrite(c a 2, 4, counter, fp);
/* save a matrix called 'posz' (joint z position) to disk
 * column 1 = desired position of joint z
 * column 2 = actual position of joint z */
         data file name = "posz";
         fwrite(&x, sizeof(Fmatrix), 1, fp);
         fwrite(data_file_name, sizeof(char), x.namlen, fp);
fwrite(c d z, 4, counter, fp);
  fwrite(c a z, 4, counter, fp);
/* save a matrix called 'posr' (joint 2 position) to disk
 * column 1 = desired position of joint r
 * column 2 = actual position of joint r */
         data file name = "posr";
         fwrite(&x, sizeof(Fmatrix), 1, fp);
         fwrite(data_file_name, sizeof(char), x.namlen, fp);
         fwrite(c_d_r, 4, counter, fp);
fwrite(c_a_r, 4, counter, fp);
         fclose(fp);
/* FREE ALLOCATED MEMORY */
terminate:
    if(s_pp) farfree(s_pp);
    if(s_c_a_1) free(s_c_a_1);
    if(s_c_d_1) free(s_c_d_1);
    if (s c a 2) free(s c a 2);
    if (s_c_d_2) free(s_c_d_2);
    if (scaz) free(scaz);
    if(s_c_d_z) free(s_c_d_z);
if(s_c_a_r) free(s_c_a_r);
    if (s c d r) free(s c d r):
    if (s_c_t_1) free(s_c_t_1);
    if(s_c_t_2) free(s_c_t_2);
    if (s_c_t_z) free(s_c_t_z);
    if (s c t r) free(s c t r);
    printf("\t\t\tPress any key to continue ");
    while (!getch());
    return;
}
```

```
* SERAJI.ROB - Decentralized Adaptive Control Algorithm */
 * From Trans. Robotics & Automation, April 1989, Vol. 5, No. 2, pp. 183-201
#include "alloc.h"
#include "stdio.h"
long int huge *choose path planner(double); /* function prototypes */
double choose_new_gain(char *name, double gain);
extern double Duration; /* Global variable declared in utility.rob */
seraji decen adapt ctrl()
     double kp_1=0, kp_2=0, kp_z=0, kp_r=0;
                                                                   /* position gains */
     double kp_11, kp_21, kp_z1, kp_r1;
double kv_1=0, kv_2=0, kv_z=0, kv_r=0;
double kv_11, kv_21, kv_z1, kv_r1;
                                                                  /* last " " */
                                                                   /* velocity gains */
                                                                   /* last " " */
                                                                   /* position errors */
     double e_1 = 0, e_2 = 0, e_z = 0, e_r = 0;
     double e_11, e_21, e_z1, e_r1=0;
double edot_1=0, edot_2=0, edot_z=0, edot_r=0;
double edot_11, edot_21, edot_z1, edot_r1;
                                                                  /* last " " */
                                                                 /* velocity errors */
/* last " " */
    double r_1=0, r_2=0, r_z=0, r_r=0;
double r_11, r_21, r_z1, r_r1;
double f_1=20, f_2=20, f_z=20, f_r=20;
double f_11, f_21, f_z1, f_r1;
                                                                    /* weighted error */
                                                                    /* last " " */
                                                                    /* auxiliary signals */
                                                                    /* last "
     static double wp_1=80, wp_2=8, wp_z=1, wp_r=1; /* weighting factors */ static double wv_1=40, wv_2=2, wv_z=1, wv_r=1; static double d_1= 175, d_2= 175, d_z= 175, d_r= 175; /* inteq. adapt gains */
     static double \overline{a0} 1=350, \overline{a0} 2=350, \overline{a0} z=350, \overline{a0} r=350;
     static double al_1=8, al_2=8, al_z=8, al_r=8;
     static double rho 1=0, rho 2=0, rho r=0; /* prop. adapt. gains */ static double b0 1= 0, b0 2= 0, b0 z= 0, b0 r= 0;
     static double b1_1=0, b1_2=0, b1_z=0, b1_r=0;
     double d_1spo2, d_2spo2, d_zspo2, d_rspo2;
double a0_1spo2, a0_2spo2, a0_zspo2, a0_rspo2;
double a1_1spo2, a1_2spo2, a1_zspo2, a1_rspo2;
     static double sample_per; /* sample period */
     double spo2;
     char ch;
     int not_enough_memory;
     long int huge *pp; /* pointer to array of desired joint positions */
     long int huge *s_pp; /* a place to save the value of this pointer */
     unsigned int far *sample_timer = 0xC0000000; /* ptr to DPR sample_period Req. *?
     double num samples;
     unsigned int num captures;
     unsigned int capture counter = 0;
     unsigned int capture_intrvl;
     int far *error = 0xC0000008;
     long int far *act_pos_1 = 0xC0000010; /* pointers to DPR actual position Regs. */
     long int far *act_pos_2 = 0xC0000014;
     long int far *act_pos_z = 0xC0000018;
     long int far *act pos r = 0xC000001C;
     int far *t_1 = 0xC0000020; /* pointers to DPR torque registers */
     int far *t 2 = 0xc00000022;
     int far *tz = 0xc00000024;
     rt far *t r = 0xC0000026;
     int far *timing_a = 0xC0000002; /* pointer to DPR timing_A register */
     int far *timing_b = 0xC0000004; /* pointer to DPR timing_B register */
int far *command = 0xC0000006; /* pointer to DPR command register */
                                      /* pointers to arrays of sampled joint positions */
     long int *c a 1;
     long int *s_c_a_1;
                                       /* a => actual, d => desired */
     long int *c_d_1;
     long int *s_c_d_1;
     long int *c a 2;
     long int *s_c_a 2;
     long int *c d 2;
     long int *s_c_d_2;
     long int *c_a z;
```

```
long int *s c_a_z;
   long int *c d z;
   long int *s_c_d_z;
   long int *c a r;
   long int *s_c_a_r;
   long int *c d r;
   long int *s_c_d_r;
   int *c_t_1;
                                      /* pointers to arrays of sampled torques */
   int *s_c_t_1;
   int *c_t_2;
   int *s_c_t_2;
   int *c_t_z;
   int *s_c_t_z;
   int *c_t_r;
   int *s_c_t_r;
   int 1, counter = 0;
   FILE *fp:
                      /* MATLAB header information */
   typedef struct(
       long type;
       long mrows;
       long ncols;
       long imagf;
       long namlen;
   } Fmatrix;
   char *data_file_name = "????";
   Fmitrix x;
/* DISPLAY PARAMETERS */
   clrscr();
   printf("\t\tSERAJI DECENTRALIZED ADAPTIVE CONTROL\n\n");
                                                                     ");
   printf("Joint\t Weighting Factors
                                        Integral Adaptation Gains
   printf("Prop Adaptation Gains");
   printf(" #\t
                    Wp\t Wv\t\tdelta alpha0 alpha1\t
                                                                  rho");
   printf(" beta0 beta1\n\n");
                                                                              %05.2f %05.2f
    printf(" 1\t
                                                               %05.2f\t
                        %05.2f %05.2f\t%05.2f
                                                   %05.2f
$05.2f\t",
       %05.2f %05.2f
    printf(" 2\t
                                                    %05.2f
                                                               %05.2f\t
105.2f\n",
       printf(" z\t
                                                               %05.2f\t
                                                                              %05.2f %05.2f
%05.2f\n",
    %05.2f\t
                                                                              %05.2f %05.2f
105.2f\n",
       wp r, wv r, d r, a0 r, a1 r, rho_r, b0_r, b1_r);
/* CHANGE PARAMETER PROMPT */
   printf("\nChange any parameters? (y/n): ");
   if(getch() == 'y') {
       printf("\n\n");
       printf("Change Weighting Factors? (y/n): ");
if(getch() == 'y') {
           printf("\n\n");
           wp 1 = choose_new_gain("wp_1", wp_1);
           wp 2 = choose_new_gain("wp_2", wp_2);
wp 2 = choose_new_gain("wp_z", wp_z);
           wp r = choose new gain("wp r", wp r);
           printf("\n");
           wv_1 = choose_new_gain("wv_1", wv_1);
wv_2 = choose_new_gain("wv_2", wv_2);
            wv_z = choose_new_gain("wv_z", wv_z);
           wv_r = choose_new_gain("wv_r", wv_r);
       printf("\n");
       printf("Change Integral Adaptation Gains? (y/n): ");
       if (getch() == 'y') {
           printf("\n\n");
            d_1 = choose_new_gain("delta_1", d_1);
           d 2 = choose new gain("delta 2", d 2);
d z = choose new gain("delta z", d z);
d r = choose new gain("delta r", d r);
```

```
printf("\n");
            a0_1 = choose_new_gain("alpha0_1", a0_1);
a0_2 = choose_new_gain("alpha0_2", a0_2);
             a0_z = choose_new_gain("alpha0_z", a0_z);
             a0_r = choose_new_gain("alpha0_r", a0_r);
            printf("\n");
             a1_1 = choose_new_gain("alpha1_1", a1_1);
            a1_2 = choose_new_gain("alpha1 2", a1_2);
a1_z = choose_new_gain("alpha1 z", a1_z);
a1_r = choose_new_gain("alpha1 r", a1_r);
       printf("\n");
       printf("Change Proportional Adaptation Gains? (y/n): ");
        if (getch() == 'y') (
             printf("\n\n");
             rho_1 = choose_new_gain("rho_1", rho_1);
             rho_2 = choose_new_gain("rho_2", rho_2);
             rho_z = choose_new_gain("rho_z", rho_z);
             rho_r = choose_new_gain("rho_r", rho_r);
             printf("\n");
             b0_1 = choose_new_gain("beta0_1", b0_1);
             b0_? = choose_new_gain("beta0_2", b0_2);
b0_z = choose_new_gain("beta0_z", b0_z);
             b0_r = choose_new_gain("beta0_r", b0_r);
             printf("\n");
             bl_1 = choose_new_gain("bcta1_1", bl_1);
bl_2 = choose_new_gain("bcta1_2", bl_2);
             bl_z = choose_new_gain("bctal_z", bl_z);
bl_r = choose_new_gain("bctal_r", bl_r);
        )
/* DISPLAY INITIAL VALUES OF THE AUXILIARY SIGNALS */
   printf("\nChange? (y/n): ");
    if(getch() == 'y') {
        printf("\n\n");
        f_1 = choose_new_gain("aux. sig. 1", f_1);
        f_2 = choose_new_gain("aux. sig. 2", f_2);
        f_z = choose_new_gain("aux. sig. z", f_z);
        f_r = choose_ncw_gain("aux. sig. r", f_r);
/* DISPLAY CURRENT SAMPLING PERIOD AND PROMPT FOR A CHANGE */
    sample_per = (double)(*sample_timer)/1000000;
    printf("\n\nThe sampling period is: %lf sec \tChange? [y/n]: ", sample per);
    if(getch() == 'y') {
        printf("\n\nEnter new sample period (in sec.): ");
        scanf ("%lf", &sample per);
        *sample_timer = (unsigned int) (sample_per * 100000^);
        printf("sample_timer = %x \n", *sample timer);
        printf("press any key to continue ");
        while (!getch());
    }
/* CHECK ROBOT FOR OVER RUN ERRORS */
    robot error check();
/* GET TRAJECTORY */
    pp = choose_path_planner(sample_per);
    s_pp = pp;
/* ALLOCATE MEMORY FOR DATA CAPTURE
 * Data from at least 1000 equally spaced samples are captured */
    num samples = Duration / sample_per;
    if (num samples < 1000) {
        capture_intrvl = 1; }
    else {
        capture_intrvl = (unsigned int) (num_samples/1000);}
```

```
num captures = (unsigned int)(num samples/capture intrvl);
    if(!(c_a_l=s_c_a_l=(long int *)malloc(num_captures * 4 + 20))){
       printf("Not enough memory available Aborting\n\n");
        goto terminate; )
    if(!(c_d_l=s_c_d_l=(long int *)malloc(num_captures * 4 + 20))){
       printf("Not enough memory available. Aborting\n\n");
       goto terminate; }
    if(!'C_a_2=s_c_a_2=(long int *)ralloc(num_captures * 4 + 20))){
    printf("Not enough memory available. Aborting\n\n");
       goto terminate; )
    if(!(c_d_2=s_c_d_2=(long int *)malloc(num_captures * 4 + 20))){
       printf("Not enough memory available. Aborting\n\n");
       goto terminate; }
    if(!(c_a_z=s_c_a_z=(long int *)malloc(num_captures * 4 + 20))){
       printf("Not enough memory available. Aborting\n\n");
       goto terminate; }
    if(!(c_d_z=s_c_d_z=(long int *)malloc(num_captures * 4 + 20)))(
       printf("Not enough memory available. Aborting\n\n");
        goto terminate; }
    if(!(c a r*s c a r=(long int *)malloc(num captures * 4 + 20))){
       printf("Not enough memory available. Aborting\n\n");
       goto terminate; }
    if(!(c_d_r=s_c_d_r=(long int *)malloc(num_captures * 4 + 20))){
       printf("Not enough memory available. Aborting\n\n");
        goto terminate; )
    if(!(c t 1 = s c t 1 = (int *)malloc(num captures * 2 + 20))){
       printf("Not enough memory available. Aborting\n\n");
       goto terminate; )
    if(!(c_t_2 = s_c_t_2 = (int *)malloc(num_captures * 2 + 20))){
       printf("Not enough memory available. Aborting\n\n");
       goto terminate; }
    if(!(c_t_z = s_c_t_z = (int *)malloc(num_captures * 2 + 20)))(
       printf("Not enough memory available. Aborting\n\n");
       goto terminate; )
    if(!(c_t_r = s_c_t_r = (int *)malloc(num_captures * 2 + 20))){
       printf("Not enough memory available. Aborting\n\n");
       goto terminate; }
/* PROMPT TO BEGIN EXECUTION */
    delline();
    printf("\r(A)bort or (B)egin the move? \n");
    do {
        switch(ch = tolower(getch())) {
            case 'a':
                        goto terminate;
            case 'b':
                        break; /* continue */
            default :
                        delline();
                        printf("\rINVALID CHOICE. TRY AGAIN: ");
    } while (ch '= 'a' && ch '= 'b');
/* BEGINNING OF THE MOVE */
    clrscr();
    printf("\t\t\EXECUTING");
    spo2 = sample_per / 2;
    if(*(pp+28) - *(pp+0) < 0) f_1 = -f_1;

if(*(pp+29) - *(pp+1) < 0) f_2 = -f_2;
    if(*(pp+30) - *(pp+2) < 0) fz = -fz;
    if(*(pp+31) - *(pp+3) < 0) f_r = -f_r;
    d_1spo2 = d_1 * sample_per / 2;
    d^2spo2 = d^2 * sample per / 2;
    d zspo2 = d z * sample per / 2;
    d_rspo2 = d_r * sample_per / 2;
    a0_1spo2= a0_1* sample_per / 2;
    a0 2spo2= a0 2* sample per / 2;
    a0_zspo2= a0_z* sample_per / 2;
    a0_rspo2= a0_r* sample_per / 2;
a1_lspo2= a1_l* sample_per / 2;
    al 2spo2= al 2* sample per / 2;
    al_zspo2= al_z* sample_per / 2;
    al_rspo2= al_r* sample_per / 2;
    *command = 0 \times 0001;
                              /* have 196 begin its i_o loop */
```

```
while( (*pp != 0xffff0000) && (*error == 0) ) {
/* wait for signal from slave to begin a new sampling period */
    while(!(*timing_a & 0x0001));
/* store current parameters as previous (last) parameters */
    f_11 = f_1, f_21 = f_2, f_21 = f_2, f_{r} = f_{r};
    r_11 = r_1, r_21 = r_2, r_21 = r_z, r_11 = r_r;

kp_11 = kp_1, kp_21 = kp_2, kp_21 = kp_z, kp_r1 = kp_r;

kv_11 = kv_1, kv_21 = kv_2, kv_21 = kv_z, kv_r1 = kv_r;

e_11 = e_1, e_21 = e_2, e_21 = e_2, e_71 = e_7;

e_21 = e_1, e_21 = e_2, e_71 = e_7;
     edot_11 = edot_1, edot_21 = edot_2, edot_z1 = edot_z, edot_r1 = edot_r;
/* wait for position */
     while(!(*timing_a & 0x0002));
/* compute joint position error in degrees and capture robot state */
    e_1 = ((*c_d_1 = *pp++) - (*c_a_1 = *act_pos_1))/872.2227;
e_2 = ((*c_d_2 = *pp++) - (*c_a_2 = *act_pos_2))/444.4444;
     e_z = ((*c_d_z = *pp++) - (*c_a_z = *act_pos_z))/380,9600;
e_r = ((*c_d_r = *pp++) - (*c_a_r = *act_pos_r))/227,5555;
/* compute joint velocity error */
     edot_1 = (e_1 - e_11)/sample_per;
     edot_2 = (e_2 - e_2)/sample_per;
     edot_z = (e_z - e_zl)/sample_per;
     edot r = (e r - e rl)/sample per;
/* compute weighting errors */
     r_1 = wp_1*e_1 + wv_1*edot_1;
r_2 = wp_2*e_2 + wv_2*edot_2;
     r_z = wp_z*e_z + wv_z*edot_z;
     r_r = wp_r*e_r + wv_r*edot_r;
/* compute auxiliary signals */
     f_1 = f_11 + d_1spo2*(r_1+r_11) + rho_1*(r_1-r_11);
f_2 = f_21 + d_2spo2*(r_2+r_21) + rho_2*(r_2-r_21);
f_z = f_z1 + d_zspo2*(r_z+r_z1) + rho_z*(r_z-r_z1);
     f[r = f[r] + d[rspo2*(r]r+r]r] + rho[r*(r]r-r]r];
/* compute gains */
     kp_1 = kp_1l + a0_1spo2*(r_1*e_1 + r_1l*e_1l) +
              b0_1*(r_1*e_1 - r_11*e_11);
     kp_2 = kp_21 + a0_2spo2*(r_2*e_2 + r_21*e_21) + b0_2*(r_2*e_2 - r_21*e_21);
     kp_z = kp_zl + a0_zspo2*(r_z*e_z + r_z)*e_zl) +
              b0_z*(r_z*e_z - r_zl*e_zl);
     kp r = kp rl + a0_rspo2*(r r*e_r + r rl*e_rl) +
b0_r*(r r*e_r - r rl*e_rl);
     kv_2 = kv_21 + al_2spo2*(r_2*edot_2 + r_21*edot_21) +
              b1_2*(r_2*edot_2 - r_21*edot_21);
     kv z = kv zl + al zspo2*(r z*edot z + r zl*edot zl) +
              bl_z*(r_z*edot_z - r_zl*edot_zl);
     kv_r = kv_rl + al_{rspo2}*(r_r*edot r + r_rl*edot_rl) +
              bl_r*(r_r*edot_r - r_rl*edot_rl);
/* compute torques */
     *c_t_1 = *t_1 = (int)(0x0800 + e_1*kp_1 + edot_1*kv_1 + f_1);
     *ct2 = *t2 = (int) (0x0800 + e2*kp2 + edot 2*kv2 + f2);
     *ctz = *tz = (int) (0x0800 + ez*kpz + edotz*kv / + f/);
*ctr = *tr = (int) (0x0800 + er*kpr + edotr*kv r + fr);
      *timing b = 0x0001; /* tell 196 that torques are ready */
 /* capture data every capture interval */
     if(!capture counter) {
          *c d 1++;
          *c_d_2++;
          *c d z++;
          *c_d_r++;
          *c a 1++;
```

```
*c_a_2++;
           *c a 2++;
           *c_a_r++;
*c_t_1++;
           *c_t_2++;
           *c_t_z++;
           *c_t_r++;
           counter++:
           capture_counter = capture_intrvl - 1;
       else (
           capture counter --;
   /* indicate end of sample periou */
       *timing_a = 0x0000;
/* INDICATE END OF COMMAND */
    *command = 0x0000;
    *timing_b = 0x0000;
/* DISPLAY SUCCESS OF EXECUTION */
   delline();
   if (*error) {
       printf("\n\n\tCommand terminated prematurely because:\n\n");
       display errors ();
   }
       printf("\n\n\t\t\tCommand executed normally.\n\n");
       printf("\t\tSaving %i data samples to disk. Please wait.\n\n", counter),
       c_a_l = s_c_a_l; /* reset the pointers */
       c_d^1 = s_c_d^2;
       c_a_2 = s_c_a_2;
       cd2 = scd2;
       caz = scaz;
       c_d_z = s_c_d_z;
       car = scar;
       cdr = scdr;
       c_t] = s_c_t_1;
       ct2 = sct2;
       c_t_z = s_c_t_z;
       c_t_r = s_:_t_r;
/* open file for saving to disk */
       if((fp=fopen("robot.mat", "wb")) ==NULL) {
           printf("cannot open file. Aborting\n");
           goto terminate;
       x.imagf = 0;
       x.namlen = 5;
       x.mrows = counter;
/* save a vector called 'torl' (joint 1 torque) to disk */
                      /* data is signed int type */
       x.type = 30;
       x.ncols = 1;
       data_file_name = "tor1";
       fwrite(&x, sizeof(Fmatrix), 1, fp);
       fwrite(data_file_name, sizeof(char), x.namlen,fp);
fwrite(&x, sizeof(Fmatrix), 1, fp);
       fwrite(data_file_name, sizeof(char), x.namlen,fp);
fwrite(&x, sizeof(Fmatrix), 1, fp);
       fwrite(data_file_name, sizeof(char), x.namlen, fp);
fwrite(c_t_z, 2, counter, fp);
/* save a vector called 'torr' (joint r torque) to disk */
       data file name = "torr";
       fwrite(&x, sizeof(Fmatrix), 1, fp);
       fwrite(data_file_name, sizeof(char), x.namlen, fp);
```

```
fwrite(c t r, 2, counter, fp);
/* save sample period and capture interval to disk */
         x.type = 0;
                            /* data is double precision */
         x.mrows = 1;
         data_file_name = "smpl";
         fwrite(&x, sizeof(Fmatrix), 1, fp);
         fwrite(data_file_name, sizeof(char), x.namlen,fp);
fwrite(&sample_per, 8, 1, fp);
         x.type = 0040; /* data is unsigned 16-bit */
data_file_name = "invl";
         fwrite(&x, sizeof(Fmatrix), 1, fp,;
         fwrite(data_file_name, sizeof(char), x.namlen,fp);
fwrite(scapture_intrv1, 2, 1, fp);
/* save a matrix called 'pos1' (joint 1 position) to disk
* column 1 = desired position of joint 1
 * column 2 = actual position of joint 1 */
         x.type = 20;
                            /* data is long int type */
         x.ncols = 2;
         x.mrows = counter;
         data file name = "pos1";
         fwrite(&x, sizeof(Fmatrix), 1, fp);
         fwrite(data_file_name, sizeof(char), x.namlen,fp);
fwrite(c_d_1, 4, counter, fp);
   fwrite(c_a_1, 4, counter, fp);
/* save a matrix called 'pos2' (joint 2 position) to disk
 * column 1 = desired position of joint 2
 * column 2 = actual position of joint 2 */
         data file name = "pos2";
         fwrite(&x, sizeof(Fmatrix), 1, fp);
         fwrite(data_file_name, sizeof(char), x.namlen,fp);
fwrite(c d 2, 4, counter, fp);
fwrite(c a 2, 4, counter, fp);
/* save a matrix called 'posz' (joint z position) to disk
 * column 1 = desired position of joint z
 * column 2 = actual position of joint z */
         data_file_name = "posz";
         fwrite(&x, sizeof(Fmatrix), 1, fp);
         fwrite(data_file_name, sizeof(char), x.namlen,fp);
         fwrite(c_d_z, 4, counter, fp);
fwrite(c_a_z, 4, counter, fp);
/* save a matrix called 'posr' (joint r position) to disk
 * column 1 = desired position of joint r
* column 2 = actual position of joint r */
         data_file_name = "posr";
         fwrite(&x, sizeof(Fmatrix), 1, fp);
         fwrite(data file name, sizeof(char), x.namlen,fp);
         fwrite(c_d_r, 4, counter, fp);
         fwrite(c_a_r, 4, counter, fp);
         fclose(fp);
/* FREE ALLOCATED MEMORY */
terminate:
     if(s_pp) farfree(s_pp);
     if (s c a l) free(s c a l);
     if(s c d 1) free(s c d 1);
if(s c a 2) free(s c a 2);
     if (s c d 2) free(s c d 2);
     if (s_c_a z) free(s_c_a z);
     if(s_c_d_z) free(s_c_d_z);
if(s_c_a_r) free(s_c_a_r);
     if (s_c_d_r) free(s_c_d_r);
     if(s_c_t_1) free(s_c_t_1);
     if (s_c_t_2) free(s c_t_2);
     if(s_c_t_z) free(s_c_t_z);
     if (s c t r) free(s c t r);
     printf("\t\t\tPress any key to continue ");
     while (!getch());
     return;
}
```

```
* CUBIC.ROB - A cubic spline path generator that incorporates via points.
* Velocity at the via points is automatically chosen.
**********************
#include "alloc.h"
#include "stdio.h"
void get cartesian position(); /* function prototypes */
void get_joint_position();
void get_time();
cmp sign(double, double);
                                /* global variable declared in utility.rob */
extern double Duration;
long int huge *cubic_spline(double sample_per)
   char ch;
   int i, j, error, num_via_points;
double s1, s2; /* slope 1, slope 2 */
   long int pos[10][4];
                               /* A two dimensional position array.
                                * A row for each specified path point.
                                * column 0 = joint 1 pulse count
                                               n 2 n
                                         1 =
                                               **
                                    **
                                                       **
                                         2 =
                                                   z
                                    **
                                               **
                                                       **
                                                             ••
                                         3 =
                                                   r
   long int position[4];
   double vel[10][4] = {
                               /* A two dimensional velocity array */
                               /* A row for each specified path point */
       0,0,0,0,
                               /* column 0 = joint 1 velocity */
       0,0,0,0,
                               /*
                                     " 1 = " 2
" 2 = " z
                                                              */
       0,0,0,0,
       0,0,0,0,
                               /*
                                                  r
                                     **
                                         3 =
                                               **
                                                               */
       0,0,0,0,
       0,0,0,0,
       0,0,0,0,
       0,0,0,0,
       0,0,0,0,
       0,0,0,0);
   double a[4][4]; /* two dimensional array of coefficients */
   double time[10];
   double t, t2, t3, tf, tf2, tf3;
long int huge *path_point; /* pointer to starting address of path */
   long int huge *s_path_point; /* a place to save this value */
   unsigned long int num bytes mem req;
   long int far *current_pos = 0xC0010000;
/* DISPLAY CURRENT POSITION */
   clrscr();
   print(("\tCUBIC SPLINE PATH GENERATOR WITH VIA POINT CAPABILITY\n\n");
   printf("Features: *current robot position is automatically taken ");
   printf("as the initial\n");
                       position of the move\n");
   printf("
                       *velocity at the via points is automatically computed\n\n");
   printf("
   current position();
/* GET CURRENT POSITION (from DPR) */
   pos[0][0] = *(current pos++);
   pos(0)(1) = *(current_pos++);
   pos[0][2] = *(current_pos++);
   pos[0][3] = *(current pos);
    time[0] = 0;
/* GET # OF VIA POINTS AND CHOICE OF SPACE */
    printf("\nWill you be entering data in (C) artesian or (J) oint space? ");
    ch = qetch();
    printf("\nEnter total number of via points (Max. 8): ");
    scanf("%i", &num via points);
/* GET DURATION OF THE MOVE AND CREATE STORAGE AREA FOR THE PATH */
```

```
printf("Enter duration of the move (in sec.): ");
   do {
        scanf("%lf", &time[num_via points+1]);
       Duration = time(num_via_points+1);
        num_bytes_mem_req=(unsigned long int)(4*4*Duration/sample per+20+100);
            /* 4 = four joints
 * 4 = four bytes per joint per sample
             * Duration/sample_per = total # of samples
             * 20 = four bytes extra per joint + 4 byte flag
             * 100 = safety margin */
        path point=(long int huge *) farmalloc(num bytes mem req);
        if('path_point) {
            printf("Insufficient memory for a %lf", Duration);
            printf("second move.\nEnter a shorter duration: ");
    } while (!path point);
/* PROMPT FOR VIA POINTS AND FINAL POINT */
    for(i=1; i<=num_via_points+1; i++) ( /* for each via point */
        if(i != num_via_points+1) printf("\nVia point no. %i:\n", i);
                                   printf("\nFinal point:\n");
        else
        if(ch == 'c') get_cartesian_position(&pos[i][0]);
                      get_joint_position(&pos[i][0]);
        else
        if(i != num_via_points+1) get_time(&time[i]);
/* COMPUTE THE VELOCITIES AT THE VIA POINTS */
    for(i=1; i<=num_via_points; i++) { /* for each via point */
                                         /* for each joint */
        for(j=0; j<=3; j++){
            s1 = (double) (pos[i+1][j]-pos[i][j]) / (time(i+1) - time(i]);
s2 = (double) (pos[i][j]-pos[i-1][j]) / (time[i] - tim (i-1]);
            if(cmp\_sign(s1,s2)) vel[i][j] = 0;
            else vel[i][j] = (s1 + s2)/2; /* the average of the two */
        }
    }
/* COMPUTE AND STORE THE PATH */
    printf("\n\n\t\tcomputing THE PATH. PLEASE WAIT");
    s path point = path point;
    t = 0.0:
    tf = 0;
    for (i=0; i<=num_via_points; i++) { /* for all points (beginning at 0) */
        t = t - tf;
        tf = t \[ [i+1] - time[i];
tf2 = _' * tf;
        tf3 = tf2 * tf;
        for(j=0; j<=3; j++) { /* for each joint */
           a[j][0] = pos[i][j];
           a[j]{1} = vel[i][j];
           a(j)(2) = 3*(pos(i+1)[j]-a[j](0))/tf2 - (2*a[j][1]+vel[i+1][j])/tf;
           a[j][3] = -2*(pos[i+1][j]-a[j][0])/tf3 + (vel[i+1][j]+a[j][1])/tf2;
        for(; t <= tf+sample_per/2; t += sample_per) {</pre>
            t2 = t * t;
            t3 = t2 * t;
            *path_point++ = (long int) (a[0][0]+a[0][1]*t+a[0][2]*t2+a[0][3]*t3+0.5);
            *path_point++ =(long int) (a[1][0]+a[1][1]*t+a[1][2]*t2+a[1][3]*t3+0.5);
            *path_point++ =(long int) (a[2][0]+a[2][1]*t+a[2][2]*t2+a[2][3]*t3+0.5);
            *path_point++ = (long int) (a[3][0]+a[3][1]*t+a[3][2]*t2+a[3][3]*t3);
        }
    *path point = 0xffff0000; /* end-of-path flag */
    return(s_path_point);
}
```

```
* CYCLOID.ROB - Cycloidal path generator
#include "alloc.h"
#include "stdlo.h"
#include "math.h"
                                        /* fucntion prototypes */
void get_cartesian_position();
void get_joint_position();
extern double Duration;
                                  /* global variabl declared in utility.rob */
long int huge *cycloid(double sample_per)
    char ch;
    long int pos[4]; /* a vector to hold desired final position */
    long int huge *path point; /* starting address of path */
    long int huge *s path point;
                                        /* place to save this value */
    unsigned long int num_bytes_mem_req;
    long int far *initial_pos = 0xC0010000;
   long int i_lp = *(initial_pos + 0);
long int i_2p = *(initial_pos + 1);
    long int i_zp = *(initial_pos + 2);
    long int i_rp = *(initial_pos + 3);
    double d1, d2, dz, dr;
    double t, w, factor;
/* DISPLAY CURRENT POSITION */
    clrscr();
    printf("\t\tCYCLOIDAL PATH GENERATOR\n\n");
    current_position();
/* GET FINAL POSITION */
    printf("\nWill you be entering data in (C) artesian or (J) oint space? ");
    ch = getch();
    printf("\n\nFinal point:\n");
    if(ch == 'c') get cartesian_position(&pos[0]);
    else
                   get joint position (&pos[0]);
/* GET DURATION OF THE MOVE AND CREATE STORAGE AREA FOR THE PATH */
        printf("\n\tEnter Duration of the move (in sec.): ");
        scanf("%lf", &Duration);
        num_bytes_mem_req-(unsigned long int) (4*4*Duration/sample_per+20+100);
            /* 4 = four joints
             * 4 = four bytes per joint per sample
             * Duration/sample per = total # of samples
             * 20 = four bytes extra per joint + 4 byte flag
             * 100 = safety margin */
        path_point=(long int huge *) farmalloc(num_bytes_mem_req);
        if (!path point) printf ("Not enough memory in far heap\n");
    ) while ('path point);
/* COMPUTE AND STORE THE PATH */
    printf("\n\n\t\tcomputing THE PATH. PLEASE WAIT");
    s_path point = path point;
    w = 2 * M_PI / Duration;
   d1 = (pos[0]-i lp)/(2*M PI), d2 = (pos[1]-i 2p)/(2*M PI);
d2 = (pos[2]-i zp)/(2*M PI), dr = (pos[3]-i rp)/(2*M PI);
for(t = 0.0; t <= Duration + sample_per/2; t += sample_per) {</pre>
        factor = (w*t - sin(w*t));
        *path point++ = (long int) (i lp + d1*factor + 0.5);
        *path_point++ = (long int)(i_2p + d2*factor + 0.5);
        *path_point++ = (long int)(i_zp + dz*factor + 0.5);
        *path_point++ = (long int) (i_rp + dr*factor + 0.5);
    *path point = Oxffff0000; /* flag at end of path */
    return(s_path_point);
}
```