

# ClockCounterClock

# Reference Manual

| Product Info    |            |
|-----------------|------------|
| Product Manager | Sven Meier |
| Author(s)       | Sven Meier |
| Reviewer(s)     | -          |
| Version         | 1.7        |
| Date            | 27.08.2020 |



# **Copyright Notice**

Copyright © 2018 NetTimeLogic GmbH, Switzerland. All rights reserved. Unauthorized duplication of this document, in whole or in part, by any means, is prohibited without the prior written permission of NetTimeLogic GmbH, Switzerland.

All referenced registered marks and trademarks are the property of their respective owners

## Disclaimer

The information available to you in this document/code may contain errors and is subject to periods of interruption. While NetTimeLogic GmbH does its best to maintain the information it offers in the document/code, it cannot be held responsible for any errors, defects, lost profits, or other consequential damages arising from the use of this document/code.

NETTIMELOGIC GMBH PROVIDES THE INFORMATION, SERVICES AND PRODUCTS AVAILABLE IN THIS DOCUMENT/CODE "AS IS," WITH NO WARRANTIES WHATSOEVER. ALL EXPRESS WARRANTIES AND ALL IMPLIED WARRANTIES, INCLUDING WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE, AND NON-INFRINGEMENT OF PROPRIETARY RIGHTS ARE HEREBY DISCLAIMED TO THE FULLEST EXTENT PERMITTED BY LAW. IN NO EVENT SHALL NETTIMELOGIC GMBH BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, CONSEQUENTIAL, SPECIAL AND EXEMPLARY DAMAGES, OR ANY DAMAGES WHATSOEVER, ARISING FROM THE USE OR PERFORMANCE OF THIS DOCUMENT/CODE OR FROM ANY INFORMATION, SERVICES OR PRODUCTS PROVIDED THROUGH THIS DOCUMENT/CODE, EVEN IF NETTIMELOGIC GMBH HAS BEEN ADVISED OF THE POSSIBILITY OF SUCH DAMAGES.

IF YOU ARE DISSATISFIED WITH THIS DOCUMENT/CODE, OR ANY PORTION THEREOF, YOUR EXCLUSIVE REMEDY SHALL BE TO CEASE USING THE DOCUMENT/CODE.



#### Overview

NetTimeLogic's Clock (CLK) Counter Clock is a full hardware (FPGA) only implementation of an adjustable counter clock with PI servo loop and spread adjustment. The whole protocol algorithms and calculations are implemented in the core, no CPU is required. This allows running synchronization completely independent and standalone from the user application. The Adjustable Counter Clock has multiple selectable input adjustments: PTP, PPS, TOD, IRIG, RTC and REG. The time and settings can be configured either by signals or by an AXI4Light-Slave Register interface.

# **Key Features:**

- 32 bit second and 32 bit nanosecond counter clock with fractional extension
- Provides time for all other cores
- 1 millisecond pulse generator aligned with the counter clock
- Allows non-integer clock periods (fractions)
- Multiplexing of multiple adjustment inputs
- 6 different adjustment sources: PTP, PPS, TOD, IRIG, RTC and REG
- Evenly spread offset and drift correction over time (offset might be set hard in case of large offsets)
- Hard setting of time possible
- Individual hardware only PI servo loops for offset and drift correction (PI parameters individually configurable)
- Runtime changeable PI parameters
- Offset correction: min 1 ns/s, max 0.5 s/s
- Drift correction: min 1 ns/s, max 0.05 s/s
- Conversion of fractional adjustments into even spread clock adjustments
- AXI4 Light register set or static configuration
- Get time and set time as well as adjusting of offset and drift possible via registers as well.
- External clock source selection
- Correction logging for Offset and Drift
- Linux Driver (PHC)



# **Revision History**

This table shows the revision history of this document.

| Version | Date       | Revision                                                                    |
|---------|------------|-----------------------------------------------------------------------------|
| 0.1     | 28.12.2015 | First draft                                                                 |
| 1.0     | 13.05.2016 | First release                                                               |
| 1.1     | 07.06.2016 | Added structured types section                                              |
| 1.2     | 24.02.2017 | Changed SNTP to RTC, added status interface and ExtSelect and CoreInfo type |
| 1.3     | 20.12.2017 | Added Fractional multiplication                                             |
| 1.4     | 09.01.2018 | Added Holdover                                                              |
| 1.5     | 09.03.2018 | Added Driver                                                                |
| 1.6     | 12.06.2018 | Added Dynmic PI Servo Paramters and Loging registers                        |
| 1.7     | 27.08.2020 | Updated Clock Select Register                                               |

Table 1: Revision History



# Content

| 1   | INTRODUCTION                 | 8  |
|-----|------------------------------|----|
| 1.1 | Context Overview             | 8  |
| 1.2 | Function                     | 9  |
| 1.3 | Architecture                 | 10 |
| 2   | CLOCK BASICS                 | 12 |
| 2.1 | Digital Counter Clock        | 12 |
| 2.2 | Drift and Offset adjustments | 12 |
| 3   | REGISTER SET                 | 14 |
| 3.1 | Register Overview            | 14 |
| 3.2 | Register Descriptions        | 16 |
| 3.2 | 2.1 General                  | 16 |
| 4   | DESIGN DESCRIPTION           | 37 |
| 4.1 | Top Level - Clk Clock        | 37 |
| 4.2 | Design Parts                 | 50 |
| 4.2 | 2.1 Clock Selector           | 50 |
| 4.2 | 2.2 Clock Adjuster           | 55 |
|     | 2.3 Clock Counter            | 61 |
|     | 2.4 Clock Timer              | 63 |
| 4.2 | 2.5 Registerset              | 65 |
| 4.3 | Configuration example        | 70 |
| 4.3 | 3.1 Static Configuration     | 70 |
| 4.3 | 3.2 AXI Configuration        | 70 |
| 4.4 | Clocking and Reset Concept   | 72 |
| 4.4 | 4.1 Clocking                 | 72 |
| 4.4 | 4.2 Reset                    | 72 |
|     |                              |    |



| 5   | RESOURCE USAGE         | 74 |
|-----|------------------------|----|
| 5.1 | Altera (Cyclone V)     | 74 |
| 5.2 | Xilinx (Artix 7)       | 74 |
| 6   | DELIVERY STRUCTURE     | 75 |
| 7   | TESTBENCH              | 76 |
| 7.1 | Run Testbench          | 76 |
| 8   | REFERENCE DESIGNS      | 77 |
| 8.1 | Altera: Terasic SocKit | 77 |
| 8.2 | Xilinx: Digilent Arty  | 78 |



# **Definitions**

| Definitions   |                                                             |
|---------------|-------------------------------------------------------------|
| Counter Clock | A counter based clock that count in the period of its fre-  |
| Counter Clock | quency in nanoseconds                                       |
| PI Servo Loop | Proportional-Integral servo loop, allows for smooth correc- |
| FI Selvo Loop | tions                                                       |
| Offset        | Phase difference between clocks                             |
| Drift         | Frequency difference between clocks                         |

Table 2: Definitions

# **Abbreviations**

| Abbreviations |                                                |
|---------------|------------------------------------------------|
| AXI           | AMBA4 Specification (Stream and Memory Mapped) |
| IRQ           | Interrupt, Signaling to e.g. a CPU             |
| PPS           | Pulse Per Second                               |
| TS            | Timestamp                                      |
| CLK           | Clock                                          |
| CC            | Counter Clock                                  |
| ТВ            | Testbench                                      |
| LUT           | Look Up Table                                  |
| FF            | Flip Flop                                      |
| RAM           | Random Access Memory                           |
| ROM           | Read Only Memory                               |
| FPGA          | Field Programmable Gate Array                  |
| VHDL          | Hardware description Language for FPGA's       |

Table 3: Abbreviations



#### 1 Introduction

#### 1.1 Context Overview

The Adjustable Counter Clock is meant as a co-processor handling clock adjustments and a time generator.

It is designed to work with all synchronization cores from NetTimeLogic. All correction inputs are multiplexed and selectable. For each correction input it allows drift, offset and time corrections. The offset and drift values are filtered via a PI servo loop and evenly spread over time to allow smooth corrections.

It is the source of time for all other cores from NetTimelogic, not only the synchronization cores which will in the end adjust the Counter Clock but also for the Signal Timestampers and Signal Generators which work aligned with this Counter Clock. It contains an AXI4Light slave for configuration, time setting, getting and adjusting from a CPU, this is however not required since the Adjustable Counter Clock can also be configured statically via signals/constants directly from within the FPGA.



Figure 1: Context Block Diagram



#### 1.2 Function

The adjustable counter clock is, as the name says already, a counter clock in the second and nanosecond format that can be frequency and phase adjusted. It adds the clock period it runs on in nanoseconds every clock cycle. If the clock period is a non-integer value e.g. for 66MHz => 15.1515... ns the fractional part of the nanoseconds can be entered as a fraction e.g. for 66MHz => 10/66 => numerator = 10 and denumerator = 66. Every clock cycle a fraction counter is adding the numerator with an overflow at the denumerator value. At each overflow of the fraction counter it will add one additional nanosecond to the nanoseconds counter. This way no error is introduced because of non-integer periods. The nanosecond counter has an overflow at 1000000000 nanoseconds. At each overflow of the nanosecond counter it will add an additional second to the seconds counter.

For adjustments additional nanoseconds can be added or subtracted from the standard period to adjust frequency and phase and the time can be overwritten to hard set the clock to a new time when small offset adjustments are not suitable e.g. for the startup phase where a jump from the 1.1.1970 to the present is required. To minimize the counter widths in other cores e.g. for timeouts or other time periods which don't need nanosecond resolution a timer is used which generates a counter clock aligned timer event every millisecond e.g. for a period count of 1 second only 10 bits instead of 30 bits are needed.

An adjustment block takes the offset, drift and time adjustments as inputs and converts them into a combined adjustment which is then applied to the counter clock. The offset and drift is converted into evenly spread adjustments which allows smooth corrections on the clock without time jumps. E.g. a drift of 1ppm is adjusted as 1 ns every 5000000 clock cycles at 50MHz. Again the adjustment is made with fractional counter for minimal computational error introduction (max computational error 1ns/s). In parallel to the correction a quality computation done to mark the in-sync state of the clock if corrections are below a certain threshold for at least 5 corrections.

Before the adjustments are converted for the counter clock, the drift and offset adjustments are passing a PI servo loop for filtering and smoother adjustments. The PI servo loop parameter for the drift and offset adjustments can be chosen individually, since frequency changes might happen quite slowly where offset adjustments probably shall be done much faster. The PI servo loops results are feed back to the other cores for correcting of the next adjustments.

Since the Adjustable Counter Clock is the heart of a synchronization solution it can take several adjustment inputs from different cores as input. Only one adjustment



input is taken as source for corrections at the time. From the Registerset the multiplexer gets the selection of the current input or if in external mode a selection from outside can be done.

The Registerset allows reading the in-sync state, taking a snapshot of the time, overwriting the current time for e.g. the startup phase and also allows adjusting the drift and offset as an additional input to the multiplexer.

#### 1.3 Architecture

The core is split up into different functional blocks for reduction of the complexity, modularity and maximum reuse of blocks. The interfaces between the functional blocks are kept as small as possible for easier understanding of the core.



Figure 2: Architecture Block Diagram

#### **Register Set**

This block allows reading status values and writing configuration.

#### Mux

This block is a multiplexer which forwards only one of the many possible adjustment inputs.

#### PI Servo Loop

This block contains two PI Servo Loops, one for the Offset and one for the Drift correction. The calculated values after the PI Servo are provided to the cores for the next calculations.

#### Adjustment



This block converts the adjustments in ratios or plain numbers to actual adjustments on the clock, smoothly spread over time. It also combines Drift and Offset adjustments and checks whether the clock is in sync.

#### Clock

This block is an adjustable counter clock with fractions. It provides the time used by all other cores.

#### 1ms Timer

This block is one millisecond timer aligned with the clock. It generates a pulse every milliseconds. This is used by other cores to keep their counters small if only relative time is needed and millisecond resolution is sufficient.



#### 2 Clock Basics

# 2.1 Digital Counter Clock

A digital counter clock is the most commonly used type of absolute time sources for digital systems. Its functionality is simple: every counter cycle it adds the period of the counter cycle to a counter. Optimally the counter period is an integer number which makes things easier. Normally such a counter clock is split into two counter parts, a sub seconds part and a seconds part, depending on the required resolution the sub second part is in nanoseconds, microseconds or milliseconds or even tens or hundreds of milliseconds. Once the sub seconds counter overflows e.g. 10^9 nanoseconds are reached, the seconds counter is incremented by one and the sub seconds counter is reset to the remainder if there is any.

The highest resolution can be achieved when the counter period is equal the clock period where the counter is run on, this is then normally a nanoseconds resolution, however with a quantitation of the clock period (this is what this core implements).

Figure 3: shows a typical high resolution counter clock with nanosecond resolution and a counter period equal the clock period and a clock of 50MHz which equals to a 20ns clock period.



Figure 3: Counter Clock

# 2.2 Drift and Offset adjustments

When a digital counter clock shall be synchronized there are two things that have to be adjusted which is frequency differences aka drift and phase differences aka offset. Normally the phase difference is only considered the phase within a second. But for absolute time also the correct second is important.



There are basically two possibilities how to adjust a digital counter clock. One is to keep the counter increment and adjust the clock frequency with e.g. load on a crystal oscillator where the load can be adjusted via a DAC. The second and the one used for this core, is to keep the clock frequency and adjust the counter increment. This has the advantages that it normally has a much higher resolution e.g. 1ns/s and it doesn't require or relies on external hardware. To adjust drift or offset additional nanoseconds are added or subtracted from the standard increment of the period. Where the first solution implicitly generates a smooth adjustment, for the second this has to be calculated and explicitly done. E.g. for a 50 MHz counter clock an offset of +100 ns could be adjusted from one clock cycle to the next: 20 => 140=>160 => ... (including 20 ns for the next clock cycle) or it could for example be spread over the next 100 clock cycles: 20 => 41 => 62 =>73 => ... which is a much smoother adjustment. The same applies to the drift which can also be set once in a period or evenly spread over time.

But why is a smooth adjustment important? If for example a PWM signal is generated from the counter clock then you don't want a time jump since the PWM would not be correct anymore, and this is exactly what would happen if the time is not corrected smoothly. The same applies for short time period measurements, these would measure wrong periods because of the adjustments.

However it is not always possible to adjust the time smoothly, e.g. at startup of a system the clock has to be adjusted by thousands of seconds to get to the time of day (TAI start with second 0 at midnight 1.1.1970) or if the adjustment is larger than the possible adjustment in a given period. This cannot be done smoothly in a reasonable time, therefore the time is then set with a time jump, and this also applies for the solution where the frequency of the clock is adjusted.

Also important is that the clock doesn't count backwards during adjustments. Data acquisition and measurement applications require for example a strongly monolithically increasing time. This requirement basically limits the maximal adjustment so the clock is still counting. E.g. at 50 MHz a norm period is 20 ns, the maximum adjustment is therefore +/-19ns per clock period so the clock would still count with lns per clock period.

All these mechanisms are implemented in this adjustable counter clock core.

When using the counter clock for signal timestamping or signal generation the quantization fault is still the clock period but with an accurate nanosecond resolution.



# **3** Register Set

This is the register set of the Adjustable Counter Clock. It is accessible via AXI4 Light Memory Mapped. All registers are 32bit wide, no burst access, no unaligned access, no byte enables, no timeouts are supported. Register address space is not contiguous. Register addresses are only offsets in the memory area where the core is mapped in the AXI inter connects. Non existing register access in the mapped memory area is answered with a slave decoding error.

# 3.1 Register Overview

| Registerset Overview            |                                                     |            |        |
|---------------------------------|-----------------------------------------------------|------------|--------|
| Name                            | Description                                         | Offset     | Access |
| Clk ClockControl Reg            | Clock Read/Write Valid and Enabled Control Register | 0x0000000  | RW     |
| Clk ClockStatus Reg             | Clock Status Register                               | 0x0000004  | RO     |
| Clk ClockSelect Reg             | Clock Adjustment Multiplexer selection Register     | 0x0000008  | RW     |
| Clk ClockVersion Reg            | Clock Version Register                              | 0x000000C  | RO     |
| Clk ClockTimeValueL Reg         | Clock current Time Nanosecond Register              | 0x0000010  | RO     |
| Clk ClockTimeValueH Reg         | Clock current Time Second Register                  | 0x00000014 | RO     |
| Clk ClockTimeAdjValueL Reg      | Clock adjust Time Nanosecond Register               | 0x00000020 | RW     |
| Clk ClockTimeAdjValueH Reg      | Clock adjust Time Second Register                   | 0x00000024 | RW     |
| Clk ClockOffsetAdjValue Reg     | Clock adjust Offset Nanosecond Register             | 0x0000030  | RW     |
| Clk ClockOffsetAdjInterval Reg  | Clock adjust Offset Interval Nanosecond Register    | 0x00000034 | RW     |
| Clk ClockDirftAdjValue Reg      | Clock adjust Drift Nanosecond Register              | 0x0000040  | RW     |
| Clk ClockDriftAdjInterval Reg   | Clock adjust Drift Interval Nanosecond Register     | 0x00000044 | RW     |
| Clk ClockInSyncThreshold Reg    | Clock adjust Drift Interval Nanosecond Register     | 0x0000050  | RW     |
| Clk_ClockServoOffsetFactorP_Reg | Clock Offset Servo P Factor Register                | 0x0000060  | RW     |
| Clk_ClockServoOffsetFactorl_Reg | Clock Offset Servo I Factor Register                | 0x0000064  | RW     |

AdjCounterClock Reference Manual 1.7 Page 14 of 81



| Clk_ClockServoDriftFactorP_Reg | Clock Drift Servo P Factor Register | 0x00000068 | RW |
|--------------------------------|-------------------------------------|------------|----|
| Clk_ClockServoDriftFactorl_Reg | Clock Drift Servo I Factor Register | 0x000006C  | RW |
| Clk_ClockStatusOffset_Reg      | Clock corrected Offset Register     | 0x00000070 | RO |
| Clk_ClockStatusDrift_Reg       | Clock corrected Drift Register      | 0x0000074  | RO |

AdjCounterClock Reference Manual 1.7



# **3.2 Register Descriptions**

#### 3.2.1 General

## 3.2.1.1 CLK Clock Control Register

Used for general control over the Adjustable Counter Clock. To get a new time snapshot the time read flag has to be set and the read done flag is asserted. Since most adjustment values are multi register values, set flags are available to mark validity of the whole value.

| CI             | c Clc     | ockC   | Cont | rol | Reg |    |    |    |    |    |    |    |    |       |      |      |     |    |    |    |    |   |           |   |   |   |   |           |            |          |        |
|----------------|-----------|--------|------|-----|-----|----|----|----|----|----|----|----|----|-------|------|------|-----|----|----|----|----|---|-----------|---|---|---|---|-----------|------------|----------|--------|
| Reg            | g Des     | script | ion  |     |     |    |    |    |    |    |    |    |    |       |      |      |     |    |    |    |    |   |           |   |   |   |   |           |            |          |        |
| 31             | 30        | 29     | 28   | 27  | 26  | 25 | 24 | 23 | 22 | 21 | 20 | 19 | 18 | 17    | 16   | 15   | 14  | 13 | 12 | 11 | 10 | 9 | 8         | 7 | 6 | 5 | 4 | 3         | 2          | 1        | 0      |
| TIME_READ_DONE | TIME_READ |        |      |     |     |    |    |    |    |    |    | ı  |    |       |      |      |     |    |    |    |    |   | SERVO_VAL |   |   |   |   | DRIFT_VAL | OFFSET_VAL | TIME_VAL | ENABLE |
| RO             | RW        |        |      |     |     |    |    |    |    |    |    | RO |    |       |      |      |     |    |    |    |    |   |           |   |   |   |   | RW        | RW         | RW       | RW     |
|                |           |        |      |     |     |    |    |    |    |    |    |    | F  | Reset |      |      |     | 0  |    |    |    |   |           |   |   |   |   |           |            |          |        |
|                | •         | •      | •    |     | •   | •  | •  |    | •  |    |    | •  | •  | Off   | set: | 0x00 | 000 |    | •  | •  |    |   | •         |   | • | • | • |           |            | •        |        |

| Name           | Description                  | Bits    | Access |
|----------------|------------------------------|---------|--------|
| TIME_READ_DONE | Time Read done (autocleared) | Bit: 31 | RO     |

AdjCounterClock Reference Manual 1.7 Page 16 of 81



| TIME_READ  | Time Read (autocleared)                                                            | Bit: 30   | RW |
|------------|------------------------------------------------------------------------------------|-----------|----|
| -          | Reserved, read 0                                                                   | Bit: 29:9 | RO |
| SERVO_VAL  | Servo Parameters Valid (autocleared), shall only be set when the clock is disabled | Bit: 8    | RW |
| -          | Reserved, read 0                                                                   | Bit: 7:4  | RO |
| DRIFT_VAL  | Drift Ajustment Valid (autocleared)                                                | Bit: 3    | RW |
| OFFSET_VAL | Offset Ajustment Valid (autocleared)                                               | Bit: 2    | RW |
| TIME_VAL   | Time Ajustment Valid (autocleared)                                                 | Bit: 1    | RW |
| ENABLE     | Enable                                                                             | Bit: 0    | RW |

AdjCounterClock Reference Manual 1.7



# 3.2.1.2 CLK Clock Status Register

Marks if the clock is in sync. In sync means 5 consecutive offset where below the threshold.

| Clk | <b>( C</b> | Clock | ιSt  | atı | ıs R | eg |    |    |    |    |    |    |    |    |    |       |      |     |    |    |    |    |   |   |   |   |   |   |   |   |             |         |
|-----|------------|-------|------|-----|------|----|----|----|----|----|----|----|----|----|----|-------|------|-----|----|----|----|----|---|---|---|---|---|---|---|---|-------------|---------|
| Reg | g D        | escri | ptic | on  |      |    |    |    |    |    |    |    |    |    |    |       |      |     |    |    |    |    |   |   |   |   |   |   |   |   |             |         |
| 31  | 3          | 0 29  | )    | 28  | 27   | 26 | 25 | 24 | 23 | 22 | 21 | 20 | 19 | 18 | 17 | 16    | 15   | 14  | 13 | 12 | 11 | 10 | 9 | 8 | 7 | 6 | 5 | 4 | 3 | 2 | 1           | 0       |
|     |            |       |      |     |      |    |    |    |    |    |    |    |    |    |    | 1     |      |     |    |    |    |    |   |   |   |   |   |   |   |   | IN_HOLDOVER | IN_SYNC |
|     |            |       |      |     |      |    |    |    |    |    |    |    |    |    |    | RO    |      |     |    |    |    |    |   |   |   |   |   |   |   |   | RO          | RO      |
|     |            |       |      |     |      |    |    |    |    |    |    |    |    | F  |    |       | 0000 |     | 0  |    |    |    |   |   |   |   |   |   |   |   |             |         |
|     |            |       |      |     |      |    |    |    |    |    |    |    |    |    | Of | fset: | 0x0( | 004 |    |    |    |    |   |   |   |   |   |   |   |   |             | _       |

| Name        | Description                                                | Bits      | Access |
|-------------|------------------------------------------------------------|-----------|--------|
| -           | Reserved, read 0                                           | Bit: 31:2 | RO     |
| IN_HOLDOVER | In holdover, after in sync for N seconds no new adjustment | Bit: 1    | RO     |
| IN_SYNC     | In sync; Last 5 corrections below threshold                | Bit: 0    | RO     |

AdjCounterClock Reference Manual 1.7 Page 18 of 81



# 3.2.1.3 CLK Clock Select Register

Which input adjustment shall be active. If external select is active the CLK\_SELECTED field shows the actual selection

| Clk ClockSelect Reg     |                         |                       |                 |  |  |  |  |  |  |  |  |
|-------------------------|-------------------------|-----------------------|-----------------|--|--|--|--|--|--|--|--|
| Reg Description         |                         |                       |                 |  |  |  |  |  |  |  |  |
| 31 30 29 28 27 26 25 24 | 23 22 21 20 19 18 17 16 | 15 14 13 12 11 10 9 8 | 7 6 5 4 3 2 1 0 |  |  |  |  |  |  |  |  |
| 1                       | CLK_SELECTED            | ı                     | CLK_SELECT      |  |  |  |  |  |  |  |  |
| RO                      | RO                      | RO                    | RW              |  |  |  |  |  |  |  |  |
|                         | Reset: 0x0              | 0000000               |                 |  |  |  |  |  |  |  |  |
|                         | Offset:                 | 0x0008                |                 |  |  |  |  |  |  |  |  |

| Name | Description      | Bits       | Access |
|------|------------------|------------|--------|
| -    | Reserved, read 0 | Bit: 31:24 | RO     |

AdjCounterClock Reference Manual 1.7 Page 19 of 81



| CLK_SELECTED | Selected Source for Clock Adjustments: None = 0 Tod = 1 Irig = 2 Pps = 3 Ptp = 4 Rtc = 5 Dcf = 6 Regs = 254 Ext = 255 | Bit: 23:16 | RO |
|--------------|-----------------------------------------------------------------------------------------------------------------------|------------|----|
| -            | Reserved, read 0                                                                                                      | Bit: 15:8  | RO |
| CLK_SELECT   | Source for Clock Adjustments: None = 0 Tod = 1 Irig = 2 Pps = 3 Ptp = 4 Rtc = 5 Dcf = 6 Regs = 254 Ext = 255          | Bit: 7:0   | RW |

# 3.2.1.4 CLK Clock Version Register

Version of the IP core, even though is seen as a 32bit value, bits 31 down to 24 represent the major, bits 23 down to 16 the minor and bits 15 down to 0 the build numbers.

| C | ik C   | lock   | Vers  | ion | Reg |    |    |    |    |    |    |    |    |    |    |    |    |    |    |    |    |   |   |   |   |   |   |   |   |   |   |
|---|--------|--------|-------|-----|-----|----|----|----|----|----|----|----|----|----|----|----|----|----|----|----|----|---|---|---|---|---|---|---|---|---|---|
| R | leg De | escrip | otion |     |     |    |    |    |    |    |    |    |    |    |    |    |    |    |    |    |    |   |   |   |   |   |   |   |   |   |   |
| 3 | 1 30   | 29     | 28    | 27  | 26  | 25 | 24 | 23 | 22 | 21 | 20 | 19 | 18 | 17 | 16 | 15 | 14 | 13 | 12 | 11 | 10 | 9 | 8 | 7 | 6 | 5 | 4 | 3 | 2 | 1 | 0 |

AdjCounterClock Reference Manual 1.7 Page 20 of 81



| Z<br>O           |  |
|------------------|--|
| ÆRSI             |  |
|                  |  |
|                  |  |
| RO               |  |
| Reset: 0xXXXXXXX |  |
| Offset: 0x000C   |  |

| Name    | Description         | Bits      | Access |
|---------|---------------------|-----------|--------|
| VERSION | Version of the core | Bit: 31:0 | RO     |

AdjCounterClock Reference Manual 1.7



# 3.2.1.5 CLK Clock Time Value Low Register

Time snapshot value nanosecond part.



| Name    | Description                | Bits      | Access |
|---------|----------------------------|-----------|--------|
| TIME_NS | Snapshoted Time Nanosecond | Bit: 31:0 | RO     |

AdjCounterClock Reference Manual 1.7 Page 22 of 81



# 3.2.1.6 CLK Clock Time Value High Register

Time snapshot value second part.



| Name   | Description            | Bits      | Access |
|--------|------------------------|-----------|--------|
| TIME_S | Snapshoted Time Second | Bit: 31:0 | RO     |

AdjCounterClock Reference Manual 1.7 Page 23 of 81



# 3.2.1.7 CLK Clock Time Adjustment Value Low Register

Time adjustment nanoseconds part value. Will overwrite the clock.



| Name        | Description              | Bits      | Access |
|-------------|--------------------------|-----------|--------|
| TIME_ADJ_NS | OverwriteTime Nanosecond | Bit: 31:0 | RW     |

AdjCounterClock Reference Manual 1.7 Page 24 of 81



# 3.2.1.8 CLK Clock Time Adjustment Value High Register

Time adjustment seconds part value. Will overwrite the clock.



| Name       | Description          | Bits      | Access |
|------------|----------------------|-----------|--------|
| TIME_ADJ_S | OverwriteTime Second | Bit: 31:0 | RW     |

AdjCounterClock Reference Manual 1.7 Page 25 of 81



#### 3.2.1.9 CLK Clock Offset Adjustment Value Register

Offset adjustment absolute value. A negative sign will slow down the clock, a positive sign therefore accelerates the clock to adjust the phase.



| Name          | Description                                   | Bits      | Access |
|---------------|-----------------------------------------------|-----------|--------|
| OFFSET_SIGN   | Sign of the Adjustment O positive, 1 negative | Bit: 31   | RW     |
| OFFSET_ADJ_NS | Offset Adjustment in Nanosecond               | Bit: 30:0 | RW     |

AdjCounterClock Reference Manual 1.7 Page 26 of 81



#### 3.2.1.10 CLK Clock Offset Adjustment Interval Register

Offset adjustment interval. It defines the interval in which the offset shall be corrected. If the offset value is larger or equal the interval, the time will be set directly.



| Name               | Description                              | Bits      | Access |
|--------------------|------------------------------------------|-----------|--------|
| OFFSET_ADJ_INTV_NS | Offset Adjustment Interval in Nanosecond | Bit: 31:0 | RW     |

AdjCounterClock Reference Manual 1.7 Page 27 of 81



## 3.2.1.11CLK Clock Drift Adjustment Value Register

Drift adjustment value. Together with the drift interval this indicates a ratio of correction in the format ns per ns which means how many nanoseconds shall be corrected in the interval. A negative sign will slow down the clock, a positive sign therefore accelerates the clock.



| Name         | Description                                   | Bits      | Access |
|--------------|-----------------------------------------------|-----------|--------|
| DRIFT_SIGN   | Sign of the Adjustment 0 positice, 1 negative | Bit: 31   | RW     |
| DRIFT_ADJ_NS | Drift Adjustment in Nanosecond                | Bit: 30:0 | RW     |

AdjCounterClock Reference Manual 1.7 Page 28 of 81



#### 3.2.1.12CLK Clock Drift Adjustment Interval Register

Drift adjustment interval. Together with the drift value this indicates a ratio of correction in the format ns per ns which means how many nanoseconds shall be corrected in the interval specified by this register.



| Name              | Description                             | Bits      | Access |
|-------------------|-----------------------------------------|-----------|--------|
| DRIFT_ADJ_INTV_NS | Drift Adjustment Interval in Nanosecond | Bit: 31:0 | RW     |

AdjCounterClock Reference Manual 1.7 Page 29 of 81



#### 3.2.1.13CLK Clock In Sync Threshold Register

Threshold of the in sync indication flag. After consecutive offset adjustments below the threshold the in sync flag is asserted. The threshold is in nanosecond and the default value is set via generic.



| Name         | Description                              | Bits      | Access |
|--------------|------------------------------------------|-----------|--------|
| THRESHOLD_NS | Threshold for InSync flag in Nanoseconds | Bit: 31:0 | RW     |

AdjCounterClock Reference Manual 1.7 Page 30 of 81



#### 3.2.1.14 CLK Clock Servo Offset Factor P Register

Proportional factor for the fractional offset multiplier. The Value is calculated the following way:

# $(MulP*2^{16})/DivP$

After reset the value is defined by the generics.

This register is only available if the generic BypassServo\_Gen is false and the generics FractionalMultiply\_Gen and DynamicServo-Parameters\_Gen are true.



| Name            | Description                                                 | Bits      | Access |
|-----------------|-------------------------------------------------------------|-----------|--------|
| OFFSET_FACTOR_P | Fractional multiplication Factor Offset P: ((Mul*2^16)/Div) | Bit: 31:0 | RW     |

AdjCounterClock Reference Manual 1.7 Page 31 of 81



## 3.2.1.15 CLK Clock Servo Offset Factor I Register

Integral factor for the fractional offset multiplier. The Value is calculated the following way:

 $(MulI * 2^{16})/DivI$ 

After reset the value is defined by the generics.

This register is only available if the generic BypassServo\_Gen is false and the generics FractionalMultiply\_Gen and DynamicServo-Parameters\_Gen are true.



| Name            | Description                                                 | Bits      | Access |
|-----------------|-------------------------------------------------------------|-----------|--------|
| OFFSET_FACTOR_I | Fractional multiplication Factor Offset I: ((Mul*2^16)/Div) | Bit: 31:0 | RW     |

AdjCounterClock Reference Manual 1.7 Page 32 of 81



#### 3.2.1.16CLK Clock Servo Drift Factor P Register

Proportional factor for the fractional drift multiplier. The Value is calculated the following way:

#### $(MulP * 2^{16})/DivP$

After reset the value is defined by the generics.

This register is only available if the generic BypassServo\_Gen is false and the generics FractionalMultiply\_Gen and DynamicServo-Parameters Gen are true.



| Name           | Description                                                | Bits      | Access |
|----------------|------------------------------------------------------------|-----------|--------|
| DRIFT_FACTOR_P | Fractional multiplication Factor Drift P: ((Mul*2^16)/Div) | Bit: 31:0 | RW     |

AdjCounterClock Reference Manual 1.7 Page 33 of 81



#### 3.2.1.17 CLK Clock Servo Drift Factor I Register

Integral factor for the fractional drift multiplier. The Value is calculated the following way:

 $(MulI * 2^{16})/DivI$ 

After reset the value is defined by the generics.

This register is only available if the generic BypassServo\_Gen is false and the generics FractionalMultiply\_Gen and DynamicServo-Parameters\_Gen are true.



| Name           | Description                                                | Bits      | Access |
|----------------|------------------------------------------------------------|-----------|--------|
| DRIFT_FACTOR_I | Fractional multiplication Factor Drift I: ((Mul*2^16)/Div) | Bit: 31:0 | RW     |

AdjCounterClock Reference Manual 1.7 Page 34 of 81



#### 3.2.1.18CLK Clock Status Offset Value Register

Last corrected offset in nanoseconds. This register is only available if the generic LogCorrections\_Gen is true.



|   | Name        | Description                                | Bits      | Access |
|---|-------------|--------------------------------------------|-----------|--------|
| ſ | OFFSET_SIGN | Sign bit: 0 = positive, 1 = negative       | Bit: 31   | RO     |
|   | OFFSET_NS   | Last corrected Offset value in Nanoseconds | Bit: 30:0 | RO     |

AdjCounterClock Reference Manual 1.7 Page 35 of 81



#### 3.2.1.19CLK Clock Status Drift Value Register

Last corrected drift in nanosecond. This register is only available if the generic LogCorrections\_Gen is true.



| Name       | Description                                            | Bits      | Access |
|------------|--------------------------------------------------------|-----------|--------|
| DRIFT_SIGN | Sign bit: 0 = positive, 1 = negative                   | Bit: 31   | RO     |
| DRIFT_NS   | Last corrected Drift value in Nanoseconds/Second (ppb) | Bit: 30:0 | RO     |

AdjCounterClock Reference Manual 1.7 Page 36 of 81



# 4 Design Description

The following chapters describe the internals of the Adjustable Counter Clock: starting with the Top Level, which is a collection of subcores, followed by the description of all subcores.

# 4.1 Top Level - Clk Clock

### 4.1.1.1 Parameters

The core must be parametrized at synthesis time. There are a couple of parameters which define the final behavior and resource usage of the core.

| Name                              | Туре    | Size | Description                                                                                                                   |
|-----------------------------------|---------|------|-------------------------------------------------------------------------------------------------------------------------------|
| StaticConfig_Gen                  | boolean | 1    | If Static Configuration or AXI is used: true = Static, false = AXI                                                            |
| ExtSelect_Gen                     | boolean | 1    | If external selection of correction shall be done or not: true = external selection is possible, false only selection via reg |
| ClockClkPeriodNano-<br>second_Gen | natural | 1    | Clock Period in Nanosecond: Default for 50 MHz = 20 ns, if non-integer only the nanosec- onds part here                       |
| ClockClkPeriodFract-<br>Num_Gen   | natural | 1    | Fractional Clock Period Nu-<br>merator (0 if integer)                                                                         |
| ClockClkPeriod-<br>FractDeNum_Gen | natural | 1    | Fractional Clock Period Denumerator (0 if integer)                                                                            |
| ClockInSyncNanosec-<br>ond_Gen    | natural | 1    | Default value for the threshold<br>when the clock is considered<br>in sync in Nanoseconds<br>Default 500 ns                   |
| ClockInHoldover TimeoutSecond_Gen | natural | 1    | Value after how many seconds after in Sync without                                                                            |



|                                |                  |    | new correction values it goes                                                                              |
|--------------------------------|------------------|----|------------------------------------------------------------------------------------------------------------|
|                                |                  |    | in holdover                                                                                                |
|                                |                  |    | Default 3 s                                                                                                |
| LogCorrections_Gen             | boolean          | 1  | Log Correction values to registers                                                                         |
| BypassServo_Gen                | boolean          | 1  | Bypass PI Servo loops, the input will directly be taken for calculations                                   |
| FractionalMultiply<br>_Gen_Gen | boolean          | 1  | Use a fractional multiplication (uses DSP slices) instead of binary multiplication and divisions (default) |
| DynamicServo Parameters_Gen    | boolean          | 1  | Allow to change the PI Servo parameters at runtime                                                         |
| DriftMulP_Gen                  | natural          | 1  | Drift Proportional part ratio<br>Numerator                                                                 |
| DriftDivP_Gen                  | natural          | 1  | Drift Proportional part ratio Denumerator                                                                  |
| DriftMull_Gen                  | natural          | 1  | Drift Integral part ratio Nu-<br>merator                                                                   |
| DriftDivl_Gen                  | natural          | 1  | Drift Integral part ratio Denu-<br>merator                                                                 |
| OffsetMulP_Gen                 | natural          | 1  | Offset Proportional part ratio<br>Numerator                                                                |
| OffsetDivP_Gen                 | natural          | 1  | Offset Proportional part ratio Denumerator                                                                 |
| OffsetMull_Gen                 | natural          | 1  | Offset Integral part ratio Nu-<br>merator                                                                  |
| OffsetDivl_Gen                 | natural          | 1  | Offset Integral part ratio De-<br>numerator                                                                |
| AxiAddressRange<br>Low_Gen     | std_logic_vector | 32 | AXI Base Address                                                                                           |
| AxiAddressRange<br>High_Gen    | std_logic_vector | 32 | AXI Base Address plus Registerset Size Default plus OxFFFF                                                 |
| Sim_Gen                        | boolean          | 1  | If in Testbench simulation mode:                                                                           |



|  | true = Simulation, false = Syn- |
|--|---------------------------------|
|  | thesis                          |

Table 4: Parameters

If an integer clock period is used all fraction parameters have to be 0.

## 4.1.1.2 Structured Types

### 4.1.1.2.1 Clk\_Time\_Type

Defined in Clk\_Package.vhd of library ClkLib

Type represents the time used everywhere. For this type overloaded operators + and - with different parameters exist.

| Field Name | Туре             | Size | Description                                              |
|------------|------------------|------|----------------------------------------------------------|
| Second     | std_logic_vector | 32   | Seconds of time                                          |
| Nanosecond | std_logic_vector | 32   | Nanoseconds of time                                      |
| Fraction   | std_logic_vector | 2    | Fraction numerator (mostly not used)                     |
| Sign       | std_logic        | 1    | Positive or negative time, 1 = negative, 0 = positive.   |
| TimeJump   | std_logic        | 1    | Marks when the clock makes a time jump (mostly not used) |

Table 5: Clk\_Time\_Type

### 4.1.1.2.2 Clk\_TimeAdjustment\_Type

Defined in Clk\_Package.vhd of library ClkLib

Type represents the time used everywhere. For this type overloaded operators + and - with different parameters exist.

| Field Name     | Туре             | Size | Description                      |
|----------------|------------------|------|----------------------------------|
| TimeAdjustment | Clk_Time_Type    | 1    | Time to adjust                   |
| Interval       |                  |      | Adjustment interval, for the     |
|                |                  | 32   | drift correction this is the de- |
|                | std_logic_vector | 32   | numerator of the rate in nano-   |
|                |                  |      | seconds (TimeAdjustment          |



| every Interval = drift rate), for |
|-----------------------------------|
| offset correction this is the pe- |
| riod in which the time shall be   |
| corrected(TimeAdjustment in       |
| Interval), for setting the time   |
| this has no mining.               |

Table 6: Clk\_TimeAdjustment\_Type

## 4.1.1.2.3 Clk\_CoreInfo\_Type

Defined in Clk\_Package.vhd of library ClkLib

This is the type used for getting info about the cores state status.

| Field Name | Туре             | Size | Description                     |
|------------|------------------|------|---------------------------------|
| State      | Clk_CoreState_T  | 1    | State of the core: Unknown_E,   |
| State      | уре              | ı    | Slave_E or Master_E             |
| Accuracy   | std_logic_vector | 8    | Accuracy of the core, higher is |
| Accuracy   | sta_logic_vector | O    | better                          |
| Enabled    | std_logic        | 1    | If the core is enabled          |
| InSync     | std_logic        | 1    | If the core is synchronized     |
| Error      | std_logic        | 1    | If the core has an error        |

Table 7: Clk\_CoreInfo\_Type

## 4.1.1.2.4 Clk\_ClockStaticConfig\_Type

Defined in Clk\_ClockAddrPackage.vhd of library ClkLib This is the type used for static configuration.

| Field Name       | Туре                         | Size | Description                             |
|------------------|------------------------------|------|-----------------------------------------|
| TimeAdjustment   | Clk_TimeAdjust-<br>ment_Type | 1    | Set the time to this value              |
| OffsetAdjustment | Clk_TimeAdjust-<br>ment_Type | 1    | Do an offset adjustment with this value |
| DriftAdjustment  | Clk_TimeAdjust-<br>ment_Type | 1    | Do an drift adjustment with this value  |
| ClockSelect      | std_logic_vector             | 32   | Which adjustment input shall be used:   |



|                 |                  |    | 0 => None                                               |
|-----------------|------------------|----|---------------------------------------------------------|
|                 |                  |    | 1 => Tod                                                |
|                 |                  |    | 2 => Irig                                               |
|                 |                  |    | 3 => Pps                                                |
|                 |                  |    | 4 => Ptp                                                |
|                 |                  |    | 5 => Rtc                                                |
|                 |                  |    | 6 => Regs                                               |
|                 |                  |    | 7 => Ext                                                |
| InSyncThreshold | std_logic_vector | 32 | Threshold of the in sync flag indication in nanoseconds |

Table 8: Clk\_ClockStaticConfig\_Type

## 4.1.1.2.5 Clk\_ClockStaticConfigVal\_Type

Defined in Clk\_ClockAddrPackage.h of library ClkLib
This is the type used for valid flags of the static configuration.

| Field Name           | Туре      | Size | Description                |
|----------------------|-----------|------|----------------------------|
| Enable_Val           | std_logic | 1    | Enables the Clock          |
| TimeAdjustment_Val   | std_logic | 1    | Set the time               |
| OffsetAdjustment_Val | std_logic | 1    | Start an offset adjustment |
| DriftAdjustment_Val  | std_logic | 1    | Start a drift adjustment   |

Table 9: Clk\_ClockStaticConfigVal\_Type

### 4.1.1.2.6 Clk\_ClockStaticStatus\_Type

Defined in Clk\_ClockAddrPackage.vhd of library ClkLib This is the type used for static status.

| Field Name | Туре                  | Size | Description                 |
|------------|-----------------------|------|-----------------------------|
| CoreInfo   | Clk_CoreInfo_<br>Type | 1    | Infor about the Cores state |
| ClockTime  | Clk_Time_Type         | 1    | Time of the Clock           |

Table 10: Clk\_ClockStaticStatus\_Type

## 4.1.1.2.7 Clk\_ClockStaticStatusVal\_Type



Defined in Clk\_ClockAddrPackage.h of library ClkLib This is the type used for valid flags of the static status.

| Field Name    | Туре      | Size | Description  |
|---------------|-----------|------|--------------|
| ClockTime_Val | std_logic | 1    | Set the time |

Table 11: Clk\_ClockStaticStatusVal\_Type

## 4.1.1.3 Entity Block Diagram



Figure 4: Adjustable Counter Clock

## 4.1.1.4 Entity Description

#### **Clock Selector**

This module multiplexes multiple adjustment inputs. Only one adjustment is active at the time. This can be selected by a configuration register. PTP, PPS, TOD, IRIG, RTC, REG and EXT (where the selection is feed from externally) are potential sources. The multiplexer can also be set to disable any adjustment input. See 4.2.1 for more details.

#### **Clock Adjuster**

This module contains the PI Servo loops for the Drift and Offset and the adjustment conversion. It takes the adjustment inputs and converts them into periodic small adjustments which are then set to the Counter Clock. E.g. a drift of 1ns per 1000ns is converted to a 1ns adjustment every 50 clock cycles at a frequency of 50MHz. For offset it works similar. E.g. 50ns shall be corrected in 2000ns which will be converted into 1ns every second clock cycle at 50MHz. The conversion of course also works for non-integer corrections with minimal error (1/- 1ns per correction cycle).



In addition it supervises the adjustments ad sets an InSync flag if 4 consecutive offset corrections were below a configurable threshold. See 4.2.2 for more details.

#### **Clock Counter**

This module is the heart of the core. It is a counter with nanosecond resolution in a 32 bit second and 32 bit nanosecond format. It can run with fractions to allow also non-integer clock periods. It normally adds the clock period to the nanoseconds counter every clock cycle but can add or subtract some extra nanosecond to do the clock adjustment. It also has an overwrite mode where the clock can be set. See 4.2.3 for more details.

#### **Timer**

This module creates a 1 millisecond timer event aligned with the Counter Clock. If the clock makes a jump in time this may lead to a loss of an event or to two events very close to each other.

See 4.2.4 for more details.

### Registerset

This module is an AXI Light Memory Mapped Slave. It provides access to all registers and allows to configure the Adjustable Counter Clock. It can be configured to either run in AXI or StaticConfig mode. If in StaticConfig mode, the configuration of the resgisters is done via signals and can be easily done from within the FPGA without CPU. If in AXI mode, an AXI Master has to configure the Datasets with AXI writes to the registers, which is typically done by a CPU. It also provides a status interface which allows similar to the static configuration to supervise the status via signals.

See 4.2.5 for more details.

## 4.1.1.5 Entity Declaration

| Name             | Dir     | Туре    | Size | Description                                 |  |
|------------------|---------|---------|------|---------------------------------------------|--|
| Generics         |         |         |      |                                             |  |
| General          | General |         |      |                                             |  |
| StaticConfig_Gen | -       | boolean | 1    | If Static Configura-<br>tion or AXI is used |  |



| ExtSelect_Gen                        | - | boolean | 1 | If external selection of sync source is used                                                                            |
|--------------------------------------|---|---------|---|-------------------------------------------------------------------------------------------------------------------------|
| ClockClkPeriodNano-<br>second_Gen    | - | natural | 1 | Clock Period in Na-<br>nosecond                                                                                         |
| ClockClkPeriodFract-<br>Num_Gen      | - | natural | 1 | Fractional Clock Period Numerator (0 if integer)                                                                        |
| ClockClkPeriod-<br>FractDeNum_Gen    | ı | natural | 1 | Fractional Clock Period Denumerator (O if integer)                                                                      |
| ClockInSyncNanosec-<br>ond_Gen       | - | natural | 1 | Default value for the threshold when the clock is considered in sync                                                    |
| ClockInHoldover<br>TimeoutSecond_Gen | - | natural | 1 | Value after how<br>many seconds after<br>in Sync without new<br>correction values it<br>goes in holdover<br>Default 3 s |
| LogCorrections_Gen                   | - | boolean | 1 | Log corrections to registers                                                                                            |
| BypassServo_Gen                      | - | boolean | 1 | Bypass PI Servo<br>loops                                                                                                |
| FractionalMultiply<br>_Gen           | - | boolean | 1 | Use a fractional multiplication (uses DSP slices) instead of binary multiplication and divisions (default)              |
| DynamicServo Parameters_Gen          | - | boolean | 1 | Allow to change the PI Servo parameters at runtime                                                                      |
| DriftMulP_Gen                        | - | natural | 1 | Drift Proportional part ratio Numera-tor                                                                                |



| DriftDivP_Gen               | -   | natural                               | 1  | Drift Proportional part ratio Denumerator    |
|-----------------------------|-----|---------------------------------------|----|----------------------------------------------|
| DriftMull_Gen               | -   | natural                               | 1  | Drift Integral part ratio Numerator          |
| DriftDivl_Gen               | -   | natural                               | 1  | Drift Integral part ratio Denumerator        |
| OffsetMulP_Gen              | -   | natural                               | 1  | Offset Proportional part ratio Numerator     |
| OffsetDivP_Gen              | -   | natural                               | 1  | Offset Proportional part ratio Denumerator   |
| OffsetMull_Gen              | -   | natural                               | 1  | Offset Integral part ratio Numerator         |
| OffsetDivl_Gen              | -   | natural                               | 1  | Offset Integral part ratio Denumerator       |
| AxiAddressRange<br>Low_Gen  | -   | std_logic_vector                      | 32 | AXI Base Address                             |
| AxiAddressRange<br>High_Gen | -   | std_logic_vector                      | 32 | AXI Base Address<br>plus Registerset<br>Size |
| Sim_Gen                     | ı   | boolean                               | 1  | If in Testbench simulation mode              |
|                             |     | Ports                                 |    |                                              |
| System SysClk_ClkIn         | in  | std_logic                             | 1  | System Clock                                 |
| SysRstN_RstIn               | in  | std_logic                             | 1  | System Reset                                 |
| Config StaticConfig_DatIn   | in  | Clk_Clock<br>StaticConfig_Type        | 1  | Static Configuration                         |
| StaticConfig_ValIn          | in  | Clk_Clock<br>StaticConfigVal<br>_Type | 1  | Static Configuration valid                   |
| Status StaticStatus_DatOut  | out | Clk_Clock<br>StaticStatus_Type        | 1  | Static Status                                |



| StaticStatus_ValOut                             | out | Clk_Clock<br>StaticStatusVal<br>_Type | 1  | Static Status valid                                                                   |
|-------------------------------------------------|-----|---------------------------------------|----|---------------------------------------------------------------------------------------|
| Timer Output  Timer1ms_EvtOut  AXI4 Light Slave | out | std_logic                             | 1  | Single clock cycle<br>event every millisec-<br>ond aligned with the<br>adjusted clock |
| AxiWriteAddrValid ValIn                         | in  | std_logic                             | 1  | Write Address Valid                                                                   |
| AxiWriteAddrReady<br>_RdyOut                    | out | std_logic                             | 1  | Write Address<br>Ready                                                                |
| AxiWriteAddrAddress<br>_AdrIn                   | in  | std_logic_vector                      | 32 | Write Address                                                                         |
| AxiWriteAddrProt<br>_DatIn                      | in  | std_logic_vector                      | 3  | Write Address Protocol                                                                |
| AxiWriteDataValid<br>_ValIn                     | in  | std_logic                             | 1  | Write Data Valid                                                                      |
| AxiWriteDataReady<br>_RdyOut                    | out | std_logic                             | 1  | Write Data Ready                                                                      |
| AxiWriteDataData<br>_DatIn                      | in  | std_logic_vector                      | 32 | Write Data                                                                            |
| AxiWriteDataStrobe<br>_DatIn                    | in  | std_logic_vector                      | 4  | Write Data Strobe                                                                     |
| AxiWriteRespValid<br>_ValOut                    | out | std_logic                             | 1  | Write Response<br>Valid                                                               |
| AxiWriteRespReady<br>_RdyIn                     | in  | std_logic                             | 1  | Write Response<br>Ready                                                               |
| AxiWriteResp<br>Response_DatOut                 | out | std_logic_vector                      | 2  | Write Response                                                                        |
| AxiReadAddrValid<br>_ValIn                      | in  | std_logic                             | 1  | Read Address Valid                                                                    |
| AxiReadAddrReady<br>_RdyOut                     | out | std_logic                             | 1  | Read Address<br>Ready                                                                 |
| AxiReadAddrAddress<br>_AdrIn                    | in  | std_logic_vector                      | 32 | Read Address                                                                          |
| AxiReadAddrProt<br>_DatIn                       | in  | std_logic_vector                      | 3  | Read Address Pro-<br>tocol                                                            |
| AxiReadDataValid<br>_ValOut                     | out | std_logic                             | 1  | Read Data Valid                                                                       |
| AxiReadDataReady<br>_RdyIn                      | in  | std_logic                             | 1  | Read Data Ready                                                                       |
| AxiReadData<br>Response_DatOut                  | out | std_logic_vector                      | 2  | Read Data                                                                             |



| AxiReadDataData<br>_DatOut   | out | std_logic_vector             | 32 | Read Data Re-<br>sponse                                                                                        |
|------------------------------|-----|------------------------------|----|----------------------------------------------------------------------------------------------------------------|
| Adjustment Selector In       | in  | Clk_Select_Type              | 1  | Which core shall be the source for the correction when Select_DatIn is external, this is feed from the outside |
| InSync Output                |     |                              | I  |                                                                                                                |
| InSync_DatOut                | out | std_logic                    | 1  | Clock Adjustments were below a cer- tain level                                                                 |
| InHoldover_DatOut            | out | std_logic                    | 1  | No new Clock Adjustments after in Sync for a defined number of seconds                                         |
| Time Output                  | ·   |                              | ı  |                                                                                                                |
| ClockTime_DatOut             | out | Clk_Time_Type                | 1  | Adjusted Clock<br>Time                                                                                         |
| ClockTime_ValOut             | out | std_logic                    | 1  | Adjusted Clock Time valid                                                                                      |
| Time Adjustment Input        |     |                              |    |                                                                                                                |
| TimeAdjustmentTod<br>_DatIn  | in  | Clk_TimeAdjust-<br>ment_Type | 1  | Calculated new Time from TOD core                                                                              |
| TimeAdjustmentTod<br>_ValIn  | in  | std_logic;                   | 1  | Calculated new Time from TOD core valid                                                                        |
| TimeAdjustmentIrig<br>_DatIn | in  | Clk_TimeAdjust-<br>ment_Type | 1  | Calculated new Time from IRIG core                                                                             |
| TimeAdjustmentIrig<br>_ValIn | in  | std_logic;                   | 1  | Calculated new Time from IRIG core valid                                                                       |
| TimeAdjustmentPps<br>_DatIn  | in  | Clk_TimeAdjust-<br>ment_Type | 1  | Calculated new Time from PPS core                                                                              |
| TimeAdjustmentPps<br>_ValIn  | in  | std_logic;                   | 1  | Calculated new Time from PPS core valid                                                                        |



| TimeAdjustmentPtp<br>_DatIn    | in  | Clk_TimeAdjust-<br>ment_Type | 1 | Calculated new Time from PTP core                  |
|--------------------------------|-----|------------------------------|---|----------------------------------------------------|
| TimeAdjustmentPtp<br>_ValIn    | in  | std_logic;                   | 1 | Calculated new Time from PTP core valid            |
| TimeAdjustmentRtc<br>_DatIn    | in  | Clk_TimeAdjust-<br>ment_Type | 1 | Calculated new Time from RTC core                  |
| TimeAdjustmentRtc<br>_ValIn    | in  | std_logic;                   | 1 | Calculated new Time from RTC core valid            |
| Offset Adjustment Inpu         | t   |                              | T |                                                    |
| OffsetAdjustmentTod<br>_DatIn  | in  | Clk_TimeAdjust-<br>ment_Type | 1 | Calculated new Off-<br>set from TOD core           |
| OffsetAdjustmentTod<br>_ValIn  | in  | std_logic;                   | 1 | Calculated new Off-<br>set from TOD core<br>valid  |
| OffsetAdjustmentIrig<br>_DatIn | in  | Clk_TimeAdjust-<br>ment_Type | 1 | Calculated new Off-<br>set from IRIG core          |
| OffsetAdjustmentIrig<br>_ValIn | in  | std_logic;                   | 1 | Calculated new Off-<br>set from IRIG core<br>valid |
| OffsetAdjustmentPps<br>_DatIn  | in  | Clk_TimeAdjust-<br>ment_Type | 1 | Calculated new Off-<br>set from PPS core           |
| OffsetAdjustmentPps<br>_ValIn  | in  | std_logic;                   | 1 | Calculated new Off-<br>set from PPS core<br>valid  |
| OffsetAdjustmentPtp<br>_DatIn  | in  | Clk_TimeAdjust-<br>ment_Type | 1 | Calculated new Off-<br>set from PTP core           |
| OffsetAdjustmentPtp<br>_ValIn  | in  | std_logic;                   | 1 | Calculated new Off-<br>set from PTP core<br>valid  |
| OffsetAdjustmentRtc<br>_DatIn  | in  | Clk_TimeAdjust-<br>ment_Type | 1 | Calculated new Off-<br>set from RTC core           |
| OffsetAdjustmentRtc<br>_ValIn  | in  | std_logic;                   | 1 | Calculated new Off-<br>set from RTC core<br>valid  |
| Offset Adjustment Outp         | out |                              |   |                                                    |



| OffsetAdjustment<br>_DatOut   | out | Clk_TimeAdjust-<br>ment_Type | 1 | Calculated new Off-<br>set after the PI<br>Servo loop       |
|-------------------------------|-----|------------------------------|---|-------------------------------------------------------------|
| OffsetAdjustment<br>_ValOut   | out | std_logic;                   | 1 | Calculated new Off-<br>set after the PI<br>Servo loop valid |
| Drift Adjustment Input        |     |                              |   |                                                             |
| DriftAdjustmentTod<br>_DatIn  | in  | Clk_TimeAdjust-<br>ment_Type | 1 | Calculated new Drift from TOD core                          |
| DriftAdjustmentTod<br>_ValIn  | in  | std_logic                    | 1 | Calculated new Drift from TOD core valid                    |
| DriftAdjustmentIrig<br>_DatIn | in  | Clk_TimeAdjust-<br>ment_Type | 1 | Calculated new Drift from IRIG core                         |
| DriftAdjustmentIrig<br>_ValIn | in  | std_logic                    | 1 | Calculated new Drift from IRIG core valid                   |
| DriftAdjustmentPps<br>_DatIn  | in  | Clk_TimeAdjust-<br>ment_Type | 1 | Calculated new Drift from PPS core                          |
| DriftAdjustmentPps<br>_ValIn  | in  | std_logic                    | 1 | Calculated new Drift from PPS core valid                    |
| DriftAdjustmentPtp<br>_DatIn  | in  | Clk_TimeAdjust-<br>ment_Type | 1 | Calculated new Drift from PTP core                          |
| DriftAdjustmentPtp<br>_ValIn  | in  | std_logic                    | 1 | Calculated new Drift from PTP core valid                    |
| DriftAdjustmentRtc<br>_DatIn  | in  | Clk_TimeAdjust-<br>ment_Type | 1 | Calculated new Drift from RTC core                          |
| DriftAdjustmentRtc<br>_ValIn  | in  | std_logic                    | 1 | Calculated new Drift from RTC core valid                    |
| Drift Adjustment Outpu        | ıt  |                              |   |                                                             |
| DriftAdjustment<br>_DatOut    | out | Clk_TimeAdjust-<br>ment_Type | 1 | Calculated new Drift<br>after the PI Servo<br>loop          |
| DriftAdjustment<br>_ValOut    | out | std_logic;                   | 1 | Calculated new Drift<br>after the PI Servo<br>loop valid    |

Table 12: Clock



## 4.2 Design Parts

The Adjustable Counter Clock core consists of a couple of subcores. Each of the subcores itself consist again of smaller function block. The following chapters describe these subcores and their functionality.

### 4.2.1 Clock Selector

## 4.2.1.1 Entity Block Diagram



Figure 5: Clock Selector

## 4.2.1.2 Entity Description

#### **Mux Selector**

This module multiplexes multiple adjustment inputs. Only one adjustment is active at the time. This can be selected by a configuration register. PTP, PPS, TOD, IRIG, RTC, REG and EXT (where the selection is feed from externally) are potential sources. The multiplexer can also be set to disable any adjustment input.

## 4.2.1.3 Entity Declaration

| Name                    | Dir | Туре      | Size | Description                                  |
|-------------------------|-----|-----------|------|----------------------------------------------|
|                         |     | Generics  |      |                                              |
| General                 |     |           |      |                                              |
| ExtSelect_Gen           | Ŧ   | boolean   | 1    | If external selection of sync source is used |
|                         |     | Ports     |      |                                              |
| System                  |     |           |      |                                              |
| SysClk_ClkIn            | in  | std_logic | 1    | System Clock                                 |
| SysRstN_RstIn           | in  | std_logic | 1    | System Reset                                 |
| Adjustment Selector Inp | out |           |      |                                              |



| ExtSelect_DatIn              | in    | Clk_Select_Type              | 1 | Which core shall be the source for the correction when Select_DatIn is external, this is feed from the outside |
|------------------------------|-------|------------------------------|---|----------------------------------------------------------------------------------------------------------------|
| Select_DatIn                 | in    | Clk_Select_Type              | 1 | Which core shall be the source for the correction                                                              |
| Adjustment Selector Ou       | utput |                              | _ |                                                                                                                |
| Selected_DatOut              | out   | Clk_Select_Type              | 1 | Which core was selected as the source for the correction                                                       |
| Time Adjustment Input        |       |                              |   |                                                                                                                |
| TimeAdjustmentTod<br>_DatIn  | in    | Clk_TimeAdjust-<br>ment_Type | 1 | Calculated new Time from TOD core                                                                              |
| TimeAdjustmentTod<br>_ValIn  | in    | std_logic;                   | 1 | Calculated new Time from TOD core valid                                                                        |
| TimeAdjustmentIrig<br>_DatIn | in    | Clk_TimeAdjust-<br>ment_Type | 1 | Calculated new Time from IRIG core                                                                             |
| TimeAdjustmentIrig<br>_ValIn | in    | std_logic;                   | 1 | Calculated new Time from IRIG core valid                                                                       |
| TimeAdjustmentPps<br>_DatIn  | in    | Clk_TimeAdjust-<br>ment_Type | 1 | Calculated new Time from PPS core                                                                              |
| TimeAdjustmentPps<br>_ValIn  | in    | std_logic;                   | 1 | Calculated new Time from PPS core valid                                                                        |
| TimeAdjustmentPtp<br>_DatIn  | in    | Clk_TimeAdjust-<br>ment_Type | 1 | Calculated new Time from PTP core                                                                              |
| TimeAdjustmentPtp<br>_ValIn  | in    | std_logic;                   | 1 | Calculated new Time from PTP core valid                                                                        |
| TimeAdjustmentRtc<br>_DatIn  | in    | Clk_TimeAdjust-<br>ment_Type | 1 | Calculated new Time from RTC core                                                                              |



|                               |     |                 |   | Calculated new      |
|-------------------------------|-----|-----------------|---|---------------------|
| TimeAdjustmentRtc             | in  | std_logic;      | 1 | Time from RTC core  |
| _Valln                        | ''' |                 |   | valid               |
|                               |     | Clk TimeAdjust- |   | Calculated new      |
| TimeAdjustmentReg<br>_DatIn   | in  | ment_Type       | 1 | Time from Registers |
|                               |     | THERE_TYPE      |   | Calculated new      |
| TimeAdjustmentReg             |     |                 | 1 |                     |
| _Valln                        | in  | std_logic;      | 1 | Time from Registers |
| T                             |     |                 |   | valid               |
| Time Adjustment Outpo         | ut  |                 |   | Calculated new      |
| TimeAdjustmentTod             |     | Clk_TimeAdjust- | 1 | Time from the se-   |
| _DatOut                       | out | ment_Type       | ' | lected core         |
|                               |     |                 |   |                     |
| TimeAdjustmentTod             |     | atal la gia     | 1 | Calculated new      |
| _ValOut                       | out | std_logic;      | 1 | Time from the se-   |
|                               |     |                 |   | lected core valid   |
| Offset Adjustment Inpu        | t   | Clk_TimeAdjust- |   | Calculated new Off- |
| OffsetAdjustmentTod<br>DatIn  | in  |                 | 1 | set from TOD core   |
|                               |     | ment_Type       |   |                     |
| OffsetAdjustmentTod           | in  | std_logic;      | 1 | Calculated new Off- |
| _Valln                        |     |                 |   | set from TOD core   |
|                               |     |                 |   | valid               |
| OffsetAdjustmentIrig          | in  | Clk_TimeAdjust- | 1 | Calculated new Off- |
| _DatIn                        | 111 | ment_Type       |   | set from IRIG core  |
|                               |     | std_logic;      | 1 | Calculated new Off- |
| OffsetAdjustmentIrig<br>ValIn | in  |                 |   | set from IRIG core  |
| _vaiiii                       |     |                 |   | valid               |
| OffsetAdjustmentPps           |     | Clk_TimeAdjust- | 1 | Calculated new Off- |
| _Datin                        | in  | ment_Type       |   | set from PPS core   |
|                               |     |                 |   | Calculated new Off- |
| OffsetAdjustmentPps           | in  | std_logic;      | 1 | set from PPS core   |
| _Valln                        | ''' |                 |   | valid               |
|                               |     | Clk TimeAdjust- |   | Calculated new Off- |
| OffsetAdjustmentPtp<br>_DatIn | in  | ment_Type       | 1 | set from PTP core   |
|                               |     | THEFTE_T YPC    |   | Calculated new Off- |
| OffsetAdjustmentPtp           |     | std logic:      | 1 | set from PTP core   |
| _Valln                        | in  | std_logic;      | 1 |                     |
|                               |     | CIL. The A.P.   |   | valid               |
| OffsetAdjustmentRtc           | in  | Clk_TimeAdjust- | 1 | Calculated new Off- |
| _DatIn                        |     | ment_Type       |   | set from RTC core   |



| OffsetAdjustmentRtc<br>_ValIn | in  | std_logic;                   | 1 | Calculated new Off-<br>set from RTC core<br>valid            |
|-------------------------------|-----|------------------------------|---|--------------------------------------------------------------|
| OffsetAdjustmentReg<br>_DatIn | in  | Clk_TimeAdjust-<br>ment_Type | 1 | Calculated new Off-<br>set from Registers                    |
| OffsetAdjustmentReg<br>_ValIn | in  | std_logic;                   | 1 | Calculated new Off-<br>set from Registers<br>valid           |
| Offset Adjustment Outp        | out |                              |   | Calculated new Off-                                          |
| OffsetAdjustment<br>_DatOut   | out | Clk_TimeAdjust-<br>ment_Type | 1 | set from the se-<br>lected core                              |
| OffsetAdjustment<br>_ValOut   | out | std_logic;                   | 1 | Calculated new Off-<br>set from the se-<br>lected core valid |
| Drift Adjustment Input        |     |                              |   |                                                              |
| DriftAdjustmentTod<br>_DatIn  | in  | Clk_TimeAdjust-<br>ment_Type | 1 | Calculated new Drift from TOD core                           |
| DriftAdjustmentTod<br>_ValIn  | in  | std_logic                    | 1 | Calculated new Drift from TOD core valid                     |
| DriftAdjustmentlrig<br>_DatIn | in  | Clk_TimeAdjust-<br>ment_Type | 1 | Calculated new Drift from IRIG core                          |
| DriftAdjustmentIrig<br>_ValIn | in  | std_logic                    | 1 | Calculated new Drift from IRIG core valid                    |
| DriftAdjustmentPps<br>_DatIn  | in  | Clk_TimeAdjust-<br>ment_Type | 1 | Calculated new Drift from PPS core                           |
| DriftAdjustmentPps<br>_ValIn  | in  | std_logic                    | 1 | Calculated new Drift from PPS core valid                     |
| DriftAdjustmentPtp<br>_DatIn  | in  | Clk_TimeAdjust-<br>ment_Type | 1 | Calculated new Drift from PTP core                           |
| DriftAdjustmentPtp<br>_ValIn  | in  | std_logic                    | 1 | Calculated new Drift from PTP core valid                     |
| DriftAdjustmentRtc<br>_DatIn  | in  | Clk_TimeAdjust-<br>ment_Type | 1 | Calculated new Drift from RTC core                           |
| DriftAdjustmentRtc<br>_ValIn  | in  | std_logic                    | 1 | Calculated new Drift from RTC core valid                     |
| DriftAdjustmentReg<br>_DatIn  | in  | Clk_TimeAdjust-<br>ment_Type | 1 | Calculated new Drift from Registers                          |



| DriftAdjustmentReg<br>_ValIn | in  | std_logic                    | 1 | Calculated new Drift from Registers valid         |
|------------------------------|-----|------------------------------|---|---------------------------------------------------|
| Drift Adjustment Outpu       | t   |                              |   |                                                   |
| DriftAdjustment<br>_DatOut   | out | Clk_TimeAdjust-<br>ment_Type | 1 | Calculated new Drift from the selected core       |
| DriftAdjustment<br>_ValOut   | out | std_logic;                   | 1 | Calculated new Drift from the selected core valid |

Table 13: Clock Selector



## 4.2.2 Clock Adjuster

### 4.2.2.1 Entity Block Diagram



Figure 6: Clock Adjuster

## 4.2.2.2 Entity Description

#### PI Servo Loops

This module contains two individual PI Servo loops; one for the Drift and one for Offset correction. Both P and I can be configured individually for each servo loop. The drift is summed up since the corrected drift is only the change to the current frequency of the clock. All PI calculated values are also made available to the other IP cores to do the correct calculations based on the real adjustments.

The multiplications and divisions needed by the PI Servo are either binary multiplications and divisions or made with a fractional multiplier based on the Fractional-Multiply\_Gen generic.

$$OffsetOut = OffsetIn * \frac{OffsetMulP_{Gen}}{OffsetDivP_{Gen}} + \int OffsetIn * \frac{OffsetMulI_{Gen}}{OffsetDivI_{Gen}}$$

$$DriftOut = OldDrift + (DriftIn * \frac{DriftMulP_{Gen}}{DriftDivP_{Gen}} + \int DriftIn * \frac{DriftMulI_{Gen}}{DriftDivI_{Gen}})$$

The following servo loop parameters are used per default in the reference designs:

Offset: 3/4 P, 3/16 I
Drift: 3/4 P, 3/16 I

The servo loop parameters can be changed at runtime if the FractionalMultiply\_Gen and DynamicServoParameters\_Gen generic are true and no static config is done. In this case the PI Mul and Div generics are only used as the default values. The servo paramteres schall only be changed when the core is disabled.



#### **Counter Adjuster**

This module takes the adjustment inputs and converts them into periodic small adjustments which are then set to the Counter Clock. E.g. a drift of 1ns per 1000ns is converted to a 1ns adjustment every 50 clock cycles at a frequency of 50MHz. For offset it works similar. E.g. 50ns shall be corrected in 2000ns which will be converted into 1ns every second clock cycle at 50MHz. The conversion of course also works for non-integer corrections with minimal error (1/- 1ns per correction cycle). In addition it supervises the adjustments ad sets an InSync flag if 4 consecutive offset corrections were below a configurable threshold, a hard set on the clock will cause immediate InSync deasertion.

### 4.2.2.3 Entity Declaration

| Name                                   | Dir | Туре    | Size | Description                                                                                                |  |
|----------------------------------------|-----|---------|------|------------------------------------------------------------------------------------------------------------|--|
| Generics                               |     |         |      |                                                                                                            |  |
| General ClockClkPeriodNano- second_Gen | -   | natural | 1    | Clock Period in Na-<br>nosecond                                                                            |  |
| Sim_Gen                                | -   | boolean | 1    | If in Testbench simulation mode                                                                            |  |
| Clock Adjuster                         |     |         |      | \/_l,fb                                                                                                    |  |
| ClockInHoldover<br>TimeoutSecond_Gen   | -   | natural | 1    | Value after how many seconds after in Sync without new correction values it goes in holdover Default 3 s   |  |
| BypassServo_Gen                        | -   | boolean | 1    | Bypass PI Servo<br>loops                                                                                   |  |
| FractionalMultiply<br>_Gen             | -   | boolean | 1    | Use a fractional multiplication (uses DSP slices) instead of binary multiplication and divisions (default) |  |



| Dy an arei of care as   |       |                 |   | Allow to change the   |
|-------------------------|-------|-----------------|---|-----------------------|
| DynamicServo            | -     | boolean         | 1 | PI Servo parameters   |
| Parameters_Gen          |       |                 |   | at runtime            |
|                         |       |                 |   | Drift Proportional    |
| DriftMulP_Gen           | -     | natural         | 1 | part ratio Numera-    |
|                         |       |                 |   | tor                   |
|                         |       |                 |   | Drift Proportional    |
| DriftDivP_Gen           | -     | natural         | 1 | part ratio Denumer-   |
|                         |       |                 |   | ator                  |
| DriftMull_Gen           |       | natural         | 1 | Drift Integral part   |
| DriitiMuii_Geri         | _     | Haturai         | I | ratio Numerator       |
| DriftDivl_Gen           | _     | natural         | 1 | Drift Integral part   |
| DIIICDIVI_OeII          |       | Haturai         | 1 | ratio Denumerator     |
|                         |       |                 |   | Offset Proportional   |
| OffsetMulP_Gen          | -     | natural         | 1 | part ratio Numera-    |
|                         |       |                 |   | tor                   |
|                         | -     | natural         | 1 | Offset Proportional   |
| OffsetDivP_Gen          |       |                 |   | part ratio Denumer-   |
|                         |       |                 |   | ator                  |
| OffsetMull_Gen          | _     | natural         | 1 | Offset Integral part  |
|                         |       | riacarar        |   | ratio Numerator       |
| OffsetDivl_Gen          | _     | natural         | 1 | Offset Integral part  |
| 3113612111_3311         |       | Tracarar        | , | ratio Denumerator     |
|                         |       | Ports           |   |                       |
| System SysClk_ClkIn     | in    | std_logic       | 1 | System Clock          |
| SysRstN RstIn           | in    | std logic       | 1 | System Reset          |
| Enable Input            |       |                 |   |                       |
| Enable_Enaln            | in    | std_logic       | 1 | Enable Correction     |
| Time Input              |       |                 |   | A discount of Charles |
| ClockTime_DatIn         | in    | Clk_Time_Type   | 1 | Adjusted Clock        |
| _                       |       |                 |   | Time                  |
| ClockTime_ValIn         | in    | std_logic       | 1 | Adjusted Clock        |
|                         | ) ut  |                 |   | Time valid            |
| Adjustment Selector Inp | out - |                 |   | Which core was se-    |
| Selected_DatIn          | in    | Clk_Select_Type | 1 | lected as the source  |
|                         | '''   |                 |   | for the correction    |
| Servo Parameters Input  |       |                 | l |                       |
|                         |       |                 |   |                       |



| ServoOffset<br>FactorP_DatIn              | in  | std_logic_vector             | 32 | Fractional Multiplier Offset P                                              |  |
|-------------------------------------------|-----|------------------------------|----|-----------------------------------------------------------------------------|--|
| ServoOffset<br>Factorl_DatIn              | in  | std_logic_vector             | 32 | Fractional Multiplier<br>Offset I                                           |  |
| ServoDrift<br>FactorP_DatIn               | in  | std_logic_vector             | 32 | Fractional Multiplier Drift P                                               |  |
| ServoDrift<br>Factorl_DatIn               | in  | std_logic_vector             | 32 | Fractional Multiplier Drift I                                               |  |
| Servo_Valln                               | in  | std_logic                    | 1  | New Parameters valid                                                        |  |
| InSync Threshold Input                    |     |                              |    | When the clock is                                                           |  |
| InSync<br>Threshold_DatIn                 | in  | std_logic_vector             | 32 | considered InSync in Nanoseconds                                            |  |
| InSync Output                             |     |                              |    |                                                                             |  |
| InSync_DatOut                             | out | std_logic                    | 1  | Clock Adjustments were below the threshold for at least 4 adjustment cycles |  |
| InHoldover_DatOut                         | out | std_logic                    | 1  | No new Clock Adjustments after in Sync for a defined number of seconds      |  |
| Time Adjustment Input                     |     |                              | 1  |                                                                             |  |
| TimeAdjustment<br>_DatIn                  | in  | Clk_TimeAdjust-<br>ment_Type | 1  | Calculated new Time from selected core                                      |  |
| TimeAdjustment<br>_ValIn                  | in  | std_logic;                   | 1  | Calculated new Time from selected core valid                                |  |
| Offset Adjustment Inpu                    | t   |                              |    | Calculated new Off-                                                         |  |
| OffsetAdjustment<br>_DatIn                | in  | Clk_TimeAdjust-<br>ment_Type | 1  | set from selected core                                                      |  |
| Offset Adjustment  Offset Adjustment Outr | in  | std_logic;                   | 1  | Calculated new Off-<br>set from selected<br>core valid                      |  |
| Offset Adjustment Output                  |     |                              |    |                                                                             |  |



| OffsetAdjustment<br>_DatOut   | out    | Clk_TimeAdjust-<br>ment_Type | 1 | Calculated new Off-<br>set after the PI<br>Servo loop        |
|-------------------------------|--------|------------------------------|---|--------------------------------------------------------------|
| OffsetAdjustment<br>_ValOut   | out    | std_logic;                   | 1 | Calculated new Off-<br>set after the PI<br>Servo loop valid  |
| Drift Adjustment Input        |        |                              |   |                                                              |
| DriftAdjustment<br>_DatIn     | in     | Clk_TimeAdjust-<br>ment_Type | 1 | Calculated new Drift from selected core                      |
| DriftAdjustment<br>_Valln     | in     | std_logic                    | 1 | Calculated new Drift<br>from selected core<br>valid          |
| Drift Adjustment Outpu        | ıt     |                              |   |                                                              |
| DriftAdjustment<br>_DatOut    | out    | Clk_TimeAdjust-<br>ment_Type | 1 | Calculated new Drift<br>after the PI Servo<br>loop           |
| DriftAdjustment<br>_ValOut    | out    | std_logic;                   | 1 | Calculated new Drift<br>after the PI Servo<br>loop valid     |
| Drift Adjustment Sum C        | Dutput |                              |   |                                                              |
| DriftAdjustmentSum<br>_DatOut | out    | Clk_TimeAdjust-<br>ment_Type | 1 | Calculated new Drift after the PI Servo loop as sum with old |
| DriftAdjustmentSum<br>_ValOut | out    | std_logic;                   | 1 | Calculated new Drift after the PI Servo loop as sum with old |
| Time Adjustment Outpu         | ut     |                              |   |                                                              |
| TimeAdjustment                |        | Clk_TimeAdjust-              | 1 | Calculated new                                               |
| _DatOut                       | out    | ment_Type                    | 1 | Time from selected core to clock                             |
| TimeAdjustment<br>_ValOut     | out    | ment_Type std_logic;         | 1 |                                                              |
| <br>TimeAdjustment            | out    |                              |   | core to clock Calculated new Time from selected              |



|                             |     |              |   | after PI servo and  |
|-----------------------------|-----|--------------|---|---------------------|
|                             |     |              |   | conversion to clock |
|                             |     |              |   | Calculated new      |
|                             |     |              |   | counter adjustment  |
| CountAdjust-<br>ment_ValOut | out | std_logic; 1 | 1 | after PI servo and  |
|                             |     |              |   | conversion to clock |
|                             |     |              |   | valid               |

Table 14: Clock Adjuster



## 4.2.3 Clock Counter

### 4.2.3.1 Entity Block Diagram



Figure 7: Clock Counter

### 4.2.3.2 Entity Description

#### **Counter Clock**

This is the adjustable counter clock with nanosecond resolution in a 32 bit second and 32 bit nanosecond format. It can take any input frequency, also non integer values with fractions. It normally adds the clock period to the nanoseconds counter every clock cycle but can add or subtract some extra nanosecond to do the clock adjustment. It also has an overwrite mode where the clock can be set. When the nanosecond counter overflows to the next second the second counter is incremented and the increment is added to the nanosecond part and a billion subtracted to keep the remainder. If fractions are used, every clock cycle the fraction numerator is summed up and when reaching the denominator value an extra nanosecond is added. The time domain is TAI, so after every reset the clock starts from 1.1.1970 at midnight which represents second 0.

## 4.2.3.3 Entity Declaration

| Name                              | Dir      | Туре    | Size | Description                                        |  |  |
|-----------------------------------|----------|---------|------|----------------------------------------------------|--|--|
|                                   | Generics |         |      |                                                    |  |  |
| Clock COunter                     |          |         |      |                                                    |  |  |
| ClockClkPeriodNano-<br>second_Gen | -        | natural | 1    | Clock Period in Na-<br>nosecond                    |  |  |
| ClockClkPeriodFract-<br>Num_Gen   | -        | natural | 1    | Fractional Clock Period Numerator (0 if integer)   |  |  |
| ClockClkPeriod-<br>FractDeNum_Gen | -        | natural | 1    | Fractional Clock Period Denumerator (O if integer) |  |  |
|                                   |          | Ports   |      |                                                    |  |  |



| System                     |     |                              |   |                                                                                                 |
|----------------------------|-----|------------------------------|---|-------------------------------------------------------------------------------------------------|
| SysClk_ClkIn               | in  | std_logic                    | 1 | System Clock                                                                                    |
| SysRstN_RstIn              | in  | std_logic                    | 1 | System Reset                                                                                    |
| Time Output                |     |                              |   |                                                                                                 |
| ClockTime_DatOut           | out | Clk_Time_Type                | 1 | Adjusted Clock<br>Time                                                                          |
| ClockTime_ValOut           | out | std_logic                    | 1 | Adjusted Clock<br>Time valid                                                                    |
| Time Adjustment Input      |     |                              |   |                                                                                                 |
| TimeAdjustment<br>_Dat     | in  | Clk_TimeAdjust-<br>ment_Type | 1 | Overwrite Time                                                                                  |
| TimeAdjustment<br>_Val     | in  | std_logic;                   | 1 | Overwrite Time valid                                                                            |
| Counter Adjustment Inp     | out |                              |   |                                                                                                 |
| CountAdjust-<br>ment_DatIn | in  | Clk_Count<br>Adjustment_Type | 1 | Calculated new counter adjustment after PI servo and conversion to clock                        |
| CountAdjust-<br>ment_Valln | in  | std_logic;                   | 1 | Calculated new counter adjustment after PI servo and conversion to clock valid, make correction |

Table 15: Clock Counter



## 4.2.4 Clock Timer

## 4.2.4.1 Entity Block Diagram



Figure 8: Clock Timer

## 4.2.4.2 Entity Description

#### 1 Millisecond Timer

This module creates a single clock cycle pulse every millisecond aligned with the adjusted counter clock. This pulse is used by all NetTimeLogic's IP cores to calculate timeouts etc. If the counter clock makes a jump in time this can lead to a loss of a pulse or two pulses next to each other.

## 4.2.4.3 Entity Declaration

| Name                | Dir  | Туре           | Size | Description         |  |  |
|---------------------|------|----------------|------|---------------------|--|--|
| Generics            |      |                |      |                     |  |  |
| General             |      |                |      |                     |  |  |
| ClockClkPeriodNano- | _    | <br>  natural  | 1    | Clock Period in Na- |  |  |
| second_Gen          |      | Hatarai        | '    | nosecond            |  |  |
| Clock Adjuster      |      |                |      |                     |  |  |
| TimerIntervalNano-  |      |                |      | Timer Interval be-  |  |  |
|                     | -    | natural        | 1    | tween two events in |  |  |
| second_Gen          |      |                |      | Nanosecond          |  |  |
|                     |      | Ports          |      |                     |  |  |
| System              |      |                |      |                     |  |  |
| SysClk_ClkIn        | in   | std_logic      | 1    | System Clock        |  |  |
| SysRstN_RstIn       | in   | std_logic      | 1    | System Reset        |  |  |
| Time Input          |      |                |      |                     |  |  |
| Clari Time Dalla    | in   | Clk_Time_Type  | 1    | Adjusted Clock      |  |  |
| ClockTime_DatIn     | 1111 | CIK_TITIE_TYPE | '    | Time                |  |  |
|                     | in   | std logic      | 1    | Adjusted Clock      |  |  |
| ClockTime_ValIn     | in   | std_logic      |      | Time valid          |  |  |
| Timer Output        |      | 1              |      |                     |  |  |



| Timer_EvtOut | out | std_logic | 1 | Single clock cycle<br>event every interval<br>aligned with the ad-<br>justed clock |
|--------------|-----|-----------|---|------------------------------------------------------------------------------------|
|--------------|-----|-----------|---|------------------------------------------------------------------------------------|

Table 16: Timer



## 4.2.5 Registerset

### 4.2.5.1 Entity Block Diagram



Figure 9: Registerset

## 4.2.5.2 Entity Description

#### **Register Set**

This module is an AXI Light Memory Mapped Slave. It provides access to all Registers and allows configuring the Adjustable Counter Clock. AXI4 Light only supports 32 bit wide data access, no byte enables, no burst, no simultaneous read and writes and no unaligned access. It can be configured to either run in AXI or StaticConfig mode. If in StaticConfig mode, the configuration of the Datasets is done via signals and can be easily done from within the FPGA without CPU. For each parameter a valid signal is available, the enable signal shall be set last (or simultaneously). To change parameters the clock has to be disabled and enabled again. If in AXI mode, an AXI Master has to configure the Datasets with AXI writes to the registers, which is typically done by a CPU. Parameters can in this case also be changed at runtime. For status supervision, similar to the static configuration the static status signals are available which allows to use the values also directly from within the FPGA without CPU.

## 4.2.5.3 Entity Declaration

| Name                | Dir | Туре    | Size | Description           |  |
|---------------------|-----|---------|------|-----------------------|--|
| Generics            |     |         |      |                       |  |
| Register Set        |     |         |      |                       |  |
| ClockInSyncNanosec- | •   | natural | 1    | Default value for the |  |
| ond_Gen             | _   | natural | I    | threshold when the    |  |



|                      |   |               |                    | clock is considered    |
|----------------------|---|---------------|--------------------|------------------------|
|                      |   |               |                    | in sync                |
| Chatia Causii a Caus |   | L L           | 1                  | If Static Configura-   |
| StaticConfig_Gen     | - | boolean       | 1                  | tion or AXI is used    |
|                      |   |               |                    | If external selection  |
| ExtSelect_Gen        | - | boolean       | 1                  | of sync source is      |
|                      |   |               |                    | used                   |
| LagCarractions Can   |   | boolean       | 1                  | Log corrections to     |
| LogCorrections_Gen   | _ | Doolean       | '                  | registers              |
| PynassSanya Gan      |   | boolean       | 1                  | Bypass PI Servo        |
| BypassServo_Gen      | _ | Doolean       | '                  | loops                  |
|                      |   |               |                    | Use a fractional mul-  |
|                      |   |               |                    | tiplication (uses DSP  |
| FractionalMultiply   | _ | l boolean     | 1                  | slices) instead of bi- |
| _Gen                 |   | Doolean       | '                  | nary multiplication    |
|                      |   |               | and divisions (de- |                        |
|                      |   |               |                    | fault)                 |
| DynamicServo         |   |               |                    | Allow to change the    |
| Parameters_Gen       | - | boolean       | 1                  | PI Servo parameters    |
| Tarameters_cen       |   |               |                    | at runtime             |
|                      |   |               |                    | Drift Proportional     |
| DriftMulP_Gen        | - | natural       | 1                  | part ratio Numera-     |
|                      |   |               |                    | tor                    |
|                      |   |               |                    | Drift Proportional     |
| DriftDivP_Gen        | - | natural       | 1                  | part ratio Denumer-    |
|                      |   |               |                    | ator                   |
| DriftMull_Gen        | _ | <br>  natural | 1                  | Drift Integral part    |
|                      |   |               |                    | ratio Numerator        |
| DriftDivl_Gen        | - | natural       | 1                  | Drift Integral part    |
|                      |   |               |                    | ratio Denumerator      |
|                      |   |               | 4                  | Offset Proportional    |
| OffsetMulP_Gen       | - | natural       | 1                  | part ratio Numera-     |
|                      |   |               |                    | tor                    |
| Official Divide Com- |   | patural       | 1                  | Offset Proportional    |
| OffsetDivP_Gen       | - | natural       | 1                  | part ratio Denumer-    |
|                      |   |               |                    | ator                   |
| OffsetMull_Gen       | - | natural       | 1                  | Offset Integral part   |
|                      |   |               |                    | ratio Numerator        |



| OffsetDivl_Gen                            | -   | natural                         | 1  | Offset Integral part ratio Denumerator       |
|-------------------------------------------|-----|---------------------------------|----|----------------------------------------------|
| AxiAddressRange<br>Low_Gen                | -   | std_logic_vector                | 32 | AXI Base Address                             |
| AxiAddressRange<br>High_Gen               | -   | std_logic_vector                | 32 | AXI Base Address<br>plus Registerset<br>Size |
|                                           |     | Ports                           |    |                                              |
| System                                    | ·   |                                 | 1  |                                              |
| SysClk_ClkIn                              | in  | std_logic                       | 1  | System Clock                                 |
| SysRstN_RstIn                             | in  | std_logic                       | 1  | System Reset                                 |
| Config StaticConfig_DatIn                 | in  | Clk_Clock<br>StaticConfig_Type  | 1  | Static Configuration                         |
| StaticConfig_ValIn                        | in  | Clk_Clock StaticConfigVal _Type | 1  | Static Configuration valid                   |
| Status                                    |     | Cll. Clast                      | ı  | Ctatic Ctatus                                |
| StaticStatus_DatOut                       | out | Clk_Clock StaticStatus_Type     | 1  | Static Status                                |
| StaticStatus_ValOut                       | out | Clk_Clock StaticStatusVal _Type | 1  | Static Status valid                          |
| In Sync Input                             |     |                                 |    |                                              |
| InSync_DatIn                              | in  | std_logic                       | 1  | InSync flag                                  |
| InHoldover_DatIn                          | in  | std_logic                       | 1  | InHoldover flag                              |
| Time Input  ClockTime_DatIn               | in  | Clk_Time_Type                   | 1  | Adjusted Clock<br>Time                       |
| ClockTime_ValIn                           | in  | std_logic                       | 1  | Adjusted Clock<br>Time valid                 |
| AXI4 Light Slave AxiWriteAddrValid _ValIn | in  | std_logic                       | 1  | Write Address Valid                          |
| AxiWriteAddrReady<br>_RdyOut              | out | std_logic                       | 1  | Write Address<br>Ready                       |
| AxiWriteAddrAddress<br>AdrIn              | in  | std_logic_vector                | 32 | Write Address                                |
| AxiWriteAddrProt<br>_DatIn                | in  | std_logic_vector                | 3  | Write Address Pro-<br>tocol                  |



| AxiWriteDataValid<br>Valln      | in    | std_logic        | 1  | Write Data Valid                                                                                                                                                                                                                                                                                                                                |
|---------------------------------|-------|------------------|----|-------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------|
| AxiWriteDataReady<br>RdyOut     | out   | std_logic        | 1  | Write Data Ready                                                                                                                                                                                                                                                                                                                                |
| AxiWriteDataData<br>_DatIn      | in    | std_logic_vector | 32 | Write Data                                                                                                                                                                                                                                                                                                                                      |
| AxiWriteDataStrobe<br>_DatIn    | in    | std_logic_vector | 4  | Write Data Strobe                                                                                                                                                                                                                                                                                                                               |
| AxiWriteRespValid<br>_ValOut    | out   | std_logic        | 1  | Write Response<br>Valid                                                                                                                                                                                                                                                                                                                         |
| AxiWriteRespReady<br>_RdyIn     | in    | std_logic        | 1  | Write Response<br>Ready                                                                                                                                                                                                                                                                                                                         |
| AxiWriteResp<br>Response_DatOut | out   | std_logic_vector | 2  | Write Response                                                                                                                                                                                                                                                                                                                                  |
| AxiReadAddrValid<br>Valln       | in    | std_logic        | 1  | Read Address Valid                                                                                                                                                                                                                                                                                                                              |
| AxiReadAddrReady<br>_RdyOut     | out   | std_logic        | 1  | Read Address<br>Ready                                                                                                                                                                                                                                                                                                                           |
| AxiReadAddrAddress<br>AdrIn     | in    | std_logic_vector | 32 | Read Address                                                                                                                                                                                                                                                                                                                                    |
| AxiReadAddrProt<br>_DatIn       | in    | std_logic_vector | 3  | Read Address Pro-<br>tocol                                                                                                                                                                                                                                                                                                                      |
| AxiReadDataValid<br>ValOut      | out   | std_logic        | 1  | Read Data Valid                                                                                                                                                                                                                                                                                                                                 |
| AxiReadDataReady<br>_RdyIn      | in    | std_logic        | 1  | Read Data Ready                                                                                                                                                                                                                                                                                                                                 |
| AxiReadData<br>Response_DatOut  | out   | std_logic_vector | 2  | Read Data                                                                                                                                                                                                                                                                                                                                       |
| AxiReadDataData<br>_DatOut      | out   | std_logic_vector | 32 | Read Data Re-<br>sponse                                                                                                                                                                                                                                                                                                                         |
| InSync Threshold Outpo          | ut    |                  | l  | When the clock is                                                                                                                                                                                                                                                                                                                               |
| InSync<br>Threshold_DatOut      | out   | std_logic_vector | 32 | considered InSync in Nanoseconds                                                                                                                                                                                                                                                                                                                |
| Adjustment Selector Ou          | utput |                  |    | \\/\big                                                                                                                                                                                                                                                                                                                                     \qq |
| Select_DatOut                   | out   | Clk_Select_Type  | 1  | Which core shall be the source for the correction                                                                                                                                                                                                                                                                                               |
| Servo Parameters Outp           | ut    |                  |    | English and M. D. D.                                                                                                                                                                                                                                                                                                                            |
| ServoOffset<br>FactorP_DatOut   | out   | std_logic_vector | 32 | Fractional Multiplier Offset P                                                                                                                                                                                                                                                                                                                  |
| ServoOffset<br>Factorl_DatOut   | out   | std_logic_vector | 32 | Fractional Multiplier<br>Offset I                                                                                                                                                                                                                                                                                                               |
| ServoDrift<br>FactorP_DatOut    | out   | std_logic_vector | 32 | Fractional Multiplier<br>Drift P                                                                                                                                                                                                                                                                                                                |



| ServoDrift<br>Factorl_DatOut                     | out | std_logic_vector             | 32 | Fractional Multiplier Drift I                            |
|--------------------------------------------------|-----|------------------------------|----|----------------------------------------------------------|
| Servo_ValOut                                     | out | std_logic                    | 1  | New Parameters valid                                     |
| Adjustment Selector In                           | out |                              | I  |                                                          |
| Selected_DatIn                                   | in  | Clk_Select_Type              | 1  | Which core was selected as the source for the correction |
| Offset Adjustment Inpu                           | t   |                              |    |                                                          |
| OffsetAdjustment<br>_DatIn                       | in  | Clk_TimeAdjust-<br>ment_Type | 1  | Calculated new Off-<br>set from selected<br>core         |
| OffsetAdjustment<br>_ValIn                       | in  | std_logic;                   | 1  | Calculated new Off-<br>set from selected<br>core valid   |
| Drift Adjustment Input                           | ı   |                              | Ī  |                                                          |
| DriftAdjustment<br>_DatIn                        | in  | Clk_TimeAdjust-<br>ment_Type | 1  | Calculated new Drift from selected core                  |
| DriftAdjustment<br>_Valln                        | in  | std_logic                    | 1  | Calculated new Drift<br>from selected core<br>valid      |
| Time Adjustment Outpo                            | ut  |                              |    |                                                          |
| TimeAdjustment<br>_DatOut                        | out | Clk_TimeAdjust-<br>ment_Type | 1  | New Time to set                                          |
| TimeAdjustment<br>_ValOut                        | out | std_logic;                   | 1  | New Time to set valid                                    |
| Offset Adjustment Out                            | out |                              |    |                                                          |
| OffsetAdjustment<br>_DatOut                      | out | Clk_TimeAdjust-<br>ment_Type | 1  | New Offset                                               |
| OffsetAdjustment _ValOut _Drift Adjustment Outpu | out | std_logic;                   | 1  | New Offset valid                                         |
| Drift Adjustment Outpu                           |     | Clk_TimeAdjust-              |    | New Drift                                                |
| DriftAdjustment<br>_DatOut                       | out | ment_Type                    | 1  |                                                          |
| DriftAdjustment _ValOut                          | out | std_logic;                   | 1  | New Drift valid                                          |
| Enable Output  ClockEnable_DatOut                | out | std_logic                    | 1  | Enable Adjustable<br>Counter Clock                       |

Table 17: Registerset



## 4.3 Configuration example

In both cases the enabling of the core shall be done last, after or together with the configuration.

## 4.3.1 Static Configuration

```
constant ClkStaticConfig Con : Clk ClockStaticConfig Type := (
  TimeAdjustment
                                 => Clk_TimeAdjustment_Type_Rst_Con,
  OffsetAdjustment
                                 => Clk TimeAdjustment Type Rst Con,
                                 => Clk TimeAdjustment Type Rst Con,
  DriftAdjustment
  ClockSelect
                                 => Clk Select Tod Con,
                                 => x"000001F4" - +/- 500 ns
  InSyncThreshold
);
constant ClkStaticConfigVal Con : Clk ClockStaticConfigVal Type := (
  Enable Val
                                 => '1',
 TimeAdjustment Val
                                 => '0',
                                 => '0',
 OffsetAdjustment Val
  DriftAdjustment Val
                                 => '0'
);
```

Figure 10: Static Configuration

The ClockSelect and InSyncThreshold should be set before enabling, but can also be changed when enabled. TimeAdjustment, DriftAdjustment and OffsetAdjustment can only be set once the clock is enabled and only have an effect if ClockSelect is set to REG.

## 4.3.2 AXI Configuration

The following code is a simplified pseudocode from the testbench: The base address of the Clock is 0x10000000.

```
-- CLOCK
-- Config
-- select: REG
AXI WRITE 10000008 00000006
-- enable adjustable counter clock
AXI WRITE 10000000 00000001
-- set time: 97000000 nanoseconds
AXI WRITE 10000020 39D10680
-- set time: 2 seconds
AXI WRITE 10000024 00000002
-- set valid bits
AXI WRITE 10000000 00000003
-- select: TOD
AXI WRITE 10000008 00000001
-- in sync threshold: 500 ns
AXI WRITE 10000050 000001F4
```



## Figure 11: AXI Configuration

The Clock Select and In Sync Threshold registers should be set before enabling but can also be changed when enabled. Time, Drift and Offset can only be set once the clock is enabled and only have an effect if Clock Select is set to REG. In the example the time is set first 2.97 seconds via register and then the TOD core selected for further adjustments.



## 4.4 Clocking and Reset Concept

## 4.4.1 Clocking

To keep the design as robust and simple as possible, the whole Ordinary Clock, including the Counter Clock and all other cores from NetTimeLogic are run in one clock domain. This is considered to be the system clock. Per default this clock is 50MHz. Where possible also the interfaces are run synchronous to this clock. For clock domain crossing asynchronous fifos with gray counters or message patterns with meat stability flip-flops are used. Clock domain crossings for the AXI interface is moved from the AXI slave to the AXI interconnect.

| Clock         | Frequency          | Description                                                         |
|---------------|--------------------|---------------------------------------------------------------------|
| System        |                    |                                                                     |
| System Clock  | 50MHz<br>(Default) | System clock where the CC runs on as well as the counter clock etc. |
| AXI Interface |                    |                                                                     |
| AXI Clock     | 50MHz<br>(Default) | Internal AXI bus clock, same as the system clock                    |

Table 18: Clocks

## 4.4.2Reset

In connection with the clocks, there is a reset signal for each clock domain. All resets are active low. All resets can be asynchronously set and shall be synchronously released with the corresponding clock domain. All resets shall be asserted for the first couple (around 8) clock cycles. All resets shall be set simultaneously and released simultaneously to avoid overflow conditions in the core. See the reference designs top file for an example of how the reset shall be handled.

| Reset         | Polarity   | Description                                                 |
|---------------|------------|-------------------------------------------------------------|
| System        |            |                                                             |
| System Reset  | Active low | Asynchronous set, synchronous release with the system clock |
| AXI Interface |            |                                                             |



| AXI Reset |            | Asynchronous set, synchronous release    |  |
|-----------|------------|------------------------------------------|--|
|           | Active low | with the AXI clock, which is the same as |  |
|           |            | the system clock                         |  |

Table 19: Resets



# 5 Resource Usage

Since the FPGA Architecture between vendors and FPGA families differ there is a split up into the two major FPGA vendors.

# 5.1 Altera (Cyclone V)

| Configuration           | FFs  | LUTs | BRAMs | DSPs |
|-------------------------|------|------|-------|------|
| Minimal (No PI Servo)   | 1868 | 3846 | 0     | 0    |
| Maximal (With PI Servo) | 3093 | 6484 | 0     | 0    |

Table 20: Resource Usage Altera

# 5.2 Xilinx (Artix 7)

| Configuration           | FFs  | LUTs | BRAMs | DSPs |
|-------------------------|------|------|-------|------|
| Minimal (No Pl Servo)   | 1872 | 3878 | 0     | 0    |
| Maximal (With PI Servo) | 3094 | 6744 | 0     | 0    |

Table 21: Resource Usage Xilinx



## **6 Delivery Structure**

AXI -- AXI library folder

CLK -- CLK library folder -- CLK library cores

|-Doc -- CLK library cores documentations

|-Driver -- CLK library driver

|-Library -- CLK library component sources |-Package -- CLK library package sources

|-Testbench -- CLK library cores testbench sources and sim/log

COMMON -- COMMON library folder

PPS -- PPS library folder

SIM -- SIM library folder

|-Testbench -- SIM library testbench template sources



## 7 Testbench

The Clock testbench consist of 1 parse/port type: AXI.

For configuration and result checks an AXI read and write port is used. The time can be written and read back.



Figure 12: Testbench Framework

For more information on the testbench framework check the Sim\_ReferenceManual documentation.

With the Sim parameter set the time base for timeouts are divided by 1000 to 100000 to speed up simulation time.

### 7.1 Run Testbench

 Run the general script first source XXX/SIM/Tools/source\_with\_args.tcl

2. Start the testbench with all test cases src XXX/CLK/Testbench/Core/ClkClock/Script/run Clk Clock Tb.tcl

3. Check the log file LogFile1.txt in the XXX/CLK/Testbench/Core/ClkClock/Log/folder for simulation results.



## 8 Reference Designs

The Adjustable Counter Clock reference design contains a PLL to generate all necessary clocks (cores are run at 50 MHz) and an instance of the Adjustable Counter Clock IP core. Optionally it also contains an instance of a PPS Master Clock IP core (has to be purchased separately). To instantiate the optional IP core, change the corresponding generic (PpsMasterAvailable\_Gen) to true via the tool specific wizards.

The Reference Design is intended to run just standalone, show the instantiation and generate a PPS output. The PPS Master Clock is used to create a PPS output which is compensated for the output delay and has a configurable duty cycle, if not available an uncompensated PPS is directly generated out of the MSB of the Time. All generics can be adapted to the specific needs.



Figure 13: Reference Design

### 8.1 Altera: Terasic SocKit

The SocKit board is an FPGA board from Terasic Inc. with a Cyclone V SoC FPGA from Altera. (<a href="http://www.terasic.com.tw/cgi-bin/page/archive.pl?Language=Eng-lish&CategoryNo=205&No=816">http://www.terasic.com.tw/cgi-bin/page/archive.pl?Language=Eng-lish&CategoryNo=205&No=816</a>)

- 1. Open Quartus 16.x
- 2. Open Project /CLK/Refdesign/Altera/SocKit/ClkClock/ClkClock.qpf
- 3. If the optional core PPS Master Clock is available add the files from the corresponding folders (PPS/Core, PPS/Library and PPS/Package)
- 4. Change the generics (PpsMasterAvailable\_Gen) in Quartus (in the settings menu, not in VHDL) to true for the optional cores that are available.
- 5. Rerun implementation
- 6. Download to FPGA via JTAG





Figure 14: SocKit (source Terasic Inc)

For the ports on the HSMC connector the GPIO to HSMC adapter from Terasic Inc. was used.

# 8.2 Xilinx: Digilent Arty

The Arty board is an FPGA board from Digilent Inc. with an Artix7 FPGA from Xilinx. (<a href="http://store.digilentinc.com/arty-board-artix-7-fpga-development-board-for-makers-and-hobbyists/">http://store.digilentinc.com/arty-board-artix-7-fpga-development-board-for-makers-and-hobbyists/</a>

- 1. Open Vivado 2017.4
- 2. Run TCL script /CLK/Refdesign/Xilinx/Arty/ClkClock/ClkClock.tcl
  - a. This has to be run only the first time and will create a new Vivado Project
- 3. If the project has been created before open the project and do not rerun the project TCL
- 4. If the optional core PPS Master Clock is available add the files from the corresponding folders (PPS/Core, PPS/Library and PPS/Package) to the corresponding Library (PpsLib).
- 5. Change the generics (PpsMasterAvailable\_Gen) in Vivado (in the settings menu, not in VHDL) to true for the optional cores that are available.



- 6. Rerun implementation
- 7. Download to FPGA via JTAG



Figure 15: Arty (source Digilent Inc)



# A List of tables

| Table 1:   | Revision History              | 4  |
|------------|-------------------------------|----|
| Table 2:   | Definitions                   | 7  |
| Table 3:   | Abbreviations                 | 7  |
| Table 4:   | Parameters                    | 39 |
| Table 5:   | Clk_Time_Type                 | 39 |
| Table 6:   | Clk_TimeAdjustment_Type       | 40 |
| Table 7:   | Clk_CoreInfo_Type             | 40 |
| Table 8:   | Clk_ClockStaticConfig_Type    | 41 |
| Table 9:   | Clk_ClockStaticConfigVal_Type | 41 |
| Table 10:  | Clk_ClockStaticStatus_Type    | 41 |
| Table 11:  | Clk_ClockStaticStatusVal_Type | 42 |
| Table 12:  | Clock                         | 49 |
| Table 13:  | Clock Selector                | 54 |
| Table 14:  | Clock Adjuster                | 60 |
| Table 15:  | Clock Counter                 | 62 |
| Table 16:  | Timer                         | 64 |
| Table 17:  | Registerset                   | 69 |
| Table 18:  | Clocks                        | 72 |
| Table 19:  | Resets                        | 73 |
| Table 20:  | Resource Usage Altera         | 74 |
| Table 21:  | Resource Usage Xilinx         | 74 |
| B List     | of figures                    |    |
| Figure 1:  | Context Block Diagram         | 8  |
| Figure 2:  | Architecture Block Diagram    | 10 |
| Figure 3:  | Counter Clock                 | 12 |
| Figure 4:  | Adjustable Counter Clock      | 42 |
| Figure 5:  | Clock Selector                | 50 |
| Figure 6:  | Clock Adjuster                | 55 |
| Figure 7:  | Clock Counter                 | 61 |
| Figure 8:  | Clock Timer                   | 63 |
| Figure 9:  | Registerset                   | 65 |
| Figure 10: | Static Configuration          | 70 |
| Figure 11: | AXI Configuration             | 71 |
| Figure 12: | Testbench Framework           | 76 |
|            |                               |    |



| Figure 13: | Reference Design            | 77 |
|------------|-----------------------------|----|
| Figure 14: | SocKit (source Terasic Inc) | 78 |
| Figure 15: | Arty (source Digilent Inc)  | 79 |