# The DDC112's Test Mode

By Jim Todsen

This application bulletin covers the DDC112's test mode. It elaborates on the explanation given in the data sheet and is written with the purpose of helping you use the test mode to its fullest capability. It does assume a basic understanding of the DDC112's operation. For a good introduction to the DDC112, see the DDC112's data sheet.

The organization of this application bulletin is as follows: First, an overview presents the basic operation of the test mode. Next are explanations of single packet and multiple packet test signals. Noise and linearity performance of the test mode are then covered followed by some special considerations for using the test mode. Finally, a program for investigating test mode performance is discussed with the listing for the program given afterwards.

## **OVERVIEW**

During normal operation, a sensor connected to each DDC112 input supplies signal current which is measured by DDC112 and converted to a 20-bit digital word. To help with debug during system development and to provide a good system-level diagnostic check, the DDC112 has a test mode which is enabled by the TEST pin. While in test mode, the inputs are disconnected and a test signal is measured instead. Figure 1 shows a simplified block diagram of the front end of the DDC112 including test mode circuitry.

You enter test mode by holding TEST HIGH while CONV toggles. Likewise, you exit test mode by holding TEST LOW while CONV toggles. The integration timing control for test mode is identical to that of normal operation: CONV still sets the side (A or B) and length of integration. As in normal operation, there is a continuous and non-continuous mode in test mode determined by the length of T<sub>INT</sub> (see Application Bulletin AB-131 for more information on the continuous and non-continuous modes).

 $C_{TEST}$  in Figure 1 supplies the test signal to the front-end integrators. Switches  $S_A$  and  $S_B$ , which in normal operation steer the sensor's current to the A or B integrator, are opened and  $S_{AT}$ ,  $S_{BT}$  and  $S_{VT}$  are used instead (these switches are only used in test mode).  $C_{TEST}$ , previously charged to  $V_{REF}$  by  $S_{VT}$ , is dumped to the appropriate side by either  $S_{AT}$  or  $S_{BT}$  depending on CONV. After the dumped charge from  $C_{TEST}$  is integrated by  $C_{INT}$ ,  $S_{AT}$  (or  $S_{BT}$ ) opens and  $C_{TEST}$  recharges to  $V_{REF}$  via  $S_{VT}$ .  $C_{TEST}$  is now ready for another charge dump. When the integration period is over, the ADC measures the signal integrated onto  $C_{INT}$  (the total test signal dumped by  $C_{TEST}$ ) just as in normal operation.  $S_{GND}$  shorts

the DDC112's inputs to ground during test mode to prevent any current from the sensor building a charge on the DDC's inputs.

 $C_{TEST}$  can be dumped once or multiple times onto the integrator during an integration period. The next two sections explain the timing necessary to control the number of dumps. Note that the magnitude of the test signal is not a direct function of the integration time  $T_{INT}$ , but rather the number of charge packet dumps. If the number of charge packets dumped remains constant, changing  $T_{INT}$  will not affect the test mode data.



FIGURE 1. Simplified Block Diagram of the DDC112's Front End.

#### SINGLE PACKET

When TEST is held HIGH, a single packet of charge from C<sub>TEST</sub> is dumped onto the integrator at the beginning of every test mode integration. Figure 2 shows the timing diagram and specifications for this case. The Integrator Status trace highlights the fact that the test mode functions the same as normal operation with respect to integration, measurement, auto-zero, reset, etc. The bottom trace of Figure 2 shows the output voltage on the side A integrator. CONV triggers the sequence that controls S<sub>AT</sub>, S<sub>BT</sub> and S<sub>VT</sub>. A short time after CONV toggles, S<sub>VT</sub> opens and switch S<sub>AT</sub> (or S<sub>AT</sub>) dumps C<sub>AT</sub> is charge onto the appropriate integra-

CONV triggers the sequence that controls  $S_{AT}$ ,  $S_{BT}$  and  $S_{VT}$ . A short time after CONV toggles,  $S_{VT}$  opens and switch  $S_{AT}$  (or  $S_{BT}$ ) dumps  $C_{TEST}$ 's charge onto the appropriate integrator side. After the charge transfers from  $C_{TEST}$  to  $C_{INT}$ , switch  $S_{AT}$  (or  $S_{BT}$ ) opens and  $C_{TEST}$  recharges to  $V_{REF}$  using  $S_{VT}$ . The test circuitry then waits until the next integration begins before dumping  $C_{TEST}$  to the other side. This cycle

continues until TEST is held LOW while CONV toggles. At that point, the test mode is over and normal operation resumes.

The size of each charge packet is  $(V_{REF})(C_{TEST})$ , approximately 13pC. If, for example, the range is set to 250pC, a single packet test signal equals approximately 5.2% of full scale. The absolute value of the charge packet can vary by 20% or more due to fabrication process variations. As previously mentioned, the readings during single packet test mode are independent of  $T_{INT}$  since the test mode supplies a fixed amount of charge ( $\approx 13pC$ ) each integration.

The A and B sides of a channel use the same  $C_{TEST}$ . Due to the nature of the switching arrangement, there is a small imbalance in the charge injection between sides A and B during test mode. This imbalance results in slightly different effective sizes for the side A and B test packets. Typical mismatch between side A and B charge packets is  $\approx 0.2 pC$ .



FIGURE 2. Single Packet Test Mode Timing.

The information provided herein is believed to be reliable; however, BURR-BROWN assumes no responsibility for inaccuracies or omissions. BURR-BROWN assumes no responsibility for the use of this information, and all use of such information shall be entirely at the user's own risk. Prices and specifications are subject to change without notice. No patent rights or licenses to any of the circuits described herein are implied or granted to any third party. BURR-BROWN does not authorize or warrant any BURR-BROWN product for use in life support devices and/or systems.

#### **MULTIPLE PACKET**

As described above, a packet of charge is dumped onto the integrator at the beginning of every integration during the test mode. Additional packets can be dumped creating a larger test signal. Multiple packets are dumped onto the integrator by toggling TEST during the integration as illustrated in Figure 3. The rising edge of TEST during a test mode integration triggers the dumping of an additional packet. Multiple toggles on TEST result in multiple packets being dumped. In order to stay in test mode, TEST must be HIGH when CONV toggles at the end of the integration.

Figure 3 shows the relevant timing specifications for multiple packet dumps. The specifications limit the maximum number of multiple packet dumps for a given  $T_{\rm INT}.$  Since the rising edges of TEST must be at least 5.4µs apart, the maximum number of dumps is  $T_{\rm INT}/5.4\mu s$  (round down when this number is a non-integer). This limitation is not a concern during the continuous mode. Even for  $T_{\rm INT}=500\mu s$ , over 90 dumps are allowed which corresponds to more than 1200pC of test signal—easily enough to reach full scale on any range. It is only for the short  $T_{\rm INT}$ 's of the noncontinous mode where you need to watch the limit.

#### NOISE

The noise of the test mode signal is very low. In fact, the noise performance in the test mode is usually better than what can be achieved in normal operation, even with very low sensor capacitance. Since the DDC112's inputs are disconnected during test mode, any coupling or interference on them (e.g., 60Hz pickup) will not affect the readings. This feature can be helpful in isolating the source of interference during system evaluation. Table I lists typical noise performance for the different internal ranges while in test mode with a single packet dump. The noise increases slightly for multiple packet dumps, particularly at the lower ranges.

| RANGE (Full-Scale Charge) | TYPICAL NOISE (rms) |
|---------------------------|---------------------|
| 1 (50pC)                  | 4.7ppm              |
| 2 (100pC)                 | 3.6ppm              |
| 3 (150pC)                 | 3.2ppm              |
| 4 (200pC)                 | 3.1ppm              |
| 5 (250pC)                 | 3.0ppm              |
| 6 (300pC)                 | 3.0ppm              |
| 7 (350pC)                 | 2.9ppm              |

TABLE I. Typical Noise Performance in Test Mode.



FIGURE 3. Multiple Packet Test Mode Timing.

#### **LINEARITY**

As with the noise performance, the linearity of the test signal is very good. Measurements made in the test mode show that the linearity of the test mode is usually limited by the performance of the DDC112 itself. That is, the linearity of the test mode is typically only slightly worse than the linearity measured in normal operation with a linear input signal. Figure 4 shows a comparison of integral non-linearity (INL) for normal operation of the DDC112 with a very linear input signal and for the test mode. INL is defined here as the deviation of the data from a line connected between the data's end points. Notice in Figure 4 how the test mode INL is only slightly worse than that of the DDC112 in normal operation. This indicates that the test mode signal is very linear and the linearity of the test mode is limited by the DDC112's inherent linearity.



FIGURE 4. Normal Mode and Test Mode INL (end-point line fit).

## SPECIAL CONSIDERATIONS

For the very best performance in test mode,  $T_{INT}$  might need to be an integer multiple of 3 CLK periods, depending on your PC board layout. For example, if CLK = 10MHz,  $T_{INT}$  might need to be a integer multiple of 300ns. If  $T_{INT}$  is not an integer multiple of 3 CLK periods, there can be excessive noise in the test mode. The reason for this is explained in Application Bulletin, AB-131. Basically, the internal slow clock (discussed in AB-131) is always left running while in test mode, even when  $T_{INT} > 4794$  CLK periods.

To prevent requiring  $T_{INT}$  to be an integer multiple of 3 CLK periods, pay close attention to the PC board layout. Place the bypass capacitors for  $V_{REF}$  and the power supplies as close to the pins as possible. Use a good analog ground plane with the analog pins on the DDC112 connecting directly to it. See the Evaluation Fixture layout for an example. In this layout, the test mode noise shows no need for  $T_{INT}$  to be an integer multiple of 3 CLK periods. That is, the same noise performance is achieved with  $T_{INT} = 500\mu s$ ,  $501\mu s$ ,  $502\mu s$ , etc.

### EXAMPLE PROGRAM

To help investigate the test mode, a Pascal program similar to the one in Application Bulletin, AB-125 is given in Listing 1. This program runs under DOS and controls the Evaluation Fixture hardware. Using the program, you can change the number of data points averaged per reading, full-scale range,  $T_{\rm INT}$ , and the number of test mode packets. With these variables set, you can then retrieve a reading. The displayed average for the retrieved reading is normalized to full scale = 1.0 and the rms noise is given in ppm.

A routine to measure the linearity of the test mode signal is also included in the program. The program determines the size of each packet by making an initial measurement with a single packet test signal. It then calculates the number of packets needed to reach full scale. A loop increments the packets up to this number while taking data at each step. Afterwards, an end-point line is connected between the first point and the final point. The program stores the linearity data in a file and reports the maximum difference between the data points and the end-point line as the INL.

### Listing 1

```
(*
                       BURR-BROWN CORPORATION
                                                         * )
( *
(*
                     Application Bulletin AB-135
( *
                       DDC112 Test Mode Program
(*
                      Written in Turbo Pascal 7.0
                                                         * )
( *
(* This program is designed to illustrate use of the DDC112's
(* test mode. This program requires the DDC112 to be connected to the
(* DDC112 Evaluation Fixture DEM-DDC112U-C. See AB-125 for more info
(* on the Evaluation Fixture.
(* The user can program the # of averages/reading, range,
(* tint and # of test packets from the menu. A single reading can be *)
(* retrieved or a complete measurement of the DDC112's linearity can be ^{\star})
(* taken. The results of the linearity measurement are saved to a file ^{\star})
(* Created by Jim Todsen 5/20/98
{$A+,B-,E+,F+,G-,I+,L-,N+,O+,R+,S+,V-,X+}
program DDC112TestMode;
uses Dos.CRT.Strings;
tvpe
Real4by1 = array[1..4] of real; {used to store 1a,1b,2a,2b data}
                             \{1->1a, 2->2a, 3->1b, 4->2b\}
Range
                     : longint;
Tint
                     : real;
Νt
                     : longint;
                     : longint;
PCPort
                    : longint;
pPrintPort
readdr, wraddr, strbaddr : word;
DataCkCode
                    : word;
SysCkCode
                     : word;
Ctrl1Code, Ctrl2Code
                    : word;
DXmitDelayCode
                    : word;
UserDDCCode
                     : word;
Yave, Yrms
                     : Real4by1;
procedure badBeep;
sound(100);
delay(200);
nosound;
end;
procedure goodBeep;
begin
sound(300);
delav(50);
nosound;
end;
```

```
procedure xilinxWrite(XilinxAddr, XilinxData : Byte);
{Write data to a specified address on the DDC112 Evaluation Fixture}
PORT[wraddr]:= $7F AND XilinxAddr;
Delav(2);
PORT[strbaddr] := 1;
Delav(1);
PORT[strbaddr] := 0;
Delay(1);
PORT[wraddr]:= $80 OR XilinxData;
Delay(2);
PORT[strbaddr] := 1;
Delay(1);
PORT[strbaddr] := 0;
Delay(1);
end;
procedure setRange(newRange: word);
{Set the DDC112 range}
begin
writeln(' setting Range = ',newRange);
Ctrl1Code := Ctrl1Code AND $78;
Ctrl1Code := Ctrl1Code OR newRange;
xilinxWrite(6,Ctrl1Code);
delay(1000); {wait for settling after change}
end;
procedure setTint(var newTint: real) ;
{Set the DDC112 Integration Time}
var
IntCountCode: longint;
begin
if newTint > 1500000 then newTint:= 1500000;
IntCountCode := round(newTint*10)-1;
                                               {10 MHz CLK}
writeln(' setting Tint = ',newTint:7:1);
xilinxWrite(7,(Ctrl2Code AND $7E));
                                                {hold conv}
                                {wait to insure CONV is stopped}
Delay(1000);
xilinxWrite(5,(IntCountCode AND $7F));
xilinxWrite(4,((IntCountCode SHR 7) AND $7F));
xilinxWrite(3,((IntCountCode SHR 14) AND $7F));
xilinxWrite(2,((IntCountCode SHR 21) AND $07));
xilinxWrite(7,(Ctrl2Code OR $1));
                                             {release conv}
end;
procedure setTstPckts(NumPckt: longint);
{*********************************
{Set the # of Test Packets dumped to the DDC112 each integration}
begin
if NumPckt > 0 then
 Ctrl1Code := Ctrl1Code OR $8;
                                    {Test ON}
 xilinxWrite(6,Ctrl1Code);
 if NumPckt = 1 then xilinxWrite(11,127)
                                   \{all \ 1's = 1 \ packet\}
 else xilinxWrite(11,(NumPckt-2));
end
else
 Ctrl1Code:= Ctrl1Code AND $F7;
                                   {Test OFF}
 xilinxWrite(6, (Ctrl1Code));
Delay(1000) ;{settling time after change}
```

```
procedure setPCport(PCport: word);
{***********************************
{Set the address of the PC's parallel port used for the Evaluation Fixture}
begin
if PCPort = 1 then pPrintPort := Ptr($40,$08)
else pPrintPort := Ptr($40,$0A);
wraddr:= pPrintPort^;
readdr:= wraddr + 1;
strbaddr:= wraddr + 2;
                     p{init Strobe at 0 (really, 1 out)}
PORT[strbaddr]:= 0;
end;
procedure xilinxRefresh;
{Refresh the Xilinx FPGAs on the Evaluation Fixture}
PORT[strbaddr] := 0;
                    {init port strobe high}
Delay(1);
xilinxWrite(0, DataCkCode);
xilinxWrite(1, SysCkCode);
xilinxWrite(6, CtrllCode);
xilinxWrite(7, Ctrl2Code);
xilinxWrite(8, ((DXmitDelayCode AND $180) SHR 7));
xilinxWrite(9, ((DXmitDelayCode AND $7F)));
xilinxWrite(10, (UserDDCCode SHL 1));
setRange(Range);
setTint(Tint);
setTstPckts(TPck);
goodBeep;
end;
procedure getInt(descrip: string; Min, Max: longint;
            var Value: longint; var Err: boolean);
{Get an INTEGER value from the user}
var
userinput
                       : string[25];
                       : integer;
check
temp
                      : longint;
begin
Err:= true;
Clrscr;
writeln;
writeln(' The current ',descrip,' is: ',Value);
writeln;
         Enter the new ',descrip,':');
writeln('
readln(userinput);
if userinput = 'x' then exit;
if userinput = 'X' then exit;
if userinput = '' then exit;
val(userinput,temp,check);
if check <> 0 then
begin
 badBeep;
 writeln(' *** Invalid ',descrip);
 writeln;
 delay(1000);
 exit;
 end
 else
 begin
  if (temp < min) or (temp > max) then
 begin
  badBeep;
  writeln;
  writeln(' *** Invalid ',descrip);
   writeln;
  delay(500);
  exit;
  end
  else
 begin
  Value := temp;
  Err := false;
  end;
end;
end;
```

```
procedure getReal(descrip: string; Min, Max: longint;
{Get a REAL value from the user}
var
userinput : string[25];
check : integer;
temp : real;
temp
begin
Err := true;
 ClrScr;
writeln;
 writeln(' The current ',descrip,' is: ',Value:9:3);
 writeln;
 writeln(' Enter the new ',descrip,':');
 readln(userinput);
 if userinput = \xspace'x' then exit;
if userinput = 'X' then exit;
if userinput = '' then exit;
 val(userinput,temp,check);
 if check <> 0 then
begin
 badBeep;
 writeln(' *** Invalid ',descrip);
 writeln;
 delay(500);
 exit;
 end
 else
 begin
  if (temp < min) or (temp > max) then
 begin
  badBeep;
  writeln;
   writeln(' *** Invalid ',descrip);
  writeln;
  delay(1000);
  exit;
  end
  else
 begin
  Value := temp;
  Err := false;
  end;
 end;
end;
{**********************************
procedure readData(var DRdError: Boolean);
{Readback "Nt" points from Evaluation Fixture; find Yave, Yrms, return status}
Var
FNum, N
                                : longint; { final number }
                               : integer; { data ready bit }
BusyBit3, Loop
 Nib, D, SideRd
                                : byte;
                               : bvte;
PortDataRd
 Ysum, Ysumsq
                                : array[1..4] of Extended;
 temp
                                : extended;
iRd, TRd, K, Count, DataTime, I, J : word;
 FScale,DelTime
                               : real;
 FirstTime
                                : boolean;
 timereq, SampleTime
                                : real;
numTints
                                : word;
begin
write(' filling Demo Board memory... ');
FirstTime := true; {flag to check for side A or B on first data point}
if Tint < 479.4 then {ncont mode}
begin
 timereq := 1501*6*1/10 - Tint; \{1501=\min \ \# \ \text{of slowclks to finish ncont meas}\}
 numTints := trunc(timereq / Tint)+1;
                                                    {# of tints skipped}
 if ((numTints MOD 2) <> 0) then numTints := numTints+1; {numTints must be even}
 SampleTime := Tint*(2+numTints);
 end {cont}
 else SampleTime := Tint*2;
```

```
I := 0;
while I < round(Nt*1.4 + 50) do {extra margin for safety}
delay(round(sampleTime/1e3));
 if KeyPressed AND (ReadKey = Chr(27)) then
begin
 goodBeep;
 I := round(Nt*1.4 + 50);
end;
I := I + 1;
end;
xilinxWrite(6, Ctrl1Code AND $5F);
                                           {enable readback}
{wait for PC Interface Board Data Valid}
K := 0;
DRdError := false;
BusyBit3 := (PORT[readdr] AND $08) SHR 3;
                                           {initialize bit 3}
While (BusyBit3 <> 1) AND (K <= 20000) do
begin
BusyBit3 := (PORT[readdr] AND $08) SHR 3;
 inc(K);
 if (K \mod 40) = 0 then delay(1);
if K = 20000 then DRdError := true;
end;
if DRdError = true then
begin
badBeep;
 clrscr;
 writeln;
 writeln('
           **** Data Retrieval Error ****');
 writeln;
 writeln('
           Check the following: ');
 writeln;
 writeln('
             DDC112 securely in socket');
            power supplies at 5V');
 writeln('
             cables properly connected');
demo board "refreshed"');
 writeln('
 writeln('
 writeln('
             PC port set correctly');
 writeln;
           writeln('
 writeln;
 writeln('
            Hit ENTER to continue ');
readln;
exit;
end
else
begin
 FScale := $FFFFF-$1000; {fullscale range = all ones - offset}
 for D := 1 to 4 do
begin
 Ysum[D] := 0;
  Ysumsq[D] := 0;
 end;
 writeln('retrieving data...');
 for N := Nt-1 downto 0 do
 for D := 1 to 4 do
  begin
   TRd := N + Nt*(D-1);
   Fnum := 0;
   PortDataRd := $6F;
                                   {decrement Xilinx RAM memory address}
   PORT[wraddr] := PortDataRd;
   PORT[strbaddr] := 1;
   PORT[strbaddr] := 0;
   PORT[strbaddr] := 1;
   PORT[strbaddr] := 0;
                                                  (PortDataRd := $6F;)}
   {prepare for RAM memory address
    for iRd := 1 to 6 do
                                                         {get 6 nibbles}
   begin
                                       {MSB, MSB-1,...,LSB nibble}
    PortDataRd := PortDataRd - $10;
    PORT[wraddr] := PortDataRd;
     PORT[strbaddr] := 1;
     PORT[strbaddr] := 0;
     {wait for PC Interface Board Data Valid}
     K := 0;
     BusyBit3 := (PORT[readdr] AND $08) SHR 3;
                                                            {init bit 3}
```

```
While (BusyBit3 <> 1) AND (K <= 20000) do
 begin
  BusyBit3 := (PORT[readdr] AND $08) SHR 3;
  inc(K);
  if (K \mod 40) = 0 then delay(1);
  if K = 20000 then
  begin
  DRdError := true;
   N := 0;
  end;
 end;
 nib := ((PORT[readdr] XOR $80) AND $F0) SHR 4;
  {XOR inverts bit 7 (-busy), AND F0 gets rid of 4 lower bits,
    shift right puts the data bits in lower nibble}
 if iRd = 1 then
 SideRd := ((nib AND $4) Xor $4) SHR 2; {bit3 lo ==> sideA}
 Fnum := Fnum SHL 4;
FNum := Fnum OR nib;
                                   {Add nibble to final number}
end;
if FirstTime then
begin
 if SideRd = 0 then FirstTime := false {first point is correct side}
 else
 begin
  FirstTime := false;
  for count := 1 to ((UserDDCCode-1)*2 + 1) do {read & throw away pts}
  begin
   PortDataRd := $6F;
                                 {decrement Xilinx RAM Memory Address}
   PORT[wraddr] := PortDataRd;
   PORT[strbaddr] := 1;
   PORT[strbaddr] := 0;
   PORT[strbaddr] := 1;
   PORT[strbaddr] := 0;
   {prepare for RAM memory address (PortDataRd := $6F;)}
   for iRd := 1 to 6 do
                                         {get 6 nibbles}
   begin
    PortDataRd := PortDataRd - $10;
                                        {MSB, MSB-1,...,LSB nibble}
    PORT[wraddr] := PortDataRd;
     PORT[strbaddr] := 1;
    PORT[strbaddr] := 0;
     {wait For PC Interface Board DValid}
     K := 0;
     BusyBit3 := (PORT[readdr] AND $08) SHR 3;
                                                 {init bit 3}
     While (BusyBit3 <> 1) AND (K <= 20000) do
    begin
      BusyBit3 := (PORT[readdr] AND $08) SHR 3;
     inc(K);
     if (K \mod 40) = 0 then delay(1);
     if K = 20000 then
     heain
      DRdError := true;
      N := 0;
     end;
    end;
   end;
  end;
  Fnum := 0;
  PortDataRd := $6F;
                              {decrement Xilinx RAM Memory Address}
  PORT[wraddr] := PortDataRd;
  PORT[strbaddr] := 1;
  PORT[strbaddr] := 0;
  PORT[strbaddr] := 1;
  PORT[strbaddr] := 0;
  {prepare for RAM memory address (PortDataRd := $6F;)}
  for iRd := 1 to 6 do
                                              {get 6 nibbles}
  begin
   PortDataRd := PortDataRd - $10; {MSB, MSB-1,...,LSB nibble}
   PORT[wraddr] := PortDataRd;
   PORT[strbaddr] := 1;
   PORT[strbaddr] := 0;
   {wait for PC Interface Board Data Valid}
   K := 0;
    BusyBit3 := (PORT[readdr] AND $08) SHR 3;
                                                {init bit 3}
```

```
While (BusyBit3 <> 1) AND (K <= 20000) do
       begin
        BusyBit3 := (PORT[readdr] AND $08) SHR 3;
        inc(K);
        if (K \mod 40) = 0 then delay(1);
        if K = 20000 then
        hegin
         DRdError := true;
         N := 0;
        end;
       nib := ((PORT[readdr] XOR $80) AND $F0) SHR 4;
        {XOR inverts bit 7 (-busy), AND F0 gets rid of 4 lower bits,
        shift right puts the data bits in lower nibble}
       if iRd = 1 then
        SideRd := ((nib AND $4) Xor $4) SHR 2; {bit3 lo ==> sideA}
       Fnum := Fnum SHL 4;
       FNum := Fnum OR nib;
                                        {add nibble to final number}
      end;
     end;
    end;
    Fnum := (FNum AND $000FFFFF); {set bits 31-21 := 0}
    Fnum := Fnum - $1000;
                              {subtract offset}
    Ysum[D] := Ysum[D] + Fnum;
    temp := Fnum;
    temp := temp * Fnum;
    Ysumsq[D] := Ysumsq[D] + Temp;
   end;
  end;
  for D := 1 to 4 do
  begin
  Yave[D] := Ysum[D]/(Nt*FScale);
  temp := Nt*Ysumsq[D]-(Ysum[D]*Ysum[D]);
  Yrms[D] := Sqrt(abs(temp)/(Nt*(Nt-1)))/Fscale;
  Yrms[D] := Yrms[D] * 1e6;
                          {convert to ppm}
 end;
end;
xilinxWrite(6,Ctrl1Code OR $20); {enable write to RAM}
procedure retrieveData(var err: boolean);
{Call readData then display results}
begin
readData(err);
if err = false then
heain
 ClrScr;
 writeln;
  writeln(' # of pts = ',Nt,' Range = ',Range,' Tint = ',Tint:8:1, ' Test pkts = ',TPck);
 writeln;
  writeln('
                 average
                          rms noise');
  writeln(' 1A: ',Yave[1]:11:7,' ',Yrms[1]:6:2,' ppm');
                             ',Yrms[2]:6:2,' ppm');
 writeln(' 2A: ',Yave[2]:11:7,'
                             ',Yrms[3]:6:2,' ppm');
  writeln(' 1B: ',Yave[3]:11:7,'
 writeln(' 2B: ',Yave[4]:11:7,' ',Yrms[4]:6:2,' ppm');
end;
procedure TestModeLinearity;
{Measure linearity of the DDC112's Test Mode}
{1st take 1 reading. Use results to determine # of steps to fullscale.}
{Fit data with endpoint line and report max deviation from that line.}
var
n.i
              : integer;
             : boolean;
err
            : string[8];
userName
             : string[12];
fileName
             : text;
F1
             : real;
tpacket.
             : word;
steps
Tdat
             : array[1..4,0..100] of real; {holds data for lin fit}
 m,b
              : real;
 INLpt, INLmax : real;
              : array[1..4] of real;
 INL
```

```
begin
 Clrscr;
 if Nt < 10 then
begin
  writeln;
  writeln('***
               The number of points / reading is low ***');
 writeln;
               Increase the number for better results ');
 badBeep;
 getInt('# of points',1,10000,Nt,err);
 end;
writeln('Enter a file name for Test Mode Linearity data (max 8 characters)');
 readln(userName);
 writeln;
fileName := userName + '.DAT';
 assign(F1,fileName);
 delav(1);
 {$I-} rewrite(F1); {$I+}
if ioresult <> 0 then
begin
  writeln('');
 writeln(' *** File I/O Problem, try a new name...');
 delay(400);
 badBeep;
  exit;
 end;
                              {set test pckts = 1 & collect 1st data}
writeln(' collecting test data 1');
 readData(err);
 if err = false then
                                  {use 1st data pt to calc # steps required}
 begin
 for n := 1 to 4 do Tdat[n,1] := Yave[n];
  tpacket := (Yave[1])*100; {calc packet size}
   writeln;
   writeln('*** Each test packet is ',tpacket:4:1,' % of full scale ***');
   steps := trunc(1/Yave[1]); {calc # of steps for fullscale}
   if steps > (trunc(Tint/5.4)) then
   hegin
    steps := trunc(Tint/5.4);
    writeln;
    writeln('Test Mode will not be able to reach full scale');
    writeln('with current Tint. Increasing Tint will allow');
    writeln('more test packets dumps during the integration.');
    badBeep;
   end;
  end
  else exit;
  for i := 2 to steps do {collect rest of data}
  begin
   writeln;
   writeln(' collecting test data ',i,' out of ',steps);
  setTstPckts(i);
  readData(err);
  if err = true then exit;
for n := 1 to 4 do Tdat[n,i] := Yave[n];
  setTstPckts(Tpck);
                              {restore user Test Package setting}
  ClrScr;
  writeln;
             Linearity Results: Endpoint Fit');
  writeln('
  writeln('
             (max deviation from endpoint line)');
  writeln;
```

```
for n := 1 to 4 do
                              {fit linearity}
  heain
   b := Tdat[n,1]; \qquad \{y = mx + b\}
   m := (Tdat[n,steps] - Tdat[n,1])/(steps-1);
   INLmax := 0;
   for i := 2 to (steps-1) do {INL for i=1 & steps =0 for endpoint fit}
   begin
   INLpt := Abs(Tdat[n,i] - (m*(i-1) + b));
   if INLpt > INLmax then INLmax := INLpt;
   end;
  INL[n] := 1e6* INLmax ; {convert to ppm}
  case n of
              INL for la = ',INL[1]:6:1,' ppm');
   1: writeln('
  2: writeln( 'INL for 2a = ',INL[2]:6:1,' ppm');
3: writeln( 'INL for 1b = ',INL[3]:6:1,' ppm');
4: writeln( 'INL for 2b = ',INL[4]:6:1,' ppm');
  end;
 end;
 writeln(F1,' Burr-Brown Corporation ');
writeln(F1,' Application Bulletin AB-135 ');
 writeln(F1,'DDC112 Test Mode linearity measurement');
 writeln(F1);
 writeln(F1,'# of averages/readings = ',Nt);
 writeln(F1,'Range = ',Range);
 writeln(F1,'Tint = ',Tint:7:1);
 writeln(F1);
 writeln(F1,'# Test');
 writeln(F1,'Packets 1a ave 2a ave 1b ave 2b ave');
 for i := 1 to steps do
 writeln(F1,i:3,' ',Tdat[1,i]:11:6,' ',Tdat[2,i]:11:6,' ',Tdat[3,i]:11:6,' ',Tdat[4,i]:11:6);
 writeln(F1);
 writeln(F1,' Endpoint Fit');
 writeln(F1,' INL(ppm): ',INL[1]:8:1, INL[2]:12:1, INL[3]:12:1, INL[4]:12:1);
goodBeep;
writeln;
           Hit ENTER to continue...');
writeln('
readln;
end;
var
userinput
                      : string[10];
                      : boolean;
goodbye,done
 err
choice
                      : word;
                       : integer; : real;
check
 minTint
begin
       := 5;
 Range
       := 500;
 Tint
       := 500;
 Nt
        := 1;
 TPck
PCPort := 1;
DataCkCode
                       := 2;
                       := 0;
:= 0;
 SysCkCode
 DXmitDelayCode
                      := 1;
UserDDCCode
 Ctrl1Code
                      := $20;
                                           {initially Read, internal DCLk}
                      := 1;
:= Ptr($40,$08);
                                           {demo bd CONV}
 Ctrl2Code
                                           {default: PC port=1}
 pPrintPort
 wraddr
                      := pPrintPort^;
                      := wraddr + 1;
:= wraddr + 2;
 readdr
 strbaddr
 PORT[strbaddr] := 0;
                                           {init Strobe at 0 (really, 1 out)}
```

```
xilinxRefresh;
Clrscr;
goodbye := false;
while (goodbye <> true) do
begin
 ClrScr;
 writeln;
 writeln('-----');
 writeln(' | Burr-Brown Corporation |');
writeln(' | Application Bulletin AB1XX |');
writeln(' | ');
 writeln('
                     (');
           DDC112 Test Mode Evaluation Program');
 writeln('
 writeln;
 writeln('
           # points/reading = `,Nt);
                         Range = ',Range);
Tint = ',Tint:7:1);
 writeln('
 writeln('
           # test packets = ',TPck);
 writeln('
 writeln;
 writeln;
 writeln('
           ----- ');
 writeln(' Choose one of the following ');
 writeln('
           writeln;
 writeln('

    Refresh Demo Board');

 writeln;
            2) Set # points averaged/reading');
 writeln('
 writeln('
           Set Range');
            4) Set Tint');
 writeln('
 writeln('
            5) Set # of Test Mode packets');
 writeln('
            Set PC port');
 writeln;
             7) Take single reading');
 writeln('
           8) Measure Linearity of Test Mode');
 writeln('
            9) exit program');
 writeln('
 writeln;
 readln(userinput);
 val(userinput,choice,check);
 if check <> 0 then badBeep
 else
 begin
  if (choice < 0) or (choice > 9) then badBeep
  else
   case choice of
    1: begin
        writeln;
        writeln('Refreshing Demo Board...');
        xilinxRefresh;
     2: getInt('# of points',1,10000,Nt,err);
     3: begin
        getInt('Range',0,7,Range,err);
        if err = false then setRange(Range);
       end;
     4: begin
        getReal('Tint',0,1500000,Tint,err);
        if err = false then setTint(Tint);
```

```
5: begin
      getInt('Test Mode Packets',0,128,TPck,err);
      if err = false then
      begin
       if TPck > Trunc(Tint/5.4) then {allow 5.4us per packet}
       begin
        writeln;
        writeln(' # of Test Packets is too high for current Tint);
        minTint := TPck*5.4;
        writeln(' Tint must be > ',minTint:7:1,' us for ',TPck,' packets.');
        TPck := Trunc(Tint/5.4);
        writeln(' Setting # of Test Packets to ',TPck);
        writeln;
        writeln('
                   Hit ENTER to continue...');
        BadBeep;
        readln;
      end;
      setTstPckts(TPck);
     end;
    end;
   6: begin
      getInt('PC port',1,2,PCport,err);
      setPCport(PCport);
     end;
   7: begin
      done := false;
      while done = false do
      begin
       retrieveData(err);
       if err = true then done := true
       else
       begin
        writeln;
        writeln(' enter "r" to repeat measurement');
        readln(userinput);
        if (userinput <> 'r') AND (userinput <> 'R')
        then done := true;
       end;
      end;
     end;
   8: TestModeLinearity;
  9: goodbye := true;
 end;
end;
```

end; end.

#### **IMPORTANT NOTICE**

Texas Instruments and its subsidiaries (TI) reserve the right to make changes to their products or to discontinue any product or service without notice, and advise customers to obtain the latest version of relevant information to verify, before placing orders, that information being relied on is current and complete. All products are sold subject to the terms and conditions of sale supplied at the time of order acknowledgment, including those pertaining to warranty, patent infringement, and limitation of liability.

TI warrants performance of its semiconductor products to the specifications applicable at the time of sale in accordance with TI's standard warranty. Testing and other quality control techniques are utilized to the extent TI deems necessary to support this warranty. Specific testing of all parameters of each device is not necessarily performed, except those mandated by government requirements.

Customers are responsible for their applications using TI components.

In order to minimize risks associated with the customer's applications, adequate design and operating safeguards must be provided by the customer to minimize inherent or procedural hazards.

TI assumes no liability for applications assistance or customer product design. TI does not warrant or represent that any license, either express or implied, is granted under any patent right, copyright, mask work right, or other intellectual property right of TI covering or relating to any combination, machine, or process in which such semiconductor products or services might be or are used. TI's publication of information regarding any third party's products or services does not constitute TI's approval, warranty or endorsement thereof.

Copyright © 2000, Texas Instruments Incorporated