HT46R47, HT46R22,
HT46R23, HT46R24
A/D Type MCU
Handbook

May, 2003
Part I  Microcontroller Profile ................................................................. 1

Chapter 1  Hardware Structure ................................................................. 3
  Introduction ........................................................................................................ 3
  Features .................................................................................................................. 4
    Technology Features ........................................................................................... 4
    Kernel Features .................................................................................................. 4
    Peripheral Features ............................................................................................ 4
  Selection Table ..................................................................................................... 5
  Block Diagram ....................................................................................................... 6
  Pin Assignment ........................................................................................................ 7
  Pin Description ....................................................................................................... 8
  Absolute Maximum Ratings ................................................................................... 12
  D.C. Characteristics ............................................................................................... 12
  A.C. Characteristics ............................................................................................... 14
  System Architecture .............................................................................................. 15
    Clocking and Pipelining ....................................................................................... 15
    Program Counter .................................................................................................. 16
    Stack .................................................................................................................... 18
    Arithmetic and Logic Unit – ALU ....................................................................... 18
  Program Memory .................................................................................................... 19
    Organization .......................................................................................................... 19
    Special Vectors ..................................................................................................... 20
    Look-up Table ...................................................................................................... 20
    Table Program Example ....................................................................................... 21
  Data Memory .......................................................................................................... 22
    Organization .......................................................................................................... 22
    General Purpose Data Memory ............................................................................. 23
    Special Purpose Data Memory .............................................................................. 24
Contents

I2C Bus Communication ................................................................. 56
Interrupts ........................................................................................... 60
  External Interrupt ................................................................. 63
  Timer/Event Counter Interrupt .................................................. 63
  A/D Interrupt ............................................................................. 63
  I2C Interrupt ............................................................................. 63
  Interrupt Priority ....................................................................... 64
Programming Considerations ..................................................... 64
Reset and Initialization .................................................................. 65
  Reset ...................................................................................... 65
Oscillator ....................................................................................... 72
  System Clock Configurations .................................................. 72
  System Crystal/Ceramic Oscillator ......................................... 72
  System RC Oscillator ............................................................. 72
  Watchdog Timer Oscillator .................................................... 73
HALT and Wake-up in Power Down Mode .................................... 73
Watchdog Timer ........................................................................... 74
Configuration Options ................................................................. 76
Application Circuits ....................................................................... 77

Part II Programming Language .................................................. 81

Chapter 2 Instruction Set Introduction ........................................ 83
  Instruction Set ........................................................................... 83
  Instruction Timing ...................................................................... 83
  Moving and Transferring Data ................................................ 84
  Arithmetic Operations ............................................................. 84
  Logical and Rotate Operations .............................................. 84
  Branches and Control Transfer .............................................. 84
  Bit Operations .......................................................................... 84
  Table Read Operations .......................................................... 85
  Other Operations ..................................................................... 85
  Instruction Set Summary .......................................................... 85
  Convention .............................................................................. 85

Chapter 3 Instruction Definition .................................................. 89

Chapter 4 Assembly Language and Cross Assembler ............... 101
  Notational Conventions .......................................................... 101
  Statement Syntax ...................................................................... 102
    Name .................................................................................. 102
    Operation ............................................................................. 102
    Operand ............................................................................... 102
    Comment .............................................................................. 103
Assembly Directives ................................................................. 103
Conditional Assembly Directives ........................................... 103
File Control Directives ........................................................... 104
Program Directives ............................................................... 105
Data Definition Directives ...................................................... 108
Macro Directives ................................................................. 110
Assembly Instructions ............................................................ 112
  Name ...................................................................................... 112
  Mnemonic ........................................................................... 112
  Operand, Operator and Expression ...................................... 112
Miscellaneous ........................................................................ 114
  Forward References .......................................................... 114
  Local Labels ........................................................................ 114
  Reserved Assembly Language Words .................................. 115
Cross Assembler Options ..................................................... 116
Assembly Listing File Format .............................................. 116
Source Program Listing ......................................................... 116
Summary of Assembly .......................................................... 117
Miscellaneous ........................................................................ 117

Part III Development Tools ...................................................... 119

Chapter 5 MCU Programming Tools ........................................ 121
  HT-IDE3000 Development Environment .............................. 121
  Holtek In-Circuit Emulator – HT-ICE ................................. 122
  HT-ICE Interface Card ........................................................ 122
  OTP Programmer ............................................................... 123
  OTP Adapter Card .............................................................. 123
  System Configuration ........................................................ 123
  Installation ......................................................................... 124
    System Requirement ...................................................... 124
    Hardware Installation ..................................................... 125
    Software Installation ..................................................... 125
    OTP HandyWriter ......................................................... 129

Chapter 6 Quick Start ............................................................ 131
  Step 1 – Create a New Project ............................................. 131
  Step 2 – Add Source Program Files to the Project .................. 131
  Step 3 – Build the Project ................................................ 131
  Step 4 – Transmit Code to Holtek ...................................... 131
  Step 5 – Programming the OTP Device ............................... 132
Appendix .................................................................................................................. 133

Appendix A Device Characteristic Graphics .............................................................. 135

Appendix B Package Information ............................................................................. 145
Since the founding of the company, Holtek Semiconductor Inc. has concentrated much of its design efforts in the area of microcontroller development. Although supplying a wide range of semiconductor devices, the microcontroller category has always been a key product category within the Holtek range, and one which will continue to expand as their devices increase in functionality and maturity. By capitalizing on the substantial accumulated skills within its dedicated microcontroller development department, Holtek has been able to release a comprehensive range of high quality low-cost microcontroller devices for a wide range of application areas. Many important applications need to process analog signals such as those which interface to external sensors. All of these applications require analog to digital signal conversion by an A/D converter before they can be processed by the microcontroller. To address these needs, Holtek has developed its range of A/D microcontrollers, which in addition to having all the features and functions of the I/O range of devices, also include integrated multi-channel A/D converters of varying resolution and channel capacity. The inclusion of PWM functions and an I2C interface further enhance the features and application possibilities of the A/D series of microcontrollers.

This handbook is divided into three parts for user convenience. Most details regarding general datasheet information and device specification is located within Part I. Information related to microcontroller programming such as device instruction set, instruction definition, and assembly language directives is found within Part II. Part III relates to the Holtek range of Development Tools where information can be found on their installation and use.

By compiling all relevant data together in one handbook, we hope users of the Holtek range of A/D Type microcontroller devices will have at their fingertips a useful, complete and simple means to efficiently implement their microcontroller applications. Holtek’s efforts to combine information on device specifications, programming and development tools into one publication have produced a handbook which with careful use by the user should result in trouble free designs and the maximum benefit being gained from the many features of Holtek microcontroller devices. We welcome feedback and comments from our customers regarding further improvements.
Part I

Microcontroller Profile
Chapter 1

Hardware Structure

This section is the main datasheet section of the A/D Type microcontroller handbook and contains all the parameters and information related to the hardware. The information contained provides designers with details on all the main hardware features of the A/D Type microcontroller range which together with the programming section contains the information to enable swift and successful implementation of user microcontroller applications. By proper consultation of the relevant parts of this section, users can ensure that they make the most efficient use of the flexible and multi-function features within the A/D Type microcontroller series.

Introduction

The HT46R47/HT46C47, HT46R22/HT46C22, HT46R23/HT46C23 and HT46R24/HT46C24 form the series of 8-bit high performance RISC architecture microcontrollers, designed especially for applications that interface directly to analog signals, such as those from sensors. All devices include an integrated multi-channel Analog to Digital Converter in addition to one or more Pulse Width Modulation outputs. Device flexibility is enhanced with the usual features of the other microcontroller range such as HALT and wake-up functions, oscillator options, programmable frequency divider etc. These features combine to ensure applications require a minimum of external components and therefore reduce overall product costs. Having the benefits of integrated A/D and PWM functions, in addition to the advantages of low power consumption, high performance, I/O flexibility, as well as low cost, these devices have the versatility to suit a wide range of application possibilities such as sensor signal processing, motor driving, industrial control, consumer products, subsystem controllers, etc. Many features are common to all devices however, they differ in areas such as I/O pin count, RAM and ROM capacity, timer number and size, A/D channels, PWM outputs, etc.

The HT46R47, HT46R22, HT46R23 and HT46R24 are OTP devices offering the advantages of easy and effective program updates, using the Holtek range of development and programming tools. These devices provide the designer with the means for fast and low cost product development cycles. However, for applications that are at a mature state in their design process, the HT46C47, HT46C22, HT46C23 and HT46C24 mask version devices offer a complementary device for products with high volume and low cost demands. Fully pin and functionally compatible with their OTP sister devices, such mask version devices provide the ideal substitute for products which have gone beyond their development cycle and are facing cost down demands.
Features

Technology Features

- High-performance RISC Architecture
- Low-power Fully Static CMOS Design
- Operating Voltage:
  \[ f_{SYS}=4\text{MHz}: 2.2V \text{ to } 5.5V \]
  \[ f_{SYS}=8\text{MHz}: 3.3V \text{ to } 5.5V \]
- Power Consumption:
  2mA Typical at 5V 4MHz (for Crystal Oscillator with ADC Disabled)
  Maximum of 1µA Standby Current at 3V with WDT Disabled
- Temperature Range:
  Operating Temperature -40°C to 85°C (Industrial Grade)
  Storage Temperature -50°C to 125°C

Kernel Features

- Program Memory:
  2K×14 OTP/Mask ROM (HT46R47/HT46C47, HT46R22/HT46C22)
  4K×15 OTP/Mask ROM (HT46R23/HT46C23)
  8K×16 OTP/Mask ROM (HT46R24/HT46C24)
- Data Memory:
  64×8 SRAM (HT46R47/HT46C47, HT46R22/HT46C22)
  192×8 SRAM (HT46R23/HT46C23)
  384×8 SRAM (HT46R24/HT46C24)
- Table Read Function
- Multi-level Hardware Stack:
  6-level (HT46R47/HT46C47, HT46R22/HT46C22)
  8-level (HT46R23/HT46C23)
  16-level (HT46R24/HT46C24)
- Direct and Indirect Data Addressing Mode
- Bit Manipulation Instructions
- 63 Powerful Instructions
- Most Instructions Implemented in 1 Machine Cycle

Peripheral Features

- From 13 to 40 Bidirectional I/O with Pull-high Options
- Multi-channel 9 or 10-bit A/D Converter
- Pulse Width Modulator Outputs
- Port A Wake-up Options
- External Interrupt Input
- Event Counter Input
- Full Timer Functions with Prescaler and Interrupt
- Watchdog Timer (WDT)
**Selection Table**

The series of A/D microcontrollers include a comprehensive range of features, some of which are standard and some of which are device dependent. Most features are common to all devices, the main feature distinguishing them is Program Memory, Data Memory capacity, I/O count, timer functions, A/D channels and PWM outputs. To assist users in their selection of the most appropriate device for their application, the following table, which summarizes the main features of each device, is provided.

<table>
<thead>
<tr>
<th>Part No.</th>
<th>VDD</th>
<th>Program Memory</th>
<th>Data Memory</th>
<th>I/O</th>
<th>Timer</th>
<th>A/D</th>
<th>PWM</th>
<th>i²C</th>
<th>Stack</th>
<th>Package Types</th>
</tr>
</thead>
<tbody>
<tr>
<td>HT46R47</td>
<td>2.2V~5.5V</td>
<td>2K×14</td>
<td>64×8</td>
<td>13</td>
<td>8-bit×1</td>
<td>9-bit×4ch</td>
<td>8-bit×1</td>
<td>---</td>
<td>6</td>
<td>18DIP, 18SOP</td>
</tr>
<tr>
<td>HT46C47</td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
</tr>
<tr>
<td>HT46R22</td>
<td>2.2V~5.5V</td>
<td>2K×14</td>
<td>64×8</td>
<td>19</td>
<td>8-bit×1</td>
<td>9-bit×8ch</td>
<td>8-bit×1</td>
<td>☑</td>
<td>6</td>
<td>24SKDIP, 24SOP</td>
</tr>
<tr>
<td>HT46C22</td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
</tr>
<tr>
<td>HT46R23</td>
<td>2.2V~5.5V</td>
<td>4K×15</td>
<td>192×8</td>
<td>23</td>
<td>16-bit×1</td>
<td>10-bit×8ch</td>
<td>8-bit×1</td>
<td>☑</td>
<td>8</td>
<td>28SKDIP, 28SOP</td>
</tr>
<tr>
<td>HT46C23</td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
</tr>
<tr>
<td>HT46R24</td>
<td>2.2V~5.5V</td>
<td>8K×16</td>
<td>384×8</td>
<td>23</td>
<td>16-bit×2</td>
<td>10-bit×8ch</td>
<td>8-bit×2</td>
<td>☑</td>
<td>16</td>
<td>48SSOP</td>
</tr>
<tr>
<td>HT46C24</td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
</tr>
</tbody>
</table>

**Note** Part numbers including “C” are mask version devices while “R” are OTP devices.
Block Diagram

The following block diagram illustrates the main functional blocks of the A/D Type microcontroller series of devices.

Note

This block diagram represents the OTP devices, for the mask device there is no Device Programming Circuitry. The HT46R47/HT46C47 does not contain an I²C interface. The Bank Pointer only exists in the HT46R24/HT46C24.
Pin Assignment

<table>
<thead>
<tr>
<th>Pin Assignment</th>
<th>HT46R47/HT46C47</th>
<th>HT46R22/HT46C22</th>
<th>HT46R23/HT46C23</th>
</tr>
</thead>
<tbody>
<tr>
<td>PB5/ANS</td>
<td>1</td>
<td>24</td>
<td>PB6/ANS</td>
</tr>
<tr>
<td>PB4/ANS</td>
<td>2</td>
<td>23</td>
<td>PB7/ANS</td>
</tr>
<tr>
<td>PA3/PFD</td>
<td>3</td>
<td>22</td>
<td>PA4/TMR</td>
</tr>
<tr>
<td>PA2</td>
<td>4</td>
<td>21</td>
<td>PA5/I/O</td>
</tr>
<tr>
<td>PA1</td>
<td>5</td>
<td>20</td>
<td>PA6/I/O</td>
</tr>
<tr>
<td>PA0</td>
<td>6</td>
<td>19</td>
<td>PA7/I/O</td>
</tr>
<tr>
<td>PB3/ANS</td>
<td>7</td>
<td>18</td>
<td>PA8/I/O</td>
</tr>
<tr>
<td>PB2/ANS</td>
<td>8</td>
<td>17</td>
<td>OSC2</td>
</tr>
<tr>
<td>PB1/ANS</td>
<td>9</td>
<td>16</td>
<td>OSC1</td>
</tr>
<tr>
<td>PB0/ANS</td>
<td>10</td>
<td>15</td>
<td>VDD</td>
</tr>
<tr>
<td>VSS</td>
<td>11</td>
<td>14</td>
<td>PDD/PWM</td>
</tr>
<tr>
<td>PC0</td>
<td>12</td>
<td>13</td>
<td>PC1</td>
</tr>
<tr>
<td>PC1</td>
<td>13</td>
<td>12</td>
<td>PC2</td>
</tr>
</tbody>
</table>

**Note** The pin compatibility features of the microcontroller SKDIP/SOP packages allow for straightforward upgrading to devices of higher functionality with minimal changes to application hardware.
Pin Description

<table>
<thead>
<tr>
<th>Pin Name</th>
<th>I/O</th>
<th>Configuration Option</th>
<th>Description</th>
</tr>
</thead>
<tbody>
<tr>
<td>PA0<del>PA2, PA3/PFD, PA4/TMR, PA5/INT, PA6</del>PA7</td>
<td>I/O</td>
<td>Pull-high Wake-up PA3 or PFD</td>
<td>Bidirectional 8-bit input/output port. Each individual bit on this port can be configured as a wake-up input by a configuration option. Software instructions determine if the pin is a CMOS output or Schmitt Trigger input. A configuration option determines which bits on the port have pull-high resistors. Pins PA3, PA4 and PA5 are pin-shared with PFD, TMR and INT respectively.</td>
</tr>
<tr>
<td>PB0/AN0, PB1/AN1, PB2/AN2, PB3/AN3</td>
<td>I/O</td>
<td>Pull-high</td>
<td>Bidirectional 4-bit input/output port. Software instructions determine if the pin is a CMOS output or Schmitt Trigger input. A configuration option determines which bits on the port have pull-high resistors. PB is pin-shared with the A/D input pins. The A/D inputs are selected via software instructions. Once selected as an A/D input, the I/O function and pull-high resistor functions are disabled automatically.</td>
</tr>
<tr>
<td>PD0/PWM</td>
<td>I/O</td>
<td>Pull-high, I/O or PWM</td>
<td>Bidirectional 1-bit input/output port. Software instructions determine if the pin is a CMOS output or Schmitt Trigger input. A configuration option determines if this pin has a pull-high resistor. The PWM output is pin-shared with pin PD0 selected via configuration option.</td>
</tr>
<tr>
<td>OSC1, OSC2</td>
<td>I/O</td>
<td>Crystal or RC</td>
<td>OSC1, OSC2 are connected to an external RC network or external crystal (determined by configuration option) for the internal system clock. For external RC system clock operation, OSC2 is an output pin for 1/4 system clock.</td>
</tr>
<tr>
<td>RES</td>
<td>I</td>
<td>—</td>
<td>Schmitt Trigger reset input. Active low.</td>
</tr>
<tr>
<td>VDD</td>
<td>—</td>
<td>—</td>
<td>Positive power supply</td>
</tr>
<tr>
<td>VSS</td>
<td>—</td>
<td>—</td>
<td>Negative power supply, ground</td>
</tr>
</tbody>
</table>

**Note**
1. Each pin on PA can be programmed through a configuration option to have a wake-up function.
2. Each pin on each port can be individually configured to have a pull-high resistor.
### Pin Name I/O Configuration Option Description

<table>
<thead>
<tr>
<th>Pin Name</th>
<th>I/O Configuration</th>
<th>Description</th>
</tr>
</thead>
<tbody>
<tr>
<td>PA0~PA2 PA3/PFD PA4/TMR PA5/INT PA6/SDA PA7/SCL</td>
<td>I/O</td>
<td>Pull-high Wake-up PA3 or PFD PA6/PA7 or SDA/SCL</td>
</tr>
<tr>
<td>PB0/AN0 PB1/AN1 PB2/AN2 PB3/AN3 PB4/AN4 PB5/AN5 PB6/AN6 PB7/AN7</td>
<td>I/O</td>
<td>Pull-high</td>
</tr>
<tr>
<td>PC0~PC1</td>
<td>I/O</td>
<td>Pull-high</td>
</tr>
<tr>
<td>PD0/PWM</td>
<td>I/O</td>
<td>Pull-high I/O or PWM</td>
</tr>
<tr>
<td>OSC1 OSC2</td>
<td>I/O</td>
<td>Crystal or RC</td>
</tr>
<tr>
<td>RES</td>
<td>I</td>
<td></td>
</tr>
<tr>
<td>VDD</td>
<td></td>
<td></td>
</tr>
<tr>
<td>VSS</td>
<td></td>
<td></td>
</tr>
</tbody>
</table>

#### Note
1. Each pin on PA can be programmed through a configuration option to have a wake-up function.
2. Individual pins on PA can be selected to have a pull-high resistors. However, individual pins on Port B and Port C cannot be selected to have pull-high resistors. If the pull-high configuration is chosen for a particular PB or PC port, then all input pins on this port will be connected to pull-high resistors.
<table>
<thead>
<tr>
<th>Pin Name</th>
<th>I/O</th>
<th>Configuration Option</th>
<th>Description</th>
</tr>
</thead>
<tbody>
<tr>
<td>PA0~PA2</td>
<td>I/O</td>
<td>Pull-high Wake-up</td>
<td>Bidirectional 8-bit input/output port. Each individual bit on this port can be configured as a wake-up input by a configuration option. Software instructions determine if the pin is a CMOS output or Schmitt Trigger input. A configuration option determines which bits on the port have pull-high resistors. Pins PA3, PA4 and PA5 are pin-shared with PFD, TMR and INT respectively. Pins PA6 and PA7 are pin-shared with SDA and SCL respectively and are used to implement the I²C bus function.</td>
</tr>
<tr>
<td>PA3/PFD</td>
<td></td>
<td>PA3 or PFD</td>
<td></td>
</tr>
<tr>
<td>PA4/TMR</td>
<td></td>
<td>PA6/PA7 or SDA/SCL</td>
<td></td>
</tr>
<tr>
<td>PA5/INT</td>
<td></td>
<td></td>
<td></td>
</tr>
<tr>
<td>PA6/SDA</td>
<td></td>
<td></td>
<td></td>
</tr>
<tr>
<td>PA7/SCL</td>
<td></td>
<td></td>
<td></td>
</tr>
<tr>
<td>PB0/AN0</td>
<td>I/O</td>
<td>Pull-high</td>
<td>Bidirectional 5-bit input/output port. Software instructions determine if the pin is a CMOS output or Schmitt Trigger input. A configuration option determines if all pins on the port have pull-high resistors. PB is pin-shared with the A/D input pins. The A/D inputs are selected via software instructions. Once selected as an A/D input, the I/O function and pull-high resistor functions are disabled automatically.</td>
</tr>
<tr>
<td>PB1/AN1</td>
<td></td>
<td></td>
<td></td>
</tr>
<tr>
<td>PB2/AN2</td>
<td></td>
<td></td>
<td></td>
</tr>
<tr>
<td>PB3/AN3</td>
<td></td>
<td></td>
<td></td>
</tr>
<tr>
<td>PB4/AN4</td>
<td></td>
<td></td>
<td></td>
</tr>
<tr>
<td>PB5/AN5</td>
<td></td>
<td></td>
<td></td>
</tr>
<tr>
<td>PB6/AN6</td>
<td></td>
<td></td>
<td></td>
</tr>
<tr>
<td>PB7/AN7</td>
<td></td>
<td></td>
<td></td>
</tr>
<tr>
<td>PC0~PC4</td>
<td>I/O</td>
<td>Pull-high</td>
<td>Bidirectional 2-bit input/output port. Software instructions determine if the pin is a CMOS output or Schmitt Trigger input. A configuration option determines if both pins on this port have pull-high resistors. The PWM0 output is pin-shared with pin PD0 and the PWM1 output is pin-shared with PD1, selected via configuration options.</td>
</tr>
<tr>
<td>PD0/PWM0</td>
<td></td>
<td>I/O or PWM</td>
<td></td>
</tr>
<tr>
<td>PD1/PWM1</td>
<td></td>
<td></td>
<td></td>
</tr>
<tr>
<td>OSC1</td>
<td>I/O</td>
<td>Crystal or RC</td>
<td>OSC1, OSC2 are connected to an external RC network or external crystal (determined by configuration option) for the internal system clock. For external RC system clock operation, OSC2 is an output pin for 1/4 system clock.</td>
</tr>
<tr>
<td>OSC2</td>
<td>I/O</td>
<td></td>
<td></td>
</tr>
<tr>
<td>RES</td>
<td>I</td>
<td>—</td>
<td>Schmitt Trigger reset input. Active low.</td>
</tr>
<tr>
<td>VDD</td>
<td>—</td>
<td>—</td>
<td>Positive power supply</td>
</tr>
<tr>
<td>VSS</td>
<td>—</td>
<td>—</td>
<td>Negative power supply, ground</td>
</tr>
</tbody>
</table>

**Note**

1. Each pin on PA can be programmed through a configuration option to have a wake-up function.
2. Individual pins on PA can be selected to have a pull-high resistors. However, individual pins on Port B, Port C and Port D cannot be selected to have pull-high resistors. If the pull-high configuration is chosen for a particular PB, PC or PD port, then all input pins on this port will be connected to pull-high resistors.
3. The pin description table is based on the 28-pin device. Due to packaging limitations some pins may not exist on the 24-pin package.
<table>
<thead>
<tr>
<th>Pin Name</th>
<th>Configuration Option</th>
<th>Description</th>
</tr>
</thead>
<tbody>
<tr>
<td>PA0~PA2</td>
<td>I/O</td>
<td>Bidirectional 8-bit input/output port. Each individual bit on this port can be configured as a wake-up input by a configuration option. Software instructions determine if the pin is a CMOS output or Schmitt Trigger input. A configuration option determines which bits on the port have pull-high resistors. Pins PA3 and PA5 are pin-shared with PFD and INT respectively. Pins PA6 and PA7 are pin-shared with SDA and SCL respectively and are used to implement the I2C bus function.</td>
</tr>
<tr>
<td>PA3/PFD</td>
<td>Pull-high</td>
<td>Wake-up PA3 or PFD PA6/PA7 or SDA/SCL</td>
</tr>
<tr>
<td>PA4</td>
<td></td>
<td></td>
</tr>
<tr>
<td>PA5/INT</td>
<td></td>
<td></td>
</tr>
<tr>
<td>PA6/SDA</td>
<td></td>
<td></td>
</tr>
<tr>
<td>PA7/SCL</td>
<td></td>
<td></td>
</tr>
<tr>
<td>PB0/AN0</td>
<td>I/O</td>
<td>Bidirectional 8-bit input/output port. Software instructions determine if the pin is a CMOS output or Schmitt Trigger input. A configuration option determines which bits on the port have pull-high resistors. PB is pin-shared with the A/D input pins. The A/D inputs are selected via software instructions. Once selected as an A/D input, the I/O function and pull-high resistor functions are disabled automatically.</td>
</tr>
<tr>
<td>PB1/AN1</td>
<td>Pull-high</td>
<td></td>
</tr>
<tr>
<td>PB2/AN2</td>
<td></td>
<td></td>
</tr>
<tr>
<td>PB3/AN3</td>
<td></td>
<td></td>
</tr>
<tr>
<td>PB4/AN4</td>
<td></td>
<td></td>
</tr>
<tr>
<td>PB5/AN5</td>
<td></td>
<td></td>
</tr>
<tr>
<td>PB6/AN6</td>
<td></td>
<td></td>
</tr>
<tr>
<td>PB7/AN7</td>
<td></td>
<td></td>
</tr>
<tr>
<td>PC0~PC7</td>
<td>I/O</td>
<td>Bidirectional 8-bit input/output port. Software instructions determine if the pin is a CMOS output or Schmitt Trigger input. A configuration option determines if all pins on this port have pull-high resistors.</td>
</tr>
<tr>
<td>PD0/PWM0</td>
<td>I/O</td>
<td>Bidirectional 8-bit input/output port. Software instructions determine if the pin is a CMOS output or Schmitt Trigger input. A configuration option determines if all pins on this port have pull-high resistors. The PWM0/PWM1/PWM2 and PWM3 output pins are pin-shared with pins PD0/PD1/PD2 and PD3 respectively, selected via configuration options.</td>
</tr>
<tr>
<td>PD1/PWM1</td>
<td>Pull-high</td>
<td>I/O or PWM</td>
</tr>
<tr>
<td>PD2/PWM2</td>
<td></td>
<td></td>
</tr>
<tr>
<td>PD3/PWM3</td>
<td></td>
<td></td>
</tr>
<tr>
<td>PD4~PD7</td>
<td></td>
<td></td>
</tr>
<tr>
<td>PF0~PF7</td>
<td>I/O</td>
<td>Bidirectional 8-bit input/output port. Software instructions determine if the pin is a CMOS output or Schmitt Trigger input. A configuration option determines if all pins on this port have pull-high resistors.</td>
</tr>
<tr>
<td>TMR0</td>
<td>I</td>
<td>—</td>
</tr>
<tr>
<td>TMR1</td>
<td>I</td>
<td>—</td>
</tr>
<tr>
<td>OSC1</td>
<td>I</td>
<td>Crystal or RC</td>
</tr>
<tr>
<td>OSC2</td>
<td>O</td>
<td>—</td>
</tr>
<tr>
<td>RES</td>
<td>I</td>
<td>—</td>
</tr>
<tr>
<td>VDD</td>
<td>—</td>
<td>—</td>
</tr>
<tr>
<td>VSS</td>
<td>—</td>
<td>—</td>
</tr>
</tbody>
</table>

Chapter 1  Hardware Structure
**Absolute Maximum Ratings**

Supply Voltage ......................................................... $V_{SS} - 0.3V$ to $V_{SS} + 6.0V$
Input Voltage ........................................................ $V_{SS} - 0.3V$ to $V_{DD} + 0.3V$
Storage Temperature .................................................. $-50^\circ C$ to $125^\circ C$
Operating Temperature .............................................. $-40^\circ C$ to $85^\circ C$

These are stress ratings only. Stresses exceeding the range specified under Absolute Maximum Ratings may cause substantial damage to the device. Functional operation of this device at other conditions beyond those listed in the specification is not implied and prolonged exposure to extreme conditions may affect device reliability.

**D.C. Characteristics**

<table>
<thead>
<tr>
<th>Symbol</th>
<th>Parameter</th>
<th>Test Conditions</th>
<th>Min.</th>
<th>Typ.</th>
<th>Max.</th>
<th>Unit</th>
</tr>
</thead>
<tbody>
<tr>
<td>$V_{DD}$</td>
<td>Operating Voltage</td>
<td>$f_{SYS}=4MHz$</td>
<td>2.2</td>
<td></td>
<td>5.5</td>
<td>V</td>
</tr>
<tr>
<td></td>
<td></td>
<td>$f_{SYS}=8MHz$</td>
<td>3.3</td>
<td></td>
<td>5.5</td>
<td>V</td>
</tr>
<tr>
<td>$I_{DD1}$</td>
<td>Operating Current (Crystal OSC)</td>
<td>$3V$ No load, $f_{SYS}=4MHz$ ADC off</td>
<td>—</td>
<td>0.6</td>
<td>1.5</td>
<td>mA</td>
</tr>
<tr>
<td></td>
<td></td>
<td>$5V$</td>
<td>—</td>
<td>2</td>
<td>4</td>
<td>mA</td>
</tr>
<tr>
<td>$I_{DD2}$</td>
<td>Operating Current (RC OSC)</td>
<td>$3V$ No load, $f_{SYS}=4MHz$ ADC off</td>
<td>—</td>
<td>0.8</td>
<td>1.5</td>
<td>mA</td>
</tr>
<tr>
<td></td>
<td></td>
<td>$5V$</td>
<td>—</td>
<td>2.5</td>
<td>4</td>
<td>mA</td>
</tr>
<tr>
<td>$I_{DD3}$</td>
<td>Operating Current</td>
<td>$5V$ No load, $f_{SYS}=8MHz$ ADC off</td>
<td>—</td>
<td>3</td>
<td>5</td>
<td>mA</td>
</tr>
<tr>
<td>$I_{STB1}$</td>
<td>Standby Current (WDT Enabled)</td>
<td>$3V$ No load, system HALT</td>
<td>—</td>
<td>—</td>
<td>5</td>
<td>µA</td>
</tr>
<tr>
<td></td>
<td></td>
<td>$5V$</td>
<td>—</td>
<td>—</td>
<td>10</td>
<td>µA</td>
</tr>
<tr>
<td>$I_{STB2}$</td>
<td>Standby Current (WDT and A/D Disabled)</td>
<td>$3V$ No load, system HALT</td>
<td>—</td>
<td>—</td>
<td>1</td>
<td>µA</td>
</tr>
<tr>
<td></td>
<td></td>
<td>$5V$</td>
<td>—</td>
<td>—</td>
<td>2</td>
<td>µA</td>
</tr>
<tr>
<td>Symbol</td>
<td>Parameter</td>
<td>Test Conditions</td>
<td>Min.</td>
<td>Typ.</td>
<td>Max.</td>
<td>Unit</td>
</tr>
<tr>
<td>--------</td>
<td>-----------</td>
<td>-----------------</td>
<td>------</td>
<td>------</td>
<td>------</td>
<td>------</td>
</tr>
<tr>
<td></td>
<td></td>
<td>VDD Conditions</td>
<td></td>
<td></td>
<td></td>
<td></td>
</tr>
<tr>
<td>VIL1</td>
<td>Input Low Voltage for I/O Ports, TMR, TMR0, TMR1, INT</td>
<td>0</td>
<td>—</td>
<td>0.3VDD</td>
<td>V</td>
<td></td>
</tr>
<tr>
<td>VIH1</td>
<td>Input High Voltage for I/O Ports, TMR, TMR0, TMR1, INT</td>
<td>0.7VDD</td>
<td>—</td>
<td>VDD</td>
<td>V</td>
<td></td>
</tr>
<tr>
<td>VIL2</td>
<td>Input Low Voltage (RES)</td>
<td>0</td>
<td>—</td>
<td>0.4VDD</td>
<td>V</td>
<td></td>
</tr>
<tr>
<td>VIH2</td>
<td>Input High Voltage (RES)</td>
<td>0.9VDD</td>
<td>—</td>
<td>VDD</td>
<td>V</td>
<td></td>
</tr>
<tr>
<td>VLR</td>
<td>Low Voltage Reset</td>
<td>—</td>
<td>—</td>
<td>2.7</td>
<td>3</td>
<td>3.3</td>
</tr>
<tr>
<td>IOL</td>
<td>I/O Port Sink Current</td>
<td>3V</td>
<td>V_{OL}=0.1VDD</td>
<td>4</td>
<td>8</td>
<td>—</td>
</tr>
<tr>
<td></td>
<td></td>
<td>5V</td>
<td>V_{OL}=0.1VDD</td>
<td>10</td>
<td>20</td>
<td>—</td>
</tr>
<tr>
<td>IOH</td>
<td>I/O Port Source Current</td>
<td>3V</td>
<td>V_{OH}=0.9VDD</td>
<td>-2</td>
<td>-4</td>
<td>—</td>
</tr>
<tr>
<td></td>
<td></td>
<td>5V</td>
<td>V_{OH}=0.9VDD</td>
<td>-5</td>
<td>-10</td>
<td>—</td>
</tr>
<tr>
<td>RPH</td>
<td>Pull-high Resistance</td>
<td>3V</td>
<td>—</td>
<td>40</td>
<td>60</td>
<td>80</td>
</tr>
<tr>
<td></td>
<td></td>
<td>5V</td>
<td>—</td>
<td>10</td>
<td>30</td>
<td>50</td>
</tr>
<tr>
<td>VAD</td>
<td>A/D Input Voltage</td>
<td>—</td>
<td>—</td>
<td>0</td>
<td>—</td>
<td>VDD</td>
</tr>
<tr>
<td>EAD</td>
<td>A/D Conversion Integral Non-Linearity Error</td>
<td>—</td>
<td>—</td>
<td>±0.5</td>
<td>±1</td>
<td>LSB</td>
</tr>
<tr>
<td>IADC</td>
<td>Additional Power Consumption if A/D Converter is Used</td>
<td>3V</td>
<td>—</td>
<td>0.5</td>
<td>1</td>
<td>mA</td>
</tr>
<tr>
<td></td>
<td></td>
<td>5V</td>
<td>—</td>
<td>1.5</td>
<td>3</td>
<td>mA</td>
</tr>
</tbody>
</table>
### A.C. Characteristics

<table>
<thead>
<tr>
<th>Symbol</th>
<th>Parameter</th>
<th>Test Conditions</th>
<th>Min.</th>
<th>Typ.</th>
<th>Max.</th>
<th>Unit</th>
</tr>
</thead>
<tbody>
<tr>
<td></td>
<td>VDD Conditions</td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
</tr>
<tr>
<td>$f_{SYS}$</td>
<td>System Clock</td>
<td>2.2V~5.5V</td>
<td>400</td>
<td></td>
<td>4000</td>
<td>kHz</td>
</tr>
<tr>
<td></td>
<td></td>
<td>3.3V~5.5V</td>
<td>400</td>
<td></td>
<td>8000</td>
<td>kHz</td>
</tr>
<tr>
<td>$f_{TIMER}$</td>
<td>Timer I/P Frequency (TMR)</td>
<td>2.2V~5.5V</td>
<td>0</td>
<td></td>
<td>4000</td>
<td>kHz</td>
</tr>
<tr>
<td></td>
<td></td>
<td>3.3V~5.5V</td>
<td>0</td>
<td></td>
<td>8000</td>
<td>kHz</td>
</tr>
<tr>
<td>$t_{WDTOSC}$</td>
<td>Watchdog Oscillator Period</td>
<td>3V</td>
<td>45</td>
<td>90</td>
<td>180</td>
<td>μs</td>
</tr>
<tr>
<td></td>
<td></td>
<td>5V</td>
<td>32</td>
<td>65</td>
<td>130</td>
<td>μs</td>
</tr>
<tr>
<td>$t_{RES}$</td>
<td>External Reset Low Pulse Width</td>
<td>—</td>
<td>1</td>
<td></td>
<td></td>
<td>μs</td>
</tr>
<tr>
<td>$t_{SST}$</td>
<td>System Start-up Timer Period</td>
<td>Wake-up from HALT</td>
<td>1024</td>
<td></td>
<td></td>
<td>$t_{SYS}$</td>
</tr>
<tr>
<td>$t_{INT}$</td>
<td>Interrupt Pulse Width</td>
<td>—</td>
<td>1</td>
<td></td>
<td></td>
<td>μs</td>
</tr>
<tr>
<td>$t_{AD}$</td>
<td>A/D Clock Period</td>
<td>—</td>
<td>1</td>
<td></td>
<td></td>
<td>μs</td>
</tr>
<tr>
<td>$t_{ADC}$</td>
<td>A/D Conversion Time</td>
<td>—</td>
<td>—</td>
<td>76</td>
<td></td>
<td>$t_{AD}$</td>
</tr>
<tr>
<td>$t_{ADCS}$</td>
<td>A/D Sampling Time</td>
<td>—</td>
<td>—</td>
<td>32</td>
<td></td>
<td>$t_{AD}$</td>
</tr>
<tr>
<td>$t_{IC}$</td>
<td>I²C Bus Clock Period</td>
<td>Connect to external pull-high resistor 2kΩ</td>
<td>64</td>
<td></td>
<td></td>
<td>$t_{SYS}$</td>
</tr>
</tbody>
</table>

*$t_{SYS} = 1/f_{SYS}$
System Architecture

A key factor in the high performance features of the Holtek range of A/D Type microcontrollers is attributed to the internal system architecture. The range of devices take advantage of the usual features found within RISC microcontrollers providing increased speed of operation and enhanced performance. The pipelining scheme is implemented in such a way that instruction fetching and instruction execution are overlapped, hence instructions are effectively executed in one cycle, with the exception of branch or call instructions. An 8-bit wide ALU is used in practically all operations of the instruction set. It carries out arithmetic operations, logic operations, rotation, increment, decrement, branch decisions, etc. The internal data path is simplified by moving data through the Accumulator and the ALU. Certain internal registers are implemented in the Data Memory and can be directly or indirectly addressed. The simple addressing methods of these registers along with additional architectural features ensure that a minimum of external components is required to provide a functional I/O and A/D control system with maximum reliability and flexibility. This makes these devices suitable for low cost, high-volume production for controller applications requiring from 2K up to 8K words of program memory and from 64 to 384 bytes of data storage.

Clocking and Pipelining

The main system clock, derived from either a Crystal/Resonator or RC oscillator is subdivided into four internally generated non-overlapping clocks, T1~T4. The Program Counter is incremented at the beginning of the T1 clock during which time a new instruction is fetched. The remaining T2~T4 clocks carry out the decoding and execution functions. In this way, one T1~T4 clock cycle forms one instruction cycle. Although the fetching and execution of instructions takes place in consecutive instruction cycles, the pipelining structure of the microcontroller ensures that instructions are effectively executed in one instruction cycle. The exception to this are instructions where the contents of the Program Counter are changed, such as subroutine calls or jumps, in which case the instruction will take one more instruction cycle to execute.

Note
When the RC oscillator is used, OSC2 is freed for use as a T1 phase clock synchronizing pin. This T1 phase clock has a frequency of fSYS/4 with a 1:3 high/low duty cycle.

System Clocking and Pipelining
For instructions involving branches, such as jump or call instructions, two machine cycles are required to complete instruction execution. An extra cycle is required as the program takes one cycle to first obtain the actual jump or call address and then another cycle to actually execute the branch. The requirement for this extra cycle should be taken into account by programmers in timing sensitive applications.

```
1  MOV A,[12H]
2  CALL DELAY
3  CPL [12H]
4  :
5  :
6  DELAY:  NOP
```

**Program Counter**

During program execution, the Program Counter is used to keep track of the address of the next instruction to be executed. It is automatically incremented by one each time an instruction is executed except for instructions such as JMP or CALL that demand a jump to a non-consecutive Program Memory address. For the A/D series of microcontrollers, note that the Program Counter width varies with the Program Memory capacity depending upon which device is selected. However, it must be noted that only the lower 8 bits, known as the Program Counter Low Register, are directly addressable by user.

When executing instructions requiring jumps to non-consecutive addresses such as a jump instruction, a subroutine call, interrupt or reset, etc., the microcontroller manages program control by loading the required address into the Program Counter. For conditional skip instructions, once the condition has been met, the next instruction, which has already been fetched during the present instruction execution, is discarded and a dummy cycle takes its place while the correct instruction is obtained.

The lower byte of the Program Counter, known as the Program Counter Low register or PCL, is available for program control and is a readable and writable register. By transferring data directly into this register, a short program jump can be executed directly, however, as only this low byte is available for manipulation, the jumps are limited to the present page of memory, that is 256 locations. When such program jumps are executed it should also be noted that a dummy cycle will be inserted.

**Note**

The lower byte of the Program Counter is fully accessible under program control. The use of the PCL might cause program branching, so an extra cycle is needed to pre-fetch. Further information on the PCL register can be found in the Special Function Register section.
<table>
<thead>
<tr>
<th>Mode</th>
<th>Program Counter Bits</th>
</tr>
</thead>
<tbody>
<tr>
<td>b12</td>
<td>b11</td>
</tr>
<tr>
<td>Initial Reset</td>
<td>0 0 0 0 0 0 0 0 0 0 0 0 0</td>
</tr>
<tr>
<td>External Interrupt</td>
<td>0 0 0 0 0 0 0 0 0 0 0 1 0 0</td>
</tr>
<tr>
<td>Timer/Event Counter 0 Overflow</td>
<td>0 0 0 0 0 0 0 0 0 0 0 1 0 0</td>
</tr>
<tr>
<td>Timer/Event Counter 1 Overflow (HT46R24/HT46C24 only)</td>
<td>0 0 0 0 0 0 0 0 0 0 0 1 1 0 0</td>
</tr>
<tr>
<td>A/D Converter Interrupt (except HT46R24/HT46C24)</td>
<td>0 0 0 0 0 0 0 0 0 0 0 1 1 0 0</td>
</tr>
<tr>
<td>A/D Converter Interrupt (HT46R24/HT46C24 only)</td>
<td>0 0 0 0 0 0 0 0 0 0 0 1 0 0 0 0</td>
</tr>
<tr>
<td>I²C Bus Interrupt (except HT46R24/HT46C24)</td>
<td>0 0 0 0 0 0 0 0 0 0 0 1 0 0 0 0</td>
</tr>
<tr>
<td>I²C Bus Interrupt (HT46R24/HT46C24 only)</td>
<td>0 0 0 0 0 0 0 0 0 0 0 1 0 1 0 0 0</td>
</tr>
<tr>
<td>Skip</td>
<td>Program Counter + 2</td>
</tr>
<tr>
<td>Loading PCL</td>
<td>PC12 PC11 PC10 PC9 PC8 @7 @6 @5 @4 @3 @2 @1 @0</td>
</tr>
<tr>
<td>Jump, Call Branch</td>
<td>#12 #11 #10 #9 #8 #7 #6 #5 #4 #3 #2 #1 #0</td>
</tr>
<tr>
<td>Return from Subroutine</td>
<td>S12 S11 S10 S9 S8 S7 S6 S5 S4 S3 S2 S1 S0</td>
</tr>
</tbody>
</table>

**Note**

1. PC12~PC8: Current Program Counter bits
2. @7~@0: PCL bits
3. #12~#0: Instruction code bits
4. S12~S0: Stack register bits
5. For the HT46R24/HT46C24, the Program Counter is 13 bits wide, i.e. from b12~b0.
6. For the HT46R23/HT46C23, since its Program Counter is 12 bits wide, the b12 column in the table is not applicable.
7. For the HT46R47/HT46C47, HT46R22/HT46C22, since its Program Counter is 11 bits wide, the b11 and b12 columns in the table are not applicable.
8. The Timer/Event Counter 1 Overflow row is available only for the HT46R24/HT46C24.
9. For the HT46R47/HT46C47, HT46R22/HT46C22 and HT46R23/HT46C23 the Timer/Event Counter 0 represents the single timer, known as TMR.
**Stack**

This is a special part of the memory which is used to save the contents of the Program Counter only. The stack can have between 6, 8 or 16 levels depending upon which device is selected and is neither part of the data nor part of the program space, and is neither readable nor writable. The activated level is indexed by the stack pointer (SP) and is neither readable nor writeable. At a subroutine call or interrupt acknowledge signal, the contents of the Program Counter are pushed onto the stack. At the end of a subroutine or an interrupt routine, signaled by a return instruction (RET or RETI), the Program Counter is restored to its previous value from the stack. After a chip reset, the SP will point to the top of the stack.

If the stack is full and a non-masked interrupt takes place, the interrupt request flag will be recorded but the acknowledge signal will be inhibited. When the stack pointer is decremented (by RET or RETI), the interrupt will be serviced. This feature prevents stack overflow allowing the programmer to use the structure more easily. However, when the stack is full, a CALL subroutine instruction can still be executed which will result in a stack overflow. Precautions should be taken to avoid such cases which might cause unpredictable program branching.

![Stack Diagram](image)

**Note**

1. For the HT46R47/HT46C47 and HT46R22/HT46C22, N=6, i.e. 6 levels of stack available.
2. For the HT46R23/HT46C23, N=8, i.e. 8 levels of stack available.
3. For the HT46R24/HT46C24, N=16, i.e. 16 levels of stack available.

**Arithmetic and Logic Unit – ALU**

The arithmetic-logic unit or ALU is a critical area of the microcontroller that carries out arithmetic and logic operations of the instruction set. Connected to the main microcontroller data bus, the ALU receives related instruction codes and performs the required arithmetic or logical operations after which the result will be placed in the specified register. As these ALU calculation or operations may result in carry, borrow or other status changes, the status register will be correspondingly updated to reflect these changes. The ALU supports the following functions:

- Arithmetic operations ADD, ADDM, ADC, ADCM, SUB, SUBM, SBC, SBCM, DAA
- Logic operations AND, OR, XOR, ANDM, ORM, XORM, CPL, CPLA
- Rotation RRA, RR, RRCA, RRC, RLA, RL, RLCA, RLC
- Increment and Decrement INCA, INC, DECA, DEC
- Branch decision, JMP, SZ, SZA, SNZ, SIZ, SDZ, SIZA, SDZA, CALL, RET, RETI
Program Memory

The Program Memory is the location where the user code or program is stored. For microcontrollers, two types of Program Memory are usually supplied. The first type is the One-Time Programmable (OTP) Memory where users can program their application code into the device. Devices with OTP memory are denoted by having an “R” within their device name. By using the appropriate programming tools, OTP devices offer users the flexibility to freely develop their applications which may be useful during debug or for products requiring frequent upgrades or program changes. OTP devices are also applicable for use in applications that require low or medium volume production runs. The other type of memory is the mask ROM memory, denoted by having a “C” within the device name. These devices offer the most cost effective solutions for high volume products.

Organization

The Program Memory has a capacity of 2K by 14 to 8K by 16 bits depending upon which device is selected. The Program Memory is addressed by the Program Counter and also contains data, table information and interrupt entries. Table data, which can be setup in any location within the Program Memory, is addressed by separate table pointer registers.

The following diagram shows the Program Memory for the A/D Type microcontroller series.
Special Vectors
Within the Program Memory, certain locations are reserved for special usage such as reset and interrupts.

- **Location 000H**
  This vector is reserved for use by the chip reset for program initialization. After a chip reset is initiated, the program will jump to this location and begin execution.

- **Location 004H**
  This vector is used by the external interrupt. If the external interrupt pin on the device goes low, the program will jump to this location and begin execution if the external interrupt is enabled and the stack is not full.

- **Location 008H**
  This internal vector is used by the Timer/Event Counter. If a counter overflow occurs, the program will jump to this location and begin execution if the timer interrupt is enabled and the stack is not full. For the HT46R24/HT46C24 devices, which has dual timers, this timer is known as Timer/Event Counter 0 or TMR0, for the other devices the timer is known as TMR.

- **Location 00CH**
  With the exception of the HT46R24/HT46C24 devices, this internal vector is used by the A/D converter. When an A/D conversion cycle is complete, the program will jump to this location and begin execution if the A/D interrupt is enabled and the stack is not full. For the HT46R24/HT46C24 devices, this internal vector is used by its Timer/Event Counter 1. If a TMR1 counter overflow occurs, the program will jump to this location and begin execution if the internal interrupt is enabled and the stack is not full.

- **Location 010H**
  With the exception of the HT46R47/HT46C47 and HT46R24/HT46C24 devices, this internal vector is used by the I²C bus interface. When the I²C bus requires data transfer, the program will jump to this location and begin execution if the I²C interrupt is enabled and the stack is not full. For the HT46R24/HT46C24 devices this internal vector is used by its A/D converter interrupt. When the A/D conversion cycle in the HT46R24/HT46C24 is complete, the program will jump to this location and begin execution if the A/D interrupt is enabled and the stack is not full.

- **Location 014H**
  This vector, only available for the HT46R24/HT46C24 devices, is used by its I²C bus interface. When the I²C bus of the HT46R24/HT46C24 requires data transfer, the program will jump to this location and begin execution if the I²C interrupt is enabled and the stack is not full.

Look-up Table
Any location within the Program Memory can be defined as a look-up table where programmers can store fixed data. To use the look-up table, the table pointer must first be setup by placing the lower-order address of the look-up data to be retrieved in the Table Pointer Register TBLP. This register defines the lower 8-bit address of the look-up table. After setting up the table pointer, the table data can be retrieved from the current Program Memory page or last Program Memory page using the "TABRDC [m]" or "TABRD [m]" instructions respectively. When these instructions are executed, the lower order table byte from the Program Memory will be transferred to the user defined Data Memory register [m] as specified in the instruction. The higher order table data byte from the Program Memory will be transferred to the TBLH special register. Any unused bits in this transferred higher order byte will be read as “0”.

20
The following diagram illustrates the addressing/data flow of the look-up table:

![Diagram](image)

**Table Program Example**

The following example shows how the table pointer and table data is defined and retrieved from the HT46R47 A/D microcontroller. This example uses raw table data located in the last page which is stored there using the ORG statement. The value at this ORG statement is “700” hex which refers to the start address of the last page within the 2K Program Memory of the HT46R47 microcontroller. The table pointer is setup here to have an initial value of 06 hex. This will ensure that the first data read from the data table will be at the Program Memory address 706 hex or 6 locations after the start of the last page. Note that the value for the table pointer is referenced to the first address of the present page if the “TABRDC [m]” instruction is being used. The high byte of the table data which in this case is equal to zero will be transferred to the TBLH register automatically when the “TABRDL [m]” instruction is executed.

```assembly
org 700h ; sets initial address of last page (for HT46R47)
dc 00Ah, 00Bh, 00Ch, 00Dh, 00Eh, 00Fh, 01Ah, 01Bh
```

```assembly
tempreg1 db ? ; temporary register #1
tempreg2 db ? ; temporary register #2
mov a,06h ; initialize table pointer - note that this address is referenced
mov tblp,a ; to the last page or present page
tabrdl tempreg1 ; transfers value in table referenced by table pointer to tempreg1
; data at prog. memory address 706H transferred to tempreg1 and TBLH
dec tblp ; reduce value of table pointer by one
```

```assembly
tabrdl tempreg2 ; transfers value in table referenced by table pointer to tempreg2
; data at prog. memory address 705H transferred to tempreg2 and TBLH
; in this example the data “1A” is transferred to tempreg1 and data “0F” to register tempreg2
; the value “$0” will be transferred to the high byte register TBLH

org 700h ; sets initial address of last page (for HT46R47)
dc 00Ah, 00Bh, 00Ch, 00Dh, 00Eh, 00Fh, 01Ah, 01Bh
```

```assembly
tempreg1 db ? ; temporary register #1
tempreg2 db ? ; temporary register #2
mov a,06h ; initialize table pointer - note that this address is referenced
mov tblp,a ; to the last page or present page
tabrdl tempreg1 ; transfers value in table referenced by table pointer to tempreg1
; data at prog. memory address 706H transferred to tempreg1 and TBLH
dec tblp ; reduce value of table pointer by one
```

```assembly
tabrdl tempreg2 ; transfers value in table referenced by table pointer to tempreg2
; data at prog. memory address 705H transferred to tempreg2 and TBLH
; in this example the data “1A” is transferred to tempreg1 and data “0F” to register tempreg2
; the value “$0” will be transferred to the high byte register TBLH
```

```assembly
tempreg1 db ? ; temporary register #1
tempreg2 db ? ; temporary register #2
mov a,06h ; initialize table pointer - note that this address is referenced
mov tblp,a ; to the last page or present page
tabrdl tempreg1 ; transfers value in table referenced by table pointer to tempreg1
; data at prog. memory address 706H transferred to tempreg1 and TBLH
dec tblp ; reduce value of table pointer by one
```

```assembly
tabrdl tempreg2 ; transfers value in table referenced by table pointer to tempreg2
; data at prog. memory address 705H transferred to tempreg2 and TBLH
; in this example the data “1A” is transferred to tempreg1 and data “0F” to register tempreg2
; the value “$0” will be transferred to the high byte register TBLH
```

```assembly
tempreg1 db ? ; temporary register #1
tempreg2 db ? ; temporary register #2
mov a,06h ; initialize table pointer - note that this address is referenced
mov tblp,a ; to the last page or present page
tabrdl tempreg1 ; transfers value in table referenced by table pointer to tempreg1
; data at prog. memory address 706H transferred to tempreg1 and TBLH
dec tblp ; reduce value of table pointer by one
```

```assembly
tabrdl tempreg2 ; transfers value in table referenced by table pointer to tempreg2
; data at prog. memory address 705H transferred to tempreg2 and TBLH
; in this example the data “1A” is transferred to tempreg1 and data “0F” to register tempreg2
; the value “$0” will be transferred to the high byte register TBLH
```

```assembly
tempreg1 db ? ; temporary register #1
tempreg2 db ? ; temporary register #2
mov a,06h ; initialize table pointer - note that this address is referenced
mov tblp,a ; to the last page or present page
tabrdl tempreg1 ; transfers value in table referenced by table pointer to tempreg1
; data at prog. memory address 706H transferred to tempreg1 and TBLH
dec tblp ; reduce value of table pointer by one
```

```assembly
tabrdl tempreg2 ; transfers value in table referenced by table pointer to tempreg2
; data at prog. memory address 705H transferred to tempreg2 and TBLH
; in this example the data “1A” is transferred to tempreg1 and data “0F” to register tempreg2
; the value “$0” will be transferred to the high byte register TBLH
```

```assembly
tempreg1 db ? ; temporary register #1
tempreg2 db ? ; temporary register #2
mov a,06h ; initialize table pointer - note that this address is referenced
mov tblp,a ; to the last page or present page
tabrdl tempreg1 ; transfers value in table referenced by table pointer to tempreg1
; data at prog. memory address 706H transferred to tempreg1 and TBLH
dec tblp ; reduce value of table pointer by one
```

```assembly
tabrdl tempreg2 ; transfers value in table referenced by table pointer to tempreg2
; data at prog. memory address 705H transferred to tempreg2 and TBLH
; in this example the data “1A” is transferred to tempreg1 and data “0F” to register tempreg2
; the value “$0” will be transferred to the high byte register TBLH
```

```assembly
tempreg1 db ? ; temporary register #1
tempreg2 db ? ; temporary register #2
mov a,06h ; initialize table pointer - note that this address is referenced
mov tblp,a ; to the last page or present page
tabrdl tempreg1 ; transfers value in table referenced by table pointer to tempreg1
; data at prog. memory address 706H transferred to tempreg1 and TBLH
dec tblp ; reduce value of table pointer by one
```

```assembly
tabrdl tempreg2 ; transfers value in table referenced by table pointer to tempreg2
; data at prog. memory address 705H transferred to tempreg2 and TBLH
; in this example the data “1A” is transferred to tempreg1 and data “0F” to register tempreg2
; the value “$0” will be transferred to the high byte register TBLH
```

```assembly
tempreg1 db ? ; temporary register #1
tempreg2 db ? ; temporary register #2
mov a,06h ; initialize table pointer - note that this address is referenced
mov tblp,a ; to the last page or present page
tabrdl tempreg1 ; transfers value in table referenced by table pointer to tempreg1
; data at prog. memory address 706H transferred to tempreg1 and TBLH
dec tblp ; reduce value of table pointer by one
```

```assembly
tabrdl tempreg2 ; transfers value in table referenced by table pointer to tempreg2
; data at prog. memory address 705H transferred to tempreg2 and TBLH
; in this example the data “1A” is transferred to tempreg1 and data “0F” to register tempreg2
; the value “$0” will be transferred to the high byte register TBLH
```

```assembly
tempreg1 db ? ; temporary register #1
tempreg2 db ? ; temporary register #2
mov a,06h ; initialize table pointer - note that this address is referenced
mov tblp,a ; to the last page or present page
tabrdl tempreg1 ; transfers value in table referenced by table pointer to tempreg1
; data at prog. memory address 706H transferred to tempreg1 and TBLH
dec tblp ; reduce value of table pointer by one
```

```assembly
tabrdl tempreg2 ; transfers value in table referenced by table pointer to tempreg2
; data at prog. memory address 705H transferred to tempreg2 and TBLH
; in this example the data “1A” is transferred to tempreg1 and data “0F” to register tempreg2
; the value “$0” will be transferred to the high byte register TBLH
```
Because the TBLH register is a read-only register and cannot be restored, care should be taken to ensure its protection if both the main routine and Interrupt Service Routine use table read instructions. If using the table read instructions, the Interrupt Service Routines may change the value of the TBLH and subsequently cause errors if used again by the main routine. As a rule it is recommended that simultaneous use of the table read instructions should be avoided. However, in situations where simultaneous use cannot be avoided, the interrupts should be disabled prior to the execution of any main routine table-read instructions. Note that all table related instructions require two instruction cycles to complete their operation.

<table>
<thead>
<tr>
<th>Instruction</th>
<th>Table Location Bits</th>
</tr>
</thead>
<tbody>
<tr>
<td>TABRDC [m]</td>
<td>b12 b11 b10 b9 b8</td>
</tr>
<tr>
<td></td>
<td>b7 b6 b5 b4 b3 b2</td>
</tr>
<tr>
<td></td>
<td>b1 b0</td>
</tr>
<tr>
<td>TABRDL [m]</td>
<td>1 1 1 1 1 @7 @6 @5</td>
</tr>
<tr>
<td></td>
<td>@4 @3 @2 @1 @0</td>
</tr>
</tbody>
</table>

Note
1. PC12~PC8: Current Program Counter bits
2. @7~@0: Table Pointer TBLP bits
3. For the HT46R24/HT46C24, the Table address location is 13 bits, i.e. from b12~b0.
4. For the HT46R23/HT46C23, the Table address location is 12 bits, i.e. from b11~b0.
5. For the HT46R47/HT46C47 and HT46R22/HT46C22, the Table address location is 11 bits, i.e. from b10~b0.

Data Memory

The Data Memory is a volatile area of 8-bit wide RAM internal memory and is the location where temporary information is stored. Divided into two sections, the first of these is an area of RAM where special function registers are located. These registers have fixed locations and are necessary for correct operation of the device. Many of these registers can be read from and written to directly under program control, however, some remain protected from user manipulation. The second area of Data Memory is reserved for general purpose use. All locations within this area are read and write accessible under program control.

Organization

The two sections of Data Memory, the Special Purpose and General Purpose Data Memory are located at consecutive locations. All are implemented in RAM and are 8 bits wide but the length of each memory section is dictated by the type of microcontroller chosen. The start address of the Data Memory for all devices is the address 00H. The last Data Memory address is 7FH for the HT46R47/HT46C47 and HT46R22/HT46C22 devices, and FFH for the HT46R23/HT46C23 and HT46R24/HT46C24 devices. Registers which are common to all microcontrollers such as ACC, PCL etc. have the same Data Memory address.
Most of the Data Memory bits can be directly manipulated using the “SET [m].i” and “CLR [m].i” with the exception of a few dedicated bits. The Data Memory can also be accessed through the memory pointer register MP.

**General Purpose Data Memory**

All microcontroller programs require an area of read/write memory where temporary data can be stored and retrieved for use later. It is this area of RAM memory that is known as General Purpose Data Memory. This area of Data Memory is fully accessible by the user program for both read and write operations. By using the “SET [m].i” and “CLR [m].i” instructions, individual bits can be set or reset under program control, giving the user a large range of flexibility for bit manipulation in the Data Memory.

Note: The 384 bytes of General Purpose Data Memory in the HT46R24/HT46C24 are stored in two individual memory banks. Before reading from or writing to the General Purpose Data Memory it is essential to first ensure that the correct Data Memory bank is selected by setting up the Bank Pointer.
Special Purpose Data Memory

This area of Data Memory is where registers, necessary for the correct operation of the microcontroller, are stored. Most of the registers are both readable and writable but some are protected and are readable only, the details of which are located under the relevant Special Function Register section. Note that for locations that are unused, any read instruction to these addresses will return the value "00H".
Special Function Registers

To ensure successful operation of the microcontroller, certain internal registers are implemented in the Data Memory area. These registers ensure correct operation of internal functions such as timers, interrupts, etc. as well as external functions such as I/O data control and A/D converter operation. The location of these registers within the Data Memory begins at the address 00H. Any unused Data Memory locations between these special function registers and the point where the General Purpose Memory begins is reserved for future expansion purposes, attempting to read data from these locations will return a value of 00H.

Indirect Addressing Registers – IAR, IAR0, IAR1

The method of indirect addressing allows data manipulation using memory pointers instead of the usual direct memory addressing method where the actual memory address is defined. Any action on the Indirect Addressing Registers will result in corresponding read/write operations to the memory location specified by the corresponding memory pointer. For the HT46R47/HT46C47 and HT46R22/HT46C22 devices, one Indirect Addressing Register, IAR, and one Memory Pointer, MP, is provided. For the HT46R23/HT46C23 and HT46R24/HT46C24 devices, two Indirect Addressing Registers, IAR0 and IAR1, and two Memory Pointers, MP0 and MP1, are provided. Note that these Indirect Addressing Registers are not physically implemented and that reading the Indirect Addressing Registers indirectly will return a result of 00H and writing to the registers indirectly will result in no operation.

Memory Pointers – MP, MP0, MP1

For the HT46R47/HT46C47 and HT46R22/HT46C22 devices, one memory pointer known as MP is provided, whereas for the HT46R23/HT46C23 and HT46R24/HT46C24 devices, two memory pointers known as MP0 and MP1 are provided. These memory pointers are physically implemented in the Data Memory and can be manipulated in the same way as normal registers providing a convenient way with which to address and track data. When any operation to the relevant Indirect Addressing Registers is carried out, the actual address that the microcontroller is directed to is the address specified by the related Memory Pointer.

Note

For the HT46R47/HT46C47 and HT46R22/HT46C22 devices, bit 7 of the memory pointer is not implemented. However, it must be noted that when the memory pointer in these devices is read, a value of "1" will be read.

The following example shows how to clear a section of four RAM locations already defined as locations adres1 to adres4.

```assembly
.data .section 'data'
adres1   db ?
adres2   db ?
adres3   db ?
adres4   db ?
block    db ?
.code .section at 0 'code'
org 00h
```
start:
  mov a,04h ; setup size of block
  mov block,a
  mov a,offset adres1 ; Accumulator loaded with first RAM address
  mov mp,a ; setup memory pointer with first RAM address

loop:
  clr [00] ; clear the data at address defined by mp
  inc mp ; increment memory pointer
  sdz block ; check if last memory location has been cleared
  jmp loop

continue:

The important point to note here is that in the example shown above, no reference is made to specific RAM addresses.

Bank Pointer – BP

The Bank Pointer only exists in the HT46R24/HT46C24 devices. The existence of the Bank Pointer enables the HT46R24/HT46C24 devices to have a higher capacity of General Purpose Data Memory compared to other devices in the A/D series. The address of the General Purpose Data Memory bank in the HT46R24/HT46C24 microcontrollers ranges from 40H to FFH, a range that would normally provide only 192 bytes of General Purpose Data Memory. However by locating the memory into two banks, known as Bank 0 and Bank 1, the General Purpose Data Memory capacity can be expanded to 384 bytes. Bit 0 of the Bank Pointer, is utilized to set the present bank of the General Purpose Data Memory. The General Purpose Data Memory is initialized to bank 0 after reset, except for the WDT Time-out reset in the HALT Mode, in which case, the General Purpose Data Memory bank remains unchanged. When it is required to read from or write to the General Purpose Data Memory in the HT46R24/HT46C24 microcontrollers, it is necessary to first setup the bank pointer to ensure that the correct memory bank is selected. It should be noted that the Special Function Data Memory is not affected by the bank selection, which means the Special Function Registers can be accessed from within either bank 0 or bank 1.

Accumulator – ACC

The Accumulator is central to the operation of any microcontroller and is closely related with operations carried out by the ALU. The Accumulator is the place where all intermediate results from the ALU are stored. Without the Accumulator it would be necessary to write the result of each calculation or logical operation such as addition, subtraction, shift, etc. to the Data Memory resulting in higher programming and timing overheads. Data transfer operations usually involve the temporary storage function of the Accumulator; for example, when transferring data between one user defined register and another, it is necessary to do this by passing the data through the Accumulator as no direct transfer between two registers is permitted.

Program Counter Low Register – PCL

To provide additional program control functions, the low byte of the Program Counter is made accessible to programmers by locating it within the Special Purpose area of the Data Memory. By manipulating this register, direct jumps to other program locations are easily implemented. Loading a value directly into this PCL register will cause a jump to the specified Program Memory location, however as the register is only 8-bit wide, only jumps within the current Program Memory page are permitted. When such operations are used, note that a dummy cycle will be inserted.
Look-up Table Registers – TBLP, TBLH

These two special function registers are used to control operation of the look-up table which is stored in the Program Memory. TBLP is the table pointer and indicates the location where the table is located. Its value must be setup before any table read commands are executed. Its value can be changed, for example using the INC or DEC instructions, allowing for easy table data pointing and reading. TBLH is the location where the high order byte of the table data is stored after a table read data instruction has been executed. Note that the lower order table data byte is transferred to a user defined location.

Status Register – STATUS

This 8-bit register (0AH) contains the zero flag (Z), carry flag (C), auxiliary carry flag (AC), overflow flag (OV), power down flag (PDF), and watchdog time-out flag (TO). It also records the status information and controls the operation sequence.

With the exception of the TO and PDF flags, bits in the status register can be altered by instructions like most other registers. Any data written into the status register will not change the TO or PDF flag. In addition, operations related to the status register may give different results due to the different instruction operations. The TO flag can be affected only by a system power-up, a WDT time-out or by executing the “CLR WDT” or “HALT” instruction. The PDF flag is affected only by executing the "HALT" or "CLR WDT" instruction or during a system power-up.

The Z, OV, AC and C flags generally reflect the status of the latest operations.

- C is set if an operation results in a carry during an addition operation or if a borrow does not take place during a subtraction operation; otherwise C is cleared. C is also affected by a rotate through carry instruction.
- AC is set if an operation results in a carry out of the low nibbles in addition, or no borrow from the high nibble into the low nibble in subtraction; otherwise AC is cleared.
- Z is set if the result of an arithmetic or logical operation is zero; otherwise Z is cleared.
- OV is set if an operation results in a carry into the highest-order bit but not a carry out of the highest-order bit, or vice versa; otherwise OV is cleared.
- PDF is cleared by a system power-up or executing the “CLR WDT” instruction. PDF is set by executing the “HALT” instruction.
- TO is cleared by a system power-up or executing the “CLR WDT” or “HALT” instruction. TO is set by a WDT time-out.

<table>
<thead>
<tr>
<th>b7</th>
<th>—</th>
<th>—</th>
<th>TO</th>
<th>PDF</th>
<th>OV</th>
<th>Z</th>
<th>AC</th>
<th>C</th>
</tr>
</thead>
</table>

**STATUS Register**

- **Arithmetic/logic operation flags**
  - Carry Flag
  - Auxiliary Carry Flag
  - Zero Flag
  - Overflow Flag

- **System management flags**
  - Power down flag
  - Watchdog time-out flag

Not implemented, read as “0”
In addition, on entering an interrupt sequence or executing a subroutine call, the status register will not be pushed onto the stack automatically. If the contents of the status registers are important and if the subroutine can corrupt the status register, precautions must be taken to correctly save it.

**Interrupt Control Registers – INTC, INTC0, INTC1**

These 8-bit registers known as INTC, INTC0 and INTC1, control the operation of the various external and internal interrupt functions. By setting various bits within these registers using standard bit manipulation instructions, the enable/disable function of each of the interrupts can be independently controlled. The various interrupt functions include those used by the internal timers, the analog to digital converter and the I²C bus in addition to the external interrupt pin INT. For the HT46R47/HT46C47 devices, only one 8-bit interrupt control register, known as INTC, is required to control all its interrupt functions, while the additional features of the other devices require two interrupt control registers, INTC0 and INTC1. A master interrupt bit within the INTC or INTC0 register, the EMI bit, acts like a global enable/disable and is used to set all of the interrupt bits either on or off. This bit is cleared when an interrupt routine is entered to disable all further interrupts and is set by executing the "RETI" instruction.

**Note**

In situations where other interrupts may require servicing within present interrupt service routines, the EMI bit can be manually set by the program after the present interrupt service routine has been entered.

**Timer/Event Counter Registers**

Depending upon which device is selected, all devices contain one or two integrated Timer/Event Counters of either 8-bit or 16-bit size. For devices with a single timer counter, an associated register, known as TMR, is the location where the timer value is located. An associated control register, known as TMRC, contains the setup information for the TMR register. For the HT46R24/HT46C24 devices which have two 16-bit timers, the individual timers are known as TMR0 and TMR1 with their respective control registers known as TMR0C and TMR1C. In the case of 16-bit timers, the actual value stored in the timer requires two bytes, a high byte and a low byte. These register pairs are known as TMRL/TMRH or TMR0L/TMR0H and TMR1L/TMR1H. Note that the timer registers can be directly written to in order to preload their contents with fixed data to allow different time intervals to be setup.

**Input/Output Ports and Control Registers**

Within the area of Special Function Registers, the I/O registers and their associated control registers play a prominent role. All I/O ports have a designated register correspondingly labeled as PA, PB, PC, etc. These labeled I/O registers are mapped to specific addresses within the Data Memory as shown in the Data Memory table which are used to transfer the appropriate output or input data on that port. With each I/O port there is an associated control register labeled PAC, PBC, PCC, etc. also mapped to specific addresses with the Data Memory. The control register specifies which pins of that port are set as inputs and which are set as outputs. To setup a pin as an input, the corresponding bit of the control register must be set high, for an output it must be set low. During program initialization, it is important to first setup the control registers to specify which pins are outputs and which are inputs before reading data from or writing data to the I/O ports. One flexible
feature of these registers is the ability to directly program single bits using the “SET [m].i” and “CLR [m].i” instructions. The ability to change I/O pins from output to input and vice-versa by manipulating specific bits of the I/O control registers during normal program operation is a useful feature of these devices.

**Pulse Width Modulator Registers – PWM, PWM0, PWM1, PWM2, PWM3**

Each device in the A/D microcontroller range contains either 1, 2 or 4 integrated Pulse Width Modulators or PWM. Each one has its own independent control register. For devices which contain a single PWM, the control register is known as PWM, for devices with two PWMs, the control registers are known as PWM0 and PWM1 while for devices with 4 PWMs, the control registers are known as PWM0–PWM3. The 8-bit contents of each of these registers define the duty cycle value for the modulation cycle of the corresponding pulse width modulator.

**I²C Bus Registers – HADR, HCR, HSR, HDR**

With the exception of the HT46R47/HT46C47, all devices contain an integrated I²C bus which interfaces to the external shared pins SDA and SCL on the microcontroller. The correct setup and data transfer operation of this 2-line bidirectional bus utilizes 4 special function registers. The HADR register sets the slave address of the device while the HCR is the control register that enables or disables the device as well as defines whether it is in transmit or receive mode. The HSR register is the status register while the HDR register is the input/output data register.

**A/D Converter Registers – ADRL, ADRH, ADCR, ADSR**

Each device in the A/D microcontroller range contains either a 4 or 8-channel A/D converter. The correct operation of the A/D requires the use of 4 registers. The high byte data register ADRH and low byte data register ADRL, are the two locations where the digital value is placed after the completion of an analog to digital conversion cycle. The channel selection and configuration of the A/D converter is setup via the control register ADCR while the A/D clock frequency is defined by the clock source register, ADSR.

**Input/Output Ports**

Holtek microcontrollers offer considerable flexibility on their I/O ports. With the input or output designation of every pin fully under user program control, pull-high options for all pins and wake up options on certain pins, the user is provided with an I/O structure to meet the needs of a wide range of application possibilities.

Depending upon which device or package is chosen, the microcontroller range provides from 13 to 40 bidirectional input/output lines labeled with port names PA, PB, PC, etc. These I/O ports are mapped to the Data Memory with specific addresses as shown in the Special Purpose Data Memory table. All of these I/O ports can be used for input and output operations. For input operation, these ports are non-latching, which means the inputs must be ready at the T2 rising edge of instruction “MOV A,[m]”, where m denotes the port address. For output operation, all the data is latched and remains unchanged until the output latch is rewritten.
Pull-high Resistors

Many product applications require pull-high resistors for their switch inputs usually requiring the use of an external resistor. To eliminate the need for these external resistors, all I/O pins, when configured as an input have the capability of being connected to an internal pull-high resistor. These pull-high resistors are selectable via configuration option and are implemented using a weak PMOS transistor. Note that on some ports, individual pins can be selected to have pull-high resistors, while on other ports all pins or no pins must be selected to have pull-high resistors.

Port A Wake-up

Each device has a HALT feature enabling the microcontroller to enter a power down mode and preserve power, a feature that is important for battery and other low power applications. Various methods exist to wake-up the microcontroller, one of which is to change the logic condition on one of the Port A pins from high to low. After a “HALT” instruction forces the microcontroller into entering a HALT condition, the processor will remain idle or in a low-power state until the logic condition of the selected wake-up pin on Port A changes from high to low. This function is especially suitable for applications that can be woken up via external switches. Note that each pin on Port A can be selected individually to have this wake-up feature.

I/O Port Control Registers

Each I/O line has its own control register (PAC, PBC, PCC, etc.) to control the input/output configuration. With this control register, each CMOS output or Schmitt Trigger input with or without pull-high resistor structures can be reconfigured dynamically under software control. Each pin of the I/O ports is directly mapped to a bit in its associated port control register. For the I/O pin to function as an input, the corresponding bit of the control register must be written as a “1”. This will then allow the logic state of the input pin to be directly read by instructions. When the corresponding bit of the control register is written as a “0”, the I/O pin will be setup as a CMOS output. If the pin is currently setup as an output, instructions can still be used to read the output register. However, it should be noted that the program will in fact only read the status of the output data latch and not the actual logic status of the output pin.

Pin-shared Functions

The flexibility of the microcontroller range is greatly enhanced by the use of pins that have more than one function. Limited numbers of pins can force serious design constraints on designers but by supplying pins with multi-functions, many of these difficulties can be overcome. For some pins, the chosen function of the multi-function I/O pins is set by configuration options while for others the function is set by application program control.

→ External Interrupt Input

The external interrupt pin INT is pin-shared with the I/O pin PA5. For applications not requiring an external interrupt input, the pin can be used as a normal I/O pin, however, to do this, the external interrupt enable bits in the INTC register must be disabled.
External Timer Clock Input
Each device in the A/D series contains either one or two timers depending upon which one is chosen. In the case of devices with a single timer, this pin is known as TMR, which is pin-shared with PA4. However, for the 48-pin package HT46R24/HT46C24 devices, which have two internal timers, there are two independent input pins known as TMR0 and TMR1. For the 28-pin package HT46R24/HT46C24 devices, which also have two internal timers, due to packaging limitations the TMR0 pin is not available. On this package only the TMR1 external timer pin is available which is pin-shared with PD1/PWM1/TMR1. If the PA4/TMR or PD1/PWM1/TMR1 pin is to be configured as a timer input, the corresponding control bits in the timer control register must be correctly set. The PA4/TMR and PD1/PWM1/TMR1 pin can be used as a normal I/O pin for applications that do not require external timer inputs. For such applications, the timer mode control bits in the timer control register must select the timer mode, which has an internal clock source, to prevent the I/O from interfering with the timer counter operation.

PFD, PWM Outputs, I²C Bus
Each device in the A/D series contains a PFD output, pin-shared with PA3, and one or more PWM outputs, pin-shared with pins PD0~PD3. The number of PWM outputs depends upon which device is chosen. With the exception of the HT46R47/HT46C47 devices, there are two pins associated with an internal I²C Bus, which are pin-shared with I/O pins PA6 and PA7. The function of all of these pins is chosen via configuration options and remains fixed after the device is programmed. Note that the correct software options within the application program must also be selected to enable correct operation. If the I²C option is chosen, then note that any pull-high resistor options associated with these pins will be automatically disconnected. For all pins, if chosen to function as I/O pins, then full pull-high options remain.

A/D inputs
Each device in the A/D series has either four or eight inputs for the A/D converter. All of these analog inputs are pin-shared with I/O pins on Port B. If these pins are to be used as A/D inputs and not as normal I/O pins then the corresponding bits in the A/D Converter Control Register, ADCR, must be properly set. There are no configuration options associated with the A/D function. If chosen as I/O pins, then full pin-high resistor configuration options remain, however if used as A/D inputs then any pull-high resistor options associated with these pins will be automatically disconnected.
PA3/PFD and PD0/PWM0~PD3/PWM3 Input/Output Ports

PA4/PA5 Input/Output Ports
Programming Considerations

Within the user program, one of the first things to consider is port initialization. After a reset, all of the I/O data and port control registers will be set high. This means that all I/O pins will default to an input state, the level of which depends on the other connected circuitry and whether pull-high options have been selected. If the port control registers PAC, PBC, PCC, etc. are then programmed to setup some pins as outputs, these output pins will have an initial high output value unless the associated port data register, PA, PB, PC, etc. is first programmed. Selecting which pins are inputs and which are outputs can be achieved byte-wide by loading the correct values into the appropriate port control register or by programming individual bits in the port control register using the “SET [m].i” and “CLR [m].i” instructions. Note that when using these bit control instructions, a read-modify-write operation takes place. The microcontroller must first read in the data on the entire port, modify it to the required new bit values and then rewrite this data back to the output ports.

Port A has the additional capability of providing wake-up functions. When the chip is in the HALT state, various methods are available to wake the device up. One of these is a high to low transition of any of the Port A pins. Single or multiple pins on Port A can be setup to have this function.

Timer/Event Counters

The provision of timers form an important part of any microcontroller, giving the designer a means of carrying out time related functions. The devices in the A/D Type MCU series contain either one or two count up timers of either 8 or 16-bit capacity depending upon which device is selected. As each timer has three different operating modes, they can be configured to operate as a general timer, an external event counter or as a pulse width measurement device. With the exception of TMR1 in the HT46R24/HT46C24 devices, the provision of an internal 8-stage prescaler to the timer clock circuitry gives added range to the timer.

There are two types of registers related to the Timer/Event Counters. The first is the register that contains the actual value of the timer and into which an initial value can be preloaded. Reading from this register retrieves the contents of the Timer/Event Counter. The second type of associated register is the timer control register which defines the timer options and determines how the timer is to be used. All devices can have the timer clock configured to come from the internal clock source. In addition, with the exception of TMR0 in the 28-pin package in the HT46R24/HT46C24 devices, the timer clock source can also be configured to come from an external timer pin. The accompanying table lists the associated timer register names.
<table>
<thead>
<tr>
<th></th>
<th>HT46R47/HT46C47</th>
<th>HT46R22/HT46C22</th>
<th>HT46R23/HT46C23</th>
<th>HT46R24/HT46R24</th>
</tr>
</thead>
<tbody>
<tr>
<td>No. of 8-bit Timers</td>
<td>1</td>
<td>1</td>
<td>—</td>
<td>—</td>
</tr>
<tr>
<td>Timer Register Name</td>
<td>TMR</td>
<td>TMR</td>
<td>—</td>
<td>—</td>
</tr>
<tr>
<td>Timer Control Register</td>
<td>TMRC</td>
<td>TMRC</td>
<td>—</td>
<td>—</td>
</tr>
<tr>
<td>No. of 16-bit Timers</td>
<td>—</td>
<td>—</td>
<td>1</td>
<td>2</td>
</tr>
<tr>
<td>Timer Register Name</td>
<td>—</td>
<td>—</td>
<td>TMR0L/TMR0H</td>
<td>TMR1L/TMR1H</td>
</tr>
<tr>
<td>Timer Control Register</td>
<td>—</td>
<td>—</td>
<td>TMRC</td>
<td>TMR0C/TMR1C</td>
</tr>
</tbody>
</table>

An external clock source is used when the timer is in the event counting mode, the clock source being provided on the external timer pin known as TMR, TMR0 or TMR1 depending on which device is selected. These external pins may be pin-shared with other I/O pins depending upon which device and package is chosen. Depending upon the condition of the TE bit in the corresponding timer control register, each high to low, or low to high transition on the external timer input pin will increment the counter by one. Note that the 28-pin package HT46R24/HT46C24 devices, although having two internal Timer/Event Counters, have only one external timer pin TMR1; due to packaging limitations the TMR0 pin is not available.

**Configuring the Timer/Event Counter Input Clock Source**

The internal timer’s clock source can originate from either the system clock or from an external clock source, with the exception of TMR0 in the 28-pin package HT46R24/HT46C24 devices. The system clock input timer source is used when the timer is in the timer mode or in the pulse width measurement mode. With the exception of TMR1 in the HT46R24/HT46C24 devices, whose timer clock source is \( f_{SYS}/4 \) and has no prescaler, the timer clock source is \( f_{SYS} \) divided by the value in the prescaler, the division ratio of which is conditioned by the bits PSC2, PSC1 and PSC0.

An external clock source is used when the timer is in the event counting mode, the clock source being provided on an external timer pin, TMR, TMR0 or TMR1 depending upon which device and which timer is used. Depending upon the condition of the TE bit, each high to low, or low to high transition on the external timer input pin will increment the counter by one. Note that as the 28-pin package HT46R24/HT46C24 devices has only one TMR1 external timer pin, its TMR0 internal timer cannot have an external clock source.

![8-bit Timer/Event Counter Structure – HT46R47/HT46C47 and HT46R22/HT46C22](image-url)
Timer Registers – TMR, TMRL/TMRH, TMR0L/TMR0H, TMR1L/TMR1H

The timer register is a special function register located in the special purpose Data Memory and is the place where the actual timer value is stored. For the 8-bit timer, this register is known as TMR. For the 16-bit timer, a pair of 8-bit registers are required to store the 16-bit timer value. In the case of the HT46R23/HT46C23 devices, this register pair is known as TMRL and TMRH. In the case of the HT46R24/HT46C24 device which has two 16-bit timers, the register pair for TMR0 is known as TMR0L and TMR0H, while the register pair for TMR1 is known as TMR1L and TMR1H.

The value in the timer registers increases by one each time an internal clock pulse is received or an external transition occurs on the external timer pin. The timer will count from the initial value loaded by the preload register to the full count of FFH for the 8-bit timer or FFFFH for the 16-bit timers at which point the timer overflows and an internal interrupt signal is generated. The timer value will then be reset with the initial preload register value and continue counting. Note that to achieve a maximum full range count of FFH for the 8-bit timer or FFFFH for the 16-bit timers, the preload registers must first be cleared to all zeros. It should be noted that after power on, the preload registers will be in an unknown condition. Note that if the Timer/Event Counters are in an OFF condition and data is written to their preload registers, this data will be immediately written into the actual counter. However, if the counter is enabled and counting, any new data written into the preload
data register during this period will remain in the preload register and will only be written into the actual counter the next time an overflow occurs. Note also that when the timer registers are read, the timer clock will be blocked to avoid errors, however, as this may result in certain timing errors, programmers must take this into account.

For devices with 16-bit timers, which have both low byte and high byte timer registers, accessing these registers is carried out in a specific way. It must be noted that when using instructions to preload data into the low byte register, namely, TMRL, TMR0L or TMR1L, the data will only be placed in a low byte buffer and not directly into the low byte register. The actual transfer of the data into the low byte register is only carried out when a write to its associated high byte register, namely, TMRH, TMR0H or TMR1H, is executed. On the other hand, using instructions to preload data into the high byte timer register will result in the data being directly written to the high byte register. At the same time the data in the low byte buffer will be transferred into its associated low byte register. For this reason, when preloading data into the 16-bit timer registers, the low byte should be written first. It must also be noted that to read the contents of the low byte register, a read to the high byte register must first be executed to latch the contents of the low byte buffer into its associated low byte register. After this has been done, the low byte register can be read in the normal way. Note that reading the low byte timer register will only result in reading the previously latched contents of the low byte buffer and not the actual contents of the low byte timer register.

Timer Control Registers – TMRC, TMR0C, TMR1C

The flexible features of the Holtek microcontroller Timer/Event Counters enable them to operate in three different modes, the options of which are determined by the contents of their respective control register. For devices with only one timer, the single timer control register is known as TMRC while for devices with two timers, there are two timer control registers known as TMR0C and TMR1C. It is the timer control register together with its corresponding timer registers that control the full operation of the Timer/Event Counters. Before the timers can be used, it is essential that the appropriate timer control register is fully programmed with the right data to ensure its correct operation, a process that is normally carried out during program initialization.

To choose which of the three modes the timer is to operate in, either in the timer mode, the event counting mode or the pulse width measurement mode, bits TM0 and TM1 must be set to the required logic levels. The timer-on bit TON or bit 4 of the Timer Control Register provides the basic on/off control of the timer, setting the bit high allows the counter to run, clearing the bit stops the counter. For timers that have prescalers, bits 0~2 of the Timer Control Register determine the division ratio of the input clock prescaler. The prescaler bit settings have no effect if an external clock source is used. If the timer is in the event count or pulse width measurement mode, the active transition edge level type is selected by the logic level of the TE or bit 3 of the TMRC register.
is SYS divided by the value programmed into the prescaler, the value of which is determined by bits the timer clock. With the exception of TMR1 in the HT46R24/HT46C24, the input clock frequency

In this mode, the timer can be utilized to measure fixed time intervals, providing an internal interrupt signal each time the counter overflows. To operate in this mode, bits TM1 (bit 7) and TM0 (bit 6) of the TMRC register must be set to 1 and 0 respectively. In this mode the internal clock is used as the timer clock. With the exception of TMR1 in the HT46R24/HT46C24, the input clock frequency is $f_{SYS}$ divided by the value programmed into the prescaler, the value of which is determined by bits PSC2–PSC0 in the timer control register. For TMR1 in the HT46R24/HT46C24, which has no prescaler, the input clock frequency is $f_{SYS}$/4. The timer-on bit, TON must be set high to enable the

The HT46R24/HT46C24 devices have two internal timers, TMR0 and TMR1, and therefore require an additional timer control register TMR1C.

Configuring the Timer Mode

In this mode, the timer can be utilized to measure fixed time intervals, providing an internal interrupt signal each time the counter overflows. To operate in this mode, bits TM1 (bit 7) and TM0 (bit 6) of the TMRC register must be set to 1 and 0 respectively. In this mode the internal clock is used as the timer clock. With the exception of TMR1 in the HT46R24/HT46C24, the input clock frequency is $f_{SYS}$ divided by the value programmed into the prescaler, the value of which is determined by bits PSC2–PSC0 in the timer control register. For TMR1 in the HT46R24/HT46C24, which has no prescaler, the input clock frequency is $f_{SYS}$/4. The timer-on bit, TON must be set high to enable the
timer to run. Each time an internal clock high to low transition occurs, the timer increments by one; when the timer is full and overflows, an interrupt signal is generated and the timer will preload the value already loaded into the preload register and continue counting.

**Configuring the Event Counter Mode**

In this mode, a number of externally changing logic events, occurring on the external timer pin, can be recorded by the internal timer. For the timer to operate in the event counting mode, bits TM1 and TM0 of the TMRC register must be set to 0 and 1 respectively. The timer-on bit, TON must be set high to enable the timer to count. With TE low, the counter will increment each time the external timer pin receives a low to high transition. If TE is high, the counter will increment each time the external timer pin receives a high to low transition. As in the case of the other two modes, when the counter is full, the timer will overflow and generate an internal interrupt signal; the counter will preload the value already loaded into the preload register. If the external timer pin is pin-shared with other I/O pins, to ensure that the pin is configured to operate as an event counter input pin, two things have to happen. The first is to ensure that the TM0 and TM1 bits place the timer/event counter in the event counting mode, the second is to ensure that the port control register configures the pin as an input. Note that the 28-pin package HT46R24/HT46C24 devices, although having two internal timers, only one TMR1 external control pin is available. As a result TMR0 cannot be used in the Event Counter Mode.

**Configuring the Pulse Width Measurement Mode**

In this mode the width of external pulses applied to the external timer pin can be measured. In the Pulse Width Measurement Mode the timer clock source is supplied by the internal clock. For the timer to operate in this mode, bits TM0 and TM1 must both be set high. If the TE bit is low, once a high to low transition has been received on the external timer pin, the timer will start counting until the external timer pin returns to its original high level. At this point the TON bit will be automatically reset to zero and the timer will stop counting. If the TE bit is high, the timer will begin counting once a low to high transition has been received on the external timer pin and stop counting when the external timer pin returns to its original low level. As before, the TON bit will be automatically reset to zero and the timer will stop counting. It is important to note that in the Pulse Width Measurement Mode, the TON bit is automatically reset to zero when the external control signal on the external timer pin returns to its original level, whereas in the other two modes the TON bit can only be reset to zero under program control. The residual value in the timer, which can now be read by the pro-
gram, therefore represents the length of the pulse received on the external timer pin. As the TON bit has now been reset, any further transitions on the external timer pin will be ignored. Not until the TON bit is again set high by the program can the timer begin further pulse width measurements. In this way, single shot pulse measurements can be easily made. It should be noted that in this mode the counter is controlled by logical transitions on the external timer pin and not by the logic level. As in the case of the other two modes, when the counter is full, the timer will overflow and generate an internal interrupt signal. The counter will also be reset to the value already loaded into the preload register. If the external timer pin is pin-shared with other I/O pins, to ensure that the pin is configured to operate as a pulse width measuring input pin, two things have to happen. The first is to ensure that the TM0 and TM1 bits place the timer/event counter in the pulse width measuring mode, the second is to ensure that the port control register configures the pin as an input. Note that the 28-pin package HT46R24/HT46C24 devices, although having two internal timers, only one TMR1 external control pin is available. As a result TMR0 cannot be used in the Pulse Width Measurement Mode.

![Pulse Width Measurement Mode Timing Chart](image)

**Programmable Frequency Divider – PFD**

The PFD output is pin-shared with the I/O pin PA3. The function is selected via configuration option, however, if not selected, the pin can operate as a normal I/O pin. Note that for the HT46R24/HT46C24 devices, which have two internal timers, the timer source for the PFD can be chosen, via configuration option, to come from either one of the two timers.

The timer overflow signal is the clock source for the PFD circuit. The output frequency is controlled by loading the required values into the timer prescaler registers to give the required division ratio. The counter, driven by the system clock which is divided by the prescaler value, will begin to count-up from this preload register value until full, at which point an overflow signal is generated, causing the PFD output to change state. The counter will then be automatically reloaded with the preload register value and continue counting-up. Refer to the relevant Timer/Event Counters section for details of its settings and operations.

For the PFD output to function, it is essential that the corresponding bit of the Port A control register PAC bit 3 is setup as an output. If setup as an input the PFD output will not function, however, the pin can still be used as a normal input pin. The PFD output will only be activated if bit PA3 is set to “1”. This output data bit is used as the on/off control bit for the PFD output. Note that the PFD output will be low if the PA3 output data bit is cleared to “0”.

40
Using this method of frequency generation, and if a crystal oscillator is used for the system clock, very precise values of frequency can be generated.

**Prescaler**

With the exception of TMR1C, bits 0~2 of the timer control register can be used to define the pre-scaling stages of the internal clock sources of the Timer/Event Counter. The Timer/Event Counter overflow signal can be used to generate signals for the PFD and as a Timer Interrupt.

**I/O Interfacing**

The Timer/Event Counter when configured to run in the event counter or pulse width measurement mode, require the use of the external timer pin for correct operation. This external timer pin may be pin-shared with other I/O pins, depending upon which device is selected. Pull-high resistors can be selected for connection to the timer input pins. The timers can also be setup to drive the PFD pin. When the PFD output is selected by selecting the correct configuration option, the output of the chosen timer can be made to drive this at a frequency determined by the contents of the timer register and the timer.

**Programming Considerations**

When configured to run in the timer mode, the internal system clock is used as the timer clock source and is therefore synchronized with the overall operation of the microcontroller. In this mode when the appropriate timer register is full, the microcontroller will generate an internal interrupt signal directing the program flow to the respective internal interrupt vector. For the pulse width measurement mode, the internal system clock is also used as the timer clock source but the timer will only run when the correct logic condition appears on the external timer input pin. As this is an external event and not synchronized with the internal timer clock, the microcontroller will only see this external event when the next timer clock pulse arrives. As a result, there may be small differences in measured values requiring programmers to take this into account during programming. The same applies if the timer is configured to be in the event counting mode which again is an external event and not synchronized with the internal system or timer clock.

**Pulse Width Modulator**

Each microcontroller in the A/D series is provided with one or more Pulse Width Modulation (PWM) outputs. Useful for such applications such as motor speed control, the PWM function provides outputs with a fixed frequency but with a duty cycle that can be varied by setting particular values into the corresponding PWM register.
A single register, located in the Data Memory is assigned to each PWM. For devices with a single PWM output, this register is known as PWM. For devices with two PWM outputs, the registers assume the names PWM0 and PWM1 while devices with four PWM outputs require a further additional two registers known as PWM2 and PWM3. It is here that the 8-bit value, which represents the overall duty cycle of one modulation cycle of the output waveform, should be placed. To increase the PWM modulation frequency, each modulation cycle is modulated into two or four individual modulation sub-sections, known as the 7+1 mode or 6+2 mode respectively. With the exception of the HT46R47/HT46C47 devices, which have a fixed 6+2 mode, each device can choose which mode to use by selecting the appropriate configuration option. When a mode configuration option is chosen, it applies to all PWM outputs on that device. Note that when using the PWM it is only necessary to write the required value into the appropriate PWM register and select the required mode configuration option, the subdivision of the waveform into its sub-modulation cycles is done automatically within the microcontroller hardware.

For all devices, the PWM clock source is the system clock \( f_{SYS} \).

### PWM Function Table

<table>
<thead>
<tr>
<th>Device</th>
<th>Channels</th>
<th>PWMMode</th>
<th>Output pin</th>
<th>PWM Register Name</th>
</tr>
</thead>
<tbody>
<tr>
<td>HT46R47/HT46C47</td>
<td>1</td>
<td>6+2</td>
<td>PD0</td>
<td>PWM</td>
</tr>
<tr>
<td>HT46R22/HT46C22</td>
<td>1</td>
<td>6+2 or 7+1</td>
<td>PD0</td>
<td>PWM</td>
</tr>
<tr>
<td>HT46R23/HT46C23</td>
<td>1 (24-pin package)</td>
<td>6+2 or 7+1</td>
<td>PD0</td>
<td>PWM0</td>
</tr>
<tr>
<td>HT46R24/HT46C24</td>
<td>2 (28-pin package)</td>
<td>6+2 or 7+1</td>
<td>PD0/PD1</td>
<td>PWM0/PWM1</td>
</tr>
<tr>
<td>HT46R24/HT46C24</td>
<td>2 (28-pin package)</td>
<td>6+2 or 7+1</td>
<td>PD0/PD1</td>
<td>PWM0/PWM1</td>
</tr>
<tr>
<td>HT46R24/HT46C24</td>
<td>4 (48-pin package)</td>
<td>6+2 or 7+1</td>
<td>PD0/PD1/PD2/PD3</td>
<td>PWM0/PWM1/PWM2/PWM3</td>
</tr>
</tbody>
</table>

This method of dividing the original modulation cycle into a further 2 or 4 sub-cycles enables the generation of higher PWM frequencies, which allow a wider range of applications to be served. As long as the periods of the generated PWM pulses are less than the time constants of the load, the PWM output will be suitable as such long time constant loads will average out the pulses of the PWM output. The difference between what is known as the PWM cycle frequency and the PWM modulation frequency should be understood. As the PWM clock is the system clock, \( f_{SYS} \), and as the PWM value is 8-bits wide, the overall PWM cycle frequency is \( f_{SYS}/256 \). However, when in the 7+1 mode of operation the PWM modulation frequency will be \( f_{SYS}/128 \), while the PWM modulation frequency for the 6+2 mode of operation will be \( f_{SYS}/64 \).

<table>
<thead>
<tr>
<th>PWM Modulation Frequency</th>
<th>PWM Cycle Frequency</th>
<th>PWM Cycle Duty</th>
</tr>
</thead>
<tbody>
<tr>
<td>( f_{SYS}/64 ) for (6+2) bits mode</td>
<td>( f_{SYS}/256 )</td>
<td>(PWM)256</td>
</tr>
<tr>
<td>( f_{SYS}/128 ) for (7+1) bits mode</td>
<td>( f_{SYS}/256 )</td>
<td>(PWM)256</td>
</tr>
</tbody>
</table>

42
**6+2 PWM Mode**

Each full PWM cycle, as it is controlled by an 8-bit PWM register, has 256 clock periods. However, in the PWM 6+2 Mode, each PWM cycle is subdivided into four individual sub-cycles known as modulation cycle 0 ~ modulation cycle 3, denoted as "i" in the table. Each one of these four sub-cycles contains 64 clock cycles. In this mode, a modulation frequency increase by a factor of four is achieved. The 8-bit PWM register value, which represents the overall duty cycle of the PWM waveform, is divided into two groups. The first group which consists of bit2~bit7 is denoted here as the DC value. The second group which consists of bit0~bit1 is known as the AC value. In the 6+2 PWM mode, the duty cycle value of each of the four modulation sub-cycles is shown in the following table.

<table>
<thead>
<tr>
<th>Parameter</th>
<th>AC (0~3)</th>
<th>DC (Duty Cycle)</th>
</tr>
</thead>
<tbody>
<tr>
<td>Modulation cycle i (i=0~3)</td>
<td>$i &lt; AC$</td>
<td>DC + 1</td>
</tr>
<tr>
<td></td>
<td>$i \geq AC$</td>
<td>64</td>
</tr>
</tbody>
</table>

**6+2 Mode Modulation Cycle Values**

The following diagram illustrates the waveforms associated with the 6+2 mode of PWM operation. It is important to note how the single PWM cycle is subdivided into 4 individual modulation cycles, numbered from 0~3 and how the AC value is related to the PWM value.

**6+2 PWM Mode**

**PWM Register for 6+2 Mode**

AC value

DC value
7+1 PWM Mode

Each full PWM cycle, as it is controlled by an 8-bit PWM register has 256 clock periods. However, in the PWM 7+1 mode, each PWM cycle is subdivided into two individual sub-cycles, known as modulation cycle 0 and modulation cycle 1, denoted as "i" in the table. Each one of these two sub-cycles contains 128 clock cycles. In this mode, a modulation frequency increase by a factor of two is achieved. The 8-bit PWM register value, which represents the overall duty cycle of the PWM waveform, is divided into two groups. The first group which consists of bit1~bit7 is denoted here as the DC value. The second group which consists of bit0 is known as the AC value. In the 7+1 PWM mode, the duty cycle value of each of the two modulation sub-cycles is shown in the following table.

<table>
<thead>
<tr>
<th>Parameter</th>
<th>AC (0~1)</th>
<th>DC (Duty Cycle)</th>
</tr>
</thead>
<tbody>
<tr>
<td>Modulation cycle i (i=0~1)</td>
<td>i&lt;AC</td>
<td>DC + 1 128</td>
</tr>
<tr>
<td></td>
<td>i≥AC</td>
<td>DC 128</td>
</tr>
</tbody>
</table>

7+1 Mode Modulation Cycle Values

The following diagram illustrates the waveforms associated with the 7+1 mode of PWM operation. It is important to note how the single PWM cycle is subdivided into 2 individual modulation cycles, numbered 0 and 1 and how the AC value is related to the PWM value.

7+1 PWM Mode

PWM Register for 7+1 Mode

PWM Register — (7+1) Mode

AC value

DC value
PWM Output Control

On all devices, the PWM outputs are pin-shared with the Port D I/O pins. To operate as PWM outputs and not as I/O pins, the correct PWM configuration options must be selected. A “0” must also be written to the corresponding bits in the I/O port control register PDC to ensure that the required PWM output pins are setup as outputs. After these two initial steps have been carried out, and of course after the required PWM value has been written into the PWM register, writing a “1” to the corresponding bit in the PD output data register will enable the PWM data to appear on the pin. Writing a “0” to the corresponding bit in the PD output data register will disable the PWM output function and force the output low. In this way, the Port D data output register can be used as an on/off control for the PWM function. Note that if the configuration options have selected the PWM function, but a “1” has been written to its corresponding bit in the PDC control register to configure the pin as an input, then the pin can still function as a normal input line, with pull-high resistor options.

```
clr PDC.0 ; set pin PD0 as output
clr PDC.1 ; set pin PD1 as output
clr PDC.2 ; set pin PD2 as output
clr PDC.3 ; set pin PD3 as output
set pd.0 ; PD.0=1; enable pin "PD0/PWM0" to be the PWM channel 0
mov a,64h ; PWM0=100D=64H
mov pwm0,a
set pd.1 ; PD.1=1; enable pin "PD1/PWM1" to be the PWM channel 1
mov a,65h ; PWM1=101D=65H
mov pwm1,a
set pd.2 ; PD.2=1; enable pin "PD2/PWM2" to be the PWM channel 2
mov a,66h ; PWM2=102D=66H
mov pwm2,a
set pd.3 ; PD.3=1; enable pin "PD3/PWM3" to be the PWM channel 3
mov a,67h ; PWM3=103D=67H
mov pwm3,a
clr pd.0 ; disable PWM0 output – PD.0 will remain low
clr pd.1 ; disable PWM1 output – PD.1 will remain low
clr pd.2 ; disable PWM2 output – PD.2 will remain low
clr pd.3 ; disable PWM3 output – PD.3 will remain low
```
Analog to Digital Converter

The need to interface to real world analog signals is a common requirement for many electronic systems. However, to properly process these signals by a microcontroller, they must first be converted into digital signals by A/D converters. By integrating the A/D conversion electronic circuitry into the microcontroller, the need for external components is reduced significantly with the corresponding follow-on benefits of lower costs and reduced component space requirements. Each of the devices in the Holtek A/D series of microcontrollers contains either a 4-channel or 8-channel analog to digital converter which can directly interface to external analog signals such as that from sensors or other control signals and convert these signals directly into either a 9-bit or 10-bit digital value.

<table>
<thead>
<tr>
<th>Device</th>
<th>Input Channels</th>
<th>Conversion Bits</th>
<th>Input Pins</th>
</tr>
</thead>
<tbody>
<tr>
<td>HT46R47/HT46C47</td>
<td>4</td>
<td>9</td>
<td>PB0~PB3</td>
</tr>
<tr>
<td>HT46R22/HT46C22</td>
<td>8</td>
<td>9</td>
<td>PB0~PB7</td>
</tr>
<tr>
<td>HT46R23/HT46C23</td>
<td>8</td>
<td>10</td>
<td>PB0~PB7</td>
</tr>
<tr>
<td>HT46R24/HT46C24</td>
<td>8</td>
<td>10</td>
<td>PB0~PB7</td>
</tr>
</tbody>
</table>

A/D Converter Data Registers – ADRL/ADRH

To store the actual 9-bit or 10-bit digital value, obtained after the completion of the conversion process, a high byte register ADRH and low byte register ADRL are assigned. After the conversion process takes place, these two registers can be directly read by the microcontroller to obtain the digitized conversion value. Note that only the high byte register ADRH utilizes its full 8-bit contents. The low byte register utilizes only 1 or 2 bits of its 8-bit contents as it contains only the lowest one or two bits of the 9 or 10-bit converted value.

In the following tables, D0~D8 or D9 are the A/D conversion data result bits.

<table>
<thead>
<tr>
<th>Register</th>
<th>Bit7</th>
<th>Bit6</th>
<th>Bit5</th>
<th>Bit4</th>
<th>Bit3</th>
<th>Bit2</th>
<th>Bit1</th>
<th>Bit0</th>
</tr>
</thead>
<tbody>
<tr>
<td>ADRL</td>
<td>D0</td>
<td>—</td>
<td>—</td>
<td>—</td>
<td>—</td>
<td>—</td>
<td>—</td>
<td>—</td>
</tr>
<tr>
<td>ADRH</td>
<td>D8</td>
<td>D7</td>
<td>D6</td>
<td>D5</td>
<td>D4</td>
<td>D3</td>
<td>D2</td>
<td>D1</td>
</tr>
</tbody>
</table>

A/D Data Register – HT46R47/HT46C47 and HT46R22/HT46C22

<table>
<thead>
<tr>
<th>Register</th>
<th>Bit7</th>
<th>Bit6</th>
<th>Bit5</th>
<th>Bit4</th>
<th>Bit3</th>
<th>Bit2</th>
<th>Bit1</th>
<th>Bit0</th>
</tr>
</thead>
<tbody>
<tr>
<td>ADRL</td>
<td>D1</td>
<td>D0</td>
<td>—</td>
<td>—</td>
<td>—</td>
<td>—</td>
<td>—</td>
<td>—</td>
</tr>
<tr>
<td>ADRH</td>
<td>D9</td>
<td>D8</td>
<td>D7</td>
<td>D6</td>
<td>D5</td>
<td>D4</td>
<td>D3</td>
<td>D2</td>
</tr>
</tbody>
</table>

A/D Data Register – HT46R23/HT46C23 and HT46R24/HT46C24
A/D Converter Control Register – ADCR

To control the function and operation of the A/D converter, a control register known as ADCR is provided. This 8-bit register defines functions such as the selection of which analog channel is connected to the internal A/D converter, which pins are used as analog inputs and which are used as normal I/Os as well as controlling and monitoring the A/D converter start and reset functions.

One section of this register contains the bits ACS2~ACS0 which define the channel number. As each of the devices contains only one actual analog to digital converter circuit, each of the individual 4 or 8 analog inputs must be routed to the converter. It is the function of the ACS2~ACS0 bits in the ADCR register to determine which analog channel is actually connected to the internal A/D converter. For the HT46R22/HT46C22, HT46R23/HT46C23 and HT46R24/HT46C24 devices which have eight analog input channels, the full three bits are required for channel selection, however, for the HT46R47/HT46C47 devices, which have only four analog input channels, bit ACS2 is not used and should be kept at a “0” value. For the HT46R47/HT46C47 devices, if ACS2 is set to “1” the function of ACS2~ACS0 will be undefined.

The ADCR control register also contains the PCR2~PCR0 bits which determine which pins on Port B are used as analog inputs for the A/D converter and which pins are to be used as normal I/Os. For the HT46R22/HT46C22, HT46R23/HT46C23 and HT46R24/HT46C24 devices which have eight analog input channels, the full three bits are required to fully configure the function of the bits on Port B. However, for the HT46R47/HT46C47 devices, which have only four analog input channels, if the 3-bit address on PCR2~PCR0 has a value of “101” or higher, then the same function as the value “100” will apply, that is AN0, AN1, AN2 and AN3 will all be set as analog inputs. Note that if the PCR2~PCR0 bits are all set to zero, then all the Port B pins will be setup as normal I/Os and the internal A/D converter circuitry will be powered off to reduce the power consumption.

<table>
<thead>
<tr>
<th>b7</th>
<th>b6</th>
<th>b5</th>
<th>b4</th>
<th>b3</th>
<th>b2</th>
<th>b1</th>
<th>b0</th>
</tr>
</thead>
<tbody>
<tr>
<td>START</td>
<td>EOCB</td>
<td>PCR2</td>
<td>PCR1</td>
<td>PCR0</td>
<td>ACS2</td>
<td>ACS1</td>
<td>ACS0</td>
</tr>
</tbody>
</table>

A/D Converter Control Register (excluding HT46R47/HT46C47)

Select A/D channel

<table>
<thead>
<tr>
<th>ACS2</th>
<th>ACS1</th>
<th>ACS0</th>
<th>AN0</th>
<th>AN1</th>
<th>AN2</th>
<th>AN3</th>
</tr>
</thead>
<tbody>
<tr>
<td>0 0 0</td>
<td>: AN0</td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
</tr>
<tr>
<td>0 0 1</td>
<td>: AN1</td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
</tr>
<tr>
<td>0 1 0</td>
<td>: AN2</td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
</tr>
<tr>
<td>0 1 1</td>
<td>: AN3</td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
</tr>
<tr>
<td>1 0 0</td>
<td>: AN4</td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
</tr>
<tr>
<td>1 0 1</td>
<td>: AN5</td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
</tr>
<tr>
<td>1 1 0</td>
<td>: AN6</td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
</tr>
<tr>
<td>1 1 1</td>
<td>: AN7</td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
</tr>
</tbody>
</table>

Port B A/D channel configurations

<table>
<thead>
<tr>
<th>PCR2</th>
<th>PCR1</th>
<th>PCR0</th>
</tr>
</thead>
<tbody>
<tr>
<td>0 0 0</td>
<td>: Port B A/D channels - all off</td>
<td></td>
</tr>
<tr>
<td>0 0 1</td>
<td>: PB0 enabled as AN0</td>
<td></td>
</tr>
<tr>
<td>0 1 0</td>
<td>: PB0-PB1 enabled as AN0-AN1</td>
<td></td>
</tr>
<tr>
<td>0 1 1</td>
<td>: PB0-PB2 enabled as AN0-AN2</td>
<td></td>
</tr>
<tr>
<td>1 0 0</td>
<td>: PB0-PB3 enabled as AN0-AN3</td>
<td></td>
</tr>
<tr>
<td>1 0 1</td>
<td>: PB0-PB4 enabled as AN0-AN4</td>
<td></td>
</tr>
<tr>
<td>1 1 0</td>
<td>: PB0-PB5 enabled as AN0-AN5</td>
<td></td>
</tr>
<tr>
<td>1 1 1</td>
<td>: PB0-PB7 enabled as AN0-AN7</td>
<td></td>
</tr>
</tbody>
</table>

Start the A/D conversion

0 → 1 → 0 : Start
0 → 1 : Reset A/D converter and set EDCB to "1"
The START bit in the ADCR register is used to start and reset the A/D converter. When the microcontroller sets this bit from low to high and then low again, an analog to digital conversion cycle will be initiated. When the START bit is brought from low to high but not low again, the EOCB bit in the ADCR register will be set to a "1" and the analog to digital converter will be reset. It is the START bit that is used to control the overall on/off operation of the internal analog to digital converter.

The EOCB bit in the ADCR register is used to indicate when the analog to digital conversion process is complete. This bit will be automatically set to "0" by the microcontroller after a conversion cycle has ended. In addition, the corresponding A/D interrupt request flag will be set in the interrupt control register, and if the interrupts are enabled, an appropriate internal interrupt signal will be generated. This A/D internal interrupt signal will direct the program flow to the associated A/D internal interrupt address for processing. If the A/D internal interrupt is disabled, the microcontroller can be used to poll the EOCB bit in the ADCR register to check whether it has been cleared as an alternative method of detecting the end of an A/D conversion cycle.

A/D Converter Clock Source Register – ACSR
The clock source for the A/D converter, which originates from the system clock f SYS, is first divided by a division ratio, the value of which is determined by the ADCS1 and ADCS0 bits in the ACSR register.
Although the A/D clock source is determined by the system clock \( f_{sys} \), and by bits ADCS1 and ADCS0, there are some limitations on the maximum A/D clock source speed that can be selected. As the minimum value of permissible A/D clock period \( t_{AD} \) is \( 1/20 \) s, for system clock speeds in excess of 2MHz, the ADCS1 and ADCS0 bits should not be set to “00”. Doing so will give A/D clock periods that are less than \( 1/20 \) s which may result in inaccurate A/D conversion values. Refer to the following table for examples, where values marked with an asterisk * are not permissible as they are less than the specified minimum A/D Clock Period.

<table>
<thead>
<tr>
<th>( f_{sys} )</th>
<th>( f_{AD} )</th>
<th>ADCS1, ADCS0=00</th>
<th>ADCS1, ADCS0=01</th>
<th>ADCS1, ADCS0=10</th>
<th>ADCS1, ADCS0=11</th>
</tr>
</thead>
<tbody>
<tr>
<td>1MHz</td>
<td>2( \mu )s</td>
<td>8( \mu )s</td>
<td>32( \mu )s</td>
<td>Undefined</td>
<td></td>
</tr>
<tr>
<td>2MHz</td>
<td>1( \mu )s</td>
<td>4( \mu )s</td>
<td>16( \mu )s</td>
<td>Undefined</td>
<td></td>
</tr>
<tr>
<td>4MHz</td>
<td>500ns*</td>
<td>2( \mu )s</td>
<td>8( \mu )s</td>
<td>Undefined</td>
<td></td>
</tr>
<tr>
<td>8MHz</td>
<td>250ns*</td>
<td>1( \mu )s</td>
<td>4( \mu )s</td>
<td>Undefined</td>
<td></td>
</tr>
</tbody>
</table>

A/D Clock Period Examples

A/D Input Pins

All of the A/D analog input pins are pin-shared with the I/O pins on Port B. The PCR2~PCR0 bits in the ADCR register, not configuration options, determine whether the input pins are setup as normal Port B input/output pins or whether they are setup as analog inputs. In this way, pins can be changed under program control to change their function from normal I/O operation to analog inputs and vice versa. Pull-high resistors, which are setup through configuration options, apply to the input pins only when they are used as normal I/O pins, if setup as A/D inputs the pull-high resistors will be automatically disconnected. Note that it is not necessary to first setup the A/D pin as an input in the PBC port control register to enable the A/D input, when the PCR2~PCR0 bits enable an A/D input, the status of the port control register will be overridden. The VDD power supply pin is used as the A/D converter reference voltage, and as such analog inputs must not be allowed to exceed this value. Appropriate measures should also be taken to ensure that the VDD pin remains as stable and noise free as possible.

Summary of A/D Conversion Steps

The following summarizes the individual steps that should be executed in order to implement an A/D conversion process.

- **Step 1**
  Select which pins on Port B are to be used as A/D inputs and configure them as A/D input pins by correctly programming the PCR2~PCR0 bits in the ADCR register.

- **Step 2**
  Select which channel is to be connected to the internal A/D converter by correctly programming the ACS2~ACS0 bits which are also contained in the ADCR register.

- **Step 3**
  Select the required A/D conversion clock by correctly programming bits ADCS1 and ADCS0 in the ACSR register.
Step 4
If the interrupts are to be used, the interrupt control registers must be correctly configured to ensure the A/D function is active. Depending upon which device is used, the master interrupt control bit, EMI, in either the INTC or INTC0 interrupt control register must be set to “1” and the A/D converter interrupt bit, EADI, in either the INTC, INTC0 or INTC1 register must also be set to “1”.

Step 5
The analog to digital conversion process can now be initialized by setting the START bit in the ADCR register to from “0” to “1” and then to “0” again. Note that this bit should have been originally set to “0”.

Step 6
To check when the analog to digital conversion process is complete, the EOCB bit in the ADCR register can be polled. The conversion process is complete when this bit goes low. When this occurs the A/D data registers ADRL and ADRH can be read to obtain the conversion value. As an alternative method if the interrupts are enabled and the stack is not full, the program can wait for an A/D interrupt to occur.

Note
When checking for the end of the conversion process, if the method of polling the EOCB bit in the ADCR register is used, step 4 above can be omitted.

The following timing diagram shows graphically the various stages involved in an analog to digital conversion process and its associated timing.

A/D Conversion Timing

The setting up and operation of the A/D converter function is fully under the control of the application program as there are no configuration options associated with the A/D converter. After an A/D conversion process has been initiated by the application program, the microcontroller internal hardware will begin to carry out the conversion, during which time the program can continue with other functions. There are two methods to determine when the A/D conversion process is com-
The first is for the application program to poll the EOCB bit in the ADCR register, while the second method is to await an A/D internal interrupt to occur. The following two short program examples illustrate both of these methods. Note that the program is based on the HT46R22/HT46C22 devices.

Example: using EOCB Polling Method to detect end of conversion

```assembly
; disable A/D interrupt in interrupt control register
clr INTC0.3
mov a,00100000B
mov ADCR,a ; setup ADCR register to configure Port PB0~PB3 as A/D inputs and select AN0 to be connected to the A/D converter
mov a,00000001B
mov ACSR,a ; setup the ACSR register to select fSYS/8 as the A/D clock

Start_conversion:
clr ADCR.7 ; reset A/D
set ADCR.7 ; start A/D

Polling_EOC:
sz ADCR.6 ; poll the ADCR register EOCB bit to detect end of A/D conversion
jmp polling_EOC ; continue polling
mov a,ADRH ; read conversion result from the high byte ADRH register
mov adrh_buffer,a ; save result to user defined register
mov a,ADRL ; read conversion result from the low byte ADRL register
mov adrl_buffer,a ; save result to user defined register
:
jmp start_conversion ; start next A/D conversion
```

Example: using Interrupt method to detect end of conversion

```assembly
set INTC0.0 ; interrupt global enable
set INTC0.3 ; enable A/D interrupt in interrupt control register
mov a,00100000B
mov ADCR,a ; setup ADCR register to configure Port PB0~PB3 as A/D inputs and select AN0 to be connected to the A/D converter
mov a,00000001B
mov ACSR,a ; setup the ACSR register to select fSYS/8 as the A/D clock

start_conversion:
clr ADCR.7 ; reset A/D
set ADCR.7 ; start A/D

; interrupt service routine
EOC_service routine:
mov a_buffer,a ; save ACC to user defined register
mov a,ADRH ; read conversion result from the high byte ADRH register
```

51
A/D Transfer Function

As the HT46R47/HT46C47 and HT46R22/HT46C22 devices each contain a 9-bit A/D converter, their full-scale converted digitized value is equal to 1FFH. Since the full-scale analog input value is equal to the VDD voltage, this gives a single bit analog input value of VDD/512. In the case of the HT46R23/HT46C23 and HT46R24/HT46C24 devices, which each contain a 10-bit A/D converter, their full-scale converted digitized value is equal to 3FFH, giving a single bit analog input value of VDD/1024. The following graphs show the ideal transfer function between the analog input value and the digitized output value for both the 9-bit and 10-bit A/D converters.
Note that to reduce the quantization error, a 0.5 LSB offset is added to the A/D Converter input. Except for the digitized value 0, the subsequent digitized values will change at a point 0.5 LSB below where they would change without the offset, and the last full scale digitized value will change at a point 1.5 LSB below the VDD.

The A/D Converter has a maximum of ±1 LSB Integral Non-Linearity Error which describes the departure from the ideal linear transfer function. For the HT46R47/HT46C47 and HT46R22/HT46C22, their 9-bit resolution A/D Converter is achieved with 8-bit accuracy while for the HT46R23/HT46C23 and HT46R24/HT46C24, their 10-bit resolution A/D Converter is achieved with 9-bit accuracy.

**I^2C Bus Serial Interface**

The I^2C bus is a bidirectional 2-wire communication interface originally developed by Philips Semiconductors. The possibility of transmitting and receiving data on only 2 lines offers many new application possibilities for microcontroller based applications and for this reason, with the exception of the HT46R47/HT46C47 devices, an I^2C bus is implemented in each of the microcontrollers in the Holtek A/D MCU range. The I^2C bus function is selectable via a configuration option.
There are two lines associated with the I²C bus, the first is known as SDA and is the Serial Data line, the second is known as SCL line and is the Serial Clock line. As many devices may be connected together on the same bus, their outputs are both open drain types. For this reason it is necessary that external pull-high resistors are connected to these outputs. Note that no chip select line exists, as each device on the I²C bus is identified by a unique address which will be transmitted and received on the I²C bus.

When two devices communicate with each other on the bidirectional I²C bus, one is known as the master device and one as the slave device. Both master and slave can transmit and receive data, however, it is the master device that has overall control of the bus. For the Holtek microcontrollers, which only operate in slave mode, there are two methods of transferring data on the I²C bus, the slave transmit mode and the slave receive mode. Four registers exist to control the I²C bus and its associated data transfer, HADR, HCR, HSR and HDR. Communication on the I²C bus requires four steps, a START signal, a slave address transmission, a data transmission and finally a STOP signal.

### I²C Bus Slave Address Register – HADR

The HADR register is the location where the slave address of the microcontroller is stored. Bits 1~7 of the HADR register define the microcontroller slave address. Bit 0 is not implemented. When a master device, which is connected to the I²C bus, sends out an address which matches the slave address in the HADR register, the microcontroller slave device will be selected.

![HADR Register](image)

Not implemented, read as "0"

Slave address

### I²C Bus Input/Output Data Register – HDR

The HDR register is the I²C bus input/output data register. Before the microcontroller writes data to the I²C bus, the actual data to be transmitted must be placed in the HDR register. After the data is received from the I²C bus, the microcontroller can read it from the HDR register. Any transmission of data to the I²C bus or reception of data from the I²C bus must be made via the HDR register.

### I²C Bus Control Register – HCR

The I²C bus control register HCR contains three bits. Bit 7, known as the HEN bit, determines if the I²C bus function is enabled or disabled, this bit must be set if the I²C bus requires data transfer. Bit 4, known as the HTX bit, determines whether the device is in the transmit mode or receive mode, and must be set high if the device is to be setup as a transmitter. Bit 3, known as the TXAK bit, is the transmit acknowledge bit. After the receipt of 8 bits of data, this bit will be transmitted to the I²C bus on the 9th clock. To continue receiving more data, this bit has to be reset to “0” before more data is received.
The I²C bus register HSR is an 8-bit status register in which five bits are utilized. Bit 7, known as HCF, is set to "0" when a data byte is being transferred, after completion of the data transfer the bit will be set to "1". The HAAS bit, which is bit 6, will be set to "1" if the transmitted address and the slave address of the device match and if the I²C interrupt request flag is set to "1". If the interrupts are enabled and the stack is not full, a subroutine call to 10H will occur. Writing data to the I²C bus will clear the HAAS bit. Also, if the transmitted address on the bus and the slave address of the device do not match, then the HAAS bit will be reset to "0".

Bit 5, known as HBB, will be set to "1" if the I²C bus is busy, which will occur when a START signal is detected. The HBB bit will be cleared to "0" if the bus is free which will occur when a STOP signal is detected. Bit 2, which is the SRW or Slave Read/Write bit, determines whether the master device wishes to transmit or receive data from the I²C bus. When the transmitted address and slave address match, that is when the HAAS bit is set to "1", the device will check the SRW bit to determine whether it should be in transmit mode or receive mode. If the SRW bit is equal to "1" the master is requesting to read data from the bus, so the device should be in transmit mode. When the SRW bit is equal to "0", the master will write data to the bus, therefore the device should be in receive mode to read this data.
Bit 0, is the Receive Acknowledge bit and known as RXAK. When the RXAK bit has been reset to "0" it means that a correct acknowledge signal has been received at the 9th clock, after 8 bits of data have been transmitted. When in the transmit mode, the transmitter checks the RXAK bit to determine if the receiver wishes to receive the next byte. The transmitter will therefore continue sending out data until the RXAK bit is set to "1". When this occurs, the transmitter will release the SDA line to allow the master to send a STOP signal to release the bus.

**I²C Bus Communication**

Communication on the I²C bus requires four separate steps, a START signal, a slave device address transmission, a data transmission and finally a STOP signal. When a START signal is placed on the I²C bus, all devices on the bus will receive this signal and be notified of the imminent arrival of data on the bus. The first seven bits of the data will be the slave address with the first bit being the MSB. If the address of the microcontroller matches that of the transmitted address, the HAAS bit in the HSR register will be set and an I²C interrupt will be generated. After entering the interrupt service routine, the microcontroller slave device must first check the condition of the HAAS bit to determine whether the interrupt source originates from an address match or from the completion of an 8-bit data transfer. During a data transfer, note that after the 7-bit slave address has been transmitted, the following bit, which is the 8th bit, is the read/write bit whose value will be placed in the SRW bit. This bit will be checked by the microcontroller to determine whether to go into transmit or receive mode. Before any transfer of data to or from the I²C bus, the microcontroller must initialize the bus, the following are steps to achieve this:

- **Step 1**
  Write the slave address of the microcontroller to the I²C bus address register HADR.
- **Step 2**
  Set the HEN bit in the I²C bus control register to "1" to enable the I²C bus.
- **Step 3**
  Set the EHI bit of the interrupt control register to enable the I²C bus interrupt.


→ **Start Signal**

The START signal can only be generated by the master device connected to the I²C bus and not by the microcontroller, which is only a slave device. This START signal will be detected by all devices connected to the I²C bus. When detected, this indicates that the I²C bus is busy and therefore the HBB bit will be set. A START condition occurs when a high to low transition on the SDA line takes place when the SCL line remains high.

→ **Slave Address**

The transmission of a START signal by the master will be detected by all devices on the I²C bus. To determine which slave device the master wishes to communicate with, the address of the slave device will be sent out immediately following the START signal. All slave devices, after receiving this 7-bit address data, will compare it with their own 7-bit slave address. If the address sent out by the master matches the internal address of the microcontroller slave device, then an internal I²C bus interrupt signal will be generated. The next bit following the address, which is the 8th bit, defines the read/write status and will be saved to the SRW bit of the HSR register. The device will then transmit an acknowledge bit, which is a low level, as the 9th bit. The microcontroller slave device will also set the status flag HAAS when the addresses match.

As an I²C bus interrupt can come from two sources, when the program enters the interrupt subroutine, the HAAS bit should be examined to see whether the interrupt source has come from a matching slave address or from the completion of a data byte transfer. When a slave address is matched, the device must be placed in either the transmit mode and then write data to the HDR register, or in the receive mode where it must implement a dummy read from the HDR register to release the SCL line.
→ **SRW Bit**

The SRW bit in the HSR register defines whether the microcontroller slave device wishes to read data from the I^2^C bus or write data to the I^2^C bus. The microcontroller should examine this bit to determine if it is to be a transmitter or a receiver. If the SRW bit is set to “1” then this indicates that the master wishes to read data from the I^2^C bus, therefore the microcontroller slave device must be setup to send data to the I^2^C bus as a transmitter. If the SRW bit is “0” then this indicates that the master wishes to send data to the I^2^C bus, therefore the microcontroller slave device must be setup to read data from the I^2^C bus as a receiver.

→ **Acknowledge Bit**

After the master has transmitted a calling address, any slave device on the I^2^C bus, whose own internal address matches the calling address, must generate an acknowledge signal. This acknowledge signal will inform the master that a slave device has accepted its calling address. If no acknowledge signal is received by the master then a STOP signal must be transmitted by the master to end the communication. When the HAAS bit is high, the addresses have matched and the microcontroller slave device must check the SRW bit to determine if it is to be a transmitter or a receiver. If the SRW bit is high, the microcontroller slave device should be setup to be a transmitter so the HTX bit in the HCR register should be set to “1”, if the SRW bit is low then the microcontroller slave device should be setup as a receiver and the HTX bit in the HCR register should be set to “0”.

→ **Data Byte**

The transmitted data is 8-bits wide and is transmitted after the slave device has acknowledged receipt of its slave address. The order of serial bit transmission is the MSB first and the LSB last. After receipt of 8-bits of data, the receiver must transmit an acknowledge signal, level “0”, before it can receive the next data byte. If the transmitter does not receive an acknowledge bit signal from the receiver, then it will release the SDA line and the master will send out a STOP signal to release control of the I^2^C bus. The corresponding data will be stored in the HDR register. If setup as a transmitter, the microcontroller slave device must first write the data to be transmitted into the HDR register. If setup as a receiver, the microcontroller slave device must read the transmitted data from the HDR register.

![Data Timing Diagram](image-url)
Chapter 1  Hardware Structure

I²C Bus Initialization Flow Chart

I²C Bus ISR Flow Chart
Interrupts

The A/D series of microcontrollers each contain a range of both external and internal interrupt functions. The external interrupt is controlled by the action of the external pin INT which is present on all devices. Internal functions such as the timer counter, A/D converter and I2C interface all utilize the internal interrupt function for their operation.

For the HT46R47/HT46C47 devices, which do not contain an internal I2C interface and contain only a single timer, one 8-bit interrupt control register, known as INTC, is sufficient to control all the required operations. As the HT46R22/HT46C22, HT46R23/HT46C23 and HT46R24/HT46C24 devices both contain an I2C interface, a single 8-bit interrupt control register is insufficient to control all the interrupt control features. For this reason two interrupt control registers are provided, known as INTC0 and INTC1.

Once an interrupt subroutine is serviced, all the other interrupts will be blocked, by clearing the EMI bit. This scheme may prevent any further interrupt nesting. Other interrupt requests may occur during this interval but only the interrupt request flag is recorded. If another interrupt requires servicing while the program is in the interrupt service routine, the EMI bit should be set after entering the routine, to allow interrupt nesting. If the stack is full, the interrupt request will not be acknowledged, even if the related interrupt is enabled, until the SP is decremented. If immediate service is desired, the stack must be prevented from becoming full.

---

**INTC Register — HT46R47/HT46C47**

**INTC0 Register — HT46R22/HT46C22**

**HT46R23/HT46C23**

<table>
<thead>
<tr>
<th>b7</th>
<th>ADF</th>
<th>TF</th>
<th>EIF</th>
<th>EAD</th>
<th>ETI</th>
<th>EEI</th>
<th>EMI</th>
</tr>
</thead>
<tbody>
<tr>
<td></td>
<td>1</td>
<td></td>
<td>1</td>
<td></td>
<td></td>
<td></td>
<td></td>
</tr>
<tr>
<td>0</td>
<td>0</td>
<td>0</td>
<td>0</td>
<td>1</td>
<td></td>
<td></td>
<td></td>
</tr>
</tbody>
</table>

Master Interrupts global enable
1: global enable
0: global disable

External interrupt enable
1: enable
0: disable

Timer/Event Counter interrupt enable
1: enable
0: disable

A/D Converter interrupt enable
1: enable
0: disable

External interrupt request flag
1: active
0: inactive

Timer/Event Counter interrupt request flag
1: active
0: inactive

A/D converter request flag
1: active
0: inactive

Not implemented, read as "0"
Differing from the other devices in the A/D series, the HT46R24/HT46C24 devices contain two internal timer counters. Although all the interrupt control functions can still be controlled by two interrupt control registers, also known as INTC0 and INTC1, they have a slightly different structure from the other devices.

**INTC1 Register — HT46R22/HT46C22 HT46R23/HT46C23**

- **b7**
  - PC Bus interrupt enable
    - 1: enable
    - 0: disable
  - Not implemented, read as "0"

- **b0**
  - PC Bus interrupt request flag
    - 1: active
    - 0: inactive
  - Not implemented, read as "0"

**INTC0 Register — HT46R24/HT46C24**

- **b7**
  - Master interrupts global enable
    - 1: global enable
    - 0: global disable

- **b6**
  - External interrupt enable
    - 1: enable
    - 0: disable

- **b5**
  - Timer/Event Counter 0 interrupt enable
    - 1: enable
    - 0: disable

- **b4**
  - Timer/Event Counter 1 interrupt enable
    - 1: enable
    - 0: disable

- **b3**
  - External interrupt request flag
    - 1: active
    - 0: inactive

- **b2**
  - Timer/Event Counter 0 request flag
    - 1: active
    - 0: inactive

- **b1**
  - Timer/Event Counter 1 request flag
    - 1: active
    - 0: inactive

- **b0**
  - Not implemented, read as "0"
The external interrupt has the capability of waking up the processor when in the HALT mode. As an interrupt is serviced, a control transfer occurs by pushing the Program Counter onto the stack, followed by a branch to a subroutine at a specified location in the Program Memory. Only the Program Counter is pushed onto the stack. If the contents of the accumulator, status register or other registers are altered by the interrupt service routine, which may corrupt the desired control sequence, then the contents should be saved in advance.

The various interrupt enable bits, together with their associated request flags, are shown in the following diagram with their order of priority.

Note: In the figure, the T1F interrupt request flag and the ET1I interrupt enable bit refer to the HT46R24/HT46C24 devices, which have two timers. For the HT46R47/HT46C47, HT46R22/HT46C22 and HT46R23/HT46C23, which only have one timer, the Timer/Event Counter 0 represents the single timer, known as TMR and has interrupt request flag known as TF and enable bit known as ETI.
External Interrupt

For an external interrupt to occur, the corresponding external interrupt enable bit must be first set. This is bit 1 of the INTC or INTC0 register and shown as EEI. External interrupts are triggered by a high to low transition of the INT line, after which the related interrupt request flag (EIF; bit 4 of the INTC or INTC0) will be set. When the interrupt is enabled, the stack is not full and the external interrupt is active, a subroutine call to location 04H will occur. The interrupt request flag EIF will be reset and EMI bits will be cleared to disable other interrupts.

Timer/Event Counter Interrupt

For a timer generated internal interrupt to occur, the corresponding internal interrupt enable bit must be first set. In the case of devices with a single timer, this is bit 2 of the INTC or INTC0 register and is known as ETI. For devices with two timers, the Timer 0 interrupt enable is bit 2 and known as ET0I while the Timer 1 interrupt enable is bit 3 and known as ET1I. An actual Timer/Event Counter interrupt will be initialized when the Timer/Event Counter interrupt request flag is set, caused by a timer overflow. In the case of devices with a single timer, this is bit 5 of the INTC or INTC0 register and is known as TF. In the case of devices with two timers, the Timer 0 request flag is bit 5 and known as T0F, while the Timer 1 request flag is bit 6 and known as T1F. When the master interrupt global enable bit is set, the stack is not full and the corresponding internal interrupt enable bit is set, an internal interrupt will be generated when the timer overflows. This will create a subroutine call to location 08H for devices with a single timer. For devices with two timers, a subroutine call to location 08H will occur for Timer 0 and a subroutine call to location 0CH for Timer 1. When an internal interrupt occurs, the interrupt request flag, TF, T0F or T1F will be reset and the EMI bit will be cleared to disable other interrupts.

A/D Interrupt

For an A/D interrupt to occur, the corresponding interrupt enable bit EADI must be first set. For the HT46R47/HT46C47 devices, this is bit 3 of the INTC register, while for the HT46R22/HT46C22 and HT46R23/HT46C23 devices, this is bit 3 of the INTC0 register. For the HT46R24/HT46C24 devices, this is bit 0 of the INTC1 register. An actual A/D interrupt will be initialized when the A/D converter request flag ADF is set, a situation that will occur when an A/D conversion process has completed. In the case of the HT46R47/HT46C47 devices, this is bit 6 of the INTC register, while for the HT46R22/HT46C22 and HT46R23/HT46C23 devices, this is bit 6 of the INTC0 register. For the HT46R24/HT46C24 devices, this is bit 4 of the INTC1 register. When the master interrupt global enable bit is set, the stack is not full and the corresponding A/D interrupt enable bit is set, an internal interrupt will be generated when the previously requested A/D conversion process finishes. With the exception of the HT46R24/HT46C24 devices, this will create a subroutine call to location 0CH. For the HT46R24/HT46C24 devices, a subroutine call to location 10H will be created. When an A/D interrupt occurs, the interrupt request flag ADF will be reset and the EMI bit will be cleared to disable other interrupts.

I²C Interrupt

For an I²C interrupt to occur, the corresponding interrupt enable bit EHI must be first set. For the HT46R22/HT46C22 and HT46R23/HT46C23 devices, this is bit 0 of the INTC1 register, while for the HT46R24/HT46C24 devices, this is bit 1 of the INTC1 register. An actual I²C interrupt will be initialized when the I²C interrupt request flag HIF is set, a situation that will occur when a matching
I²C slave address is received or from the completion of an I²C data byte transfer. In the case of the HT46R22/HT46C22 and HT46R23/HT46C23 devices, this is bit 4 of the INTC1 register, while for the HT46R24/HT46C24 devices, this is bit 5 of the INTC1 register. Note that as the HT46R47/HT46C47 devices do not contain an I²C interface, their interrupt control register INTC has no associated I²C enable bit or request flag. When the master interrupt global enable bit is set, the stack is not full and the corresponding I²C interrupt enable bit is set, an internal interrupt will be generated when a matching I²C slave address is received or from the completion of an I²C data byte transfer. For the HT46R22/HT46C22 and HT46R23 HT46C23 devices, this will create a subroutine call to location 10H, while for the HT46R24/HT46C24 devices, a subroutine call to location 14H will be created. When an I²C interrupt occurs, the interrupt request flag HIF will be reset and the EMI bit will be cleared to disable other interrupts.

Interrupt Priority

Interrupts, occurring in the interval between the rising edges of two consecutive T2 pulses, will be serviced on the latter of the two T2 pulses, if the corresponding interrupts are enabled. In case of simultaneous requests, the following table shows the priority that is applied. These can be masked by resetting the EMI bit.

<table>
<thead>
<tr>
<th>Interrupt Source</th>
<th>HT46R47 Priority</th>
<th>HT46C47 Priority</th>
<th>HT46R22 Priority</th>
<th>HT46C22 Priority</th>
<th>HT46R23 Priority</th>
<th>HT46C23 Priority</th>
<th>HT46R24 Priority</th>
<th>HT46C24 Priority</th>
</tr>
</thead>
<tbody>
<tr>
<td>External Interrupt</td>
<td>1</td>
<td>1</td>
<td>1</td>
<td>1</td>
<td></td>
<td></td>
<td></td>
<td></td>
</tr>
<tr>
<td>TMR/TMR0 Overflow</td>
<td>2</td>
<td>2</td>
<td>2</td>
<td>2</td>
<td></td>
<td></td>
<td></td>
<td></td>
</tr>
<tr>
<td>TMR1 Overflow</td>
<td>N/A</td>
<td>N/A</td>
<td>N/A</td>
<td>3</td>
<td></td>
<td></td>
<td></td>
<td></td>
</tr>
<tr>
<td>A/D Converter Interrupt</td>
<td>3</td>
<td>3</td>
<td>3</td>
<td>4</td>
<td></td>
<td></td>
<td></td>
<td></td>
</tr>
<tr>
<td>I²C Bus Interrupt</td>
<td>N/A</td>
<td>4</td>
<td>4</td>
<td>5</td>
<td></td>
<td></td>
<td></td>
<td></td>
</tr>
</tbody>
</table>

Note

Only the HT46R24/HT46C24 devices have two internal timers, known as TMR0 and TMR1. The other devices in the series have only one internal timer, which is known as TMR.

In cases where both external and internal interrupts are enabled and where an external and internal interrupt occurs simultaneously, the external interrupt will always have priority and will therefore be serviced first. Suitable masking of the individual interrupts using the INTC register can prevent simultaneous occurrences.

Programming Considerations

The interrupt request flags, TF, T0F, T1F, EIF, ADF and HIF together with the interrupt enable bits ET1, ET0I, ET1I, EEI, EADI and EHI form the interrupt control registers INTC, INTC0 and INTC1, which are located in the Data Memory. By disabling the interrupt enable bits, a requested interrupt can be prevented from being serviced, however, once an interrupt request flag is set, it will remain in this condition in the INTC, INTC0 or INTC1 register until the corresponding interrupt is serviced or until the request flag is cleared by a software instruction.

It is recommended that programs do not use the "CALL subroutine" instruction within the interrupt subroutine. Interrupts often occur in an unpredictable manner or need to be serviced immediately.
in some applications. If only one stack is left and the interrupt is not well controlled, the original control sequence will be damaged once a “CALL subroutine” is executed in the interrupt subroutine.

**Reset and Initialization**

A reset function is a fundamental part of any microcontroller ensuring that the device can be set to some predetermined condition irrespective of outside parameters. The most important reset condition is after power is first applied to the microcontroller. In this case, internal circuitry will ensure that the microcontroller, after a short delay, will be in a well defined state and ready to execute the first program instruction. After this power-on reset, certain important internal registers will be set to defined states before the program commences. One of these registers is the Program Counter, which will be reset to zero forcing the microcontroller to begin program execution from the lowest Program Memory address.

In addition to the power-on reset, situations may arise where it is necessary to forcefully apply a reset condition when the microcontroller is running. One example of this is where after power has been applied and the microcontroller is already running, the RES line is forcefully pulled low. In such a case, known as a normal operation reset, some of the microcontroller registers remain unchanged allowing the microcontroller to proceed with normal operation after the reset line is allowed to return high. Another type of reset is when the Watchdog Timer overflows and resets the microcontroller. All types of reset operations result in different register conditions being setup.

Another reset exists in the form of a Low Voltage or LVR, where a full reset, similar to the RES reset is implemented in situations where the power supply voltage falls below a certain threshold.

**Reset**

There are five ways in which a microcontroller reset can occur, through events occurring both internally and externally:

→ **Power-on Reset**

  The most fundamental and unavoidable reset is the one that occurs after power is first applied to the microcontroller. As well as ensuring that the Program Memory begins execution from the first memory address, a power-on reset also ensures that certain other registers are preset to known conditions. All the I/O port and port control registers will power up in a high condition ensuring that all pins will be first set to inputs.

  Although the microcontroller has an internal RC reset function, due to unstable power on conditions, an external RC network connected to the RES pin is generally recommended. This time delay created by the RC network ensures that the RES pin remains low for an extended period while the power supply stabilizes. During this time, normal operation of the microcontroller is inhibited. After the RES line reaches a certain voltage value, the reset delay time \( t_{\text{STD}} \) is invoked to provide an extra delay time after which the microcontroller can begin normal operation. The abbreviation SST in the figures stands for System Start-up Timer.
RES Pin Reset

This type of reset occurs when the microcontroller is already running and the RES pin is forcefully pulled low by external hardware such as an external switch. In this case as in the case of other resets, the Program Counter will reset to zero and program execution initiated from this point.

Low Voltage Reset – LVR

The microcontroller contains a low voltage reset circuit in order to monitor the supply voltage of the device. If the supply voltage of the device drops to within a range of 0.9V~V_LVR such as might occur when changing the battery, the LVR will automatically reset the device internally. The LVR includes the following specifications: For a valid LVR signal, a low voltage, i.e. a voltage in the range between 0.9V~V_LVR must exist for greater than 1ms. If the low voltage state does not exceed 1ms, the LVR will ignore it and will not perform a reset function.
Watchdog Time-out Reset during Normal Operation

The Watchdog Time-out Reset during normal operation is the same as RES reset except that the Watchdog Time-out flag TO will be set to 1.

![WDT Time-out Reset during Normal Operation Timing Chart]

Watchdog Time-out Reset during HALT

The Watchdog Time-out Reset during HALT is a little different from other kinds of reset. Most of the conditions remain unchanged except that the Program Counter and the Stack Pointer will be cleared to 0 and the TO flag will be set to 1. Refer to the A.C. Characteristics for tSST details.

![WDT Time-out Reset during HALT Timing Chart]

The different types of resets described affect the reset flags in different ways. These flags known as PDF and TO are located in the status register and are controlled by various microcontroller operations such as the HALT function or Watchdog Timer. The reset flags are shown in the table:

<table>
<thead>
<tr>
<th>TO</th>
<th>PDF</th>
<th>RESET Conditions</th>
</tr>
</thead>
<tbody>
<tr>
<td>0</td>
<td>0</td>
<td>RES reset during power on</td>
</tr>
<tr>
<td>u</td>
<td>u</td>
<td>RES or LVR reset during normal operation</td>
</tr>
<tr>
<td>1</td>
<td>u</td>
<td>WDT time-out reset during normal operation</td>
</tr>
<tr>
<td>1</td>
<td>1</td>
<td>WDT time-out reset during HALT</td>
</tr>
</tbody>
</table>

"u" stands for unchanged

The following table indicates the way in which the various components of the microcontroller are affected after a reset occurs.

<table>
<thead>
<tr>
<th>Item</th>
<th>Condition After RESET</th>
</tr>
</thead>
<tbody>
<tr>
<td>Program Counter</td>
<td>Reset to zero</td>
</tr>
<tr>
<td>Interrupts</td>
<td>All interrupts will be disabled</td>
</tr>
<tr>
<td>WDT</td>
<td>Clear after reset, WDT begins counting</td>
</tr>
<tr>
<td>Timer/Event Counter</td>
<td>All Timer Counters will be turned off</td>
</tr>
<tr>
<td>Prescaler</td>
<td>The Timer Counter Prescaler will be cleared</td>
</tr>
<tr>
<td>Input/Output Ports</td>
<td>All I/O ports will be setup as inputs</td>
</tr>
<tr>
<td>Stack Pointer</td>
<td>Stack pointer will point to the top of the stack</td>
</tr>
</tbody>
</table>
The different kinds of reset all affect the internal registers of the microcontroller in different ways. To ensure reliable continuation of normal program execution after a reset occurs, it is important to know what condition the microcontroller is in after a particular reset occurs. The following table describes how each type of reset affects each of the microcontroller internal registers.

<table>
<thead>
<tr>
<th>Register</th>
<th>Reset (Power On)</th>
<th>RES or LVR Reset</th>
<th>WDT Time-out (Normal Operation)</th>
<th>WDT Time-out (HALT)</th>
</tr>
</thead>
<tbody>
<tr>
<td>MP</td>
<td>- xxx xxxx</td>
<td>- uuu uuuu</td>
<td>- uuu uuuu</td>
<td>- uuu uuuu</td>
</tr>
<tr>
<td>ACC</td>
<td>xxxxxxx</td>
<td>uuuu uuuu</td>
<td>uuuu uuuu</td>
<td>uuuu uuuu</td>
</tr>
<tr>
<td>PCL</td>
<td>0000 0000</td>
<td>0000 0000</td>
<td>0000 0000</td>
<td>0000 0000</td>
</tr>
<tr>
<td>TBLP</td>
<td>xxxxxxx</td>
<td>uuuu uuuu</td>
<td>uuuu uuuu</td>
<td>uuuu uuuu</td>
</tr>
<tr>
<td>TBLH</td>
<td>--xx xxxx</td>
<td>--uu uuuu</td>
<td>--uu uuuu</td>
<td>--uu uuuu</td>
</tr>
<tr>
<td>STATUS</td>
<td>--00 xxxx</td>
<td>--uu uuuu</td>
<td>--1u uuuu</td>
<td>--11 uuuu</td>
</tr>
<tr>
<td>INTC</td>
<td>-000 0000</td>
<td>-000 0000</td>
<td>-000 0000</td>
<td>-uuu uuuu</td>
</tr>
<tr>
<td>TMR</td>
<td>xxxxxxx</td>
<td>xxxxxxx</td>
<td>xxxxxxx</td>
<td>xxxxxxx uuuu uuuu</td>
</tr>
<tr>
<td>TMRC</td>
<td>00-0 1000</td>
<td>00-0 1000</td>
<td>00-0 1000</td>
<td>uu-u uuuu</td>
</tr>
<tr>
<td>PA</td>
<td>1111 1111</td>
<td>1111 1111</td>
<td>1111 1111</td>
<td>1111 1111 uuuu uuuu</td>
</tr>
<tr>
<td>PAC</td>
<td>1111 1111</td>
<td>1111 1111</td>
<td>1111 1111</td>
<td>1111 1111 uuuu uuuu</td>
</tr>
<tr>
<td>PB</td>
<td>----- 1111</td>
<td>----- 1111</td>
<td>----- 1111</td>
<td>----- uuuu</td>
</tr>
<tr>
<td>PBC</td>
<td>----- 1111</td>
<td>----- 1111</td>
<td>----- 1111</td>
<td>----- uuuu</td>
</tr>
<tr>
<td>PD</td>
<td>----- ---1</td>
<td>----- ---1</td>
<td>----- ---1</td>
<td>----- --uu</td>
</tr>
<tr>
<td>PDC</td>
<td>----- ---1</td>
<td>----- ---1</td>
<td>----- ---1</td>
<td>----- --uu</td>
</tr>
<tr>
<td>PWM</td>
<td>xxxxxxx</td>
<td>xxxxxxx</td>
<td>xxxxxxx</td>
<td>xxxxxxx uuuu uuuu</td>
</tr>
<tr>
<td>ADRL</td>
<td>x---- ----</td>
<td>x---- ----</td>
<td>x---- ----</td>
<td>u---- ----</td>
</tr>
<tr>
<td>ADRH</td>
<td>xxxxxxx</td>
<td>xxxxxxx</td>
<td>xxxxxxx</td>
<td>xxxxxxx uuuu uuuu</td>
</tr>
<tr>
<td>ADCR</td>
<td>0100 0000</td>
<td>0100 0000</td>
<td>0100 0000</td>
<td>0100 0000 uuuu uuuu</td>
</tr>
<tr>
<td>ACSR</td>
<td>1--- --00</td>
<td>1--- --00</td>
<td>1--- --00</td>
<td>1--- --00 u--- --uu</td>
</tr>
</tbody>
</table>

*“u” stands for unchanged
*“x” stands for unknown
*“-” stands for unimplemented
<table>
<thead>
<tr>
<th>Register</th>
<th>Reset (Power On)</th>
<th>RES or LVR Reset</th>
<th>WDT Time-out (Normal Operation)</th>
<th>WDT Time-out (HALT)</th>
</tr>
</thead>
<tbody>
<tr>
<td>MP</td>
<td>-xxxx xxxx</td>
<td>-uuu uuuu</td>
<td>-uuu uuuu</td>
<td>-uuu uuuu</td>
</tr>
<tr>
<td>ACC</td>
<td>xxxxxxxxx</td>
<td>uuuu uuuu</td>
<td>uuuu uuuu</td>
<td>uuuu uuuu</td>
</tr>
<tr>
<td>PCL</td>
<td>0000 0000</td>
<td>0000 0000</td>
<td>0000 0000</td>
<td>0000 0000</td>
</tr>
<tr>
<td>TBLP</td>
<td>xxxxxxxxx</td>
<td>uuuu uuuu</td>
<td>uuuu uuuu</td>
<td>uuuu uuuu</td>
</tr>
<tr>
<td>TB LH</td>
<td>-xxxx xxxx</td>
<td>-uuu uuuu</td>
<td>-uuu uuuu</td>
<td>-uuu uuuu</td>
</tr>
<tr>
<td>STATUS</td>
<td>--00 xxxxx</td>
<td>--uu uuuu</td>
<td>--1uu uuuu</td>
<td>--11 uuuu</td>
</tr>
<tr>
<td>INTC0</td>
<td>-000 0000</td>
<td>-000 0000</td>
<td>-000 0000</td>
<td>-uuu uuuu</td>
</tr>
<tr>
<td>INTC1</td>
<td>--0 --0</td>
<td>--0 --0</td>
<td>--0 --0</td>
<td>--0 --0</td>
</tr>
<tr>
<td>TMR</td>
<td>xxxxxxxxx xxxxxx xxxxxxx</td>
<td>xxxxxxxxx uuuu uuuu</td>
<td>xxxxxxxxx uuuu uuuu</td>
<td>xxxxxxxxx uuuu uuuu</td>
</tr>
<tr>
<td>TMRC</td>
<td>00-0 1000</td>
<td>00-0 1000</td>
<td>00-0 1000</td>
<td>uu-uu uuu</td>
</tr>
<tr>
<td>PA</td>
<td>1111 1111</td>
<td>1111 1111</td>
<td>1111 1111</td>
<td>uuuu uuuu</td>
</tr>
<tr>
<td>PAC</td>
<td>1111 1111</td>
<td>1111 1111</td>
<td>1111 1111</td>
<td>uuuu uuuu</td>
</tr>
<tr>
<td>PB</td>
<td>1111 1111</td>
<td>1111 1111</td>
<td>1111 1111</td>
<td>uuuu uuuu</td>
</tr>
<tr>
<td>PBC</td>
<td>1111 1111</td>
<td>1111 1111</td>
<td>1111 1111</td>
<td>uuuu uuuu</td>
</tr>
<tr>
<td>PC</td>
<td>----- --11</td>
<td>----- --11</td>
<td>----- --11</td>
<td>----- --uu</td>
</tr>
<tr>
<td>PCC</td>
<td>----- --11</td>
<td>----- --11</td>
<td>----- --11</td>
<td>----- --uu</td>
</tr>
<tr>
<td>PD</td>
<td>----- --1</td>
<td>----- --1</td>
<td>----- --1</td>
<td>----- --uu</td>
</tr>
<tr>
<td>PDC</td>
<td>----- --1</td>
<td>----- --1</td>
<td>----- --1</td>
<td>----- --uu</td>
</tr>
<tr>
<td>PWM</td>
<td>xxxxxxxxx xxxxxx xxxxxxx</td>
<td>xxxxxxxxx uuuu uuuu</td>
<td>xxxxxxxxx uuuu uuuu</td>
<td>xxxxxxxxx uuuu uuuu</td>
</tr>
<tr>
<td>HADR</td>
<td>xxxxxxxxx xxxxxx xxxxxxx</td>
<td>xxxxxxxxx uuuu uuuu</td>
<td>xxxxxxxxx uuuu uuuu</td>
<td>xxxxxxxxx uuuu uuuu</td>
</tr>
<tr>
<td>HCR</td>
<td>0--0 0--0</td>
<td>0--0 0--0</td>
<td>0--0 0--0</td>
<td>uu--uu u--uu</td>
</tr>
<tr>
<td>HSR</td>
<td>100--0-0-1</td>
<td>100--0-0-1</td>
<td>100--0-0-1</td>
<td>uu--uu u--uu</td>
</tr>
<tr>
<td>HDR</td>
<td>xxxxxxxxx xxxxxx xxxxxxx</td>
<td>xxxxxxxxx uuuu uuuu</td>
<td>xxxxxxxxx uuuu uuuu</td>
<td>xxxxxxxxx uuuu uuuu</td>
</tr>
<tr>
<td>ADRL</td>
<td>-- -- -- --</td>
<td>-- -- -- --</td>
<td>-- -- -- --</td>
<td>-- -- -- --</td>
</tr>
<tr>
<td>ADRH</td>
<td>xxxxxxxxx xxxxxx xxxxxxx</td>
<td>xxxxxxxxx uuuu uuuu</td>
<td>xxxxxxxxx uuuu uuuu</td>
<td>xxxxxxxxx uuuu uuuu</td>
</tr>
<tr>
<td>ADCR</td>
<td>0100 0000</td>
<td>0100 0000</td>
<td>0100 0000</td>
<td>uuuu uuuuu</td>
</tr>
<tr>
<td>ACSR</td>
<td>1-- --0-0</td>
<td>1-- --0-0</td>
<td>1-- --0-0</td>
<td>uu----uu</td>
</tr>
</tbody>
</table>

"u" stands for unchanged
"x" stands for unknown
"-" stands for unimplemented
<table>
<thead>
<tr>
<th>Register</th>
<th>Reset (Power On)</th>
<th>RES or LVR Reset</th>
<th>WDT Time-out (Normal Operation)</th>
<th>WDT Time-out (HALT)</th>
</tr>
</thead>
<tbody>
<tr>
<td>MP0</td>
<td>xxxx xxxx</td>
<td>uuuu uuuu</td>
<td>uuuu uuuu uuuu uuuu uuuu</td>
<td></td>
</tr>
<tr>
<td>MP1</td>
<td>xxxx xxxx</td>
<td>uuuu uuuu</td>
<td>uuuu uuuu uuuu uuuu uuuu</td>
<td></td>
</tr>
<tr>
<td>ACC</td>
<td>xxxx xxxx</td>
<td>uuuu uuuu</td>
<td>uuuu uuuu uuuu uuuu uuuu</td>
<td></td>
</tr>
<tr>
<td>PCL</td>
<td>0000 0000</td>
<td>0000 0000</td>
<td>0000 0000 0000 0000</td>
<td></td>
</tr>
<tr>
<td>TBLP</td>
<td>xxxx xxxx</td>
<td>uuuu uuuu</td>
<td>uuuu uuuu uuuu uuuu uuuu</td>
<td></td>
</tr>
<tr>
<td>TBLH</td>
<td>-xxx xxxx</td>
<td>-uuu uuuu</td>
<td>-uuu uuuu uuuu uuuu uuuu</td>
<td></td>
</tr>
<tr>
<td>STATUS</td>
<td>--00 xxxx</td>
<td>--uu uuuu</td>
<td>--1u uuuu --11 uuuu</td>
<td></td>
</tr>
<tr>
<td>INTC0</td>
<td>-000 0000</td>
<td>-000 0000</td>
<td>-000 0000 0000 0000</td>
<td></td>
</tr>
<tr>
<td>INTC1</td>
<td>--0 -- --0</td>
<td>--0 -- --0</td>
<td>--0 -- --0 -- --u -- --u</td>
<td></td>
</tr>
<tr>
<td>TMRL</td>
<td>xxxx xxxx xxxx</td>
<td>xxxx xxxx xxxx</td>
<td>xxxx xxxx uuuu uuuu</td>
<td></td>
</tr>
<tr>
<td>TMRH</td>
<td>xxxx xxxx xxxx</td>
<td>xxxx xxxx xxxx</td>
<td>xxxx xxxx uuuu uuuu</td>
<td></td>
</tr>
<tr>
<td>TMRC</td>
<td>00-0 1000</td>
<td>00-0 1000</td>
<td>00-0 1000 uuu uuuu</td>
<td></td>
</tr>
<tr>
<td>PA</td>
<td>1111 1111</td>
<td>1111 1111</td>
<td>1111 1111 uuuu uuuu</td>
<td></td>
</tr>
<tr>
<td>PAC</td>
<td>1111 1111</td>
<td>1111 1111</td>
<td>1111 1111 uuuu uuuu</td>
<td></td>
</tr>
<tr>
<td>PB</td>
<td>1111 1111</td>
<td>1111 1111</td>
<td>1111 1111 uuuu uuuu</td>
<td></td>
</tr>
<tr>
<td>PBC</td>
<td>1111 1111</td>
<td>1111 1111</td>
<td>1111 1111 uuuu uuuu</td>
<td></td>
</tr>
<tr>
<td>PC</td>
<td>---1 1111</td>
<td>---1 1111</td>
<td>---1 1111 -- --u -- --u</td>
<td></td>
</tr>
<tr>
<td>PCC</td>
<td>---1 1111</td>
<td>---1 1111</td>
<td>---1 1111 -- --u -- --u</td>
<td></td>
</tr>
<tr>
<td>PD</td>
<td>--- -- --11</td>
<td>--- -- --11</td>
<td>--- -- --11 -- --u</td>
<td></td>
</tr>
<tr>
<td>PDC</td>
<td>--- -- --11</td>
<td>--- -- --11</td>
<td>--- -- --11 -- --u</td>
<td></td>
</tr>
<tr>
<td>PWM0</td>
<td>xxxx xxxx xxxx</td>
<td>xxxx xxxx xxxx</td>
<td>xxxx xxxx uuuu uuuu</td>
<td></td>
</tr>
<tr>
<td>PWM1</td>
<td>xxxx xxxx xxxx</td>
<td>xxxx xxxx xxxx</td>
<td>xxxx xxxx uuuu uuuu</td>
<td></td>
</tr>
<tr>
<td>HADR</td>
<td>xxxx xxxx</td>
<td>xxxx xxxx xxxx</td>
<td>xxxx xxxx-- uuuu uuuu--</td>
<td></td>
</tr>
<tr>
<td>HCR</td>
<td>0--0 0--0</td>
<td>0--0 0--0</td>
<td>0--0 0--0 -- --u -- --u</td>
<td></td>
</tr>
<tr>
<td>HSR</td>
<td>100--0--0--1</td>
<td>100--0--0--1</td>
<td>100--0--0--1 uuu--u--u</td>
<td></td>
</tr>
<tr>
<td>HDR</td>
<td>xxxx xxxx xxxx</td>
<td>xxxx xxxx xxxx</td>
<td>xxxx xxxx uuuu uuuu</td>
<td></td>
</tr>
<tr>
<td>ADRL</td>
<td>xx-- xxxx--</td>
<td>xx-- xxxx--</td>
<td>xx-- xxxx-- uuu--u--u</td>
<td></td>
</tr>
<tr>
<td>ADRH</td>
<td>xxxx xxxx xxxx</td>
<td>xxxx xxxx xxxx</td>
<td>xxxx xxxx uuuu uuuu</td>
<td></td>
</tr>
<tr>
<td>ACSR</td>
<td>0100 0000</td>
<td>0100 0000</td>
<td>0100 0000 uuuu uuuu</td>
<td></td>
</tr>
</tbody>
</table>

*"u" stands for unchanged*

*'x" stands for unknown

*"-" stands for unimplemented*
# HT46R24/HT46C24

## Register Reset

<table>
<thead>
<tr>
<th>Register</th>
<th>Reset (Power On)</th>
<th>RES or LVR Reset</th>
<th>WDT Time-out (Normal Operation)</th>
<th>WDT Time-out (HALT)</th>
</tr>
</thead>
<tbody>
<tr>
<td>MP0</td>
<td>xxxxxxxxxxxxxxx</td>
<td>uuuuuuuuuuuu</td>
<td>uuuuuuuuuuuu</td>
<td>uuuuuuuuuuuu</td>
</tr>
<tr>
<td>MP1</td>
<td>xxxxxxxxxxxxxxx</td>
<td>uuuuuuuuuuuu</td>
<td>uuuuuuuuuuuu</td>
<td>uuuuuuuuuuuu</td>
</tr>
<tr>
<td>ACC</td>
<td>xxxxxxxxxxxxxxx</td>
<td>uuuuuuuuuuuu</td>
<td>uuuuuuuuuuuu</td>
<td>uuuuuuuuuuuu</td>
</tr>
<tr>
<td>PCL</td>
<td>0000000000</td>
<td>0000000000</td>
<td>0000000000</td>
<td>0000000000</td>
</tr>
<tr>
<td>BP</td>
<td>-------000000000</td>
<td>-------000000000</td>
<td>-------000000000</td>
<td>-------000000000</td>
</tr>
<tr>
<td>TBLP</td>
<td>xxxxxxxxxxxxxxx</td>
<td>uuuuuuuuuuuu</td>
<td>uuuuuuuuuuuu</td>
<td>uuuuuuuuuuuu</td>
</tr>
<tr>
<td>TBLH</td>
<td>xxxxxxxxxxxxxxx</td>
<td>uuuuuuuuuuuu</td>
<td>uuuuuuuuuuuu</td>
<td>uuuuuuuuuuuu</td>
</tr>
<tr>
<td>STATUS</td>
<td>--00xxxxxxxxxxx</td>
<td>--uuuuuuuuuuu</td>
<td>--uuuuuuuuuuu</td>
<td>--11uuuuuuuuuu</td>
</tr>
<tr>
<td>INTC0</td>
<td>--00000000</td>
<td>--00000000</td>
<td>--00000000</td>
<td>--uuuuuuuuuuu</td>
</tr>
<tr>
<td>INTC1</td>
<td>--00--00--00</td>
<td>--00--00--00</td>
<td>--00--00--00</td>
<td>--uu--uuuuuuu</td>
</tr>
<tr>
<td>TMR0H</td>
<td>xxxxxxxxxxxxxxx</td>
<td>xxxxxxxxxxxxxxx</td>
<td>xxxxxxxxxxxxxxx</td>
<td>xxxxxxxxxxxxxxx</td>
</tr>
<tr>
<td>TMR0L</td>
<td>xxxxxxxxxxxxxxx</td>
<td>xxxxxxxxxxxxxxx</td>
<td>xxxxxxxxxxxxxxx</td>
<td>xxxxxxxxxxxxxxx</td>
</tr>
<tr>
<td>TMR0C</td>
<td>0000100000</td>
<td>0000100000</td>
<td>0000100000</td>
<td>uuuuuuuuuuuuu</td>
</tr>
<tr>
<td>TMR1H</td>
<td>xxxxxxxxxxxxxxx</td>
<td>xxxxxxxxxxxxxxx</td>
<td>xxxxxxxxxxxxxxx</td>
<td>xxxxxxxxxxxxxxx</td>
</tr>
<tr>
<td>TMR1L</td>
<td>xxxxxxxxxxxxxxx</td>
<td>xxxxxxxxxxxxxxx</td>
<td>xxxxxxxxxxxxxxx</td>
<td>xxxxxxxxxxxxxxx</td>
</tr>
<tr>
<td>TMR1C</td>
<td>00001----</td>
<td>00001----</td>
<td>00001----</td>
<td>uu--uuuuuuuuuuu</td>
</tr>
<tr>
<td>PA</td>
<td>1111111111</td>
<td>1111111111</td>
<td>1111111111</td>
<td>uuuuuuuuuuuu</td>
</tr>
<tr>
<td>PAC</td>
<td>1111111111</td>
<td>1111111111</td>
<td>1111111111</td>
<td>uuuuuuuuuuuu</td>
</tr>
<tr>
<td>PB</td>
<td>1111111111</td>
<td>1111111111</td>
<td>1111111111</td>
<td>uuuuuuuuuuuu</td>
</tr>
<tr>
<td>PBC</td>
<td>1111111111</td>
<td>1111111111</td>
<td>1111111111</td>
<td>uuuuuuuuuuuu</td>
</tr>
<tr>
<td>PC</td>
<td>1111111111</td>
<td>1111111111</td>
<td>1111111111</td>
<td>uuuuuuuuuuuu</td>
</tr>
<tr>
<td>PCC</td>
<td>1111111111</td>
<td>1111111111</td>
<td>1111111111</td>
<td>uuuuuuuuuuuu</td>
</tr>
<tr>
<td>PD</td>
<td>1111111111</td>
<td>1111111111</td>
<td>1111111111</td>
<td>uuuuuuuuuuuu</td>
</tr>
<tr>
<td>PDC</td>
<td>1111111111</td>
<td>1111111111</td>
<td>1111111111</td>
<td>uuuuuuuuuuuu</td>
</tr>
<tr>
<td>PF</td>
<td>1111111111</td>
<td>1111111111</td>
<td>1111111111</td>
<td>uuuuuuuuuuuu</td>
</tr>
<tr>
<td>PFC</td>
<td>1111111111</td>
<td>1111111111</td>
<td>1111111111</td>
<td>uuuuuuuuuuuu</td>
</tr>
<tr>
<td>PWM0</td>
<td>xxxxxxxxxxxxxxx</td>
<td>xxxxxxxxxxxxxxx</td>
<td>xxxxxxxxxxxxxxx</td>
<td>xxxxxxxxxxxxxxx</td>
</tr>
<tr>
<td>PWM1</td>
<td>xxxxxxxxxxxxxxx</td>
<td>xxxxxxxxxxxxxxx</td>
<td>xxxxxxxxxxxxxxx</td>
<td>xxxxxxxxxxxxxxx</td>
</tr>
<tr>
<td>PWM2</td>
<td>xxxxxxxxxxxxxxx</td>
<td>xxxxxxxxxxxxxxx</td>
<td>xxxxxxxxxxxxxxx</td>
<td>xxxxxxxxxxxxxxx</td>
</tr>
<tr>
<td>PWM3</td>
<td>xxxxxxxxxxxxxxx</td>
<td>xxxxxxxxxxxxxxx</td>
<td>xxxxxxxxxxxxxxx</td>
<td>xxxxxxxxxxxxxxx</td>
</tr>
<tr>
<td>HADR</td>
<td>xxxxxxxxxx-----</td>
<td>xxxxxxxxxx-----</td>
<td>xxxxxxxxxx-----</td>
<td>xxxxxxxxxx-----</td>
</tr>
<tr>
<td>HCR</td>
<td>00--00--00</td>
<td>00--00--00</td>
<td>00--00--00</td>
<td>uu--uuuuuuuuu</td>
</tr>
<tr>
<td>HSR</td>
<td>100--0--01</td>
<td>100--0--01</td>
<td>100--0--01</td>
<td>uuuuuuuuuuuuuuu</td>
</tr>
<tr>
<td>HDR</td>
<td>xxxxxxxxxxxxxxx</td>
<td>xxxxxxxxxxxxxxx</td>
<td>xxxxxxxxxxxxxxx</td>
<td>xxxxxxxxxxxxxxx</td>
</tr>
<tr>
<td>ADRL</td>
<td>xx-----xx-----</td>
<td>xx-----xx-----</td>
<td>xx-----xx-----</td>
<td>uu-----uuuuuuuuu</td>
</tr>
<tr>
<td>ADRH</td>
<td>xxxxxxxxxxxxxxx</td>
<td>xxxxxxxxxxxxxxx</td>
<td>xxxxxxxxxxxxxxx</td>
<td>xxxxxxxxxxxxxxx</td>
</tr>
<tr>
<td>ADCR</td>
<td>0100000000</td>
<td>0100000000</td>
<td>0100000000</td>
<td>uuuuuuuuuuuu</td>
</tr>
<tr>
<td>ACSR</td>
<td>1111111111</td>
<td>1111111111</td>
<td>1111111111</td>
<td>uu--uuuuuuuuu</td>
</tr>
</tbody>
</table>

*“u” stands for unchanged*

*“x” stands for unknown*

*“--” stands for unimplemented*
Oscillator

Various oscillator options offer the user a wide range of functions according to their various application requirements. Two types of system clocks can be selected while various clock source options for the Watchdog Timer are provided for maximum flexibility. All oscillator options are selected through the configuration options.

System Clock Configurations

There are two methods of generating the system clock, using an external crystal/ceramic oscillator or an external RC network. The chosen method is selected through the configuration options.

System Crystal/Ceramic Oscillator

For the crystal oscillator configuration, the simple connection of a crystal across OSC1 and OSC2 will create the necessary phase shift and feedback for oscillation with no other external components required. A ceramic resonator can be used instead of a crystal but two small value capacitors should be connected between OSC1, OSC2 and ground.

```
<table>
<thead>
<tr>
<th>fOSC (Hz)</th>
<th>C1 (pF)</th>
<th>C2 (pF)</th>
<th>C1 (pF)</th>
<th>C2 (pF)</th>
</tr>
</thead>
<tbody>
<tr>
<td>400k</td>
<td>—</td>
<td>—</td>
<td>300</td>
<td>300</td>
</tr>
<tr>
<td>455k</td>
<td>—</td>
<td>—</td>
<td>300</td>
<td>300</td>
</tr>
<tr>
<td>480k</td>
<td>—</td>
<td>—</td>
<td>300</td>
<td>300</td>
</tr>
<tr>
<td>640k</td>
<td>—</td>
<td>—</td>
<td>300</td>
<td>300</td>
</tr>
<tr>
<td>1M</td>
<td>0</td>
<td>0</td>
<td>300</td>
<td>300</td>
</tr>
<tr>
<td>2M</td>
<td>0</td>
<td>0</td>
<td>0</td>
<td>0</td>
</tr>
<tr>
<td>3.58M</td>
<td>0</td>
<td>0</td>
<td>0</td>
<td>0</td>
</tr>
<tr>
<td>4M</td>
<td>0</td>
<td>0</td>
<td>0</td>
<td>0</td>
</tr>
<tr>
<td>6M</td>
<td>0</td>
<td>0</td>
<td>0</td>
<td>0</td>
</tr>
<tr>
<td>8M</td>
<td>0</td>
<td>0</td>
<td>0</td>
<td>0</td>
</tr>
</tbody>
</table>
```

*“0” stands for no capacitor required.

“—” stands for crystal oscillator frequencies normally not available.
System RC Oscillator

Using the external RC network as an oscillator requires that a resistor, with a value between 30kΩ and 750kΩ, is connected between OSC1 and GND. The generated system clock divided by 4 will be provided on OSC2 as an output which can be used for external synchronization purposes. Although this is a cost effective oscillator configuration, the oscillation frequency can vary with VDD, temperature and process variations on the chip itself and is therefore not suitable for applications where timing is critical or where accurate oscillator frequencies are required. For the value of the external resistor \( R_{\text{OSC}} \) please refer to the Appendix section for typical RC Oscillator vs. Temperature and \( V_{\text{DD}} \) characteristics graphics.

![RC Oscillator Diagram]

Note
An internal capacitor together with the external resistor, \( R_{\text{OSC}} \), are the components which determine the frequency of the oscillator. The external capacitor shown on the diagram does not influence the frequency of oscillation. This external capacitor should be added to improve oscillator stability if the open-drain OSC2 output is utilized in the application circuit.

Watchdog Timer Oscillator

The WDT oscillator is a fully self-contained free running on-chip RC oscillator with a typical period of 65\( \mu \)s at 5V requiring no external components. When the device enters the power down mode, the system clock will stop running but the WDT oscillator continues to free-run and to keep the watchdog active. However, to preserve power in certain applications the WDT oscillator can be disabled via a configuration option.

HALT and Wake-up in Power Down Mode

The HALT mode is initialized by the “HALT” instruction and results in the following:

- The system oscillator will be turned off
- The contents of the on chip RAM and registers remain unchanged
- The WDT will be cleared and resume counting if the WDT clock source is selected to come from the WDT oscillator
- All of the I/O ports remain unchanged
- The PDF flag is set and the TO flag is cleared
When the system enters the HALT mode the system oscillator will be stopped to reduce power consumption. However, it is important to remember that if the internal WDT oscillator is enabled this will keep running and result in a small amount of power being consumed. In addition if the A/D converter is used, even though the system oscillator has been stopped there will still be some power consumption associated with the A/D circuitry. Therefore to minimize power consumption when in the HALT mode, the A/D converter should be first disabled by clearing all the PCR bits in the ADCR register.

The system can leave the HALT mode by means of a reset, an external interrupt, an external falling edge signal on Port A or a WDT overflow. A reset will initialize a chip reset and a WDT overflow will initialize a WDT time-out Reset from HALT but by examining the TO and PDF flags the source of the reset can be determined. The PDF flag is cleared by a system power-up or executing the "CLR WDT" instruction and is set when executing the "HALT" instruction. The TO flag is set if a WDT time-out occurs, and causes a wake-up that only resets the PC and SP; the other flags remain in their original status.

Port A wake-up and external interrupt wake-up methods can be considered as a continuation of normal execution. Each bit in Port A can be independently selected to wake-up the device by configuration option. Awakening from an I/O port stimulus, the program will resume execution at the next instruction. If the system is woken up via an external interrupt, two possibilities may occur. If the external interrupt is disabled or the external interrupt is enabled but the stack is full, the program will resume execution at the next instruction. If the external interrupt is enabled and the stack is not full, the regular interrupt response takes place. If the external interrupt request flag is set to "1" before entering the HALT mode, the wake-up function of the related interrupt will be disabled. Once a wake-up event occurs, it takes 1024 system clock periods to resume normal operation. In other words, a dummy period will be inserted after a wake-up. If the wake-up results from an external interrupt acknowledge signal, the actual interrupt subroutine execution will be delayed by one or more cycles. If the wake-up results in the next instruction execution, this will be executed immediately after the dummy period is finished.

Watchdog Timer

The Watchdog Timer is provided to prevent program malfunctions or sequences from jumping to unknown locations, due to certain uncontrollable external events such as electrical noise. It operates by providing a "chip reset" when the WDT counter overflows. The WDT clock is supplied by one of two sources selected by configuration option: its own self contained dedicated internal WDT oscillator or the instruction clock which is the system clock divided by 4. Note that if the WDT configuration option has been disabled, then any instruction relating to its operation will result in no operation.

In the A/D series of microcontrollers, all watchdog timer options, such as enable/disable, WDT clock source, and if applicable clock source division ratios are all selected through configuration options. There are no internal registers associated with the WDT in the A/D series. One of the WDT clock sources is an internal oscillator which has an approximate period of 65μs at a supply voltage of 5V. However, it should be noted that this specified internal clock period can vary with VDD, temperature and process variations. The other WDT clock source option is the instruction clock which is the system clock divided by four (f_{sys}/4). Whether the WDT clock source comes from its own internal WDT oscillator, or from the instruction clock, it is further divided by an internal
counter to give longer watchdog time-outs. In the case of the HT46R47/HT46C47 devices, this division ratio is fixed by an internal counter which gives a $2^{15}$ fixed division ratio. In the case of the other devices, the division ratio can be varied by selecting different configuration options to give a $2^{12}$ to $2^{15}$ division ratio range. As the clear instruction only resets the last stage of the counter chain, for this reason the actual division ratio and corresponding WDT time-out can vary by a factor of two. The exact division ratio depends upon the residual value in the WDT counter before the clear instruction is executed. As an example, if a WDT time-out value of $2^{12}$ (4096) is chosen in the configuration options, the actual time-out value can range from $f_S/2^{12}$ to $f_S/2^{15}$, where $f_S$ represents the WDT clock source. As mentioned earlier this clock source can come from either the internal WDT oscillator or from the system clock divided by four.

If the instruction clock is used as the clock source, it should be noted that when the system enters the power-down mode, then the instruction clock is stopped and the WDT will lose its protecting purposes. In such cases, the system can only be restarted via external logic. For systems that operate in noisy environments, using the internal WDT oscillator is strongly recommended.

Under normal program operation, the WDT time-out will initialize a “chip reset” and set the status bit “TO”. However, if the system is in the power-down mode, only a WDT time-out reset from “HALT” will be initialized which will only reset the Program Counter and SP. Three methods can be adopted to clear the contents of the WDT. The first is an external hardware reset (a low level on the RES pin), the second is via software instructions and the third is via a “HALT” instruction. There are two methods of using software instructions to clear the Watchdog Timer, one of which must be chosen by configuration option. The first option is to use the single “CLR WDT” instruction while the second is to use the two commands “CLR WDT1” and “CLR WDT2”. For the first option, a simple execution of “CLR WDT” will clear the WDT while for the second option, both “CLR WDT1” and “CLR WDT2” must both be executed to successfully clear the WDT. Note that for this second option, if “CLR WDT1” is used to clear the WDT, successive executions of this instruction will have no effect, only the execution of a “CLR WDT2” instruction will clear the WDT. Similarly after the “CLR WDT2” instruction has been executed, only a successive “CLR WDT1” instruction can clear the Watchdog Timer.

**Watchdog Timer**

![Watchdog Timer Diagram](image)

**Note**

1. The 4-to-1 configuration option to select $f_S/2^{12}$, $f_S/2^{13}$, $f_S/2^{14}$ or $f_S/2^{15}$ is not applicable in the HT46R47/HT46C47, which has a fixed $f_S/2^{15}$ division ratio.
2. Because only the last stage of counter chain is cleared by instructions, the WDT time-out period varies. As an example, the selected value of $2^{15}/f_S$ may range from $2^{16}/f_S$ to $2^{15}/f_S$. 

75
## Configuration Options

The various microcontroller configuration options selected using the HT-IDE are stored in the option memory. All bits must be defined for proper system function, the details of which are shown in the table. After the configuration options have been programmed into the microcontroller by the user, it is important to note that they cannot be altered later by the application program. For the mask version devices, these configuration options, once defined, are implemented into the microcontroller during the manufacturing process and therefore cannot be reconfigured by the user.

<table>
<thead>
<tr>
<th>No.</th>
<th>Option</th>
</tr>
</thead>
<tbody>
<tr>
<td>1</td>
<td>WDT clock source: WDT oscillator or fSYS/4 or disable</td>
</tr>
<tr>
<td>2</td>
<td>CLRWDI instructions: 1 or 2 instructions</td>
</tr>
<tr>
<td>3</td>
<td>PA0~PA7 wake-up: enable or disable (by bit)</td>
</tr>
<tr>
<td>4</td>
<td>PA, PB, PC, PD, PF pull-high enable or disable</td>
</tr>
<tr>
<td></td>
<td>(Number of ports is device dependent. Pull-high bit or port is also device dependent.)</td>
</tr>
<tr>
<td>5</td>
<td>PD0~PD3: PWM function selection. Number of PWM channels are device dependent.</td>
</tr>
<tr>
<td>6</td>
<td>PWM mode selection: (7+1) or (6+2) mode</td>
</tr>
<tr>
<td></td>
<td>(excluding HT46R47/HT46C47, which is fixed at (6+2) mode)</td>
</tr>
<tr>
<td>7</td>
<td>OSC type selection: RC or crystal</td>
</tr>
<tr>
<td>8</td>
<td>PA3 PFD function: enable or disable</td>
</tr>
<tr>
<td></td>
<td>PFD source selection: from timer 0 or timer 1 PFD output (for HT46R24/HT46C24 only)</td>
</tr>
<tr>
<td>9</td>
<td>WDT division ratio: $2^{12}$, $2^{13}$, $2^{14}$ or $2^{15}$ (excluding HT46R47/HT46C47)</td>
</tr>
<tr>
<td>10</td>
<td>PA6, PA7 I2C bus function: enable or disable (excluding HT46R47/HT46C47)</td>
</tr>
<tr>
<td>11</td>
<td>LVR function: enable or disable</td>
</tr>
</tbody>
</table>
Application Circuits
Part II

Programming Language
Instruction Set

Central to the successful operation of any microcontroller is its instruction set, which is a set of program instruction codes that directs the microcontroller to perform certain operations. In the case of Holtek microcontrollers, a comprehensive and flexible set of over 60 instructions is provided to enable programmers to implement their application with the minimum of programming overheads.

For easier understanding of the various instruction codes, they have been subdivided into several functional groupings.

Instruction Timing

Most instructions are implemented within one instruction cycle. The exceptions to this are branch, call, or table read instructions where two instruction cycles are required. One instruction cycle is equal to 4 system clock cycles, therefore in the case of an 8MHz system oscillator, most instructions would be implemented within 0.5 μs and branch or call instructions would be implemented within 1 μs. Although instructions which require one more cycle to implement are generally limited to the JMP, CALL, RET, RETI and table read instructions, it is important to realize that any other instructions which involve manipulation of the Program Counter Low register or PCL will also take one more cycle to implement. As instructions which change the contents of the PCL will imply a direct jump to that new address, one more cycle will be required. Examples of such instructions would be “CLR PCL” or “MOV PCL, A”. For the case of skip instructions, it must be noted that if the result of the comparison involves a skip operation then this will also take one more cycle, if no skip is involved then only one cycle is required.
Moving and Transferring Data
The transfer of data within the microcontroller program is one of the most frequently used operations. Making use of three kinds of MOV instructions, data can be transferred from registers to the Accumulator and vice-versa as well as being able to move specific immediate data directly into the Accumulator. One of the most important data transfer applications is to receive data from the input ports and transfer data to the output ports.

Arithmetic Operations
The ability to perform certain arithmetic operations and data manipulation is a necessary feature of most microcontroller applications. Within the Holtek microcontroller instruction set are a range of add and subtract instruction mnemonics to enable the necessary arithmetic to be carried out. Care must be taken to ensure correct handling of Carry and borrow data when results exceed 255 for addition and less than 0 for subtraction. The increment and decrement instructions INC, INCA, DEC and DECA provide a simple means of increasing or decreasing by a value of one of the values in the destination specified.

Logical and Rotate Operations
The standard logical operations such as AND, OR, XOR and CPL all have their own instruction within the Holtek microcontroller instruction set. As with the case of most instructions involving data manipulation, data must pass through the Accumulator which may involve additional programming steps. In all logical data operations, the zero flag may be set if the result of the operation is zero. Another form of logical data manipulation comes from the rotate instructions such as RR, RL, RRC and RLC which provide a simple means of rotating one bit right or left. Different rotate instructions exist depending on program requirements. Rotate instructions are useful for serial port programming applications where data can be rotated from an internal register into the Carry bit from where it can be examined and the necessary serial bit set high or low. Another application where rotate data operations are used is to implement multiplication and division calculations.

Branches and Control Transfer
Program branching takes the form of either jumps to specified locations using the JMP instruction or to a subroutine using the CALL instruction. They differ in the sense that in the case of a subroutine call, the program must return to the instruction immediately when the subroutine has been carried out. This is done by placing a return instruction RET in the subroutine which will cause the program to jump back to the address right after the CALL instruction. In the case of a JMP instruction, the program simply jumps to the desired location. There is no requirement to jump back to the original jumping off point as in the case of the CALL. One special and extremely useful set of branch instructions are the conditional branches. Here a decision is first made regarding the condition of a certain data memory or individual bits. Depending upon the conditions, the program will continue with the next instruction or skip over it and jump to the following instruction. These instructions are the key to decision making and branching within the program perhaps determined by the condition of certain input switches or by the condition of internal data bits.

Bit Operations
The ability to provide single bit operations on Data Memory is an extremely flexible feature of all Holtek microcontrollers. This feature is especially useful for output port bit programming where in-
individual bits or port pins can be directly set high or low using either the “SET [m].i” or “CLR [m].i” instructions respectively. The feature removes the need for programmers to first read the 8-bit output port, manipulate the input data to ensure that other bits are not changed and then output the port with the correct new data. This read-modify-write process is taken care of automatically when these bit operation instructions are used.

Table Read Operations
Data storage is normally implemented by using registers. However, when working with large amounts of fixed data, the volume involved often makes it inconvenient to store the fixed data in individual memory. To overcome this problem, Holtek microcontrollers allow an area of Program Memory to be setup as a table where data can be directly stored. A set of easy to use instructions provides the means by which this fixed data can be referenced and retrieved from the Program Memory.

Other Operations
In addition to the above functional instructions, a range of other instructions also exist such as “HALT” instruction for Power-down operation and instructions to control the operation of the Watchdog Timer for reliable program operations under extreme electric or electromagnetic environment. For their relevant operations, refer to the functional related sections.

Instruction Set Summary

Convention
x : bits immediate data
m: Data Memory address
A: Accumulator
i: 0~7 number of bits
addr: Program memory address

<table>
<thead>
<tr>
<th>Mnemonic</th>
<th>Description</th>
<th>Cycles</th>
<th>Flag Affected</th>
</tr>
</thead>
<tbody>
<tr>
<td>ADD A,[m]</td>
<td>Add Data Memory to ACC</td>
<td>1</td>
<td>Z, C, AC, OV</td>
</tr>
<tr>
<td>ADDM A,[m]</td>
<td>Add ACC to Data Memory</td>
<td>Note</td>
<td>Z, C, AC, OV</td>
</tr>
<tr>
<td>ADD A,x</td>
<td>Add immediate data to ACC</td>
<td>1</td>
<td>Z, C, AC, OV</td>
</tr>
<tr>
<td>ADC A,[m]</td>
<td>Add Data Memory to ACC with Carry</td>
<td>1</td>
<td>Z, C, AC, OV</td>
</tr>
<tr>
<td>ADCM A,[m]</td>
<td>Add ACC to Data memory with Carry</td>
<td>Note</td>
<td>Z, C, AC, OV</td>
</tr>
<tr>
<td>SUB A,x</td>
<td>Subtract immediate data from the ACC</td>
<td>1</td>
<td>Z, C, AC, OV</td>
</tr>
<tr>
<td>SUB A,[m]</td>
<td>Subtract Data Memory from ACC</td>
<td>1</td>
<td>Z, C, AC, OV</td>
</tr>
<tr>
<td>SUBM A,[m]</td>
<td>Subtract Data Memory from ACC with result in Data Memory</td>
<td>Note</td>
<td>Z, C, AC, OV</td>
</tr>
<tr>
<td>SBC A,[m]</td>
<td>Subtract Data Memory from ACC with Carry</td>
<td>1</td>
<td>Z, C, AC, OV</td>
</tr>
<tr>
<td>SBCM A,[m]</td>
<td>Subtract Data Memory from ACC with Carry, result in Data Memory</td>
<td>Note</td>
<td>Z, C, AC, OV</td>
</tr>
<tr>
<td>DAA [m]</td>
<td>Decimal adjust ACC for Addition with result in Data Memory</td>
<td>Note</td>
<td>C</td>
</tr>
<tr>
<td>Mnemonic</td>
<td>Description</td>
<td>Cycles</td>
<td>Flag Affected</td>
</tr>
<tr>
<td>-----------</td>
<td>-----------------------------------------</td>
<td>--------</td>
<td>---------------</td>
</tr>
<tr>
<td><strong>Logic Operation</strong></td>
<td></td>
<td></td>
<td></td>
</tr>
<tr>
<td>AND A,[m]</td>
<td>Logical AND Data Memory to ACC</td>
<td>1</td>
<td>Z</td>
</tr>
<tr>
<td>OR A,[m]</td>
<td>Logical OR Data Memory to ACC</td>
<td>1</td>
<td>Z</td>
</tr>
<tr>
<td>XOR A,[m]</td>
<td>Logical XOR Data Memory to ACC</td>
<td>1</td>
<td>Z</td>
</tr>
<tr>
<td>ANDM A,[m]</td>
<td>Logical AND ACC to Data Memory</td>
<td>$^\text{Note}$</td>
<td>Z</td>
</tr>
<tr>
<td>ORM A,[m]</td>
<td>Logical OR ACC to Data Memory</td>
<td>$^\text{Note}$</td>
<td>Z</td>
</tr>
<tr>
<td>XORM A,[m]</td>
<td>Logical XOR ACC to Data Memory</td>
<td>$^\text{Note}$</td>
<td>Z</td>
</tr>
<tr>
<td>AND A,x</td>
<td>Logical AND immediate Data to ACC</td>
<td>1</td>
<td>Z</td>
</tr>
<tr>
<td>OR A,x</td>
<td>Logical OR immediate Data to ACC</td>
<td>1</td>
<td>Z</td>
</tr>
<tr>
<td>XOR A,x</td>
<td>Logical XOR immediate Data to ACC</td>
<td>1</td>
<td>Z</td>
</tr>
<tr>
<td>CPL [m]</td>
<td>Complement Data Memory</td>
<td></td>
<td>Z</td>
</tr>
<tr>
<td>CPLA [m]</td>
<td>Complement Data Memory with result in ACC</td>
<td>1</td>
<td>Z</td>
</tr>
<tr>
<td><strong>Increment &amp; Decrement</strong></td>
<td></td>
<td></td>
<td></td>
</tr>
<tr>
<td>INCA [m]</td>
<td>Increment Data Memory with result in ACC</td>
<td>1</td>
<td>Z</td>
</tr>
<tr>
<td>INC [m]</td>
<td>Increment Data Memory</td>
<td>$^\text{Note}$</td>
<td>Z</td>
</tr>
<tr>
<td>DECA [m]</td>
<td>Decrement Data Memory with result in ACC</td>
<td>1</td>
<td>Z</td>
</tr>
<tr>
<td>DEC [m]</td>
<td>Decrement Data Memory</td>
<td>$^\text{Note}$</td>
<td>Z</td>
</tr>
<tr>
<td><strong>Rotate</strong></td>
<td></td>
<td></td>
<td></td>
</tr>
<tr>
<td>RRA [m]</td>
<td>Rotate Data Memory right with result in ACC</td>
<td>1</td>
<td>None</td>
</tr>
<tr>
<td>RR [m]</td>
<td>Rotate Data Memory right</td>
<td>$^\text{Note}$</td>
<td>None</td>
</tr>
<tr>
<td>RRC [m]</td>
<td>Rotate Data Memory right through Carry with result in ACC</td>
<td>$^\text{Note}$</td>
<td>C</td>
</tr>
<tr>
<td>RL [m]</td>
<td>Rotate Data Memory left</td>
<td>$^\text{Note}$</td>
<td>None</td>
</tr>
<tr>
<td>RLC [m]</td>
<td>Rotate Data Memory left through Carry with result in ACC</td>
<td>$^\text{Note}$</td>
<td>C</td>
</tr>
<tr>
<td><strong>Data Move</strong></td>
<td></td>
<td></td>
<td></td>
</tr>
<tr>
<td>MOV A,[m]</td>
<td>Move Data Memory to ACC</td>
<td>1</td>
<td>None</td>
</tr>
<tr>
<td>MOV [m],A</td>
<td>Move ACC to Data Memory</td>
<td>$^\text{Note}$</td>
<td>None</td>
</tr>
<tr>
<td>MOV A,x</td>
<td>Move immediate data to ACC</td>
<td>1</td>
<td>None</td>
</tr>
<tr>
<td><strong>Bit Operation</strong></td>
<td></td>
<td></td>
<td></td>
</tr>
<tr>
<td>CLR [m],i</td>
<td>Clear bit of Data Memory</td>
<td>$^\text{Note}$</td>
<td>None</td>
</tr>
<tr>
<td>SET [m],i</td>
<td>Set bit of Data Memory</td>
<td>$^\text{Note}$</td>
<td>None</td>
</tr>
<tr>
<td>Mnemonic</td>
<td>Description</td>
<td>Cycles</td>
<td>Flag Affected</td>
</tr>
<tr>
<td>----------</td>
<td>-------------</td>
<td>--------</td>
<td>---------------</td>
</tr>
<tr>
<td><strong>Branch</strong></td>
<td></td>
<td></td>
<td></td>
</tr>
<tr>
<td>JMP addr</td>
<td>Jump unconditionally</td>
<td>2</td>
<td>None</td>
</tr>
<tr>
<td>S [m]</td>
<td>Skip if Data Memory is zero</td>
<td>1</td>
<td>Note None</td>
</tr>
<tr>
<td>SZA [m]</td>
<td>Skip if Data Memory is zero with data movement to ACC</td>
<td>1</td>
<td>Note None</td>
</tr>
<tr>
<td>SZ [m].i</td>
<td>Skip if bit i of Data Memory is zero</td>
<td>1</td>
<td>Note None</td>
</tr>
<tr>
<td>SNZ [m].i</td>
<td>Skip if bit i of Data Memory is not zero</td>
<td>1</td>
<td>Note None</td>
</tr>
<tr>
<td>SIZ [m]</td>
<td>Skip if increment Data Memory is zero</td>
<td>1</td>
<td>Note None</td>
</tr>
<tr>
<td>SDZ [m]</td>
<td>Skip if decrement Data Memory is zero</td>
<td>1</td>
<td>Note None</td>
</tr>
<tr>
<td>SIZA [m]</td>
<td>Skip if increment Data Memory is zero with result in ACC</td>
<td>1</td>
<td>Note None</td>
</tr>
<tr>
<td>SDZA [m]</td>
<td>Skip if decrement Data Memory is zero with result in ACC</td>
<td>1</td>
<td>Note None</td>
</tr>
<tr>
<td>CALL addr</td>
<td>Subroutine call</td>
<td>2</td>
<td>None</td>
</tr>
<tr>
<td>RET</td>
<td>Return from subroutine</td>
<td>2</td>
<td>None</td>
</tr>
<tr>
<td>RETA,x</td>
<td>Return from subroutine and load immediate data to ACC</td>
<td>2</td>
<td>None</td>
</tr>
<tr>
<td>RETI</td>
<td>Return from interrupt</td>
<td>2</td>
<td>None</td>
</tr>
<tr>
<td><strong>Table Read</strong></td>
<td></td>
<td></td>
<td></td>
</tr>
<tr>
<td>TABRDC [m]</td>
<td>Read table (current page) to TBLH and Data Memory</td>
<td>2</td>
<td>Note None</td>
</tr>
<tr>
<td>TABRDL [m]</td>
<td>Read table (last page) to TBLH and Data Memory</td>
<td>2</td>
<td>Note None</td>
</tr>
<tr>
<td><strong>Miscellaneous</strong></td>
<td></td>
<td></td>
<td></td>
</tr>
<tr>
<td>NOP</td>
<td>No operation</td>
<td>1</td>
<td>None</td>
</tr>
<tr>
<td>CLR [m]</td>
<td>Clear Data Memory</td>
<td>1</td>
<td>Note None</td>
</tr>
<tr>
<td>SET [m]</td>
<td>Set Data Memory</td>
<td>1</td>
<td>Note None</td>
</tr>
<tr>
<td>CLR WDT</td>
<td>Clear Watchdog Timer</td>
<td>1</td>
<td>TO, PDF</td>
</tr>
<tr>
<td>CLR WDT1</td>
<td>Pre-clear Watchdog Timer</td>
<td>1</td>
<td>TO, PDF</td>
</tr>
<tr>
<td>CLR WDT2</td>
<td>Pre-clear Watchdog Timer</td>
<td>1</td>
<td>TO, PDF</td>
</tr>
<tr>
<td>SWAP [m]</td>
<td>Swap nibbles of Data Memory</td>
<td>1</td>
<td>Note None</td>
</tr>
<tr>
<td>SWAP A [m]</td>
<td>Swap nibbles of Data Memory with result in ACC</td>
<td>1</td>
<td>None</td>
</tr>
<tr>
<td>HALT</td>
<td>Enter power down mode</td>
<td>1</td>
<td>TO, PDF</td>
</tr>
</tbody>
</table>

**Note**

1. For skip instructions, if the result of the comparison involves a skip then two cycles are required, if no skip takes place only one cycle is required.
2. Any instruction which changes the contents of the PCL will also require 2 cycles for execution.
3. For the “CLR WDT1” and “CLR WDT2” instructions the TO and PDF flags may be affected by the execution status. The TO and PDF flags are cleared after both “CLR WDT1” or “CLR WDT2” instructions are executed. Otherwise the TO and PDF flags remain unchanged.
## Chapter 3 Instruction Definition

### ADC A,[m]
**Description:** Add Data Memory to ACC with Carry
The contents of the specified Data Memory, Accumulator and the carry flag are added. The result is stored in the Accumulator.

**Operation:** 
$\text{ACC} \leftarrow \text{ACC} + [m] + C$

**Affected flag(s):** OV, Z, AC, C

### ADCM A,[m]
**Description:** Add ACC to Data Memory with Carry
The contents of the specified Data Memory, Accumulator and the carry flag are added. The result is stored in the specified Data Memory.

**Operation:** 
$[m] \leftarrow \text{ACC} + [m] + C$

**Affected flag(s):** OV, Z, AC, C

### ADD A,[m]
**Description:** Add Data Memory to ACC
The contents of the specified Data Memory and the Accumulator are added. The result is stored in the Accumulator.

**Operation:** 
$\text{ACC} \leftarrow \text{ACC} + [m]$

**Affected flag(s):** OV, Z, AC, C

### ADD A,x
**Description:** Add immediate data to ACC
The contents of the Accumulator and the specified immediate data are added. The result is stored in the Accumulator.

**Operation:** 
$\text{ACC} \leftarrow \text{ACC} + x$

**Affected flag(s):** OV, Z, AC, C

### ADDM A,[m]
**Description:** Add ACC to Data Memory
The contents of the specified Data Memory and the Accumulator are added. The result is stored in the specified Data Memory.

**Operation:** 
$[m] \leftarrow \text{ACC} + [m]$

**Affected flag(s):** OV, Z, AC, C
<table>
<thead>
<tr>
<th>Instruction</th>
<th>Description</th>
<th>Operation</th>
<th>Affected flag(s)</th>
</tr>
</thead>
<tbody>
<tr>
<td><strong>AND A,[m]</strong></td>
<td>Logical AND Data Memory to ACC</td>
<td>ACC $\leftarrow$ ACC “AND” [m]</td>
<td>Z</td>
</tr>
<tr>
<td><strong>AND A,x</strong></td>
<td>Logical AND immediate data to ACC</td>
<td>ACC $\leftarrow$ ACC “AND” x</td>
<td>Z</td>
</tr>
<tr>
<td><strong>ANDM A,[m]</strong></td>
<td>Logical AND ACC to Data Memory</td>
<td>[m] $\leftarrow$ ACC “AND” [m]</td>
<td>Z</td>
</tr>
<tr>
<td><strong>CALL addr</strong></td>
<td>Subroutine call</td>
<td>Stack $\leftarrow$ Program Counter + 1</td>
<td>None</td>
</tr>
<tr>
<td><strong>CLR [m]</strong></td>
<td>Clear Data Memory</td>
<td>[m] $\leftarrow$ 00H</td>
<td>None</td>
</tr>
<tr>
<td><strong>CLR [m].i</strong></td>
<td>Clear bit of Data Memory</td>
<td>[m].i $\leftarrow$ 0</td>
<td>None</td>
</tr>
</tbody>
</table>
CLR WDT
Clear Watchdog Timer
Description The TO, PDF flags and the WDT are all cleared.
Operation WDT cleared
TO ← 0
PDF ← 0
Affected flag(s) TO, PDF

CLR WDT1
Pre-clear Watchdog Timer
Description The TO, PDF flags and the WDT are all cleared. Note that this instruction works in conjunc-
tion with CLR WDT2 and must be executed alternately with CLR WDT2 to have effect. Re-
petitively executing this instruction without alternately executing CLR WDT2 will have no
effect.
Operation WDT cleared
TO ← 0
PDF ← 0
Affected flag(s) TO, PDF

CLR WDT2
Pre-clear Watchdog Timer
Description The TO, PDF flags and the WDT are all cleared. Note that this instruction works in conjunc-
tion with CLR WDT1 and must be executed alternately with CLR WDT1 to have effect. Re-
petitively executing this instruction without alternately executing CLR WDT1 will have no
effect.
Operation WDT cleared
TO ← 0
PDF ← 0
Affected flag(s) TO, PDF

CPL \[m\]
Complement Data Memory
Description Each bit of the specified Data Memory is logically complemented (1’s complement). Bits
which previously contained a 1 are changed to 0 and vice versa.
Operation \[m\] ← \[\overline{m}\]
Affected flag(s) Z

CPLA \[m\]
Complement Data Memory with result in ACC
Description Each bit of the specified Data Memory is logically complemented (1’s complement). Bits
which previously contained a 1 are changed to 0 and vice versa. The complemented result
is stored in the Accumulator and the contents of the Data Memory remain unchanged.
Operation ACC ← \[\overline{m}\]
Affected flag(s) Z
DAA [m]  Decimal-Adjust ACC for addition with result in Data Memory
Description  Convert the contents of the Accumulator value to a BCD (Binary Coded Decimal) value resulting from the previous addition of two BCD variables. If the low nibble is greater than 9 or if AC flag is set, then a value of 6 will be added to the low nibble. Otherwise the low nibble remains unchanged. If the high nibble is greater than 9 or if the C flag is set, then a value of 6 will be added to the high nibble. Essentially, the decimal conversion is performed by adding 00H, 06H, 60H or 66H depending on the Accumulator and flag conditions. Only the C flag may be affected by this instruction which indicates that if the original BCD sum is greater than 100, it allows multiple precision decimal addition.

Operation  [m] ← ACC + 00H or 
[m] ← ACC + 06H or 
[m] ← ACC + 60H or 
[m] ← ACC + 66H

Affected flag(s)  C

DEC [m]  Decrement Data Memory
Description  Data in the specified Data Memory is decremented by 1.

Operation  [m] ← [m] – 1

Affected flag(s)  Z

DECA [m]  Decrement Data Memory with result in ACC
Description  Data in the specified Data Memory is decremented by 1. The result is stored in the Accumulator. The contents of the Data Memory remain unchanged.

Operation  ACC ← [m] – 1

Affected flag(s)  Z

HALT  Enter power down mode
Description  This instruction stops the program execution and turns off the system clock. The contents of the Data Memory and registers are retained. The WDT and prescaler are cleared. The power down flag PDF is set and the WDT time-out flag TO is cleared.

Operation  TO ← 0 
PDF ← 1

Affected flag(s)  TO, PDF

INC [m]  Increment Data Memory
Description  Data in the specified Data Memory is incremented by 1.

Operation  [m] ← [m] + 1

Affected flag(s)  Z
| **INCA [m]** | Increment Data Memory with result in ACC |
| **Description** | Data in the specified Data Memory is incremented by 1. The result is stored in the Accumulator. The contents of the Data Memory remain unchanged. |
| **Operation** | ACC ← [m] + 1 |
| **Affected flag(s)** | Z |

| **JMP addr** | Jump unconditionally |
| **Description** | The contents of the Program Counter are replaced with the specified address. Program execution then continues from this new address. As this requires the insertion of a dummy instruction while the new address is loaded, it is a two cycle instruction. |
| **Operation** | Program Counter ← addr |
| **Affected flag(s)** | None |

| **MOV A,[m]** | Move Data Memory to ACC |
| **Description** | The contents of the specified Data Memory are copied to the Accumulator. |
| **Operation** | ACC ← [m] |
| **Affected flag(s)** | None |

| **MOV A,x** | Move immediate data to ACC |
| **Description** | The immediate data specified is loaded into the Accumulator. |
| **Operation** | ACC ← x |
| **Affected flag(s)** | None |

| **MOV [m],A** | Move ACC to Data Memory |
| **Description** | The contents of the Accumulator are copied to the specified Data Memory. |
| **Operation** | [m] ← ACC |
| **Affected flag(s)** | None |

| **NOP** | No operation |
| **Description** | No operation is performed. Execution continues with the next instruction. |
| **Operation** | No operation |
| **Affected flag(s)** | None |

<p>| <strong>OR A,[m]</strong> | Logical OR Data Memory to ACC |
| <strong>Description</strong> | Data in the Accumulator and the specified Data Memory perform a bitwise logical OR operation. The result is stored in the Accumulator. |
| <strong>Operation</strong> | ACC ← ACC “OR” [m] |
| <strong>Affected flag(s)</strong> | Z |</p>
<table>
<thead>
<tr>
<th>Instruction</th>
<th>Description</th>
<th>Operation</th>
<th>Affected flag(s)</th>
</tr>
</thead>
<tbody>
<tr>
<td><strong>OR A,x</strong></td>
<td>Logical OR immediate data to ACC</td>
<td>ACC ← ACC “OR” x</td>
<td>Z</td>
</tr>
<tr>
<td><strong>ORM A,[m]</strong></td>
<td>Logical OR ACC to Data Memory</td>
<td>[m] ← ACC “OR” [m]</td>
<td>Z</td>
</tr>
<tr>
<td><strong>RET</strong></td>
<td>Return from subroutine</td>
<td>Program Counter ← Stack</td>
<td>None</td>
</tr>
<tr>
<td><strong>RET A,x</strong></td>
<td>Return from subroutine and load immediate data to ACC</td>
<td>Program Counter ← Stack, ACC ← x</td>
<td>None</td>
</tr>
<tr>
<td><strong>RETI</strong></td>
<td>Return from interrupt</td>
<td>Program Counter ← Stack, EMI ← 1</td>
<td>None</td>
</tr>
<tr>
<td><strong>RL [m]</strong></td>
<td>Rotate Data Memory left</td>
<td>[m].(i+1) ← [m].i; (i = 0–6)</td>
<td>[m].0 ← [m].7</td>
</tr>
</tbody>
</table>
### RLA [m]
**Description**
The contents of the specified Data Memory are rotated left by 1 bit with bit 7 rotated into bit 0. The rotated result is stored in the Accumulator and the contents of the Data Memory remain unchanged.

**Operation**
\[
\begin{align*}
\text{ACC}.(i+1) &\leftarrow [m].i; \quad (i = 0\text{–}6) \\
\text{ACC}.0 &\leftarrow [m].7
\end{align*}
\]

**Affected flag(s)**
None

### RLC [m]
**Description**
The contents of the specified Data Memory and the carry flag are rotated left by 1 bit. Bit 7 replaces the Carry bit and the original carry flag is rotated into bit 0.

**Operation**
\[
\begin{align*}
[m].(i+1) &\leftarrow [m].i; \quad (i = 0\text{–}6) \\
[m].0 &\leftarrow C \\
C &\leftarrow [m].7
\end{align*}
\]

**Affected flag(s)**
C

### RLCA [m]
**Description**
Data in the specified Data Memory and the carry flag are rotated left by 1 bit. Bit 7 replaces the Carry bit and the original carry flag is rotated into the bit 0. The rotated result is stored in the Accumulator and the contents of the Data Memory remain unchanged.

**Operation**
\[
\begin{align*}
\text{ACC}.(i+1) &\leftarrow [m].i; \quad (i = 0\text{–}6) \\
\text{ACC}.0 &\leftarrow C \\
C &\leftarrow [m].7
\end{align*}
\]

**Affected flag(s)**
C

### RR [m]
**Description**
The contents of the specified Data Memory are rotated right by 1 bit with bit 0 rotated into bit 7.

**Operation**
\[
\begin{align*}
[m].i &\leftarrow [m].(i+1); \quad (i = 0\text{–}6) \\
[m].7 &\leftarrow [m].0
\end{align*}
\]

**Affected flag(s)**
None

### RRA [m]
**Description**
Data in the specified Data Memory and the carry flag are rotated right by 1 bit with bit 0 rotated into bit 7. The rotated result is stored in the Accumulator and the contents of the Data Memory remain unchanged.

**Operation**
\[
\begin{align*}
\text{ACC}.i &\leftarrow [m].(i+1); \quad (i = 0\text{–}6) \\
\text{ACC}.7 &\leftarrow [m].0
\end{align*}
\]

**Affected flag(s)**
None
RRC [m] Rotate Data Memory right through Carry
Description The contents of the specified Data Memory and the carry flag are rotated right by 1 bit. Bit 0 replaces the Carry bit and the original carry flag is rotated into bit 7.
Operation 
\[
\begin{align*}
  [m].i & \leftarrow [m].(i+1); \ (i = 0 \sim 6) \\
  [m].7 & \leftarrow C \\
  C & \leftarrow [m].0
\end{align*}
\]
Affected flag(s) C

RRCA [m] Rotate Data Memory right through Carry with result in ACC
Description Data in the specified Data Memory and the carry flag are rotated right by 1 bit. Bit 0 replaces the Carry bit and the original carry flag is rotated into bit 7. The rotated result is stored in the Accumulator and the contents of the Data Memory remain unchanged.
Operation 
\[
\begin{align*}
  ACC.i & \leftarrow [m].(i+1); \ (i = 0 \sim 6) \\
  ACC.7 & \leftarrow C \\
  C & \leftarrow [m].0
\end{align*}
\]
Affected flag(s) C

SBC A,[m] Subtract Data Memory from ACC with Carry
Description The contents of the specified Data Memory and the complement of the carry flag are subtracted from the Accumulator. The result is stored in the Accumulator. Note that if the result of subtraction is negative, the C flag will be cleared to 0, otherwise if the result is positive or zero, the C flag will be set to 1.
Operation 
\[
\begin{align*}
  ACC & \leftarrow ACC - [m] - \overline{C}
\end{align*}
\]
Affected flag(s) OV, Z, AC, C

SBCM A,[m] Subtract Data Memory from ACC with Carry and result in Data Memory
Description The contents of the specified Data Memory and the complement of the carry flag are subtracted from the Accumulator. The result is stored in the Data Memory. Note that if the result of subtraction is negative, the C flag will be cleared to 0, otherwise if the result is positive or zero, the C flag will be set to 1.
Operation 
\[
\begin{align*}
  [m] & \leftarrow ACC - [m] - \overline{C}
\end{align*}
\]
Affected flag(s) OV, Z, AC, C

SDZ [m] Skip if decrement Data Memory is 0
Description The contents of the specified Data Memory are first decremented by 1. If the result is 0 the following instruction is skipped. As this requires the insertion of a dummy instruction while the next instruction is fetched, it is a two cycle instruction. If the result is not 0 the program proceeds with the following instruction.
Operation 
\[
\begin{align*}
  [m] & \leftarrow [m] - 1 \\
  \text{Skip if } [m] = 0
\end{align*}
\]
Affected flag(s) None
**SDZA [m]**

**Description**
The contents of the specified Data Memory are first decremented by 1. If the result is 0, the following instruction is skipped. The result is stored in the Accumulator but the specified Data Memory contents remain unchanged. As this requires the insertion of a dummy instruction while the next instruction is fetched, it is a two cycle instruction. If the result is not 0, the program proceeds with the following instruction.

**Operation**

\[ \text{ACC} \leftarrow [m] - 1 \]

Skip if ACC = 0

**Affected flag(s)**
None

---

**SET [m]**

**Description**
Set Data Memory

Each bit of the specified Data Memory is set to 1.

**Operation**

\[ [m] \leftarrow \text{FFH} \]

**Affected flag(s)**
None

---

**SET [m].i**

**Description**
Set bit of Data Memory

Bit i of the specified Data Memory is set to 1.

**Operation**

\[ [m].i \leftarrow 1 \]

**Affected flag(s)**
None

---

**SIZ [m]**

**Description**
Skip if increment Data Memory is 0

The contents of the specified Data Memory are first incremented by 1. If the result is 0, the following instruction is skipped. As this requires the insertion of a dummy instruction while the next instruction is fetched, it is a two cycle instruction. If the result is not 0 the program proceeds with the following instruction.

**Operation**

\[ [m] \leftarrow [m] + 1 \]

Skip if [m] = 0

**Affected flag(s)**
None

---

**SIZA [m]**

**Description**
Skip if increment Data Memory is zero with result in ACC

The contents of the specified Data Memory are first incremented by 1. If the result is 0, the following instruction is skipped. The result is stored in the Accumulator but the specified Data Memory contents remain unchanged. As this requires the insertion of a dummy instruction while the next instruction is fetched, it is a two cycle instruction. If the result is not 0 the program proceeds with the following instruction.

**Operation**

\[ \text{ACC} \leftarrow [m] + 1 \]

Skip if ACC = 0

**Affected flag(s)**
None
<table>
<thead>
<tr>
<th>Instruction</th>
<th>Description</th>
<th>Operation</th>
<th>Affected flag(s)</th>
</tr>
</thead>
<tbody>
<tr>
<td><strong>SNZ [m].i</strong></td>
<td>Skip if bit i of Data Memory is not 0</td>
<td>Skip if [m].i ≠ 0</td>
<td>None</td>
</tr>
<tr>
<td><strong>SUB A,[m]</strong></td>
<td>Subtract Data Memory from ACC</td>
<td>ACC ← ACC − [m]</td>
<td>OV, Z, AC, C</td>
</tr>
<tr>
<td><strong>SUBM A,[m]</strong></td>
<td>Subtract Data Memory from ACC with result in Data Memory</td>
<td>[m] ← ACC − [m]</td>
<td>OV, Z, AC, C</td>
</tr>
<tr>
<td><strong>SUB A,x</strong></td>
<td>Subtract immediate data from ACC</td>
<td>ACC ← ACC − x</td>
<td>OV, Z, AC, C</td>
</tr>
<tr>
<td><strong>SWAP [m]</strong></td>
<td>Swap nibbles of Data Memory</td>
<td>[m].3~[m].0 ↔ [m].7~[m].4</td>
<td>None</td>
</tr>
<tr>
<td><strong>SWAPA [m]</strong></td>
<td>Swap nibbles of Data Memory with result in ACC</td>
<td>ACC.3 ~ ACC.0 ← [m].7 ~ [m].4</td>
<td>None</td>
</tr>
</tbody>
</table>

The immediate data specified by the code is subtracted from the contents of the Accumulator. Note that if the result of subtraction is negative, the C flag will be cleared to 0, otherwise if the result is positive or zero, the C flag will be set to 1.

The specified Data Memory is subtracted from the contents of the Accumulator. Note that if the result of subtraction is negative, the C flag will be cleared to 0, otherwise if the result is positive or zero, the C flag will be set to 1.

The low-order and high-order nibbles of the specified Data Memory are interchanged.

The low-order and high-order nibbles of the specified Data Memory are interchanged. The result is stored in the Accumulator. The contents of the Data Memory remain unchanged.
**Chapter 3  Instruction Definition**

**SZ [m]**  
Skip if Data Memory is 0  
Description: If the contents of the specified Data Memory is 0, the following instruction is skipped. As this requires the insertion of a dummy instruction while the next instruction is fetched, it is a two cycle instruction. If the result is not 0 the program proceeds with the following instruction.  
Operation: Skip if \([m] = 0\)  
Affected flag(s): None  

**SZA [m]**  
Skip if Data Memory is 0 with data movement to ACC  
Description: The contents of the specified Data Memory are copied to the Accumulator. If the value is zero, the following instruction is skipped. As this requires the insertion of a dummy instruction while the next instruction is fetched, it is a two cycle instruction. If the result is not 0, the program proceeds with the following instruction.  
Operation:  
\[
\text{ACC} \leftarrow [m]  
\]  
Affected flag(s): None  

**SZ [m].i**  
Skip if bit i of Data Memory is 0  
Description: If bit i of the specified Data Memory is 0, the following instruction is skipped. As this requires the insertion of a dummy instruction while the next instruction is fetched, it is a two cycle instruction. If the result is not 0, the program proceeds with the following instruction.  
Operation: Skip if \([m].i = 0\)  
Affected flag(s): None  

**TABRDC [m]**  
Read table (current page) to TBLH and Data Memory  
Description: The low byte of the program code (current page) addressed by the table pointer (TBLP) is moved to the specified Data Memory and the high byte moved to TBLH.  
Operation:  
\[
[m] \leftarrow \text{program code (low byte)}  
\text{TBLH} \leftarrow \text{program code (high byte)}  
\]  
Affected flag(s): None  

**TABRDL [m]**  
Read table (last page) to TBLH and Data Memory  
Description: The low byte of the program code (last page) addressed by the table pointer (TBLP) is moved to the specified Data Memory and the high byte moved to TBLH.  
Operation:  
\[
[m] \leftarrow \text{program code (low byte)}  
\text{TBLH} \leftarrow \text{program code (high byte)}  
\]  
Affected flag(s): None
<table>
<thead>
<tr>
<th>Instruction</th>
<th>Description</th>
<th>Operation</th>
<th>Affected flag(s)</th>
</tr>
</thead>
<tbody>
<tr>
<td>XOR A,[m]</td>
<td>Logical XOR Data Memory to ACC</td>
<td>ACC ← ACC &quot;XOR&quot; [m]</td>
<td>Z</td>
</tr>
<tr>
<td>XORM A,[m]</td>
<td>Logical XOR ACC to Data Memory</td>
<td>[m] ← ACC &quot;XOR&quot; [m]</td>
<td>Z</td>
</tr>
<tr>
<td>XOR A,x</td>
<td>Logical XOR immediate data to ACC</td>
<td>ACC ← ACC &quot;XOR&quot; x</td>
<td>Z</td>
</tr>
</tbody>
</table>
Chapter 4

Assembly Language and Cross Assembler

Assembly-Language programs are written as source files. They can be assembled into object files by the Holtek Cross Assembler. Object files are combined by the Cross Linker to generate a task file.

A source program is made up of statements and look up tables, giving directions to the Cross Assembler at assembly time or to the processor at run time. Statements are constituted by mnemonics (operations), operands and comments.

Notational Conventions

The following list describes the notations used by this document.

<table>
<thead>
<tr>
<th>Example of Convention</th>
<th>Description of Convention</th>
</tr>
</thead>
<tbody>
<tr>
<td>[[optional items]]</td>
<td>Syntax elements that are enclosed by a pair of brackets are optional. For example, the syntax of the command line is as follows:</td>
</tr>
<tr>
<td>[HASM [options] [filename [;] ] ]</td>
<td>In the above command line, options and semicolon; are both optional, but filename is required, except for the following case:</td>
</tr>
<tr>
<td>{choice1 | choice2}</td>
<td>Braces and vertical bars stand for a choice between two or more items. Braces enclose the choices whereas vertical bars separate the choices. Only one item can be chosen.</td>
</tr>
</tbody>
</table>
### Example of Convention

<table>
<thead>
<tr>
<th>Description of Convention</th>
</tr>
</thead>
<tbody>
<tr>
<td>Three dots following an item signify that more items with the same form may be entered. For example, the directive PUBLIC has the following form: PUBLIC name1 [,name2 [...]</td>
</tr>
</tbody>
</table>

Repeating elements...

In the above form, the three dots following name2 indicate that many names can be entered as long as each is preceded by a comma.

---

**Statement Syntax**

The construction of each statement is as follows:

\[
\text{name} \ [\text{operation}] \ [\text{operands}] \ [; \text{comment}]
\]

- All fields are optional.
- Each field (except the comment field) must be separated from other fields by at least one space or one tab character.
- Fields are not case-sensitive, i.e., lower-case characters are changed to upper-case characters before processing.

**Name**

Statements can be assigned labels to enable easy access by other statements. A name consists of the following characters:

- A–Z a–z 0–9 ? _ @

with the following restrictions:

- 0–9 cannot be the first character of a name
- ? cannot stand alone as a name
- Only the first 31 characters are recognized

**Operation**

The operation defines the statement action of which two types exist, directives and instructions. Directives give directions to the Cross Assembler, specifying the manner in which the Cross Assembler is to generate the object code at assembly time. Instructions, on the other hand, give directions to the processor. They are translated to object code at assembly time, the object code in turn controls the behavior of the processor at run time.

**Operand**

Operands define the data used by directives and instructions. They can be made up of symbols, constants, expressions and registers.
Comment

Comments are the descriptions of codes. They are used for documentation only and are ignored by the Cross Assembler. Any text following a semicolon is considered a comment.

Assembly Directives

Directives give direction to the Cross Assembler, specifying the manner in which the Cross Assembler generates object code at assembly time. Directives can be further classified according to their behavior as described below.

Conditional Assembly Directives

The conditional block has the following form:

```
IF
  statements
[ELSE
  statements]
ENDIF
```

→ Syntax

```
IF expression
IFE expression
```

• Description

The directives IF and IFE test the expression following them.

The IF directive grants assembly if the value of the expression is true, i.e. non-zero.

The IFE directive grants assembly if the value of the expression is false, i.e. zero.

• Example

```
IF debugcase
  ACC1 equ 5
  extern username: byte
ENDIF
```

In this example, the value of the variable ACC1 is set to 5 and the username is declared as an external variable if the symbol debugcase is evaluated as true, i.e. nonzero.

→ Syntax

```
IFDEF name
IFNDEF name
```

• Description

The directivesIFDEF andIFNDEF test whether or not the given name has been defined. TheIFDEF directive grants assembly only if the name is a label, a variable or a symbol. TheIFNDEF directive grants assembly only if the name has not yet been defined. The conditional assembly directives support a nesting structure, with a maximum nesting level of 7.

• Example

```
IFDEF buf_flag
  buffer DB 20 dup(?)
ENDIF
```

In this example, the buffer is allocated only if thebuf_flag has been previously defined.
File Control Directives

→ Syntax
   INCLUDE file-name
or
   INCLUDE "file-name"

• Description
   This directive inserts source codes from the source file given by file-name into the current source file during assembly. Cross Assembler supports at most 7 nesting levels.

• Example
   INCLUDE macro.def
   In this example, the Cross Assembler inserts the source codes from the file macro.def into the current source file.

→ Syntax
   PAGE size

• Description
   This directive specifies the number of the lines in a page of the program listing file. The page size must be within the range from 10 to 255, the default page size is 60.

• Example
   PAGE 57
   This example sets the maximum page size of the listing file to 57 lines.

→ Syntax
   .LIST
   .Nolist

• Description
   The directives .LIST and .NOLIST decide whether or not the source program lines are to be copied to the program listing file. .NOLIST suppresses copying of subsequent source lines to the program listing file. .LIST restores the copying of subsequent source lines to the program listing file. The default is .LIST.

• Example
   .NOLIST
   mov a, 1
   mov b1, a
   .LIST
   In this example, the two instructions in the block enclosed by .NOLIST and .LIST are suppressed from copying to the source listing file.

→ Syntax
   .LISTMACRO
   .NOLISTMACRO

• Description
   The directive .LISTMACRO causes the Cross Assembler to list all the source statements, including comments, in a macro. The directive .NOLISTMACRO suppresses the listing of all macro expansions. The default is .NOLISTMACRO.
Syntax

.LISTINCLUDE
.Nolistinclude

• Description
  The directive .LISTINCLUDE inserts the contents of all included files into the program listing. The directive .Nolistinclude suppresses the addition of included files. The default is .Nolistinclude.

Syntax

MESSAGE 'text-string'

• Description
  The directive MESSAGE directs the Cross Assembler to display the text-string on the screen. The characters in the text-string must be enclosed by a pair of single quotation marks.

Syntax

ERRMESSAGE 'error-string'

• Description
  The directive ERRMESSAGE directs the Cross Assembler to issue an error. The characters in the error-string must be enclosed by a pair of single quotation marks.

Program Directives

Syntax (comment)

; text

• Description
  A comment consists of characters preceded by a semicolon (;) and terminated by an embedded carriage-return/line-feed.

Syntax

name .SECTION [align] [combine] 'class'

• Description
  The .SECTION directive marks the beginning of a program section. A program section is a collection of instructions and/or data whose addresses are relative to the section beginning with the name which defines that section. The name of a section can be unique or be the same as the name given to other sections in the program. Sections with the same complete names are treated as the same section.
  The optional align type defines the alignment of the given section. It can be one of the following:
  
<table>
<thead>
<tr>
<th>Class</th>
<th>Description</th>
</tr>
</thead>
<tbody>
<tr>
<td>BYTE</td>
<td>uses any byte address (the default align type)</td>
</tr>
<tr>
<td>WORD</td>
<td>uses any word address</td>
</tr>
<tr>
<td>PARA</td>
<td>uses a paragraph address</td>
</tr>
<tr>
<td>PAGE</td>
<td>uses a page address</td>
</tr>
</tbody>
</table>

  For the CODE section, the byte address is in a single instruction unit. BYTE aligns the section at any instruction address, WORD aligns the section at any even instruction address, PARA aligns the section at any instruction address which is a multiple of 16, and PAGE aligns the section at any instruction address with a multiple of 256.
For DATA sections, the byte address is in one byte units (8 bits/byte). BYTE aligns the section at any byte address, WORD aligns the section at any even address, PARA aligns the section at any address which is a multiple of 16, and PAGE aligns the section at any address which is a multiple of 256.

The optional combine type defines the way of combining sections having the same complete name (section and class name). It can be any one of the following:

- COMMON
  Creates overlapping sections by placing the start of all sections with the same complete name at the same address. The length of the resulting area is the length of the longest section.

- AT address
  Causes all label and variable addresses defined in a section to be relative to the given address. The address can be any valid expression except a forward reference. It is an absolute address in a specified ROM/RAM bank and must be within the ROM/RAM range.

If no combine type is given, the section is combinative, i.e., this section can be concatenated with all sections having the same complete name to form a single, contiguous section.

The class type defines the sections that are to be loaded in the contiguous memory. Sections with the same class name are loaded into the memory one after another. The class name CODE is used for sections stored in ROM, and the class name DATA is used for sections stored in RAM. The complete name of a section consists of a section name and a class name. The named section includes all codes and data below (after) it until the next section is defined.

### Syntax

For DATA sections, the byte address is in one byte units (8 bits/byte). BYTE aligns the section at any byte address, WORD aligns the section at any even address, PARA aligns the section at any address which is a multiple of 16, and PAGE aligns the section at any address which is a multiple of 256.

The optional combine type defines the way of combining sections having the same complete name (section and class name). It can be any one of the following:

- COMMON
  Creates overlapping sections by placing the start of all sections with the same complete name at the same address. The length of the resulting area is the length of the longest section.

- AT address
  Causes all label and variable addresses defined in a section to be relative to the given address. The address can be any valid expression except a forward reference. It is an absolute address in a specified ROM/RAM bank and must be within the ROM/RAM range.

If no combine type is given, the section is combinative, i.e., this section can be concatenated with all sections having the same complete name to form a single, contiguous section.

The class type defines the sections that are to be loaded in the contiguous memory. Sections with the same class name are loaded into the memory one after another. The class name CODE is used for sections stored in ROM, and the class name DATA is used for sections stored in RAM. The complete name of a section consists of a section name and a class name. The named section includes all codes and data below (after) it until the next section is defined.

### Syntax

- **ROMBANK**
  
  **Syntax**
  ```
  ROMBANK banknum section-name [,section-name,...]
  ```

  **Description**
  This directive declares which sections are allocated to the specified ROM bank. The `banknum` specifies the ROM bank, ranging from 0 to the maximum bank number of the destination MCU. The `section-name` is the name of the section defined previously in the program. More than one section can be declared in a bank as long as the total size of the sections does not exceed the bank size of 8K words. If this directive is not declared, bank 0 is assumed and all CODE sections defined in this program will be in bank 0. If a CODE section is not declared in any ROM bank, then bank 0 is assumed.

- **RAMBANK**
  
  **Syntax**
  ```
  RAMBANK banknum section-name [,section-name,...]
  ```

  **Description**
  This directive is similar to **ROMBANK** except that it specifies the RAM bank, the size of RAM bank is 256 bytes.

- **END**
  
  **Syntax**
  ```
  END
  ```

  **Description**
  This directive marks the end of a program. Adding this directive to any included file should be avoided.
Syntax
ORG \texttt{expression}

- Description
  This directive sets the location counter to \texttt{expression}. The subsequent code and data offsets begin at the new offset specified by \texttt{expression}. The code or data offset is relative to the beginning of the section where the directive \texttt{ORG} is defined. The attribute of a section determines the actual value of offset, absolute or relative.

- Example
  \begin{verbatim}
  ORG 8
  mov A, 1
  \end{verbatim}

  In this example, the statement \texttt{mov A, 1} begins at location 8 in the current section.

Syntax
PUBLIC \texttt{name1 [,name2 [,...]]}
EXTERN \texttt{name1:type [,name2:type [, ...]]}

- Description
  The \texttt{PUBLIC} directive marks the variable or label specified by a name that is available to other modules in the program. The \texttt{EXTERN} directive, on the other hand, declares an external variable, label or symbol of the specified name and type. The type can be one of the four types: \texttt{BYTE}, \texttt{WORD} and \texttt{BIT} (these three types are for data variables), and \texttt{NEAR} (a label type and used by call or jmp).

- Example
  \begin{verbatim}
  PUBLIC start, setflag
  EXTERN tmpbuf:byte
  CODE .SECTION 'CODE'
  start:
    mov a, 55h
    call setflag
    ...
  setflag proc
    mov tmpbuf, a
    ret
  setflag endp
  end
  \end{verbatim}

  In this example, both the label \texttt{start} and the procedure \texttt{setflag} are declared as public variables. Programs in other sources may refer to these variables. The variable \texttt{tmpbuf} is also declared as external. There should be a source file defining a byte that is named \texttt{tmpbuf} and is declared as a public variable.
Syntax

name PROC
name ENDP

- Description
  The PROC and ENDP directives mark a block of code which can be called or jumped to from other modules. The PROC creates a label name which stands for the address of the first instruction of a procedure. The Cross Assembler will set the value of the label to the current value of the location counter.

- Example
  ```
toggle PROC
  mov tmpbuf, a
  mov a, 1
  xorm a, flag
  mov a, tmpbuf
  ret
  toggle ENDP
  ```

Syntax

[label:] DC expression1 [,expression2 [,...]]

- Description
  The DC directive stores the value of expression1, expression2 etc. in consecutive memory locations. This directive is used for the CODE section only. The bit size of the result value is dependent on the ROM size of the MCU. The Cross Assembler will clear any redundant bits; expression1 has to be a value or a label. This directive may also be employed to setup the table in the code section.

- Example
  ```
table1: DC 0128h, 025CH
  ```
  In this example, the Cross Assembler reserves two units of ROM space and also stores 0128H and 025CH into these two ROM units.

Data Definition Directives

An assembly language program consists of one or more statements and comments. A statement or comment is a composition of characters, numbers, and names. The assembly language supports integer numbers. An integer number is a collection of binary, octal, decimal, or hexadecimal digits along with an optional radix. If no radix is given, the Cross Assembler uses the default radix (decimal). The table lists the digits that can be used with each radix.

<table>
<thead>
<tr>
<th>Radix</th>
<th>Type</th>
<th>Digits</th>
</tr>
</thead>
<tbody>
<tr>
<td>B</td>
<td>Binary</td>
<td>01</td>
</tr>
<tr>
<td>O</td>
<td>Octal</td>
<td>01234567</td>
</tr>
<tr>
<td>D</td>
<td>Decimal</td>
<td>0123456789</td>
</tr>
<tr>
<td>H</td>
<td>Hexadecimal</td>
<td>0123456789ABCDEF</td>
</tr>
</tbody>
</table>
Chapter 4: Assembly Language and Cross Assembler

→ Syntax
- **DB** - Defines bytes.
  - Syntax: `name DB value1 [,value2 [, ...]]`
  - Description: These directives reserve the number of bytes/words specified by the repeated-count or reserve bytes/words only. `value1` and `value2` should be `?` due to the microcontroller RAM. The Cross Assembler will not initialize the RAM data. `DB` reserves a bit. The content `?` denotes uninitialized data, i.e., reserves the space of the data. The Cross Assembler will gather every 8 `DBIT` together and reserve a byte for these 8 `DBIT` variables.

- **DW** - Defines words.
  - Syntax: `name DW value1 [,value2 [, ...]]`

- **DBIT** - Defines bit.
  - Syntax: `name DBIT`

→ Example
- **DATA** - Defines data section.
  - Syntax: `DATA .SECTION 'DATA'
  - Example:
    ```
    DATA .SECTION 'DATA'
    tbuf DB ?
    checksum DW ?
    flag1 DBIT
    sbuf DB ?
    cflag DBIT
    ```
  - In this example, the Cross Assembler reserves byte location 0 for `tbuf`, location 1 and 2 for `checksum`, bit 0 of location 3 for `flag1`, location 4 for `sbuf` and bit 1 of location 3 for `cflag`.

→ Syntax
- **LABEL** - Defines label.
  - Syntax: `name LABEL {BIT|BYTE|WORD}
  - Description: The `name` with the data type has the same address as the following data variable.

→ Example
- **lab1** - Defines word label.
  - Syntax: `lab1 LABEL WORD
  - Example:
    ```
    lab1 LABEL WORD
d1 DB ?
d2 DB ?
    ```
  - In this example, `d1` is the low byte of `lab1` and `d2` is the high byte of `lab1`.

→ Syntax
- **EQU** - Defines equate.
  - Syntax: `name EQU expression
  - Description: The **EQU** directive creates absolute symbols, aliases, or text symbols by assigning an expression to `name`. An absolute symbol is a name standing for a 16-bit value; an alias is a name representing another symbol; a text symbol is a name for another combination of characters. The `name` must be unique, i.e., not having been defined previously. The `expression` can be an integer, a string constant, an instruction mnemonic, a constant expression, or an address expression.

→ Example
- **accreg** - Defines equate.
  - Syntax: `accreg EQU 5
  - Example:
    ```
    bmove EQU mov
    ```
  - In this example, the variable `accreg` is equal to 5, and `bmove` is equal to the instruction `mov`.

109
Macro Directives

Macro directives enable a block of source statements to be named, and then that name to be re-used in the source file to represent the statements. During assembly, the Cross Assembler automatically replaces each occurrence of the macro name with the statements in the macro definition.

A macro can be defined at any place in the source file as long as the definition precedes the first source line that calls this macro. In the macro definition, the macro to be defined may refer to other macros which have been previously defined. The Cross Assembler supports a maximum of 7 nesting levels.

→ Syntax
   
   name  MACRO [dummy-parameter [, ...]]
   statements
   ENDM

The Cross Assembler supports a directive LOCAL for the macro definition.

→ Syntax
   
   name  LOCAL dummy-name [, ...]

• Description
   
   The LOCAL directive defines symbols available only in the defined macro. It must be the first line following the MACRO directive, if it is present. The dummy-name is a temporary name that is replaced by a unique name when the macro is expanded. The Cross Assembler creates a new actual name for dummy-name each time the macro is expanded. The actual name has the form ??digit, where digit is a hexadecimal number within the range from 0000 to FFFF. A label should be added to the LOCAL directive when labels are used within the MACRO/ENDM block. Otherwise, the Cross Assembler will issue an error if this MACRO is referred to more than once in the source file.

In the following example, tmp1 and tmp2 are both dummy parameters, and are replaced by actual parameters when calling this macro. label1 and label2 are both declared LOCAL, and are replaced by ??0000 and ??0001 respectively at the first reference, if no other MACRO is referred. If no LOCAL declaration takes place, label1 and label2 will be referred to labels, similar to the declaration in the source program. At the second reference of this macro, a multiple define error message is displayed.

Delay  MACRO  tmp1, tmp2
   LOCAL    label1, label2
   mov    a, 70h
   mov    tmp1, a
   label1:
   mov    tmp2, a
   label2:
   clr    wdt1
   clr    wdt2
   sdz    tmp2
   jmp    label2
   sdz    tmp1
   jmp    label1
   ENDM
The following source program refers to the macro Delay ...

```
; T.ASM
; Sample program for MACRO.
.ListMacro
Delay MACRO tmp1, tmp2
  LOCAL label1, label2
  mov a, 70h
  mov tmp1, a
  label1:
    mov tmp2, a
  label2:
    clr wdt1
    clr wdt2
    sdz tmp2
    jmp label1
  sdz tmp1
  jmp label1
ENDM

data .section 'data'
  BCnt db ?
  SCnt db ?

code .section at 0 'code'
Delay BCnt, SCnt
end
```

The Cross Assembler will expand the macro Delay as shown in the following listing file. Note that the offset of each line in the macro body, from line 4 to line 17, is 0000. Line 24 is expanded to 11 lines and forms the macro body. In addition the formal parameters, tmp1 and tmp2, are replaced with the actual parameters, BCnt and SCnt, respectively.

```
File: T.asm     Holtek Cross-Assembler Version 2.80     Page 1

1  0000     ; T.ASM
2  0000     ; Sample program for MACRO.
3  0000     .ListMacro
4  0000     Delay MACRO tmp1, tmp2
5  0000     LOCAL label1, label2
6  0000     mov a, 70h
7  0000     mov tmp1, a
8  0000     label1:
9  0000     mov tmp2, a
10 0000     label2:
11 0000     clr wdt1
12 0000     clr wdt2
13 0000     sdz tmp2
14 0000     jmp label1
15 0000     sdz tmp1
16 0000     jmp label1
17 0000     ENDM
18 0000
19 0000     data .section 'data'
20 0000 00  BCnt db ?
21 0001 00  SCnt db ?
22 0002     code .section at 0 'code'
23 0000     Delay BCnt, SCnt
24 0000 0F70 1  mov a, 70h
24 0001 0090  R1  mov BCnt, a
24 0002 1  770000:
24 0002 0080  R1  mov SCnt, a
24 0003 1  770001:
24 0003 0001  1  clr wdt1
24 0004 0005  1  clr wdt2
24 0005 1780  R1  sdz SCnt
24 0006 2803  1  jmp 770001
24 0007 1780  R1  sdz BCnt
24 0008 2802  1  jmp 770000
25 0009     end

0 Errors
```
Assembly Instructions

The syntax of an instruction has the following form:

\[
\text{name:} \ \text{mnemonic} \ [\text{operand1},\text{operand2}] \ [\text{comment}]
\]

where

- **name:** → label name
- **mnemonic:** → instruction name (keywords)
- **operand1:** → registers
  - memory address
- **operand2:** → registers
  - memory address
  - immediate value

Name

A name is made up of letters, digits, and special characters, and is used as a label.

Mnemonic

Mnemonic is an instruction name dependent upon the type of the MCU used in the source program.

Operand, Operator and Expression

Operands (source or destination) are the argument defining values that are to be acted on by instructions. They can be constants, variables, registers, expressions or keywords. When using the instruction statements, care must be taken to select the correct operand type, i.e. source operand or destination operand. The dollar sign $ is a special operand, namely the current location operand.

An expression consists of many operands that are combined to describe a value or a memory location. The combined operators are evaluated at assembly time. They can contain constants, symbols, or any combination of constants and symbols that are separated by arithmetic operators.

Operators specify the operations to be performed while combining the operands of an expression. The Cross Assembler provides many operators to combine and evaluate operands. Some operators work with integer constants, some with memory values, and some with both. Operators handle the calculation of constant values that are known at the assembly time. The following are some operators provided by the Cross Assembler:

- Arithmetic operators: $+$ $-$ $*$ $/$ $(\text{MOD})$
- SHL and SHR operators
  - Syntax
    
    \[
    \text{expression SHR count}
    \]
    
    \[
    \text{expression SHL count}
    \]
The values of these shift bit operators are all constant values. The expression is shifted right SHR or left SHL by the number of bits specified by count. If bits are shifted out of position, the corresponding bits that are shifted in are zero-filled. The following are such examples:

- `mov A, 01110111b SHR 3 ; result ACC=00001110b`
- `mov A, 01110111b SHL 4 ; result ACC=01110000b`

- **Bitwise operators NOT, AND, OR, XOR**
  - **Syntax**
    - `NOT expression`
    - `expression1 AND expression2`
    - `expression1 OR expression2`
    - `expression1 XOR expression2`
  - **NOT** is a bitwise complement.
  - **AND** is a bitwise AND.
  - **OR** is a bitwise inclusive OR.
  - **XOR** is a bitwise exclusive OR.

- **OFFSET operator**
  - **Syntax**
    - `OFFSET expression`
  - The `OFFSET` operator returns the offset address of an `expression`. The `expression` can be a label, a variable, or other direct memory operand. The value returned by the `OFFSET` operator is an immediate operand.

- **LOW, MID and HIGH operator**
  - **Syntax**
    - `LOW expression`
    - `MID expression`
    - `HIGH expression`
  - The `LOW/MID/HIGH` operator returns the value of an `expression` if the result of the `expression` is an immediate value. The `LOW/MID/HIGH` operators will then take the low/middle/high byte of this value. But if the `expression` is a label, the `LOW/MID/HIGH` operator will take the values of the low/middle/high byte of the program count of this label.

- **BANK operator**
  - **Syntax**
    - `BANK name`
  - The `BANK` operator returns the bank number allocated to the section of the `name` declared. If the `name` is a label then it returns the rom bank number. If the `name` is a data variable then it returns the ram bank number. The format of the bank number is the same as the BP defined. For more information of the format please refer to the data sheets of the corresponding MCUs. (Note: The format of the BP might be different between MCUs.)

**Example 1:**

```
mov A, BANK start
mov BP, A
jmp start
```
Example 2:

```
mov A, BANK var
mov BP, A
mov A, OFFSET var
mov MP1, A
mov A, IAR1
```

- Operator precedence

<table>
<thead>
<tr>
<th>Precedence</th>
<th>Operators</th>
</tr>
</thead>
<tbody>
<tr>
<td>1 (Highest)</td>
<td>(), []</td>
</tr>
<tr>
<td>2</td>
<td>+, – (unary), LOW, MID, HIGH, OFFSET, BANK</td>
</tr>
<tr>
<td>3</td>
<td>*, /, %, SHL, SHR</td>
</tr>
<tr>
<td>4</td>
<td>+, – (binary)</td>
</tr>
<tr>
<td>5</td>
<td>&gt; (greater than), &gt;= (greater than or equal to), &lt; (less than), &lt;= (less than or equal to)</td>
</tr>
<tr>
<td>6</td>
<td>== (equal to), != (not equal to)</td>
</tr>
<tr>
<td>7</td>
<td>! (bitwise NOT)</td>
</tr>
<tr>
<td>8</td>
<td>&amp; (bitwise AND)</td>
</tr>
<tr>
<td>9 (Lowest)</td>
<td></td>
</tr>
</tbody>
</table>

Miscellaneous

Forward References

The Cross Assembler allows reference to labels, variable names, and other symbols before they are declared in the source code (forward named references). But symbols to the right of EQU are not allowed to be forward referenced.

Local Labels

A local label is a label with a fixed form such as $number. The number can be 0~29. The function of a local label is the same as a label except that the local label can be used repeatedly. The local label should be used between any two consecutive labels and the same local label name may used between other two consecutive labels. The Cross Assembler will transfer every local label into a unique label before assembling the source file. At most 30 local labels can be defined between two consecutive labels.

Example.

```
Label1: ; label
    $1: ; local label
        mov a, 1
        jmp $3
    $2: ; local label
        mov a, 2
        jmp $1
    $3: ; local label
        jmp $2

Label2: ; label
        jmp $1
    $0: ; local label
        jmp Label1
    $1: ; local label

Label3:
```

114
Reserved Assembly Language Words

The following tables list all reserved words used by the assembly language.

- **Reserved Names (directives, operators)**

<table>
<thead>
<tr>
<th>Symbol</th>
<th>Word 1</th>
<th>Word 2</th>
<th>Word 3</th>
</tr>
</thead>
<tbody>
<tr>
<td>$</td>
<td>DUP</td>
<td>INCLUDE</td>
<td>NOT</td>
</tr>
<tr>
<td>*</td>
<td>DW</td>
<td>LABEL</td>
<td>OFFSET</td>
</tr>
<tr>
<td>+</td>
<td>ELSE</td>
<td>.LIST</td>
<td>OR</td>
</tr>
<tr>
<td>–</td>
<td>END</td>
<td>.LISTINCLUDE</td>
<td>ORG</td>
</tr>
<tr>
<td>.</td>
<td>ENDIF</td>
<td>.LISTMACRO</td>
<td>PAGE</td>
</tr>
<tr>
<td>/</td>
<td>ENDM</td>
<td>LOCAL</td>
<td>PARA</td>
</tr>
<tr>
<td>=</td>
<td>ENDP</td>
<td>LOW</td>
<td>PROC</td>
</tr>
<tr>
<td>?</td>
<td>EQU</td>
<td>MACRO</td>
<td>PUBLIC</td>
</tr>
<tr>
<td>[ ]</td>
<td>ERRMESSAGE</td>
<td>MESSAGE</td>
<td>RAMBANK</td>
</tr>
<tr>
<td>AND</td>
<td>EXTERN</td>
<td>MID</td>
<td>ROMBANK</td>
</tr>
<tr>
<td>BANK</td>
<td>HIGH</td>
<td>MOD</td>
<td>.SECTION</td>
</tr>
<tr>
<td>BYTE</td>
<td>IF</td>
<td>NEAR</td>
<td>SHL</td>
</tr>
<tr>
<td>DB</td>
<td>IFDEF</td>
<td>.NOLIST</td>
<td>SHR</td>
</tr>
<tr>
<td>DBIT</td>
<td>IFE</td>
<td>.NOLISTINCLUDE</td>
<td>WORD</td>
</tr>
<tr>
<td>DC</td>
<td>IFNDEF</td>
<td>.NOLISTMACRO</td>
<td>XOR</td>
</tr>
</tbody>
</table>

- **Reserved Names (instruction mnemonics)**

<table>
<thead>
<tr>
<th>Instruction</th>
<th>Word 1</th>
<th>Word 2</th>
<th>Word 3</th>
</tr>
</thead>
<tbody>
<tr>
<td>ADC</td>
<td>HALT</td>
<td>RLCA</td>
<td>SUB</td>
</tr>
<tr>
<td>ADCM</td>
<td>INC</td>
<td>RR</td>
<td>SUBM</td>
</tr>
<tr>
<td>ADD</td>
<td>INCA</td>
<td>RRA</td>
<td>SWAP</td>
</tr>
<tr>
<td>ADDM</td>
<td>JMP</td>
<td>RRC</td>
<td>SWAPA</td>
</tr>
<tr>
<td>AND</td>
<td>MOV</td>
<td>RRCA</td>
<td>SZ</td>
</tr>
<tr>
<td>ANDM</td>
<td>NOP</td>
<td>SBC</td>
<td>SZA</td>
</tr>
<tr>
<td>CALL</td>
<td>OR</td>
<td>SBCM</td>
<td>TABRDC</td>
</tr>
<tr>
<td>CLR</td>
<td>ORM</td>
<td>SDZ</td>
<td>TABRDL</td>
</tr>
<tr>
<td>CPL</td>
<td>RET</td>
<td>SDZA</td>
<td>XOR</td>
</tr>
<tr>
<td>CPLA</td>
<td>RETI</td>
<td>SET</td>
<td>XORM</td>
</tr>
<tr>
<td>DAA</td>
<td>RL</td>
<td>SIZ</td>
<td></td>
</tr>
<tr>
<td>DEC</td>
<td>RLA</td>
<td>SIZA</td>
<td></td>
</tr>
<tr>
<td>DECA</td>
<td>RLC</td>
<td>SNZ</td>
<td></td>
</tr>
</tbody>
</table>

- **Reserved Names (registers names)**

<table>
<thead>
<tr>
<th>Register</th>
<th>Word 1</th>
<th>Word 2</th>
<th>Word 3</th>
</tr>
</thead>
<tbody>
<tr>
<td>A</td>
<td>WDT</td>
<td>WDT1</td>
<td>WDT2</td>
</tr>
</tbody>
</table>
Cross Assembler Options

The Cross Assembler options can be set via the Options menu Project command in HT-IDE3000. The Cross Assembler Options is located on the center part of the Project Option dialog box.

The symbols could be defined in the Define Symbol edit box.

→ Syntax

symbol1[=value1] [, symbol2[=value2] [, ...]]

→ Example,

ddebugflag=1, newver=3

The check box of the Generate listing file is used to decide whether the listing file should be generated or not. If the check box is checked, the listing file will be generated. Otherwise, it won't be generated.

Assembly Listing File Format

The Assembly Listing File contains the source program listing and summary information. The first line of each page is a title line which include company name, the Cross Assembler version number, source file name, date/time of assembly and page number.

Source Program Listing

Each line in the source program has the following syntax:

line-number offset [code] statement

→ Line-number is the number of the line starting from the first statement in the assembly source file (4 decimal digits).

→ The 2nd field – offset – is the offset from the beginning of the current section to the code (4 hexadecimal digits)

→ The 3rd field – code – is present only if the statement generates code or data (two hexadecimal 4-digit data)

The code shows the numeric value in hexadecimal if the value is known at assembly time. Otherwise, a proper flag will indicate the action required to compute the value. The following two flags may appear behind the code field.

R → relocatable address (Cross Linker must resolve)

E → external symbol (Cross Linker must resolve)

The following flag may appear before the code field

= → EQU or equal-sign directive

The following 2 flags may appear in the code field

---- → section address (Cross Linker must resolve)

nn[xx] → DUP expression: nn DUP(?)

→ The 4th field – statement – is the source statement shown exactly as it appears in the source file, or as expanded by a macro. The following flags may appear before a statement.

n → Macro-expansion nesting level

C → line from INCLUDE file
• Summary

<table>
<thead>
<tr>
<th>Line number (4 digits, right alignment)</th>
<th>Offset of code (4 digits)</th>
<th>Two 4-digits for opcode</th>
<th>External reference</th>
<th>Statement from included file</th>
<th>Relocatable name</th>
<th>Macro-expansion nesting level</th>
</tr>
</thead>
</table>

Summary of Assembly

The total warning number and total error number is the information provided at the end of the Cross Assembler listing file.

Miscellaneous

If any errors occur during assembly, each error message and error number will appear directly below the statement where the error occurred.
Example of assembly listing file

File: SAMPLE.ASM  Holtek Cross-Assembler Version 2.86  Page 1

1 0000                  page 60
2 0000                  message 'Sample Program 1'
3 0000
4 0000                  .listinclude
5 0000                  .listmacro
6 0000
7 0000                  #include "sample.inc"
1 0000                  C pa equ [12h]
2 0000                  C pac equ [13h]
3 0000                  C pb equ [14h]
4 0000                  C pbc equ [15h]
5 0000                  C pc equ [16h]
6 0000                  C pcc equ [17h]
7 0000                  C
8 0000
9 0000                  extern extlab : near
10 0000                  extern extbl : byte
11 0000
12 0000                  clrpb macro
13 0000
14 0000                  endm
15 0000
16 0000                  clrpa macro
17 0000
18 0000                  mov a, 00h
19 0000                  mov pa, a
20 0000
21 0000                  endm
22 0000
23 0000 00                  b1 db ?
24 0000 00                  b2 db ?
25 0002 00                  bit1 db 1t
26 0003
27 0000                  code .section "code"
28 0000 0F55                  mov a, 055h
29 0001 0080                  R mov b1, a
30 0002 0080                  E mov extbl, a
31 0003 0FAA                  mov a, 0Aah
32 0004 0093                  mov pac, a
33 0005
34 0005 0F00                  I mov a, 00h
35 0006 0092                  I mov [12h], a
36 0007 0093                  I clrpb
37 0007 1F14                  2 clr [14h]
38 0008 0700                  R mov a, b1
39 0009 0F00                  E mov a, bank extlab
40 000A 0F00                  E mov a, offset extbl
41 000B 2800                  E jmp extlab
42 000C 1234 5678               dw 1234h, 5678h, 0abcdh, 0ef12h
43 000C 1234h, 5678h, 0abcdh, 0ef12h
44 0000                  end
45
0 Errors

118
Part III

Development Tools
Chapter 5

MCU Programming Tools

To ease the process of application development, the importance and availability of supporting tools for microcontrollers cannot be underestimated. To support its range of MCUs, Holtek is fully committed to the development and release of easy to use and fully functional tools for its full range of devices. Known as the HT-IDE3000 development system, the software provides a user friendly Windows based approach for program editing and debugging while the hardware provides full real time emulation with multi functional trace, stepping and breakpoint functions. With a complete set of interface cards for its full device range and regular software Service Pack updates, the HT-IDE3000 ensures that designers have the best tools to maximize efficiency in the design and release of their microcontroller applications.

HT-IDE3000 Development Environment

The HT-IDE3000 (Holtek Integrated Development Environment) is a high performance integrated development environment designed around Holtek’s series of 8-bit microcontroller (MCU) chips. Incorporated within the system is the hardware and software tools necessary for rapid and easy development of applications based on the Holtek range of 8-bit MCUs. The key component within the HT-IDE3000 system is the HT-ICE or In-Circuit Emulator, capable of emulating the Holtek 8-bit MCU in real time, in addition to providing powerful debugging and trace features.

As for the software, the HT-IDE3000 provides a friendly workbench to ease the process of application program development, by integrating all of the software tools, such as editor, Cross Assembler, Cross Linker, library and symbolic debugger into a user friendly Windows based environment. In addition, the HT-IDE3000 provides a software simulator which is capable of simulating the behavior of Holtek’s 8-bit MCU range without using the HT-ICE. All fundamental functions of the HT-ICE hardware are valid for the simulator.

More detailed information on the HT-IDE3000 development system is contained within the HT-IDE3000 User’s Guide. Installed in conjunction with the HT-IDE3000 and to ensure that the development system contains information on new microcontrollers and software updates, Holtek provides regular HT-IDE3000 Service Packs. These Service Packs do not replace the HT-IDE3000 but are installed after the HT-IDE3000 system software has been installed.
Some of the special features provided by the HT-IDE3000 include:

→ **Emulation**
  - Real-time program instruction emulation

→ **Hardware**
  - Easy installation and usage
  - Either internal or external oscillator
  - Breakpoint mechanism
  - Trace functions and trigger qualification supported by trace emulation chip
  - Printer port for connecting the HT-ICE to a host computer
  - I/O interface card for connecting the user’s application board to the HT-ICE

→ **Software**
  - Windows based software utilities
  - Source program level debugger (symbolic debugger)
  - Workbench for multiple source program files (more than one source program file in one application project)
  - All tools are included for the development, debug, evaluation and generation of the final application program code (mask ROM file)
  - Library for the setting up of common procedures which can be linked at a later date to other projects.
  - Simulator can simulate and debug programs without connection to the HT-ICE hardware
  - Virtual Peripheral Manager (VPM) simulates the behavior of the peripheral devices.
  - LCD simulator simulates the behavior of the LCD panel.

**Holtek In-Circuit Emulator – HT-ICE**

Developed alongside the Holtek 8-bit microcontroller device range, the Holtek ICE is a fully functional in-circuit emulator for Holtek’s 8-bit microcontroller devices. Incorporated within the system are a comprehensive set of hardware and software tools for rapid and easy development of user applications. Central to the system is the in-circuit hardware emulator, capable of emulating all of Holtek’s 8-bit devices in real-time, while also providing a range of powerful debugging and trace facilities. Regarding software functions, the system incorporates a user-friendly Windows based workbench which integrates together functions such as program editor, Cross Assembler, Cross Linker and library manager. In addition, the system is capable of running in software simulation mode without connection to the HT-ICE hardware.

**HT-ICE Interface Card**

The interface cards supplied with the HT-ICE can be used for most applications, however it is possible for the user to omit the supplied interface card and design their own interface card. By including the necessary interface circuitry on their own interface card, the user has a means of directly connecting their target boards to the CN1 and CN2 connectors of the HT-ICE.
OTP Programmer

Holtek’s OTP devices are fully supported by a range of programmers. For engineering level OTP device programming, Holtek supplies its HandyWriter programming tool which provides a quick and efficient means for low volume OTP programming. More programmers from other suppliers are available which provide more efficient and higher volume production capability.

OTP Adapter Card

The Holtek OTP programmers are supplied with a standard Textool chip socket. The OTP Adapter Card is used to connect the Holtek OTP programmers to the various sizes of available OTP chip packages that are unable to use this supplied socket.

System Configuration

The HT-IDE3000 system configuration is shown below, in which the host computer is a Pentium compatible machine with Windows 95/98/2000/NT/XP or later. Note that if Windows 2000/NT/XP or later systems are used, then the HT-IDE3000 must be installed in the Supervisor Privilege mode.

The HT-IDE3000 system contains the following hardware components
- The HT-ICE box contains the emulator box with 1 printer port connector for connecting to the host machine, I/O signal connector and one power-on LED, Fig below
- I/O interface card for connecting the target board to the HT-ICE box
- Power Adapter, output 9V
- 25-pin D-type printer cable
- HandyWriter
I/O interface card

The I/O interface card (Fig. below) is a PCB which is used to connect the HT-ICE system to the user's target board. It provides the following functions:

- External clock source
- External signal trace input
- MCU socket pin assignment

The external clock source has two modes, RC & Crystal. For use with a crystal clock, short positions 2 and 3 on Jumper JP1. Otherwise, for an RC clock, short positions 1 and 2, and adjust the system frequency with VR1. Refer to the Tools/Mask Option Menu of the HT-IDE3000 User's Guide for the choice of the clock source and system frequency.

The 4 external signal trace inputs, marked as ET0 to ET3 at jumper location JP3, exist to help the user trace and digitize signals and analyze their behavior. For more information, refer to the chapters on Breakpoint and Trace the Application Program of Holtek's HT-IDE3000 User's Guide.

The MCU socket pin assignments in location U2, U3 are defined in the same manner as the pin assignment of the MCU series according to the Holtek 8-bit MCU datasheets. The VME connector at location CN1 is used to connect the I/O interface card to the HT-ICE.

Installation

System Requirement

The hardware and software requirements for installing HT-IDE3000 system are as follows:

- PC/AT compatible machine with Pentium or higher CPU
- SVGA color monitor
- At least 32M RAM for best performance
- CD ROM drive (for CD installation)
- At least 20M free disk space
- Parallel port to connect PC and HT-ICE
- Windows 95/98/NT/2000/XP

Windows 95/98/NT/2000/XP are trademarks of Microsoft Corporation.
Hardware Installation

- Step 1
  Plug the power adapter into the power connector of the HT-ICE
- Step 2
  Connect the target board to the HT-ICE by using the I/O interface card or flat cable
- Step 3
  Connect the HT-ICE to the host machine using the printer cable

The LED on the HT-ICE should now be lighted, if not, there is an error and your dealer should be contacted.

Caution
Exercise care when using the power adapter. Do not use a power adapter whose output voltage is not 9V, otherwise the HT-ICE may be damaged. It is strongly recommended that only the power adapter supplied by Holtek be used. First plug the power adapter to the power connector of the HT-ICE.

Software Installation

- Step 1
  Insert the HT-IDE3000 CD into the CD ROM drive, the following dialog will be shown.
Click <HT-IDE3000> button and the following dialog will be shown.

Click <HT-IDE3000> or <Service Pack> as you want.

Here's an Example of installing HT-IDE3000
Click <HT-IDE3000> button.

- Step 2
  Press the <Next> button to continue setup or press <Cancel> button to abort.
- **Step 3**
  The following dialog will be shown to ask the user to enter a directory name.

- **Step 4**
  Specify the path you want to install the HT-IDE3000 and click <Next> button.

- **Step 5**
  SETUP will copy all files to the specified directory.
• **Step 6**
  If the process is successful a dialog will be shown.

• **Step 7**
  Press Finish button and restart the computer system. Then you can run HT-IDE3000 now.
  SETUP will create four subdirectories, BIN, INCLUDE, LIB, SAMPLE, under the destination directory you specified in Step 4. The BIN subdirectory contains all the system executables (EXE), dynamic link libraries (DLL) and configuration files (CFG, FMT) for all supported MCU. The INCLUDE subdirectory contains all the include files (.H, .INC) provided by Holtek. The LIB subdirectory contains the library files (.LIB) provided by Holtek. The SAMPLE subdirectory contains some sample programs.

Note before running the HT-IDE3000 for the first time, the system will ask for company information as shown in the figure below. Select appropriate area and fill in the company name and ID. The HT-IDE3000 provider can be requested to supply an ID number.
OTP HandyWriter

The Holtek HandyWriter was specifically developed to program the range of Holtek OTP (One Time Programmable) MCU devices allowing users to easily and efficiently burn their programming code into the OTP devices. The advantages of this writer include its small and easy to manage size, ease of installation and easy to use special features. The structure of the writer includes the following components:

- Single 40-pin DIP TEXTOOL
- Single 25-pin printer port D-type female connector
- Single 96-pin VME connector

To use the HandyWriter requires the following:

- 16V power adapter with minimum current rating of 500mA. For best purposes please use the adapter included with the HandyWriter carton.
- IBM386 compatible or higher spec. PC
- Win95/98/NT/2000/XP Windows operating system
- HT-IDE3000 Integrated Development Environment
- If the writer is directly connected to the PC, the HT-ICE is not required.
Installation of the HandyWriter:

- To directly connect to a PC, use the printer cable to connect from the HandyWriter’s 25-pin D-type connector to the printer port of the PC as shown below:

![Diagram of HandyWriter connecting to PC directly]

- To connect via the HT-ICE, first connect the HandyWriter to the VME 96-pin socket CN1 on the HT-ICE then connect the HT-ICE to the PC’s printer port using the printer cable as shown in the figure below.

![Diagram of HandyWriter connecting to PC via HT-ICE]
Chapter 6

Quick Start

This chapter gives a brief description of using HT-IDE3000 to develop an application project.

Step 1 – Create a New Project

- Click on Project menu and select New command
- Enter your project name and select an MCU from the combo box
- Click OK button and the system will ask you to setup the mask options
- Setup all mask options and click Save button

Step 2 – Add Source Program Files to the Project

- Create your source files by using File/New command
- Write your program and save them with a file name, say TEST.ASM
- Click on Project menu and select Edit command
- An Edit Project dialog will ask you to add/delete files to/from the project
- Select a source file name, say TEST.ASM, and click Add button
- Click OK button after you setup all files in the project

Step 3 – Build the Project

- Click on Project menu and select Build command
- The system will assemble/compile all source files in the project
  - If there are some errors in the programs, double click on the error message line and the system will prompt you the position where the error happened.
  - If all the program files are error free, the system will create a Task file and download to the HT-ICE for debug.
- You may repeat this step before you finish debugging your programs

Step 4 – Transmit Code to Holtek

- Build the project for creating the .COD file
- Click on Project menu and select Print Option Table command
- Send the .COD file and the Option Approval Sheet to Holtek
Step 5 – Programming the OTP Device

- Build the project for creating the .OTP file
- Click on Tools menu and select HandyWriter command or use HT-HandyWriter in program group to program the OTP devices

The Programming and data flow is illustrated by the following diagram:
Appendix A

Device Characteristic Graphics

The following characteristic graphics depicts typical device behavior. The data presented here is a statistical summary of data gathered on units from different lots over a period of time. This is for information only and the figures were not tested during manufacturing.

In some of the graphs, the data exceeding the specified operating range are shown for information purposes only. The device will operate properly only within the specified range.
Typical RC Oscillator Frequency vs. V\textsubscript{DD}

Typical RC OSC vs. Temperature
Appendix A  Device Characteristic Graphics

$I_{OH}$ vs. $V_{OH}$, $V_{DD}=3V$

$I_{OH}$ vs. $V_{OH}$, $V_{DD}=5V$
Appendix A  Device Characteristic Graphics

Typical $R_{PH}$ vs. $V_{DD}$

![Graph showing $R_{PH}$ vs. $V_{DD}$]

Typical $V_{IH}, V_{IL}$ vs. $V_{DD}$ in -40°C to +85°C

![Graph showing $V_{IH}, V_{IL}$ vs. $V_{DD}$]

139
Appendix A  Device Characteristic Graphics

Typical $I_D$ vs. Frequency (External Clock, $T_a=40^\circ C$)

Typical $I_D$ vs. Frequency (External Clock, $T_a=0^\circ C$)
Typical $I_{DD}$ vs. Frequency (External Clock, $T_a=\pm 25^\circ C$)

Typical $I_{DD}$ vs. Frequency (External Clock, $T_a=\pm 85^\circ C$)
Typical $V_{LVR}$ vs. Temperature

![Typical $V_{LVR}$ vs. Temperature graph](image-url)
### 18-pin DIP (300mil) Outline Dimensions

<table>
<thead>
<tr>
<th>Symbol</th>
<th>Dimensions in mil</th>
</tr>
</thead>
<tbody>
<tr>
<td></td>
<td>Min.</td>
</tr>
<tr>
<td>A</td>
<td>895</td>
</tr>
<tr>
<td>B</td>
<td>240</td>
</tr>
<tr>
<td>C</td>
<td>125</td>
</tr>
<tr>
<td>D</td>
<td>125</td>
</tr>
<tr>
<td>E</td>
<td>16</td>
</tr>
<tr>
<td>F</td>
<td>50</td>
</tr>
<tr>
<td>G</td>
<td>—</td>
</tr>
<tr>
<td>H</td>
<td>295</td>
</tr>
<tr>
<td>I</td>
<td>335</td>
</tr>
<tr>
<td>α</td>
<td>0°</td>
</tr>
</tbody>
</table>
18-pin SOP (300mil) Outline Dimensions

<table>
<thead>
<tr>
<th>Symbol</th>
<th>Dimensions in mil</th>
</tr>
</thead>
<tbody>
<tr>
<td></td>
<td>Min.</td>
</tr>
<tr>
<td>A</td>
<td>394</td>
</tr>
<tr>
<td>B</td>
<td>290</td>
</tr>
<tr>
<td>C</td>
<td>14</td>
</tr>
<tr>
<td>C'</td>
<td>447</td>
</tr>
<tr>
<td>D</td>
<td>92</td>
</tr>
<tr>
<td>E</td>
<td>—</td>
</tr>
<tr>
<td>F</td>
<td>4</td>
</tr>
<tr>
<td>G</td>
<td>32</td>
</tr>
<tr>
<td>H</td>
<td>4</td>
</tr>
<tr>
<td>α</td>
<td>0°</td>
</tr>
</tbody>
</table>
### 24-pin SKDIP (300mil) Outline Dimensions

<table>
<thead>
<tr>
<th>Symbol</th>
<th>Dimensions in mil</th>
</tr>
</thead>
<tbody>
<tr>
<td></td>
<td>Min.</td>
</tr>
<tr>
<td>A</td>
<td>1235</td>
</tr>
<tr>
<td>B</td>
<td>255</td>
</tr>
<tr>
<td>C</td>
<td>125</td>
</tr>
<tr>
<td>D</td>
<td>125</td>
</tr>
<tr>
<td>E</td>
<td>16</td>
</tr>
<tr>
<td>F</td>
<td>50</td>
</tr>
<tr>
<td>G</td>
<td>—</td>
</tr>
<tr>
<td>H</td>
<td>295</td>
</tr>
<tr>
<td>I</td>
<td>345</td>
</tr>
<tr>
<td>α</td>
<td>0°</td>
</tr>
</tbody>
</table>
24-pin SOP (300mil) Outline Dimensions

<table>
<thead>
<tr>
<th>Symbol</th>
<th>Dimensions in mil</th>
</tr>
</thead>
<tbody>
<tr>
<td></td>
<td>394</td>
</tr>
<tr>
<td>B</td>
<td>290</td>
</tr>
<tr>
<td>C</td>
<td>14</td>
</tr>
<tr>
<td>C'</td>
<td>590</td>
</tr>
<tr>
<td>D</td>
<td>92</td>
</tr>
<tr>
<td>E</td>
<td>—</td>
</tr>
<tr>
<td>F</td>
<td>4</td>
</tr>
<tr>
<td>G</td>
<td>32</td>
</tr>
<tr>
<td>H</td>
<td>4</td>
</tr>
<tr>
<td>α</td>
<td>0°</td>
</tr>
</tbody>
</table>
### 28-pin SKDIP (300mil) Outline Dimensions

<table>
<thead>
<tr>
<th>Symbol</th>
<th>Dimensions in mil</th>
</tr>
</thead>
<tbody>
<tr>
<td>Min.</td>
<td>Nom.</td>
</tr>
<tr>
<td>A</td>
<td>1375</td>
</tr>
<tr>
<td>B</td>
<td>278</td>
</tr>
<tr>
<td>C</td>
<td>125</td>
</tr>
<tr>
<td>D</td>
<td>125</td>
</tr>
<tr>
<td>E</td>
<td>16</td>
</tr>
<tr>
<td>F</td>
<td>50</td>
</tr>
<tr>
<td>G</td>
<td>—</td>
</tr>
<tr>
<td>H</td>
<td>295</td>
</tr>
<tr>
<td>I</td>
<td>330</td>
</tr>
<tr>
<td>α</td>
<td>0°</td>
</tr>
</tbody>
</table>
### 28-pin SOP (300mil) Outline Dimensions

<table>
<thead>
<tr>
<th>Symbol</th>
<th>Dimensions in mil</th>
</tr>
</thead>
<tbody>
<tr>
<td></td>
<td>Min.</td>
</tr>
<tr>
<td>A</td>
<td>394</td>
</tr>
<tr>
<td>B</td>
<td>290</td>
</tr>
<tr>
<td>C</td>
<td>14</td>
</tr>
<tr>
<td>C'</td>
<td>697</td>
</tr>
<tr>
<td>D</td>
<td>92</td>
</tr>
<tr>
<td>E</td>
<td>—</td>
</tr>
<tr>
<td>F</td>
<td>4</td>
</tr>
<tr>
<td>G</td>
<td>32</td>
</tr>
<tr>
<td>H</td>
<td>4</td>
</tr>
<tr>
<td>(\alpha)</td>
<td>0°</td>
</tr>
</tbody>
</table>
### 48-pin SSOP (300mil) Outline Dimensions

<table>
<thead>
<tr>
<th>Symbol</th>
<th>Dimensions in mil</th>
</tr>
</thead>
<tbody>
<tr>
<td></td>
<td>Min.</td>
</tr>
<tr>
<td>A</td>
<td>395</td>
</tr>
<tr>
<td>B</td>
<td>291</td>
</tr>
<tr>
<td>C</td>
<td>8</td>
</tr>
<tr>
<td>C'</td>
<td>613</td>
</tr>
<tr>
<td>D</td>
<td>85</td>
</tr>
<tr>
<td>E</td>
<td>—</td>
</tr>
<tr>
<td>F</td>
<td>4</td>
</tr>
<tr>
<td>G</td>
<td>25</td>
</tr>
<tr>
<td>H</td>
<td>4</td>
</tr>
<tr>
<td>α</td>
<td>0°</td>
</tr>
</tbody>
</table>
Copyright © 2003 by HOLTEK SEMICONDUCTOR INC.

The information appearing in this handbook is believed to be accurate at the time of publication. However, Holtek assumes no responsibility arising from the use of the specifications described. The applications mentioned herein are used solely for the purpose of illustration and Holtek makes no warranty or representation that such applications will be suitable without further modification, nor recommends the use of its products for application that may present a risk to human life due to malfunction or otherwise. Holtek’s products are not authorized for use as critical components in life support devices or systems. Holtek reserves the right to alter its products without prior notification. For the most up-to-date information, please visit our web site at http://www.holtek.com.tw.