Adding upstream version 4.6.1.
Signed-off-by: Daniel Baumann <daniel.baumann@progress-linux.org>
This commit is contained in:
parent
98773d3bb9
commit
1fb0d187c9
305 changed files with 87945 additions and 0 deletions
339
COPYING
Normal file
339
COPYING
Normal file
|
@ -0,0 +1,339 @@
|
||||||
|
GNU GENERAL PUBLIC LICENSE
|
||||||
|
Version 2, June 1991
|
||||||
|
|
||||||
|
Copyright (C) 1989, 1991 Free Software Foundation, Inc.,
|
||||||
|
51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA
|
||||||
|
Everyone is permitted to copy and distribute verbatim copies
|
||||||
|
of this license document, but changing it is not allowed.
|
||||||
|
|
||||||
|
Preamble
|
||||||
|
|
||||||
|
The licenses for most software are designed to take away your
|
||||||
|
freedom to share and change it. By contrast, the GNU General Public
|
||||||
|
License is intended to guarantee your freedom to share and change free
|
||||||
|
software--to make sure the software is free for all its users. This
|
||||||
|
General Public License applies to most of the Free Software
|
||||||
|
Foundation's software and to any other program whose authors commit to
|
||||||
|
using it. (Some other Free Software Foundation software is covered by
|
||||||
|
the GNU Lesser General Public License instead.) You can apply it to
|
||||||
|
your programs, too.
|
||||||
|
|
||||||
|
When we speak of free software, we are referring to freedom, not
|
||||||
|
price. Our General Public Licenses are designed to make sure that you
|
||||||
|
have the freedom to distribute copies of free software (and charge for
|
||||||
|
this service if you wish), that you receive source code or can get it
|
||||||
|
if you want it, that you can change the software or use pieces of it
|
||||||
|
in new free programs; and that you know you can do these things.
|
||||||
|
|
||||||
|
To protect your rights, we need to make restrictions that forbid
|
||||||
|
anyone to deny you these rights or to ask you to surrender the rights.
|
||||||
|
These restrictions translate to certain responsibilities for you if you
|
||||||
|
distribute copies of the software, or if you modify it.
|
||||||
|
|
||||||
|
For example, if you distribute copies of such a program, whether
|
||||||
|
gratis or for a fee, you must give the recipients all the rights that
|
||||||
|
you have. You must make sure that they, too, receive or can get the
|
||||||
|
source code. And you must show them these terms so they know their
|
||||||
|
rights.
|
||||||
|
|
||||||
|
We protect your rights with two steps: (1) copyright the software, and
|
||||||
|
(2) offer you this license which gives you legal permission to copy,
|
||||||
|
distribute and/or modify the software.
|
||||||
|
|
||||||
|
Also, for each author's protection and ours, we want to make certain
|
||||||
|
that everyone understands that there is no warranty for this free
|
||||||
|
software. If the software is modified by someone else and passed on, we
|
||||||
|
want its recipients to know that what they have is not the original, so
|
||||||
|
that any problems introduced by others will not reflect on the original
|
||||||
|
authors' reputations.
|
||||||
|
|
||||||
|
Finally, any free program is threatened constantly by software
|
||||||
|
patents. We wish to avoid the danger that redistributors of a free
|
||||||
|
program will individually obtain patent licenses, in effect making the
|
||||||
|
program proprietary. To prevent this, we have made it clear that any
|
||||||
|
patent must be licensed for everyone's free use or not licensed at all.
|
||||||
|
|
||||||
|
The precise terms and conditions for copying, distribution and
|
||||||
|
modification follow.
|
||||||
|
|
||||||
|
GNU GENERAL PUBLIC LICENSE
|
||||||
|
TERMS AND CONDITIONS FOR COPYING, DISTRIBUTION AND MODIFICATION
|
||||||
|
|
||||||
|
0. This License applies to any program or other work which contains
|
||||||
|
a notice placed by the copyright holder saying it may be distributed
|
||||||
|
under the terms of this General Public License. The "Program", below,
|
||||||
|
refers to any such program or work, and a "work based on the Program"
|
||||||
|
means either the Program or any derivative work under copyright law:
|
||||||
|
that is to say, a work containing the Program or a portion of it,
|
||||||
|
either verbatim or with modifications and/or translated into another
|
||||||
|
language. (Hereinafter, translation is included without limitation in
|
||||||
|
the term "modification".) Each licensee is addressed as "you".
|
||||||
|
|
||||||
|
Activities other than copying, distribution and modification are not
|
||||||
|
covered by this License; they are outside its scope. The act of
|
||||||
|
running the Program is not restricted, and the output from the Program
|
||||||
|
is covered only if its contents constitute a work based on the
|
||||||
|
Program (independent of having been made by running the Program).
|
||||||
|
Whether that is true depends on what the Program does.
|
||||||
|
|
||||||
|
1. You may copy and distribute verbatim copies of the Program's
|
||||||
|
source code as you receive it, in any medium, provided that you
|
||||||
|
conspicuously and appropriately publish on each copy an appropriate
|
||||||
|
copyright notice and disclaimer of warranty; keep intact all the
|
||||||
|
notices that refer to this License and to the absence of any warranty;
|
||||||
|
and give any other recipients of the Program a copy of this License
|
||||||
|
along with the Program.
|
||||||
|
|
||||||
|
You may charge a fee for the physical act of transferring a copy, and
|
||||||
|
you may at your option offer warranty protection in exchange for a fee.
|
||||||
|
|
||||||
|
2. You may modify your copy or copies of the Program or any portion
|
||||||
|
of it, thus forming a work based on the Program, and copy and
|
||||||
|
distribute such modifications or work under the terms of Section 1
|
||||||
|
above, provided that you also meet all of these conditions:
|
||||||
|
|
||||||
|
a) You must cause the modified files to carry prominent notices
|
||||||
|
stating that you changed the files and the date of any change.
|
||||||
|
|
||||||
|
b) You must cause any work that you distribute or publish, that in
|
||||||
|
whole or in part contains or is derived from the Program or any
|
||||||
|
part thereof, to be licensed as a whole at no charge to all third
|
||||||
|
parties under the terms of this License.
|
||||||
|
|
||||||
|
c) If the modified program normally reads commands interactively
|
||||||
|
when run, you must cause it, when started running for such
|
||||||
|
interactive use in the most ordinary way, to print or display an
|
||||||
|
announcement including an appropriate copyright notice and a
|
||||||
|
notice that there is no warranty (or else, saying that you provide
|
||||||
|
a warranty) and that users may redistribute the program under
|
||||||
|
these conditions, and telling the user how to view a copy of this
|
||||||
|
License. (Exception: if the Program itself is interactive but
|
||||||
|
does not normally print such an announcement, your work based on
|
||||||
|
the Program is not required to print an announcement.)
|
||||||
|
|
||||||
|
These requirements apply to the modified work as a whole. If
|
||||||
|
identifiable sections of that work are not derived from the Program,
|
||||||
|
and can be reasonably considered independent and separate works in
|
||||||
|
themselves, then this License, and its terms, do not apply to those
|
||||||
|
sections when you distribute them as separate works. But when you
|
||||||
|
distribute the same sections as part of a whole which is a work based
|
||||||
|
on the Program, the distribution of the whole must be on the terms of
|
||||||
|
this License, whose permissions for other licensees extend to the
|
||||||
|
entire whole, and thus to each and every part regardless of who wrote it.
|
||||||
|
|
||||||
|
Thus, it is not the intent of this section to claim rights or contest
|
||||||
|
your rights to work written entirely by you; rather, the intent is to
|
||||||
|
exercise the right to control the distribution of derivative or
|
||||||
|
collective works based on the Program.
|
||||||
|
|
||||||
|
In addition, mere aggregation of another work not based on the Program
|
||||||
|
with the Program (or with a work based on the Program) on a volume of
|
||||||
|
a storage or distribution medium does not bring the other work under
|
||||||
|
the scope of this License.
|
||||||
|
|
||||||
|
3. You may copy and distribute the Program (or a work based on it,
|
||||||
|
under Section 2) in object code or executable form under the terms of
|
||||||
|
Sections 1 and 2 above provided that you also do one of the following:
|
||||||
|
|
||||||
|
a) Accompany it with the complete corresponding machine-readable
|
||||||
|
source code, which must be distributed under the terms of Sections
|
||||||
|
1 and 2 above on a medium customarily used for software interchange; or,
|
||||||
|
|
||||||
|
b) Accompany it with a written offer, valid for at least three
|
||||||
|
years, to give any third party, for a charge no more than your
|
||||||
|
cost of physically performing source distribution, a complete
|
||||||
|
machine-readable copy of the corresponding source code, to be
|
||||||
|
distributed under the terms of Sections 1 and 2 above on a medium
|
||||||
|
customarily used for software interchange; or,
|
||||||
|
|
||||||
|
c) Accompany it with the information you received as to the offer
|
||||||
|
to distribute corresponding source code. (This alternative is
|
||||||
|
allowed only for noncommercial distribution and only if you
|
||||||
|
received the program in object code or executable form with such
|
||||||
|
an offer, in accord with Subsection b above.)
|
||||||
|
|
||||||
|
The source code for a work means the preferred form of the work for
|
||||||
|
making modifications to it. For an executable work, complete source
|
||||||
|
code means all the source code for all modules it contains, plus any
|
||||||
|
associated interface definition files, plus the scripts used to
|
||||||
|
control compilation and installation of the executable. However, as a
|
||||||
|
special exception, the source code distributed need not include
|
||||||
|
anything that is normally distributed (in either source or binary
|
||||||
|
form) with the major components (compiler, kernel, and so on) of the
|
||||||
|
operating system on which the executable runs, unless that component
|
||||||
|
itself accompanies the executable.
|
||||||
|
|
||||||
|
If distribution of executable or object code is made by offering
|
||||||
|
access to copy from a designated place, then offering equivalent
|
||||||
|
access to copy the source code from the same place counts as
|
||||||
|
distribution of the source code, even though third parties are not
|
||||||
|
compelled to copy the source along with the object code.
|
||||||
|
|
||||||
|
4. You may not copy, modify, sublicense, or distribute the Program
|
||||||
|
except as expressly provided under this License. Any attempt
|
||||||
|
otherwise to copy, modify, sublicense or distribute the Program is
|
||||||
|
void, and will automatically terminate your rights under this License.
|
||||||
|
However, parties who have received copies, or rights, from you under
|
||||||
|
this License will not have their licenses terminated so long as such
|
||||||
|
parties remain in full compliance.
|
||||||
|
|
||||||
|
5. You are not required to accept this License, since you have not
|
||||||
|
signed it. However, nothing else grants you permission to modify or
|
||||||
|
distribute the Program or its derivative works. These actions are
|
||||||
|
prohibited by law if you do not accept this License. Therefore, by
|
||||||
|
modifying or distributing the Program (or any work based on the
|
||||||
|
Program), you indicate your acceptance of this License to do so, and
|
||||||
|
all its terms and conditions for copying, distributing or modifying
|
||||||
|
the Program or works based on it.
|
||||||
|
|
||||||
|
6. Each time you redistribute the Program (or any work based on the
|
||||||
|
Program), the recipient automatically receives a license from the
|
||||||
|
original licensor to copy, distribute or modify the Program subject to
|
||||||
|
these terms and conditions. You may not impose any further
|
||||||
|
restrictions on the recipients' exercise of the rights granted herein.
|
||||||
|
You are not responsible for enforcing compliance by third parties to
|
||||||
|
this License.
|
||||||
|
|
||||||
|
7. If, as a consequence of a court judgment or allegation of patent
|
||||||
|
infringement or for any other reason (not limited to patent issues),
|
||||||
|
conditions are imposed on you (whether by court order, agreement or
|
||||||
|
otherwise) that contradict the conditions of this License, they do not
|
||||||
|
excuse you from the conditions of this License. If you cannot
|
||||||
|
distribute so as to satisfy simultaneously your obligations under this
|
||||||
|
License and any other pertinent obligations, then as a consequence you
|
||||||
|
may not distribute the Program at all. For example, if a patent
|
||||||
|
license would not permit royalty-free redistribution of the Program by
|
||||||
|
all those who receive copies directly or indirectly through you, then
|
||||||
|
the only way you could satisfy both it and this License would be to
|
||||||
|
refrain entirely from distribution of the Program.
|
||||||
|
|
||||||
|
If any portion of this section is held invalid or unenforceable under
|
||||||
|
any particular circumstance, the balance of the section is intended to
|
||||||
|
apply and the section as a whole is intended to apply in other
|
||||||
|
circumstances.
|
||||||
|
|
||||||
|
It is not the purpose of this section to induce you to infringe any
|
||||||
|
patents or other property right claims or to contest validity of any
|
||||||
|
such claims; this section has the sole purpose of protecting the
|
||||||
|
integrity of the free software distribution system, which is
|
||||||
|
implemented by public license practices. Many people have made
|
||||||
|
generous contributions to the wide range of software distributed
|
||||||
|
through that system in reliance on consistent application of that
|
||||||
|
system; it is up to the author/donor to decide if he or she is willing
|
||||||
|
to distribute software through any other system and a licensee cannot
|
||||||
|
impose that choice.
|
||||||
|
|
||||||
|
This section is intended to make thoroughly clear what is believed to
|
||||||
|
be a consequence of the rest of this License.
|
||||||
|
|
||||||
|
8. If the distribution and/or use of the Program is restricted in
|
||||||
|
certain countries either by patents or by copyrighted interfaces, the
|
||||||
|
original copyright holder who places the Program under this License
|
||||||
|
may add an explicit geographical distribution limitation excluding
|
||||||
|
those countries, so that distribution is permitted only in or among
|
||||||
|
countries not thus excluded. In such case, this License incorporates
|
||||||
|
the limitation as if written in the body of this License.
|
||||||
|
|
||||||
|
9. The Free Software Foundation may publish revised and/or new versions
|
||||||
|
of the General Public License from time to time. Such new versions will
|
||||||
|
be similar in spirit to the present version, but may differ in detail to
|
||||||
|
address new problems or concerns.
|
||||||
|
|
||||||
|
Each version is given a distinguishing version number. If the Program
|
||||||
|
specifies a version number of this License which applies to it and "any
|
||||||
|
later version", you have the option of following the terms and conditions
|
||||||
|
either of that version or of any later version published by the Free
|
||||||
|
Software Foundation. If the Program does not specify a version number of
|
||||||
|
this License, you may choose any version ever published by the Free Software
|
||||||
|
Foundation.
|
||||||
|
|
||||||
|
10. If you wish to incorporate parts of the Program into other free
|
||||||
|
programs whose distribution conditions are different, write to the author
|
||||||
|
to ask for permission. For software which is copyrighted by the Free
|
||||||
|
Software Foundation, write to the Free Software Foundation; we sometimes
|
||||||
|
make exceptions for this. Our decision will be guided by the two goals
|
||||||
|
of preserving the free status of all derivatives of our free software and
|
||||||
|
of promoting the sharing and reuse of software generally.
|
||||||
|
|
||||||
|
NO WARRANTY
|
||||||
|
|
||||||
|
11. BECAUSE THE PROGRAM IS LICENSED FREE OF CHARGE, THERE IS NO WARRANTY
|
||||||
|
FOR THE PROGRAM, TO THE EXTENT PERMITTED BY APPLICABLE LAW. EXCEPT WHEN
|
||||||
|
OTHERWISE STATED IN WRITING THE COPYRIGHT HOLDERS AND/OR OTHER PARTIES
|
||||||
|
PROVIDE THE PROGRAM "AS IS" WITHOUT WARRANTY OF ANY KIND, EITHER EXPRESSED
|
||||||
|
OR IMPLIED, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED WARRANTIES OF
|
||||||
|
MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE. THE ENTIRE RISK AS
|
||||||
|
TO THE QUALITY AND PERFORMANCE OF THE PROGRAM IS WITH YOU. SHOULD THE
|
||||||
|
PROGRAM PROVE DEFECTIVE, YOU ASSUME THE COST OF ALL NECESSARY SERVICING,
|
||||||
|
REPAIR OR CORRECTION.
|
||||||
|
|
||||||
|
12. IN NO EVENT UNLESS REQUIRED BY APPLICABLE LAW OR AGREED TO IN WRITING
|
||||||
|
WILL ANY COPYRIGHT HOLDER, OR ANY OTHER PARTY WHO MAY MODIFY AND/OR
|
||||||
|
REDISTRIBUTE THE PROGRAM AS PERMITTED ABOVE, BE LIABLE TO YOU FOR DAMAGES,
|
||||||
|
INCLUDING ANY GENERAL, SPECIAL, INCIDENTAL OR CONSEQUENTIAL DAMAGES ARISING
|
||||||
|
OUT OF THE USE OR INABILITY TO USE THE PROGRAM (INCLUDING BUT NOT LIMITED
|
||||||
|
TO LOSS OF DATA OR DATA BEING RENDERED INACCURATE OR LOSSES SUSTAINED BY
|
||||||
|
YOU OR THIRD PARTIES OR A FAILURE OF THE PROGRAM TO OPERATE WITH ANY OTHER
|
||||||
|
PROGRAMS), EVEN IF SUCH HOLDER OR OTHER PARTY HAS BEEN ADVISED OF THE
|
||||||
|
POSSIBILITY OF SUCH DAMAGES.
|
||||||
|
|
||||||
|
END OF TERMS AND CONDITIONS
|
||||||
|
|
||||||
|
How to Apply These Terms to Your New Programs
|
||||||
|
|
||||||
|
If you develop a new program, and you want it to be of the greatest
|
||||||
|
possible use to the public, the best way to achieve this is to make it
|
||||||
|
free software which everyone can redistribute and change under these terms.
|
||||||
|
|
||||||
|
To do so, attach the following notices to the program. It is safest
|
||||||
|
to attach them to the start of each source file to most effectively
|
||||||
|
convey the exclusion of warranty; and each file should have at least
|
||||||
|
the "copyright" line and a pointer to where the full notice is found.
|
||||||
|
|
||||||
|
<one line to give the program's name and a brief idea of what it does.>
|
||||||
|
Copyright (C) <year> <name of author>
|
||||||
|
|
||||||
|
This program is free software; you can redistribute it and/or modify
|
||||||
|
it under the terms of the GNU General Public License as published by
|
||||||
|
the Free Software Foundation; either version 2 of the License, or
|
||||||
|
(at your option) any later version.
|
||||||
|
|
||||||
|
This program is distributed in the hope that it will be useful,
|
||||||
|
but WITHOUT ANY WARRANTY; without even the implied warranty of
|
||||||
|
MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
||||||
|
GNU General Public License for more details.
|
||||||
|
|
||||||
|
You should have received a copy of the GNU General Public License along
|
||||||
|
with this program; if not, write to the Free Software Foundation, Inc.,
|
||||||
|
51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA.
|
||||||
|
|
||||||
|
Also add information on how to contact you by electronic and paper mail.
|
||||||
|
|
||||||
|
If the program is interactive, make it output a short notice like this
|
||||||
|
when it starts in an interactive mode:
|
||||||
|
|
||||||
|
Gnomovision version 69, Copyright (C) year name of author
|
||||||
|
Gnomovision comes with ABSOLUTELY NO WARRANTY; for details type `show w'.
|
||||||
|
This is free software, and you are welcome to redistribute it
|
||||||
|
under certain conditions; type `show c' for details.
|
||||||
|
|
||||||
|
The hypothetical commands `show w' and `show c' should show the appropriate
|
||||||
|
parts of the General Public License. Of course, the commands you use may
|
||||||
|
be called something other than `show w' and `show c'; they could even be
|
||||||
|
mouse-clicks or menu items--whatever suits your program.
|
||||||
|
|
||||||
|
You should also get your employer (if you work as a programmer) or your
|
||||||
|
school, if any, to sign a "copyright disclaimer" for the program, if
|
||||||
|
necessary. Here is a sample; alter the names:
|
||||||
|
|
||||||
|
Yoyodyne, Inc., hereby disclaims all copyright interest in the program
|
||||||
|
`Gnomovision' (which makes passes at compilers) written by James Hacker.
|
||||||
|
|
||||||
|
<signature of Ty Coon>, 1 April 1989
|
||||||
|
Ty Coon, President of Vice
|
||||||
|
|
||||||
|
This General Public License does not permit incorporating your program into
|
||||||
|
proprietary programs. If your program is a subroutine library, you may
|
||||||
|
consider it more useful to permit linking proprietary applications with the
|
||||||
|
library. If this is what you want to do, use the GNU Lesser General
|
||||||
|
Public License instead of this License.
|
165
INSTALL
Normal file
165
INSTALL
Normal file
|
@ -0,0 +1,165 @@
|
||||||
|
Installation
|
||||||
|
|
||||||
|
The software is distributed as source code which has to be compiled. The source
|
||||||
|
code is supplied in the form of a gzipped tar file, which unpacks to a
|
||||||
|
subdirectory identifying the name and version of the program.
|
||||||
|
|
||||||
|
A C compiler (e.g. gcc or clang) and GNU Make are needed to build chrony. The
|
||||||
|
following libraries with their development files, and programs, are needed to
|
||||||
|
enable optional features:
|
||||||
|
|
||||||
|
o pkg-config: detection of development libraries
|
||||||
|
|
||||||
|
o Nettle, GnuTLS, NSS, or LibTomCrypt: secure hash functions (SECHASH)
|
||||||
|
|
||||||
|
o libcap: dropping root privileges on Linux (DROPROOT)
|
||||||
|
|
||||||
|
o libseccomp: system call filter on Linux (SCFILTER)
|
||||||
|
|
||||||
|
o GnuTLS and Nettle: Network Time Security (NTS)
|
||||||
|
|
||||||
|
o Editline: line editing in chronyc (READLINE)
|
||||||
|
|
||||||
|
o timepps.h header: PPS reference clock
|
||||||
|
|
||||||
|
o Asciidoctor: documentation in HTML format
|
||||||
|
|
||||||
|
o Bash: test suite
|
||||||
|
|
||||||
|
The following programs are needed when building chrony from the git repository
|
||||||
|
instead of a released tar file:
|
||||||
|
|
||||||
|
o Asciidoctor: manual pages
|
||||||
|
|
||||||
|
o Bison: parser for chronyc settime command
|
||||||
|
|
||||||
|
After unpacking the source code, change directory into it, and type
|
||||||
|
|
||||||
|
./configure
|
||||||
|
|
||||||
|
This is a shell script that automatically determines the system type. There is
|
||||||
|
an optional parameter --prefix, which indicates the directory tree where the
|
||||||
|
software should be installed. For example,
|
||||||
|
|
||||||
|
./configure --prefix=/opt/free
|
||||||
|
|
||||||
|
will install the chronyd daemon into /opt/free/sbin and the chronyc control
|
||||||
|
program into /opt/free/bin. The default value for the prefix is /usr/local.
|
||||||
|
|
||||||
|
The configure script assumes you want to use gcc as your compiler. If you want
|
||||||
|
to use a different compiler, you can configure this way:
|
||||||
|
|
||||||
|
CC=cc ./configure --prefix=/opt/free
|
||||||
|
|
||||||
|
for Bourne-family shells, or
|
||||||
|
|
||||||
|
setenv CC cc
|
||||||
|
setenv CFLAGS -O
|
||||||
|
./configure --prefix=/opt/free
|
||||||
|
|
||||||
|
for C-family shells.
|
||||||
|
|
||||||
|
If the software cannot (yet) be built on your system, an error message will be
|
||||||
|
shown. Otherwise, Makefile will be generated.
|
||||||
|
|
||||||
|
On Linux, if development files for the libcap library are available, chronyd
|
||||||
|
will be built with support for dropping root privileges. On other systems no
|
||||||
|
extra library is needed. The default user which chronyd should run as can be
|
||||||
|
specified with the --with-user option of the configure script.
|
||||||
|
|
||||||
|
If development files for the POSIX threads library are available, chronyd will
|
||||||
|
be built with support for asynchronous resolving of hostnames specified in the
|
||||||
|
server, peer, and pool directives. This allows chronyd operating as a server to
|
||||||
|
respond to client requests when resolving a hostname. If you don't want to
|
||||||
|
enable the support, specify the --disable-asyncdns flag to configure.
|
||||||
|
|
||||||
|
If development files for the Nettle, NSS, or libtomcrypt library are available,
|
||||||
|
chronyd will be built with support for other cryptographic hash functions than
|
||||||
|
MD5, which can be used for NTP authentication with a symmetric key. If you
|
||||||
|
don't want to enable the support, specify the --disable-sechash flag to
|
||||||
|
configure.
|
||||||
|
|
||||||
|
If development files for the editline library are available, chronyc will be
|
||||||
|
built with line editing support. If you don't want this, specify the
|
||||||
|
--disable-readline flag to configure.
|
||||||
|
|
||||||
|
If a timepps.h header is available (e.g. from the LinuxPPS project), chronyd
|
||||||
|
will be built with PPS API reference clock driver. If the header is installed
|
||||||
|
in a location that isn't normally searched by the compiler, you can add it to
|
||||||
|
the searched locations by setting the CPPFLAGS variable to -I/path/to/timepps.
|
||||||
|
|
||||||
|
The --help option can be specified to configure to print all options supported
|
||||||
|
by the script.
|
||||||
|
|
||||||
|
Now type
|
||||||
|
|
||||||
|
make
|
||||||
|
|
||||||
|
to build the programs.
|
||||||
|
|
||||||
|
If you want to build the manual in HTML, type
|
||||||
|
|
||||||
|
make docs
|
||||||
|
|
||||||
|
Once the programs have been successfully compiled, they need to be installed in
|
||||||
|
their target locations. This step normally needs to be performed by the
|
||||||
|
superuser, and requires the following command to be entered.
|
||||||
|
|
||||||
|
make install
|
||||||
|
|
||||||
|
This will install the binaries and man pages.
|
||||||
|
|
||||||
|
To install the HTML version of the manual, enter the command
|
||||||
|
|
||||||
|
make install-docs
|
||||||
|
|
||||||
|
Now that the software is successfully installed, the next step is to set up a
|
||||||
|
configuration file. The default location of the file is /etc/chrony.conf.
|
||||||
|
Several examples of configuration with comments are included in the examples
|
||||||
|
directory. Suppose you want to use public NTP servers from the pool.ntp.org
|
||||||
|
project as your time reference. A minimal useful configuration file could be
|
||||||
|
|
||||||
|
pool pool.ntp.org iburst
|
||||||
|
makestep 1.0 3
|
||||||
|
rtcsync
|
||||||
|
|
||||||
|
Then, chronyd can be run. For security reasons, it's recommended to create an
|
||||||
|
unprivileged user for chronyd and specify it with the -u command-line option or
|
||||||
|
the user directive in the configuration file, or set the default user with the
|
||||||
|
--with-user configure option before building.
|
||||||
|
|
||||||
|
Support for system call filtering
|
||||||
|
|
||||||
|
chronyd can be built with support for the Linux secure computing (seccomp)
|
||||||
|
facility. This requires development files for the libseccomp library and the
|
||||||
|
--enable-scfilter option specified to configure. The -F option of chronyd will
|
||||||
|
enable a system call filter, which should significantly reduce the kernel
|
||||||
|
attack surface and possibly prevent kernel exploits from chronyd if it is
|
||||||
|
compromised.
|
||||||
|
|
||||||
|
Extra options for package builders
|
||||||
|
|
||||||
|
The configure and make procedures have some extra options that may be useful if
|
||||||
|
you are building a distribution package for chrony.
|
||||||
|
|
||||||
|
The --mandir=DIR option to configure specifies an installation directory for
|
||||||
|
the man pages. This overrides the man subdirectory of the argument to the
|
||||||
|
--prefix option.
|
||||||
|
|
||||||
|
./configure --prefix=/usr --mandir=/usr/share/man
|
||||||
|
|
||||||
|
to set both options together.
|
||||||
|
|
||||||
|
The final option is the DESTDIR option to the make command. For example, you
|
||||||
|
could use the commands
|
||||||
|
|
||||||
|
./configure --prefix=/usr --mandir=/usr/share/man
|
||||||
|
make all docs
|
||||||
|
make install DESTDIR=./tmp
|
||||||
|
cd tmp
|
||||||
|
tar cvf - . | gzip -9 > chrony.tar.gz
|
||||||
|
|
||||||
|
to build a package. When untarred within the root directory, this will install
|
||||||
|
the files to the intended final locations.
|
||||||
|
|
||||||
|
Last updated 2024-10-08 14:49:43 +0200
|
143
Makefile.in
Normal file
143
Makefile.in
Normal file
|
@ -0,0 +1,143 @@
|
||||||
|
##################################################
|
||||||
|
#
|
||||||
|
# chronyd/chronyc - Programs for keeping computer clocks accurate.
|
||||||
|
#
|
||||||
|
# Copyright (C) Richard P. Curnow 1997-2003
|
||||||
|
#
|
||||||
|
# This program is free software; you can redistribute it and/or modify
|
||||||
|
# it under the terms of version 2 of the GNU General Public License as
|
||||||
|
# published by the Free Software Foundation.
|
||||||
|
#
|
||||||
|
# This program is distributed in the hope that it will be useful, but
|
||||||
|
# WITHOUT ANY WARRANTY; without even the implied warranty of
|
||||||
|
# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
|
||||||
|
# General Public License for more details.
|
||||||
|
#
|
||||||
|
# You should have received a copy of the GNU General Public License along
|
||||||
|
# with this program; if not, write to the Free Software Foundation, Inc.,
|
||||||
|
# 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA.
|
||||||
|
#
|
||||||
|
# =======================================================================
|
||||||
|
#
|
||||||
|
# Makefile template
|
||||||
|
|
||||||
|
SYSCONFDIR = @SYSCONFDIR@
|
||||||
|
BINDIR = @BINDIR@
|
||||||
|
SBINDIR = @SBINDIR@
|
||||||
|
LOCALSTATEDIR = @LOCALSTATEDIR@
|
||||||
|
CHRONYVARDIR = @CHRONYVARDIR@
|
||||||
|
DESTDIR =
|
||||||
|
|
||||||
|
CC = @CC@
|
||||||
|
CFLAGS = @CFLAGS@
|
||||||
|
CPPFLAGS = @CPPFLAGS@
|
||||||
|
LDFLAGS = @LDFLAGS@
|
||||||
|
|
||||||
|
GETDATE_CFLAGS = @GETDATE_CFLAGS@
|
||||||
|
|
||||||
|
EXTRA_OBJS = @EXTRA_OBJS@
|
||||||
|
|
||||||
|
OBJS = array.o cmdparse.o conf.o leapdb.o local.o logging.o main.o memory.o quantiles.o \
|
||||||
|
reference.o regress.o rtc.o samplefilt.o sched.o socket.o sources.o sourcestats.o \
|
||||||
|
stubs.o smooth.o sys.o sys_null.o tempcomp.o util.o $(EXTRA_OBJS)
|
||||||
|
|
||||||
|
EXTRA_CLI_OBJS = @EXTRA_CLI_OBJS@
|
||||||
|
|
||||||
|
CLI_OBJS = array.o client.o cmdparse.o getdate.o memory.o nameserv.o \
|
||||||
|
pktlength.o socket.o util.o $(EXTRA_CLI_OBJS)
|
||||||
|
|
||||||
|
ALL_OBJS = $(OBJS) $(CLI_OBJS)
|
||||||
|
|
||||||
|
LIBS = @LIBS@
|
||||||
|
EXTRA_LIBS = @EXTRA_LIBS@
|
||||||
|
EXTRA_CLI_LIBS = @EXTRA_CLI_LIBS@
|
||||||
|
|
||||||
|
# Until we have a main procedure we can link, just build object files
|
||||||
|
# to test compilation
|
||||||
|
|
||||||
|
all : chronyd chronyc
|
||||||
|
|
||||||
|
chronyd : $(OBJS)
|
||||||
|
$(CC) $(CFLAGS) -o chronyd $(OBJS) $(LDFLAGS) $(LIBS) $(EXTRA_LIBS)
|
||||||
|
|
||||||
|
chronyc : $(CLI_OBJS)
|
||||||
|
$(CC) $(CFLAGS) -o chronyc $(CLI_OBJS) $(LDFLAGS) $(LIBS) $(EXTRA_CLI_LIBS)
|
||||||
|
|
||||||
|
getdate.o: CFLAGS += $(GETDATE_CFLAGS)
|
||||||
|
|
||||||
|
distclean : clean
|
||||||
|
$(MAKE) -C doc distclean
|
||||||
|
$(MAKE) -C test/unit distclean
|
||||||
|
-rm -f .DS_Store
|
||||||
|
-rm -f Makefile config.h config.log
|
||||||
|
|
||||||
|
clean :
|
||||||
|
$(MAKE) -C test/unit clean
|
||||||
|
-rm -f *.o *.s chronyc chronyd core.* *~
|
||||||
|
-rm -f *.gcda *.gcno
|
||||||
|
-rm -rf .deps
|
||||||
|
-rm -rf *.dSYM
|
||||||
|
|
||||||
|
getdate.c : getdate.y
|
||||||
|
bison -o getdate.c getdate.y
|
||||||
|
|
||||||
|
# This can be used to force regeneration of getdate.c
|
||||||
|
getdate :
|
||||||
|
bison -o getdate.c getdate.y
|
||||||
|
|
||||||
|
# For install, don't use the install command, because its switches
|
||||||
|
# seem to vary between systems.
|
||||||
|
|
||||||
|
install: chronyd chronyc
|
||||||
|
[ -d $(DESTDIR)$(SYSCONFDIR) ] || mkdir -p $(DESTDIR)$(SYSCONFDIR)
|
||||||
|
[ -d $(DESTDIR)$(SBINDIR) ] || mkdir -p $(DESTDIR)$(SBINDIR)
|
||||||
|
[ -d $(DESTDIR)$(BINDIR) ] || mkdir -p $(DESTDIR)$(BINDIR)
|
||||||
|
[ -d $(DESTDIR)$(CHRONYVARDIR) ] || mkdir -p $(DESTDIR)$(CHRONYVARDIR)
|
||||||
|
if [ -f $(DESTDIR)$(SBINDIR)/chronyd ]; then rm -f $(DESTDIR)$(SBINDIR)/chronyd ; fi
|
||||||
|
if [ -f $(DESTDIR)$(BINDIR)/chronyc ]; then rm -f $(DESTDIR)$(BINDIR)/chronyc ; fi
|
||||||
|
cp chronyd $(DESTDIR)$(SBINDIR)/chronyd
|
||||||
|
chmod 755 $(DESTDIR)$(SBINDIR)/chronyd
|
||||||
|
cp chronyc $(DESTDIR)$(BINDIR)/chronyc
|
||||||
|
chmod 755 $(DESTDIR)$(BINDIR)/chronyc
|
||||||
|
$(MAKE) -C doc install
|
||||||
|
|
||||||
|
docs :
|
||||||
|
$(MAKE) -C doc docs
|
||||||
|
|
||||||
|
install-docs :
|
||||||
|
$(MAKE) -C doc install-docs
|
||||||
|
|
||||||
|
%.o : %.c
|
||||||
|
$(CC) $(CFLAGS) $(CPPFLAGS) -c $<
|
||||||
|
|
||||||
|
%.s : %.c
|
||||||
|
$(CC) $(CFLAGS) $(CPPFLAGS) -S $<
|
||||||
|
|
||||||
|
quickcheck : chronyd chronyc
|
||||||
|
$(MAKE) -C test/unit check
|
||||||
|
cd test/simulation && ./run
|
||||||
|
cd test/system && ./run
|
||||||
|
|
||||||
|
check : chronyd chronyc
|
||||||
|
$(MAKE) -C test/unit check
|
||||||
|
cd test/simulation && ./run -i 20 -m 2
|
||||||
|
cd test/system && ./run
|
||||||
|
|
||||||
|
print-chronyd-objects :
|
||||||
|
@echo $(OBJS)
|
||||||
|
|
||||||
|
Makefile : Makefile.in configure
|
||||||
|
@echo
|
||||||
|
@echo Makefile needs to be regenerated, run ./configure
|
||||||
|
@echo
|
||||||
|
@exit 1
|
||||||
|
|
||||||
|
.deps:
|
||||||
|
@mkdir .deps
|
||||||
|
|
||||||
|
.deps/%.d: %.c | .deps
|
||||||
|
@$(CC) -MM $(CPPFLAGS) -MT '$(<:%.c=%.o) $@' $< -o $@
|
||||||
|
|
||||||
|
ifndef NODEPS
|
||||||
|
-include $(ALL_OBJS:%.o=.deps/%.d)
|
||||||
|
endif
|
145
README
Normal file
145
README
Normal file
|
@ -0,0 +1,145 @@
|
||||||
|
This is the README for chrony.
|
||||||
|
|
||||||
|
What is chrony?
|
||||||
|
===============
|
||||||
|
|
||||||
|
chrony is a versatile implementation of the Network Time Protocol (NTP).
|
||||||
|
It can synchronise the system clock with NTP servers, reference clocks
|
||||||
|
(e.g. GPS receiver), and manual input using wristwatch and keyboard.
|
||||||
|
It can also operate as an NTPv4 (RFC 5905) server and peer to provide
|
||||||
|
a time service to other computers in the network.
|
||||||
|
|
||||||
|
It is designed to perform well in a wide range of conditions, including
|
||||||
|
intermittent network connections, heavily congested networks, changing
|
||||||
|
temperatures (ordinary computer clocks are sensitive to temperature),
|
||||||
|
and systems that do not run continuously, or run on a virtual machine.
|
||||||
|
|
||||||
|
Typical accuracy between two machines synchronised over the Internet is
|
||||||
|
within a few milliseconds; on a LAN, accuracy is typically in tens of
|
||||||
|
microseconds. With hardware timestamping, or a hardware reference clock,
|
||||||
|
sub-microsecond accuracy may be possible.
|
||||||
|
|
||||||
|
Two programs are included in chrony, chronyd is a daemon that can be
|
||||||
|
started at boot time and chronyc is a command-line interface program
|
||||||
|
which can be used to monitor chronyd's performance and to change various
|
||||||
|
operating parameters whilst it is running.
|
||||||
|
|
||||||
|
What will chrony run on?
|
||||||
|
========================
|
||||||
|
|
||||||
|
The software is known to work on Linux, FreeBSD, NetBSD, macOS and
|
||||||
|
illumos. Closely related systems may work too. Any other system will
|
||||||
|
likely require a porting exercise.
|
||||||
|
|
||||||
|
How do I set it up?
|
||||||
|
===================
|
||||||
|
|
||||||
|
The file INSTALL gives instructions. On supported systems the
|
||||||
|
compilation process should be automatic. You will need a C compiler,
|
||||||
|
e.g. gcc or clang.
|
||||||
|
|
||||||
|
What documentation is there?
|
||||||
|
============================
|
||||||
|
|
||||||
|
The distribution includes manual pages and a document containing
|
||||||
|
Frequently Asked Questions (FAQ).
|
||||||
|
|
||||||
|
The documentation is also available on the chrony web pages, accessible
|
||||||
|
through the URL
|
||||||
|
|
||||||
|
https://chrony-project.org/
|
||||||
|
|
||||||
|
License
|
||||||
|
=======
|
||||||
|
|
||||||
|
chrony is distributed under the GNU General Public License version 2.
|
||||||
|
|
||||||
|
Authors
|
||||||
|
=======
|
||||||
|
|
||||||
|
Richard P. Curnow <rc@rc0.org.uk>
|
||||||
|
Miroslav Lichvar <mlichvar@redhat.com>
|
||||||
|
|
||||||
|
Acknowledgements
|
||||||
|
================
|
||||||
|
|
||||||
|
In writing the chronyd program, extensive use has been made of the NTPv3 (RFC
|
||||||
|
1305) and NTPv4 (RFC 5905) specification. The source code of the xntpd/ntpd
|
||||||
|
implementation written by Dennis Fergusson, Lars Mathiesen, David Mills, and
|
||||||
|
others has been used to check the details of the protocol.
|
||||||
|
|
||||||
|
The following people have provided patches and other major contributions
|
||||||
|
to chrony:
|
||||||
|
|
||||||
|
Lonnie Abelbeck <lonnie@abelbeck.com>
|
||||||
|
Benny Lyne Amorsen <benny@amorsen.dk>
|
||||||
|
Andrew Bishop <amb@gedanken.demon.co.uk>
|
||||||
|
Vincent Blut <vincent.debian@free.fr>
|
||||||
|
Luca Boccassi <bluca@debian.org>
|
||||||
|
Stephan I. Boettcher <stephan@nevis1.columbia.edu>
|
||||||
|
David Bohman <debohman@gmail.com>
|
||||||
|
Goswin Brederlow <brederlo@informatik.uni-tuebingen.de>
|
||||||
|
Leigh Brown <leigh@solinno.co.uk>
|
||||||
|
Erik Bryer <ebryer@spots.ab.ca>
|
||||||
|
Jonathan Cameron <jic23@cam.ac.uk>
|
||||||
|
Bryan Christianson <bryan@whatroute.net>
|
||||||
|
Juliusz Chroboczek <jch@pps.jussieu.fr>
|
||||||
|
Dan Drown <dan-ntp@drown.org>
|
||||||
|
Kamil Dudka <kdudka@redhat.com>
|
||||||
|
Christian Ehrhardt <christian.ehrhardt@canonical.com>
|
||||||
|
Paul Elliott <pelliott@io.com>
|
||||||
|
Robert Fairley <rfairley@redhat.com>
|
||||||
|
Stefan R. Filipek <srfilipek@gmail.com>
|
||||||
|
Andy Fiddaman <illumos@fiddaman.net>
|
||||||
|
Mike Fleetwood <mike@rockover.demon.co.uk>
|
||||||
|
Rob Gill <rrobgill@protonmail.com>
|
||||||
|
Alexander Gretencord <arutha@gmx.de>
|
||||||
|
Andrew Griffiths <agriffit@redhat.com>
|
||||||
|
Walter Haidinger <walter.haidinger@gmx.at>
|
||||||
|
Juergen Hannken-Illjes <hannken@eis.cs.tu-bs.de>
|
||||||
|
John Hasler <john@dhh.gt.org>
|
||||||
|
Tjalling Hattink <t.hattink@fugro.nl>
|
||||||
|
Liam Hatton <me@liamhatton.com>
|
||||||
|
Holger Hoffstätte <holger@applied-asynchrony.com>
|
||||||
|
Jachym Holecek <jakym@volny.cz>
|
||||||
|
Håkan Johansson <f96hajo@chalmers.se>
|
||||||
|
Jim Knoble <jmknoble@pobox.com>
|
||||||
|
Antti Jrvinen <costello@iki.fi>
|
||||||
|
Uwe Kleine-König <u.kleine-koenig@pengutronix.de>
|
||||||
|
Eric Lammerts <eric@lammerts.org>
|
||||||
|
Stefan Lucke <stefan@lucke.in-berlin.de>
|
||||||
|
Victor Lum <viclum@vanu.com>
|
||||||
|
Kevin Lyda <kevin@ie.suberic.net>
|
||||||
|
Paul Menzel <paulepanter@users.sourceforge.net>
|
||||||
|
Vladimir Michl <vladimir.michl@seznam.cz>
|
||||||
|
Victor Moroz <vim@prv.adlum.ru>
|
||||||
|
Kalle Olavi Niemitalo <tosi@stekt.oulu.fi>
|
||||||
|
Patrick Oppenlander <patrick.oppenlander@gmail.com>
|
||||||
|
Frank Otto <sandwichmacher@web.de>
|
||||||
|
Denny Page <dennypage@me.com>
|
||||||
|
Rupesh Patel <rupatel@redhat.com>
|
||||||
|
Chris Perl <cperl@janestreet.com>
|
||||||
|
Gautier PHILIPPON <gautier.philippon@ensimag.grenoble-inp.fr>
|
||||||
|
Andreas Piesk <apiesk@virbus.de>
|
||||||
|
Mike Ryan <msr@hsilop.net>
|
||||||
|
Baruch Siach <baruch@tkos.co.il>
|
||||||
|
Josef 'Jeff' Sipek <jeffpc@josefsipek.net>
|
||||||
|
Foster Snowhill <forst@forstwoof.ru>
|
||||||
|
Andreas Steinmetz <ast@domdv.de>
|
||||||
|
NAKAMURA Takumi <takumi@ps.sakura.ne.jp>
|
||||||
|
Timo Teras <timo.teras@iki.fi>
|
||||||
|
Bill Unruh <unruh@physics.ubc.ca>
|
||||||
|
Luke Valenta <lvalenta@cloudflare.com>
|
||||||
|
Stephen Wadeley <swadeley@redhat.com>
|
||||||
|
Bernhard Weiss <lisnablagh@web.de>
|
||||||
|
Wolfgang Weisselberg <weissel@netcologne.de>
|
||||||
|
Bernhard M. Wiedemann <bwiedemann@suse.de>
|
||||||
|
Joachim Wiedorn <ad_debian@joonet.de>
|
||||||
|
Ralf Wildenhues <Ralf.Wildenhues@gmx.de>
|
||||||
|
Ulrich Windl <ulrich.windl@rz.uni-regensburg.de>
|
||||||
|
Michael Witten <mfwitten@gmail.com>
|
||||||
|
Doug Woodward <dougw@whistler.com>
|
||||||
|
Thomas Zajic <zlatko@zlatko.fdns.net>
|
||||||
|
|
||||||
|
Many other people have contributed bug reports and suggestions. We are sorry
|
||||||
|
we cannot identify all of you individually.
|
67
addressing.h
Normal file
67
addressing.h
Normal file
|
@ -0,0 +1,67 @@
|
||||||
|
/*
|
||||||
|
chronyd/chronyc - Programs for keeping computer clocks accurate.
|
||||||
|
|
||||||
|
**********************************************************************
|
||||||
|
* Copyright (C) Richard P. Curnow 1997-2002
|
||||||
|
*
|
||||||
|
* This program is free software; you can redistribute it and/or modify
|
||||||
|
* it under the terms of version 2 of the GNU General Public License as
|
||||||
|
* published by the Free Software Foundation.
|
||||||
|
*
|
||||||
|
* This program is distributed in the hope that it will be useful, but
|
||||||
|
* WITHOUT ANY WARRANTY; without even the implied warranty of
|
||||||
|
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
|
||||||
|
* General Public License for more details.
|
||||||
|
*
|
||||||
|
* You should have received a copy of the GNU General Public License along
|
||||||
|
* with this program; if not, write to the Free Software Foundation, Inc.,
|
||||||
|
* 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA.
|
||||||
|
*
|
||||||
|
**********************************************************************
|
||||||
|
|
||||||
|
=======================================================================
|
||||||
|
|
||||||
|
Types used for addressing sources etc
|
||||||
|
*/
|
||||||
|
|
||||||
|
#ifndef GOT_ADDRESSING_H
|
||||||
|
#define GOT_ADDRESSING_H
|
||||||
|
|
||||||
|
#include "sysincl.h"
|
||||||
|
|
||||||
|
/* This type is used to represent an IPv4 address or IPv6 address.
|
||||||
|
Addresses which are not resolved yet can be represented with an ID.
|
||||||
|
All parts are in HOST order, NOT network order. */
|
||||||
|
|
||||||
|
#define IPADDR_UNSPEC 0
|
||||||
|
#define IPADDR_INET4 1
|
||||||
|
#define IPADDR_INET6 2
|
||||||
|
#define IPADDR_ID 3
|
||||||
|
|
||||||
|
typedef struct {
|
||||||
|
union {
|
||||||
|
uint32_t in4;
|
||||||
|
uint8_t in6[16];
|
||||||
|
uint32_t id;
|
||||||
|
} addr;
|
||||||
|
uint16_t family;
|
||||||
|
uint16_t _pad;
|
||||||
|
} IPAddr;
|
||||||
|
|
||||||
|
typedef struct {
|
||||||
|
IPAddr ip_addr;
|
||||||
|
uint16_t port;
|
||||||
|
} IPSockAddr;
|
||||||
|
|
||||||
|
typedef IPSockAddr NTP_Remote_Address;
|
||||||
|
|
||||||
|
#define INVALID_IF_INDEX -1
|
||||||
|
|
||||||
|
typedef struct {
|
||||||
|
IPAddr ip_addr;
|
||||||
|
int if_index;
|
||||||
|
int sock_fd;
|
||||||
|
} NTP_Local_Address;
|
||||||
|
|
||||||
|
#endif /* GOT_ADDRESSING_H */
|
||||||
|
|
405
addrfilt.c
Normal file
405
addrfilt.c
Normal file
|
@ -0,0 +1,405 @@
|
||||||
|
/*
|
||||||
|
chronyd/chronyc - Programs for keeping computer clocks accurate.
|
||||||
|
|
||||||
|
**********************************************************************
|
||||||
|
* Copyright (C) Richard P. Curnow 1997,1998,1999,2000,2001,2002,2005
|
||||||
|
* Copyright (C) Miroslav Lichvar 2009, 2015
|
||||||
|
*
|
||||||
|
* This program is free software; you can redistribute it and/or modify
|
||||||
|
* it under the terms of version 2 of the GNU General Public License as
|
||||||
|
* published by the Free Software Foundation.
|
||||||
|
*
|
||||||
|
* This program is distributed in the hope that it will be useful, but
|
||||||
|
* WITHOUT ANY WARRANTY; without even the implied warranty of
|
||||||
|
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
|
||||||
|
* General Public License for more details.
|
||||||
|
*
|
||||||
|
* You should have received a copy of the GNU General Public License along
|
||||||
|
* with this program; if not, write to the Free Software Foundation, Inc.,
|
||||||
|
* 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA.
|
||||||
|
*
|
||||||
|
**********************************************************************
|
||||||
|
|
||||||
|
=======================================================================
|
||||||
|
|
||||||
|
This module provides a set of routines for checking IP addresses
|
||||||
|
against a set of rules and deciding whether they are allowed or
|
||||||
|
disallowed.
|
||||||
|
|
||||||
|
*/
|
||||||
|
|
||||||
|
#include "config.h"
|
||||||
|
|
||||||
|
#include "sysincl.h"
|
||||||
|
|
||||||
|
#include "addrfilt.h"
|
||||||
|
#include "memory.h"
|
||||||
|
|
||||||
|
/* Define the number of bits which are stripped off per level of
|
||||||
|
indirection in the tables */
|
||||||
|
#define NBITS 4
|
||||||
|
|
||||||
|
/* Define the table size */
|
||||||
|
#define TABLE_SIZE (1UL<<NBITS)
|
||||||
|
|
||||||
|
typedef enum {DENY, ALLOW, AS_PARENT} State;
|
||||||
|
|
||||||
|
typedef struct _TableNode {
|
||||||
|
State state;
|
||||||
|
struct _TableNode *extended;
|
||||||
|
} TableNode;
|
||||||
|
|
||||||
|
struct ADF_AuthTableInst {
|
||||||
|
TableNode base4; /* IPv4 node */
|
||||||
|
TableNode base6; /* IPv6 node */
|
||||||
|
};
|
||||||
|
|
||||||
|
/* ================================================== */
|
||||||
|
|
||||||
|
static void
|
||||||
|
split_ip6(IPAddr *ip, uint32_t *dst)
|
||||||
|
{
|
||||||
|
int i;
|
||||||
|
|
||||||
|
for (i = 0; i < 4; i++)
|
||||||
|
dst[i] = (uint32_t)ip->addr.in6[i * 4 + 0] << 24 |
|
||||||
|
ip->addr.in6[i * 4 + 1] << 16 |
|
||||||
|
ip->addr.in6[i * 4 + 2] << 8 |
|
||||||
|
ip->addr.in6[i * 4 + 3];
|
||||||
|
}
|
||||||
|
|
||||||
|
/* ================================================== */
|
||||||
|
|
||||||
|
inline static uint32_t
|
||||||
|
get_subnet(uint32_t *addr, unsigned int where)
|
||||||
|
{
|
||||||
|
int off;
|
||||||
|
|
||||||
|
off = where / 32;
|
||||||
|
where %= 32;
|
||||||
|
|
||||||
|
return (addr[off] >> (32 - NBITS - where)) & ((1UL << NBITS) - 1);
|
||||||
|
}
|
||||||
|
|
||||||
|
/* ================================================== */
|
||||||
|
|
||||||
|
ADF_AuthTable
|
||||||
|
ADF_CreateTable(void)
|
||||||
|
{
|
||||||
|
ADF_AuthTable result;
|
||||||
|
result = MallocNew(struct ADF_AuthTableInst);
|
||||||
|
|
||||||
|
/* Default is that nothing is allowed */
|
||||||
|
result->base4.state = DENY;
|
||||||
|
result->base4.extended = NULL;
|
||||||
|
result->base6.state = DENY;
|
||||||
|
result->base6.extended = NULL;
|
||||||
|
|
||||||
|
return result;
|
||||||
|
}
|
||||||
|
|
||||||
|
/* ================================================== */
|
||||||
|
/* This function deletes all definitions of child nodes, in effect
|
||||||
|
pruning a whole subnet definition back to a single parent
|
||||||
|
record. */
|
||||||
|
static void
|
||||||
|
close_node(TableNode *node)
|
||||||
|
{
|
||||||
|
int i;
|
||||||
|
TableNode *child_node;
|
||||||
|
|
||||||
|
if (node->extended != NULL) {
|
||||||
|
for (i=0; i<TABLE_SIZE; i++) {
|
||||||
|
child_node = &(node->extended[i]);
|
||||||
|
close_node(child_node);
|
||||||
|
}
|
||||||
|
Free(node->extended);
|
||||||
|
node->extended = NULL;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
/* ================================================== */
|
||||||
|
/* Allocate the extension field in a node, and set all the children's
|
||||||
|
states to default to that of the node being extended */
|
||||||
|
|
||||||
|
static void
|
||||||
|
open_node(TableNode *node)
|
||||||
|
{
|
||||||
|
int i;
|
||||||
|
TableNode *child_node;
|
||||||
|
|
||||||
|
if (node->extended == NULL) {
|
||||||
|
|
||||||
|
node->extended = MallocArray(struct _TableNode, TABLE_SIZE);
|
||||||
|
|
||||||
|
for (i=0; i<TABLE_SIZE; i++) {
|
||||||
|
child_node = &(node->extended[i]);
|
||||||
|
child_node->state = AS_PARENT;
|
||||||
|
child_node->extended = NULL;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/* ================================================== */
|
||||||
|
|
||||||
|
static ADF_Status
|
||||||
|
set_subnet(TableNode *start_node,
|
||||||
|
uint32_t *ip,
|
||||||
|
int ip_len,
|
||||||
|
int subnet_bits,
|
||||||
|
State new_state,
|
||||||
|
int delete_children)
|
||||||
|
{
|
||||||
|
int bits_to_go, bits_consumed;
|
||||||
|
uint32_t subnet;
|
||||||
|
TableNode *node;
|
||||||
|
|
||||||
|
bits_consumed = 0;
|
||||||
|
bits_to_go = subnet_bits;
|
||||||
|
node = start_node;
|
||||||
|
|
||||||
|
if ((subnet_bits < 0) ||
|
||||||
|
(subnet_bits > 32 * ip_len)) {
|
||||||
|
|
||||||
|
return ADF_BADSUBNET;
|
||||||
|
|
||||||
|
} else {
|
||||||
|
|
||||||
|
if ((bits_to_go & (NBITS-1)) == 0) {
|
||||||
|
|
||||||
|
while (bits_to_go > 0) {
|
||||||
|
subnet = get_subnet(ip, bits_consumed);
|
||||||
|
if (!(node->extended)) {
|
||||||
|
open_node(node);
|
||||||
|
}
|
||||||
|
node = &(node->extended[subnet]);
|
||||||
|
bits_to_go -= NBITS;
|
||||||
|
bits_consumed += NBITS;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (delete_children) {
|
||||||
|
close_node(node);
|
||||||
|
}
|
||||||
|
node->state = new_state;
|
||||||
|
|
||||||
|
} else { /* Have to set multiple entries */
|
||||||
|
int N, i, j;
|
||||||
|
TableNode *this_node;
|
||||||
|
|
||||||
|
while (bits_to_go >= NBITS) {
|
||||||
|
subnet = get_subnet(ip, bits_consumed);
|
||||||
|
if (!(node->extended)) {
|
||||||
|
open_node(node);
|
||||||
|
}
|
||||||
|
node = &(node->extended[subnet]);
|
||||||
|
bits_to_go -= NBITS;
|
||||||
|
bits_consumed += NBITS;
|
||||||
|
}
|
||||||
|
|
||||||
|
/* How many subnet entries to set : 1->8, 2->4, 3->2 */
|
||||||
|
N = 1 << (NBITS-bits_to_go);
|
||||||
|
|
||||||
|
subnet = get_subnet(ip, bits_consumed) & ~(N - 1);
|
||||||
|
assert(subnet + N <= TABLE_SIZE);
|
||||||
|
|
||||||
|
if (!(node->extended)) {
|
||||||
|
open_node(node);
|
||||||
|
}
|
||||||
|
|
||||||
|
for (i=subnet, j=0; j<N; i++, j++) {
|
||||||
|
this_node = &(node->extended[i]);
|
||||||
|
if (delete_children) {
|
||||||
|
close_node(this_node);
|
||||||
|
}
|
||||||
|
this_node->state = new_state;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
return ADF_SUCCESS;
|
||||||
|
}
|
||||||
|
|
||||||
|
}
|
||||||
|
|
||||||
|
/* ================================================== */
|
||||||
|
|
||||||
|
static ADF_Status
|
||||||
|
set_subnet_(ADF_AuthTable table,
|
||||||
|
IPAddr *ip_addr,
|
||||||
|
int subnet_bits,
|
||||||
|
State new_state,
|
||||||
|
int delete_children)
|
||||||
|
{
|
||||||
|
uint32_t ip6[4];
|
||||||
|
|
||||||
|
switch (ip_addr->family) {
|
||||||
|
case IPADDR_INET4:
|
||||||
|
return set_subnet(&table->base4, &ip_addr->addr.in4, 1, subnet_bits, new_state, delete_children);
|
||||||
|
case IPADDR_INET6:
|
||||||
|
split_ip6(ip_addr, ip6);
|
||||||
|
return set_subnet(&table->base6, ip6, 4, subnet_bits, new_state, delete_children);
|
||||||
|
case IPADDR_UNSPEC:
|
||||||
|
/* Apply to both, subnet_bits has to be 0 */
|
||||||
|
if (subnet_bits != 0)
|
||||||
|
return ADF_BADSUBNET;
|
||||||
|
memset(ip6, 0, sizeof (ip6));
|
||||||
|
if (set_subnet(&table->base4, ip6, 1, 0, new_state, delete_children) == ADF_SUCCESS &&
|
||||||
|
set_subnet(&table->base6, ip6, 4, 0, new_state, delete_children) == ADF_SUCCESS)
|
||||||
|
return ADF_SUCCESS;
|
||||||
|
break;
|
||||||
|
default:
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
|
||||||
|
return ADF_BADSUBNET;
|
||||||
|
}
|
||||||
|
|
||||||
|
ADF_Status
|
||||||
|
ADF_Allow(ADF_AuthTable table,
|
||||||
|
IPAddr *ip,
|
||||||
|
int subnet_bits)
|
||||||
|
{
|
||||||
|
return set_subnet_(table, ip, subnet_bits, ALLOW, 0);
|
||||||
|
}
|
||||||
|
|
||||||
|
/* ================================================== */
|
||||||
|
|
||||||
|
|
||||||
|
ADF_Status
|
||||||
|
ADF_AllowAll(ADF_AuthTable table,
|
||||||
|
IPAddr *ip,
|
||||||
|
int subnet_bits)
|
||||||
|
{
|
||||||
|
return set_subnet_(table, ip, subnet_bits, ALLOW, 1);
|
||||||
|
}
|
||||||
|
|
||||||
|
/* ================================================== */
|
||||||
|
|
||||||
|
ADF_Status
|
||||||
|
ADF_Deny(ADF_AuthTable table,
|
||||||
|
IPAddr *ip,
|
||||||
|
int subnet_bits)
|
||||||
|
{
|
||||||
|
return set_subnet_(table, ip, subnet_bits, DENY, 0);
|
||||||
|
}
|
||||||
|
|
||||||
|
/* ================================================== */
|
||||||
|
|
||||||
|
ADF_Status
|
||||||
|
ADF_DenyAll(ADF_AuthTable table,
|
||||||
|
IPAddr *ip,
|
||||||
|
int subnet_bits)
|
||||||
|
{
|
||||||
|
return set_subnet_(table, ip, subnet_bits, DENY, 1);
|
||||||
|
}
|
||||||
|
|
||||||
|
/* ================================================== */
|
||||||
|
|
||||||
|
void
|
||||||
|
ADF_DestroyTable(ADF_AuthTable table)
|
||||||
|
{
|
||||||
|
close_node(&table->base4);
|
||||||
|
close_node(&table->base6);
|
||||||
|
Free(table);
|
||||||
|
}
|
||||||
|
|
||||||
|
/* ================================================== */
|
||||||
|
|
||||||
|
static int
|
||||||
|
check_ip_in_node(TableNode *start_node, uint32_t *ip)
|
||||||
|
{
|
||||||
|
uint32_t subnet;
|
||||||
|
int bits_consumed = 0;
|
||||||
|
int result = 0;
|
||||||
|
int finished = 0;
|
||||||
|
TableNode *node;
|
||||||
|
State state=DENY;
|
||||||
|
|
||||||
|
node = start_node;
|
||||||
|
|
||||||
|
do {
|
||||||
|
if (node->state != AS_PARENT) {
|
||||||
|
state = node->state;
|
||||||
|
}
|
||||||
|
if (node->extended) {
|
||||||
|
subnet = get_subnet(ip, bits_consumed);
|
||||||
|
node = &(node->extended[subnet]);
|
||||||
|
bits_consumed += NBITS;
|
||||||
|
} else {
|
||||||
|
/* Make decision on this node */
|
||||||
|
finished = 1;
|
||||||
|
}
|
||||||
|
} while (!finished);
|
||||||
|
|
||||||
|
switch (state) {
|
||||||
|
case ALLOW:
|
||||||
|
result = 1;
|
||||||
|
break;
|
||||||
|
case DENY:
|
||||||
|
result = 0;
|
||||||
|
break;
|
||||||
|
case AS_PARENT:
|
||||||
|
assert(0);
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
|
||||||
|
return result;
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
/* ================================================== */
|
||||||
|
|
||||||
|
int
|
||||||
|
ADF_IsAllowed(ADF_AuthTable table,
|
||||||
|
IPAddr *ip_addr)
|
||||||
|
{
|
||||||
|
uint32_t ip6[4];
|
||||||
|
|
||||||
|
switch (ip_addr->family) {
|
||||||
|
case IPADDR_INET4:
|
||||||
|
return check_ip_in_node(&table->base4, &ip_addr->addr.in4);
|
||||||
|
case IPADDR_INET6:
|
||||||
|
split_ip6(ip_addr, ip6);
|
||||||
|
return check_ip_in_node(&table->base6, ip6);
|
||||||
|
default:
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/* ================================================== */
|
||||||
|
|
||||||
|
static int
|
||||||
|
is_any_allowed(TableNode *node, State parent)
|
||||||
|
{
|
||||||
|
State state;
|
||||||
|
int i;
|
||||||
|
|
||||||
|
state = node->state != AS_PARENT ? node->state : parent;
|
||||||
|
assert(state != AS_PARENT);
|
||||||
|
|
||||||
|
if (node->extended) {
|
||||||
|
for (i = 0; i < TABLE_SIZE; i++) {
|
||||||
|
if (is_any_allowed(&node->extended[i], state))
|
||||||
|
return 1;
|
||||||
|
}
|
||||||
|
} else if (state == ALLOW) {
|
||||||
|
return 1;
|
||||||
|
}
|
||||||
|
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
/* ================================================== */
|
||||||
|
|
||||||
|
int
|
||||||
|
ADF_IsAnyAllowed(ADF_AuthTable table, int family)
|
||||||
|
{
|
||||||
|
switch (family) {
|
||||||
|
case IPADDR_INET4:
|
||||||
|
return is_any_allowed(&table->base4, AS_PARENT);
|
||||||
|
case IPADDR_INET6:
|
||||||
|
return is_any_allowed(&table->base6, AS_PARENT);
|
||||||
|
default:
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
}
|
80
addrfilt.h
Normal file
80
addrfilt.h
Normal file
|
@ -0,0 +1,80 @@
|
||||||
|
/*
|
||||||
|
chronyd/chronyc - Programs for keeping computer clocks accurate.
|
||||||
|
|
||||||
|
**********************************************************************
|
||||||
|
* Copyright (C) Richard P. Curnow 1997-2002
|
||||||
|
*
|
||||||
|
* This program is free software; you can redistribute it and/or modify
|
||||||
|
* it under the terms of version 2 of the GNU General Public License as
|
||||||
|
* published by the Free Software Foundation.
|
||||||
|
*
|
||||||
|
* This program is distributed in the hope that it will be useful, but
|
||||||
|
* WITHOUT ANY WARRANTY; without even the implied warranty of
|
||||||
|
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
|
||||||
|
* General Public License for more details.
|
||||||
|
*
|
||||||
|
* You should have received a copy of the GNU General Public License along
|
||||||
|
* with this program; if not, write to the Free Software Foundation, Inc.,
|
||||||
|
* 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA.
|
||||||
|
*
|
||||||
|
**********************************************************************
|
||||||
|
|
||||||
|
=======================================================================
|
||||||
|
|
||||||
|
Module for providing an authorisation filter on IP addresses
|
||||||
|
*/
|
||||||
|
|
||||||
|
#ifndef GOT_ADDRFILT_H
|
||||||
|
#define GOT_ADDRFILT_H
|
||||||
|
|
||||||
|
#include "addressing.h"
|
||||||
|
|
||||||
|
typedef struct ADF_AuthTableInst *ADF_AuthTable;
|
||||||
|
|
||||||
|
typedef enum {
|
||||||
|
ADF_SUCCESS,
|
||||||
|
ADF_BADSUBNET
|
||||||
|
} ADF_Status;
|
||||||
|
|
||||||
|
|
||||||
|
/* Create a new table. The default rule is deny for everything */
|
||||||
|
extern ADF_AuthTable ADF_CreateTable(void);
|
||||||
|
|
||||||
|
/* Allow anything in the supplied subnet, EXCEPT for any more specific
|
||||||
|
subnets that are already defined */
|
||||||
|
extern ADF_Status ADF_Allow(ADF_AuthTable table,
|
||||||
|
IPAddr *ip,
|
||||||
|
int subnet_bits);
|
||||||
|
|
||||||
|
/* Allow anything in the supplied subnet, overwriting existing
|
||||||
|
definitions for any more specific subnets */
|
||||||
|
extern ADF_Status ADF_AllowAll(ADF_AuthTable table,
|
||||||
|
IPAddr *ip,
|
||||||
|
int subnet_bits);
|
||||||
|
|
||||||
|
/* Deny anything in the supplied subnet, EXCEPT for any more specific
|
||||||
|
subnets that are already defined */
|
||||||
|
extern ADF_Status ADF_Deny(ADF_AuthTable table,
|
||||||
|
IPAddr *ip,
|
||||||
|
int subnet_bits);
|
||||||
|
|
||||||
|
/* Deny anything in the supplied subnet, overwriting existing
|
||||||
|
definitions for any more specific subnets */
|
||||||
|
extern ADF_Status ADF_DenyAll(ADF_AuthTable table,
|
||||||
|
IPAddr *ip,
|
||||||
|
int subnet_bits);
|
||||||
|
|
||||||
|
/* Clear up the table */
|
||||||
|
extern void ADF_DestroyTable(ADF_AuthTable table);
|
||||||
|
|
||||||
|
/* Check whether a given IP address is allowed by the rules in
|
||||||
|
the table */
|
||||||
|
extern int ADF_IsAllowed(ADF_AuthTable table,
|
||||||
|
IPAddr *ip);
|
||||||
|
|
||||||
|
/* Check if at least one address from a given family is allowed by
|
||||||
|
the rules in the table */
|
||||||
|
extern int ADF_IsAnyAllowed(ADF_AuthTable table,
|
||||||
|
int family);
|
||||||
|
|
||||||
|
#endif /* GOT_ADDRFILT_H */
|
145
array.c
Normal file
145
array.c
Normal file
|
@ -0,0 +1,145 @@
|
||||||
|
/*
|
||||||
|
chronyd/chronyc - Programs for keeping computer clocks accurate.
|
||||||
|
|
||||||
|
**********************************************************************
|
||||||
|
* Copyright (C) Miroslav Lichvar 2014
|
||||||
|
*
|
||||||
|
* This program is free software; you can redistribute it and/or modify
|
||||||
|
* it under the terms of version 2 of the GNU General Public License as
|
||||||
|
* published by the Free Software Foundation.
|
||||||
|
*
|
||||||
|
* This program is distributed in the hope that it will be useful, but
|
||||||
|
* WITHOUT ANY WARRANTY; without even the implied warranty of
|
||||||
|
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
|
||||||
|
* General Public License for more details.
|
||||||
|
*
|
||||||
|
* You should have received a copy of the GNU General Public License along
|
||||||
|
* with this program; if not, write to the Free Software Foundation, Inc.,
|
||||||
|
* 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA.
|
||||||
|
*
|
||||||
|
**********************************************************************
|
||||||
|
|
||||||
|
=======================================================================
|
||||||
|
|
||||||
|
Functions implementing an array with automatic memory allocation.
|
||||||
|
|
||||||
|
*/
|
||||||
|
|
||||||
|
#include "config.h"
|
||||||
|
|
||||||
|
#include "sysincl.h"
|
||||||
|
|
||||||
|
#include "array.h"
|
||||||
|
#include "memory.h"
|
||||||
|
|
||||||
|
struct ARR_Instance_Record {
|
||||||
|
void *data;
|
||||||
|
unsigned int elem_size;
|
||||||
|
unsigned int used;
|
||||||
|
unsigned int allocated;
|
||||||
|
};
|
||||||
|
|
||||||
|
ARR_Instance
|
||||||
|
ARR_CreateInstance(unsigned int elem_size)
|
||||||
|
{
|
||||||
|
ARR_Instance array;
|
||||||
|
|
||||||
|
assert(elem_size > 0);
|
||||||
|
|
||||||
|
array = MallocNew(struct ARR_Instance_Record);
|
||||||
|
|
||||||
|
array->data = NULL;
|
||||||
|
array->elem_size = elem_size;
|
||||||
|
array->used = 0;
|
||||||
|
array->allocated = 0;
|
||||||
|
|
||||||
|
return array;
|
||||||
|
}
|
||||||
|
|
||||||
|
void
|
||||||
|
ARR_DestroyInstance(ARR_Instance array)
|
||||||
|
{
|
||||||
|
Free(array->data);
|
||||||
|
Free(array);
|
||||||
|
}
|
||||||
|
|
||||||
|
static void
|
||||||
|
realloc_array(ARR_Instance array, unsigned int min_size)
|
||||||
|
{
|
||||||
|
assert(min_size <= 2 * min_size);
|
||||||
|
if (array->allocated >= min_size && array->allocated <= 2 * min_size)
|
||||||
|
return;
|
||||||
|
|
||||||
|
if (array->allocated < min_size) {
|
||||||
|
while (array->allocated < min_size)
|
||||||
|
array->allocated = array->allocated ? 2 * array->allocated : 1;
|
||||||
|
} else {
|
||||||
|
array->allocated = min_size;
|
||||||
|
}
|
||||||
|
|
||||||
|
array->data = Realloc2(array->data, array->allocated, array->elem_size);
|
||||||
|
}
|
||||||
|
|
||||||
|
void *
|
||||||
|
ARR_GetNewElement(ARR_Instance array)
|
||||||
|
{
|
||||||
|
array->used++;
|
||||||
|
realloc_array(array, array->used);
|
||||||
|
return ARR_GetElement(array, array->used - 1);
|
||||||
|
}
|
||||||
|
|
||||||
|
void *
|
||||||
|
ARR_GetElement(ARR_Instance array, unsigned int index)
|
||||||
|
{
|
||||||
|
assert(index < array->used);
|
||||||
|
return (void *)((char *)array->data + (size_t)index * array->elem_size);
|
||||||
|
}
|
||||||
|
|
||||||
|
void *
|
||||||
|
ARR_GetElements(ARR_Instance array)
|
||||||
|
{
|
||||||
|
/* Return a non-NULL pointer when the array has zero size */
|
||||||
|
if (!array->data) {
|
||||||
|
assert(!array->used);
|
||||||
|
return array;
|
||||||
|
}
|
||||||
|
|
||||||
|
return array->data;
|
||||||
|
}
|
||||||
|
|
||||||
|
void
|
||||||
|
ARR_AppendElement(ARR_Instance array, void *element)
|
||||||
|
{
|
||||||
|
void *e;
|
||||||
|
|
||||||
|
e = ARR_GetNewElement(array);
|
||||||
|
memcpy(e, element, array->elem_size);
|
||||||
|
}
|
||||||
|
|
||||||
|
void
|
||||||
|
ARR_RemoveElement(ARR_Instance array, unsigned int index)
|
||||||
|
{
|
||||||
|
void *e, *l;
|
||||||
|
|
||||||
|
e = ARR_GetElement(array, index);
|
||||||
|
l = ARR_GetElement(array, array->used - 1);
|
||||||
|
|
||||||
|
if (e < l)
|
||||||
|
memmove(e, (char *)e + array->elem_size, (char *)l - (char *)e);
|
||||||
|
array->used--;
|
||||||
|
|
||||||
|
realloc_array(array, array->used);
|
||||||
|
}
|
||||||
|
|
||||||
|
void
|
||||||
|
ARR_SetSize(ARR_Instance array, unsigned int size)
|
||||||
|
{
|
||||||
|
realloc_array(array, size);
|
||||||
|
array->used = size;
|
||||||
|
}
|
||||||
|
|
||||||
|
unsigned int
|
||||||
|
ARR_GetSize(ARR_Instance array)
|
||||||
|
{
|
||||||
|
return array->used;
|
||||||
|
}
|
59
array.h
Normal file
59
array.h
Normal file
|
@ -0,0 +1,59 @@
|
||||||
|
/*
|
||||||
|
chronyd/chronyc - Programs for keeping computer clocks accurate.
|
||||||
|
|
||||||
|
**********************************************************************
|
||||||
|
* Copyright (C) Miroslav Lichvar 2014
|
||||||
|
*
|
||||||
|
* This program is free software; you can redistribute it and/or modify
|
||||||
|
* it under the terms of version 2 of the GNU General Public License as
|
||||||
|
* published by the Free Software Foundation.
|
||||||
|
*
|
||||||
|
* This program is distributed in the hope that it will be useful, but
|
||||||
|
* WITHOUT ANY WARRANTY; without even the implied warranty of
|
||||||
|
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
|
||||||
|
* General Public License for more details.
|
||||||
|
*
|
||||||
|
* You should have received a copy of the GNU General Public License along
|
||||||
|
* with this program; if not, write to the Free Software Foundation, Inc.,
|
||||||
|
* 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA.
|
||||||
|
*
|
||||||
|
**********************************************************************
|
||||||
|
|
||||||
|
=======================================================================
|
||||||
|
|
||||||
|
Header file for array functions.
|
||||||
|
*/
|
||||||
|
|
||||||
|
#ifndef GOT_ARRAY_H
|
||||||
|
#define GOT_ARRAY_H
|
||||||
|
|
||||||
|
typedef struct ARR_Instance_Record *ARR_Instance;
|
||||||
|
|
||||||
|
/* Create a new array with given element size */
|
||||||
|
extern ARR_Instance ARR_CreateInstance(unsigned int elem_size);
|
||||||
|
|
||||||
|
/* Destroy the array */
|
||||||
|
extern void ARR_DestroyInstance(ARR_Instance array);
|
||||||
|
|
||||||
|
/* Return pointer to a new element added to the end of the array */
|
||||||
|
extern void *ARR_GetNewElement(ARR_Instance array);
|
||||||
|
|
||||||
|
/* Return element with given index */
|
||||||
|
extern void *ARR_GetElement(ARR_Instance array, unsigned int index);
|
||||||
|
|
||||||
|
/* Return pointer to the internal array of elements */
|
||||||
|
extern void *ARR_GetElements(ARR_Instance array);
|
||||||
|
|
||||||
|
/* Add a new element to the end of the array */
|
||||||
|
extern void ARR_AppendElement(ARR_Instance array, void *element);
|
||||||
|
|
||||||
|
/* Remove element with given index */
|
||||||
|
extern void ARR_RemoveElement(ARR_Instance array, unsigned int index);
|
||||||
|
|
||||||
|
/* Set the size of the array */
|
||||||
|
extern void ARR_SetSize(ARR_Instance array, unsigned int size);
|
||||||
|
|
||||||
|
/* Return current size of the array */
|
||||||
|
extern unsigned int ARR_GetSize(ARR_Instance array);
|
||||||
|
|
||||||
|
#endif
|
869
candm.h
Normal file
869
candm.h
Normal file
|
@ -0,0 +1,869 @@
|
||||||
|
/*
|
||||||
|
chronyd/chronyc - Programs for keeping computer clocks accurate.
|
||||||
|
|
||||||
|
**********************************************************************
|
||||||
|
* Copyright (C) Richard P. Curnow 1997-2003
|
||||||
|
*
|
||||||
|
* This program is free software; you can redistribute it and/or modify
|
||||||
|
* it under the terms of version 2 of the GNU General Public License as
|
||||||
|
* published by the Free Software Foundation.
|
||||||
|
*
|
||||||
|
* This program is distributed in the hope that it will be useful, but
|
||||||
|
* WITHOUT ANY WARRANTY; without even the implied warranty of
|
||||||
|
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
|
||||||
|
* General Public License for more details.
|
||||||
|
*
|
||||||
|
* You should have received a copy of the GNU General Public License along
|
||||||
|
* with this program; if not, write to the Free Software Foundation, Inc.,
|
||||||
|
* 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA.
|
||||||
|
*
|
||||||
|
**********************************************************************
|
||||||
|
|
||||||
|
=======================================================================
|
||||||
|
|
||||||
|
Definitions for the network protocol used for command and monitoring
|
||||||
|
of the timeserver.
|
||||||
|
|
||||||
|
*/
|
||||||
|
|
||||||
|
#ifndef GOT_CANDM_H
|
||||||
|
#define GOT_CANDM_H
|
||||||
|
|
||||||
|
#include "sysincl.h"
|
||||||
|
#include "addressing.h"
|
||||||
|
|
||||||
|
/* This is the default port to use for CANDM, if no alternative is
|
||||||
|
defined */
|
||||||
|
#define DEFAULT_CANDM_PORT 323
|
||||||
|
|
||||||
|
/* Request codes */
|
||||||
|
#define REQ_NULL 0
|
||||||
|
#define REQ_ONLINE 1
|
||||||
|
#define REQ_OFFLINE 2
|
||||||
|
#define REQ_BURST 3
|
||||||
|
#define REQ_MODIFY_MINPOLL 4
|
||||||
|
#define REQ_MODIFY_MAXPOLL 5
|
||||||
|
#define REQ_DUMP 6
|
||||||
|
#define REQ_MODIFY_MAXDELAY 7
|
||||||
|
#define REQ_MODIFY_MAXDELAYRATIO 8
|
||||||
|
#define REQ_MODIFY_MAXUPDATESKEW 9
|
||||||
|
#define REQ_LOGON 10
|
||||||
|
#define REQ_SETTIME 11
|
||||||
|
#define REQ_LOCAL 12
|
||||||
|
#define REQ_MANUAL 13
|
||||||
|
#define REQ_N_SOURCES 14
|
||||||
|
#define REQ_SOURCE_DATA 15
|
||||||
|
#define REQ_REKEY 16
|
||||||
|
#define REQ_ALLOW 17
|
||||||
|
#define REQ_ALLOWALL 18
|
||||||
|
#define REQ_DENY 19
|
||||||
|
#define REQ_DENYALL 20
|
||||||
|
#define REQ_CMDALLOW 21
|
||||||
|
#define REQ_CMDALLOWALL 22
|
||||||
|
#define REQ_CMDDENY 23
|
||||||
|
#define REQ_CMDDENYALL 24
|
||||||
|
#define REQ_ACCHECK 25
|
||||||
|
#define REQ_CMDACCHECK 26
|
||||||
|
#define REQ_ADD_SERVER 27
|
||||||
|
#define REQ_ADD_PEER 28
|
||||||
|
#define REQ_DEL_SOURCE 29
|
||||||
|
#define REQ_WRITERTC 30
|
||||||
|
#define REQ_DFREQ 31
|
||||||
|
#define REQ_DOFFSET 32
|
||||||
|
#define REQ_TRACKING 33
|
||||||
|
#define REQ_SOURCESTATS 34
|
||||||
|
#define REQ_RTCREPORT 35
|
||||||
|
#define REQ_TRIMRTC 36
|
||||||
|
#define REQ_CYCLELOGS 37
|
||||||
|
#define REQ_SUBNETS_ACCESSED 38
|
||||||
|
#define REQ_CLIENT_ACCESSES 39
|
||||||
|
#define REQ_CLIENT_ACCESSES_BY_INDEX 40
|
||||||
|
#define REQ_MANUAL_LIST 41
|
||||||
|
#define REQ_MANUAL_DELETE 42
|
||||||
|
#define REQ_MAKESTEP 43
|
||||||
|
#define REQ_ACTIVITY 44
|
||||||
|
#define REQ_MODIFY_MINSTRATUM 45
|
||||||
|
#define REQ_MODIFY_POLLTARGET 46
|
||||||
|
#define REQ_MODIFY_MAXDELAYDEVRATIO 47
|
||||||
|
#define REQ_RESELECT 48
|
||||||
|
#define REQ_RESELECTDISTANCE 49
|
||||||
|
#define REQ_MODIFY_MAKESTEP 50
|
||||||
|
#define REQ_SMOOTHING 51
|
||||||
|
#define REQ_SMOOTHTIME 52
|
||||||
|
#define REQ_REFRESH 53
|
||||||
|
#define REQ_SERVER_STATS 54
|
||||||
|
#define REQ_CLIENT_ACCESSES_BY_INDEX2 55
|
||||||
|
#define REQ_LOCAL2 56
|
||||||
|
#define REQ_NTP_DATA 57
|
||||||
|
#define REQ_ADD_SERVER2 58
|
||||||
|
#define REQ_ADD_PEER2 59
|
||||||
|
#define REQ_ADD_SERVER3 60
|
||||||
|
#define REQ_ADD_PEER3 61
|
||||||
|
#define REQ_SHUTDOWN 62
|
||||||
|
#define REQ_ONOFFLINE 63
|
||||||
|
#define REQ_ADD_SOURCE 64
|
||||||
|
#define REQ_NTP_SOURCE_NAME 65
|
||||||
|
#define REQ_RESET_SOURCES 66
|
||||||
|
#define REQ_AUTH_DATA 67
|
||||||
|
#define REQ_CLIENT_ACCESSES_BY_INDEX3 68
|
||||||
|
#define REQ_SELECT_DATA 69
|
||||||
|
#define REQ_RELOAD_SOURCES 70
|
||||||
|
#define REQ_DOFFSET2 71
|
||||||
|
#define REQ_MODIFY_SELECTOPTS 72
|
||||||
|
#define REQ_MODIFY_OFFSET 73
|
||||||
|
#define REQ_LOCAL3 74
|
||||||
|
#define N_REQUEST_TYPES 75
|
||||||
|
|
||||||
|
/* Structure used to exchange timespecs independent of time_t size */
|
||||||
|
typedef struct {
|
||||||
|
uint32_t tv_sec_high;
|
||||||
|
uint32_t tv_sec_low;
|
||||||
|
uint32_t tv_nsec;
|
||||||
|
} Timespec;
|
||||||
|
|
||||||
|
/* This is used in tv_sec_high for 32-bit timestamps */
|
||||||
|
#define TV_NOHIGHSEC 0x7fffffff
|
||||||
|
|
||||||
|
/* Structure for 64-bit integers (not requiring 64-bit alignment) */
|
||||||
|
typedef struct {
|
||||||
|
uint32_t high;
|
||||||
|
uint32_t low;
|
||||||
|
} Integer64;
|
||||||
|
|
||||||
|
/* 32-bit floating-point format consisting of 7-bit signed exponent
|
||||||
|
and 25-bit signed coefficient without hidden bit.
|
||||||
|
The result is calculated as: 2^(exp - 25) * coef */
|
||||||
|
typedef struct {
|
||||||
|
int32_t f;
|
||||||
|
} Float;
|
||||||
|
|
||||||
|
/* The EOR (end of record) fields are used by the offsetof operator in
|
||||||
|
pktlength.c, to get the number of bytes that ought to be
|
||||||
|
transmitted for each packet type. */
|
||||||
|
|
||||||
|
typedef struct {
|
||||||
|
int32_t EOR;
|
||||||
|
} REQ_Null;
|
||||||
|
|
||||||
|
typedef struct {
|
||||||
|
IPAddr mask;
|
||||||
|
IPAddr address;
|
||||||
|
int32_t EOR;
|
||||||
|
} REQ_Online;
|
||||||
|
|
||||||
|
typedef struct {
|
||||||
|
IPAddr mask;
|
||||||
|
IPAddr address;
|
||||||
|
int32_t EOR;
|
||||||
|
} REQ_Offline;
|
||||||
|
|
||||||
|
typedef struct {
|
||||||
|
IPAddr mask;
|
||||||
|
IPAddr address;
|
||||||
|
int32_t n_good_samples;
|
||||||
|
int32_t n_total_samples;
|
||||||
|
int32_t EOR;
|
||||||
|
} REQ_Burst;
|
||||||
|
|
||||||
|
typedef struct {
|
||||||
|
IPAddr address;
|
||||||
|
int32_t new_minpoll;
|
||||||
|
int32_t EOR;
|
||||||
|
} REQ_Modify_Minpoll;
|
||||||
|
|
||||||
|
typedef struct {
|
||||||
|
IPAddr address;
|
||||||
|
int32_t new_maxpoll;
|
||||||
|
int32_t EOR;
|
||||||
|
} REQ_Modify_Maxpoll;
|
||||||
|
|
||||||
|
typedef struct {
|
||||||
|
int32_t pad;
|
||||||
|
int32_t EOR;
|
||||||
|
} REQ_Dump;
|
||||||
|
|
||||||
|
typedef struct {
|
||||||
|
IPAddr address;
|
||||||
|
Float new_max_delay;
|
||||||
|
int32_t EOR;
|
||||||
|
} REQ_Modify_Maxdelay;
|
||||||
|
|
||||||
|
typedef struct {
|
||||||
|
IPAddr address;
|
||||||
|
Float new_max_delay_ratio;
|
||||||
|
int32_t EOR;
|
||||||
|
} REQ_Modify_Maxdelayratio;
|
||||||
|
|
||||||
|
typedef struct {
|
||||||
|
IPAddr address;
|
||||||
|
Float new_max_delay_dev_ratio;
|
||||||
|
int32_t EOR;
|
||||||
|
} REQ_Modify_Maxdelaydevratio;
|
||||||
|
|
||||||
|
typedef struct {
|
||||||
|
IPAddr address;
|
||||||
|
int32_t new_min_stratum;
|
||||||
|
int32_t EOR;
|
||||||
|
} REQ_Modify_Minstratum;
|
||||||
|
|
||||||
|
typedef struct {
|
||||||
|
IPAddr address;
|
||||||
|
int32_t new_poll_target;
|
||||||
|
int32_t EOR;
|
||||||
|
} REQ_Modify_Polltarget;
|
||||||
|
|
||||||
|
typedef struct {
|
||||||
|
Float new_max_update_skew;
|
||||||
|
int32_t EOR;
|
||||||
|
} REQ_Modify_Maxupdateskew;
|
||||||
|
|
||||||
|
typedef struct {
|
||||||
|
int32_t limit;
|
||||||
|
Float threshold;
|
||||||
|
int32_t EOR;
|
||||||
|
} REQ_Modify_Makestep;
|
||||||
|
|
||||||
|
typedef struct {
|
||||||
|
Timespec ts;
|
||||||
|
int32_t EOR;
|
||||||
|
} REQ_Logon;
|
||||||
|
|
||||||
|
typedef struct {
|
||||||
|
Timespec ts;
|
||||||
|
int32_t EOR;
|
||||||
|
} REQ_Settime;
|
||||||
|
|
||||||
|
typedef struct {
|
||||||
|
int32_t on_off;
|
||||||
|
int32_t stratum;
|
||||||
|
Float distance;
|
||||||
|
int32_t orphan;
|
||||||
|
Float activate;
|
||||||
|
uint32_t reserved[2];
|
||||||
|
int32_t EOR;
|
||||||
|
} REQ_Local;
|
||||||
|
|
||||||
|
typedef struct {
|
||||||
|
int32_t option;
|
||||||
|
int32_t EOR;
|
||||||
|
} REQ_Manual;
|
||||||
|
|
||||||
|
typedef struct {
|
||||||
|
int32_t index;
|
||||||
|
int32_t EOR;
|
||||||
|
} REQ_Source_Data;
|
||||||
|
|
||||||
|
typedef struct {
|
||||||
|
IPAddr ip;
|
||||||
|
int32_t subnet_bits;
|
||||||
|
int32_t EOR;
|
||||||
|
} REQ_Allow_Deny;
|
||||||
|
|
||||||
|
typedef struct {
|
||||||
|
IPAddr ip;
|
||||||
|
int32_t EOR;
|
||||||
|
} REQ_Ac_Check;
|
||||||
|
|
||||||
|
/* Source types in NTP source requests */
|
||||||
|
#define REQ_ADDSRC_SERVER 1
|
||||||
|
#define REQ_ADDSRC_PEER 2
|
||||||
|
#define REQ_ADDSRC_POOL 3
|
||||||
|
|
||||||
|
/* Flags used in NTP source requests */
|
||||||
|
#define REQ_ADDSRC_ONLINE 0x1
|
||||||
|
#define REQ_ADDSRC_AUTOOFFLINE 0x2
|
||||||
|
#define REQ_ADDSRC_IBURST 0x4
|
||||||
|
#define REQ_ADDSRC_PREFER 0x8
|
||||||
|
#define REQ_ADDSRC_NOSELECT 0x10
|
||||||
|
#define REQ_ADDSRC_TRUST 0x20
|
||||||
|
#define REQ_ADDSRC_REQUIRE 0x40
|
||||||
|
#define REQ_ADDSRC_INTERLEAVED 0x80
|
||||||
|
#define REQ_ADDSRC_BURST 0x100
|
||||||
|
#define REQ_ADDSRC_NTS 0x200
|
||||||
|
#define REQ_ADDSRC_COPY 0x400
|
||||||
|
#define REQ_ADDSRC_EF_EXP_MONO_ROOT 0x800
|
||||||
|
#define REQ_ADDSRC_EF_EXP_NET_CORRECTION 0x1000
|
||||||
|
#define REQ_ADDSRC_IPV4 0x2000
|
||||||
|
#define REQ_ADDSRC_IPV6 0x4000
|
||||||
|
|
||||||
|
typedef struct {
|
||||||
|
uint32_t type;
|
||||||
|
uint8_t name[256];
|
||||||
|
uint32_t port;
|
||||||
|
int32_t minpoll;
|
||||||
|
int32_t maxpoll;
|
||||||
|
int32_t presend_minpoll;
|
||||||
|
uint32_t min_stratum;
|
||||||
|
uint32_t poll_target;
|
||||||
|
uint32_t version;
|
||||||
|
uint32_t max_sources;
|
||||||
|
int32_t min_samples;
|
||||||
|
int32_t max_samples;
|
||||||
|
uint32_t authkey;
|
||||||
|
uint32_t nts_port;
|
||||||
|
Float max_delay;
|
||||||
|
Float max_delay_ratio;
|
||||||
|
Float max_delay_dev_ratio;
|
||||||
|
Float min_delay;
|
||||||
|
Float asymmetry;
|
||||||
|
Float offset;
|
||||||
|
uint32_t flags;
|
||||||
|
int32_t filter_length;
|
||||||
|
uint32_t cert_set;
|
||||||
|
Float max_delay_quant;
|
||||||
|
uint32_t reserved[1];
|
||||||
|
int32_t EOR;
|
||||||
|
} REQ_NTP_Source;
|
||||||
|
|
||||||
|
typedef struct {
|
||||||
|
IPAddr ip_addr;
|
||||||
|
int32_t EOR;
|
||||||
|
} REQ_Del_Source;
|
||||||
|
|
||||||
|
typedef struct {
|
||||||
|
Float dfreq;
|
||||||
|
int32_t EOR;
|
||||||
|
} REQ_Dfreq;
|
||||||
|
|
||||||
|
typedef struct {
|
||||||
|
Float doffset;
|
||||||
|
int32_t EOR;
|
||||||
|
} REQ_Doffset;
|
||||||
|
|
||||||
|
typedef struct {
|
||||||
|
uint32_t index;
|
||||||
|
int32_t EOR;
|
||||||
|
} REQ_Sourcestats;
|
||||||
|
|
||||||
|
/* This is based on the response size rather than the
|
||||||
|
request size */
|
||||||
|
#define MAX_CLIENT_ACCESSES 8
|
||||||
|
|
||||||
|
typedef struct {
|
||||||
|
uint32_t first_index;
|
||||||
|
uint32_t n_clients;
|
||||||
|
uint32_t min_hits;
|
||||||
|
uint32_t reset;
|
||||||
|
int32_t EOR;
|
||||||
|
} REQ_ClientAccessesByIndex;
|
||||||
|
|
||||||
|
typedef struct {
|
||||||
|
int32_t index;
|
||||||
|
int32_t EOR;
|
||||||
|
} REQ_ManualDelete;
|
||||||
|
|
||||||
|
typedef struct {
|
||||||
|
Float distance;
|
||||||
|
int32_t EOR;
|
||||||
|
} REQ_ReselectDistance;
|
||||||
|
|
||||||
|
#define REQ_SMOOTHTIME_RESET 0
|
||||||
|
#define REQ_SMOOTHTIME_ACTIVATE 1
|
||||||
|
|
||||||
|
typedef struct {
|
||||||
|
int32_t option;
|
||||||
|
int32_t EOR;
|
||||||
|
} REQ_SmoothTime;
|
||||||
|
|
||||||
|
typedef struct {
|
||||||
|
IPAddr ip_addr;
|
||||||
|
int32_t EOR;
|
||||||
|
} REQ_NTPData;
|
||||||
|
|
||||||
|
typedef struct {
|
||||||
|
IPAddr ip_addr;
|
||||||
|
int32_t EOR;
|
||||||
|
} REQ_NTPSourceName;
|
||||||
|
|
||||||
|
typedef struct {
|
||||||
|
IPAddr ip_addr;
|
||||||
|
int32_t EOR;
|
||||||
|
} REQ_AuthData;
|
||||||
|
|
||||||
|
typedef struct {
|
||||||
|
uint32_t index;
|
||||||
|
int32_t EOR;
|
||||||
|
} REQ_SelectData;
|
||||||
|
|
||||||
|
/* Mask and options reuse the REQ_ADDSRC flags */
|
||||||
|
typedef struct {
|
||||||
|
IPAddr address;
|
||||||
|
uint32_t ref_id;
|
||||||
|
uint32_t mask;
|
||||||
|
uint32_t options;
|
||||||
|
int32_t EOR;
|
||||||
|
} REQ_Modify_SelectOpts;
|
||||||
|
|
||||||
|
typedef struct {
|
||||||
|
IPAddr address;
|
||||||
|
uint32_t ref_id;
|
||||||
|
Float new_offset;
|
||||||
|
int32_t EOR;
|
||||||
|
} REQ_Modify_Offset;
|
||||||
|
|
||||||
|
/* ================================================== */
|
||||||
|
|
||||||
|
#define PKT_TYPE_CMD_REQUEST 1
|
||||||
|
#define PKT_TYPE_CMD_REPLY 2
|
||||||
|
|
||||||
|
/* This version number needs to be incremented whenever the packet
|
||||||
|
size and/or the format of any of the existing messages is changed.
|
||||||
|
Other changes, e.g. new command types, should be handled cleanly by
|
||||||
|
client.c and cmdmon.c anyway, so the version can stay the same.
|
||||||
|
|
||||||
|
Version 1 : original version with fixed size packets
|
||||||
|
|
||||||
|
Version 2 : both command and reply packet sizes made capable of
|
||||||
|
being variable length.
|
||||||
|
|
||||||
|
Version 3 : NTP_Source message lengthened (auto_offline)
|
||||||
|
|
||||||
|
Version 4 : IPv6 addressing added, 64-bit time values, sourcestats
|
||||||
|
and tracking reports extended, added flags to NTP source request,
|
||||||
|
trimmed source report, replaced fixed-point format with floating-point
|
||||||
|
and used also instead of integer microseconds, new commands: modify stratum,
|
||||||
|
modify polltarget, modify maxdelaydevratio, reselect, reselectdistance
|
||||||
|
|
||||||
|
Version 5 : auth data moved to the end of the packet to allow hashes with
|
||||||
|
different sizes, extended sources, tracking and activity reports, dropped
|
||||||
|
subnets accessed and client accesses
|
||||||
|
|
||||||
|
Version 6 : added padding to requests to prevent amplification attack,
|
||||||
|
changed maximum number of samples in manual list to 16, new commands: modify
|
||||||
|
makestep, smoothing, smoothtime
|
||||||
|
|
||||||
|
Support for authentication was removed later in version 6 of the protocol
|
||||||
|
and commands that required authentication are allowed only locally over Unix
|
||||||
|
domain socket.
|
||||||
|
|
||||||
|
Version 6 (no authentication) : changed format of client accesses by index
|
||||||
|
(two times), delta offset, and manual timestamp, added new fields and
|
||||||
|
flags to NTP source request and report, made length of manual list constant,
|
||||||
|
added new commands: authdata, ntpdata, onoffline, refresh, reset,
|
||||||
|
selectdata, serverstats, shutdown, sourcename
|
||||||
|
*/
|
||||||
|
|
||||||
|
#define PROTO_VERSION_NUMBER 6
|
||||||
|
|
||||||
|
/* The oldest protocol versions that are compatible enough with the current
|
||||||
|
version to report a version mismatch for the server and the client */
|
||||||
|
#define PROTO_VERSION_MISMATCH_COMPAT_SERVER 5
|
||||||
|
#define PROTO_VERSION_MISMATCH_COMPAT_CLIENT 4
|
||||||
|
|
||||||
|
/* The first protocol version using padding in requests */
|
||||||
|
#define PROTO_VERSION_PADDING 6
|
||||||
|
|
||||||
|
/* The maximum length of padding in request packet, currently
|
||||||
|
defined by CLIENT_ACCESSES_BY_INDEX3 */
|
||||||
|
#define MAX_PADDING_LENGTH 484
|
||||||
|
|
||||||
|
/* ================================================== */
|
||||||
|
|
||||||
|
typedef struct {
|
||||||
|
uint8_t version; /* Protocol version */
|
||||||
|
uint8_t pkt_type; /* What sort of packet this is */
|
||||||
|
uint8_t res1;
|
||||||
|
uint8_t res2;
|
||||||
|
uint16_t command; /* Which command is being issued */
|
||||||
|
uint16_t attempt; /* How many resends the client has done
|
||||||
|
(count up from zero for same sequence
|
||||||
|
number) */
|
||||||
|
uint32_t sequence; /* Client's sequence number */
|
||||||
|
uint32_t pad1;
|
||||||
|
uint32_t pad2;
|
||||||
|
|
||||||
|
union {
|
||||||
|
REQ_Null null;
|
||||||
|
REQ_Online online;
|
||||||
|
REQ_Offline offline;
|
||||||
|
REQ_Burst burst;
|
||||||
|
REQ_Modify_Minpoll modify_minpoll;
|
||||||
|
REQ_Modify_Maxpoll modify_maxpoll;
|
||||||
|
REQ_Dump dump;
|
||||||
|
REQ_Modify_Maxdelay modify_maxdelay;
|
||||||
|
REQ_Modify_Maxdelayratio modify_maxdelayratio;
|
||||||
|
REQ_Modify_Maxdelaydevratio modify_maxdelaydevratio;
|
||||||
|
REQ_Modify_Minstratum modify_minstratum;
|
||||||
|
REQ_Modify_Polltarget modify_polltarget;
|
||||||
|
REQ_Modify_Maxupdateskew modify_maxupdateskew;
|
||||||
|
REQ_Modify_Makestep modify_makestep;
|
||||||
|
REQ_Logon logon;
|
||||||
|
REQ_Settime settime;
|
||||||
|
REQ_Local local;
|
||||||
|
REQ_Manual manual;
|
||||||
|
REQ_Source_Data source_data;
|
||||||
|
REQ_Allow_Deny allow_deny;
|
||||||
|
REQ_Ac_Check ac_check;
|
||||||
|
REQ_NTP_Source ntp_source;
|
||||||
|
REQ_Del_Source del_source;
|
||||||
|
REQ_Dfreq dfreq;
|
||||||
|
REQ_Doffset doffset;
|
||||||
|
REQ_Sourcestats sourcestats;
|
||||||
|
REQ_ClientAccessesByIndex client_accesses_by_index;
|
||||||
|
REQ_ManualDelete manual_delete;
|
||||||
|
REQ_ReselectDistance reselect_distance;
|
||||||
|
REQ_SmoothTime smoothtime;
|
||||||
|
REQ_NTPData ntp_data;
|
||||||
|
REQ_NTPSourceName ntp_source_name;
|
||||||
|
REQ_AuthData auth_data;
|
||||||
|
REQ_SelectData select_data;
|
||||||
|
REQ_Modify_SelectOpts modify_select_opts;
|
||||||
|
REQ_Modify_Offset modify_offset;
|
||||||
|
} data; /* Command specific parameters */
|
||||||
|
|
||||||
|
/* Padding used to prevent traffic amplification. It only defines the
|
||||||
|
maximum size of the packet, there is no hole after the data field. */
|
||||||
|
uint8_t padding[MAX_PADDING_LENGTH];
|
||||||
|
|
||||||
|
} CMD_Request;
|
||||||
|
|
||||||
|
/* ================================================== */
|
||||||
|
/* Authority codes for command types */
|
||||||
|
|
||||||
|
#define PERMIT_OPEN 0
|
||||||
|
#define PERMIT_LOCAL 1
|
||||||
|
#define PERMIT_AUTH 2
|
||||||
|
|
||||||
|
/* ================================================== */
|
||||||
|
|
||||||
|
/* Reply codes */
|
||||||
|
#define RPY_NULL 1
|
||||||
|
#define RPY_N_SOURCES 2
|
||||||
|
#define RPY_SOURCE_DATA 3
|
||||||
|
#define RPY_MANUAL_TIMESTAMP 4
|
||||||
|
#define RPY_TRACKING 5
|
||||||
|
#define RPY_SOURCESTATS 6
|
||||||
|
#define RPY_RTC 7
|
||||||
|
#define RPY_SUBNETS_ACCESSED 8
|
||||||
|
#define RPY_CLIENT_ACCESSES 9
|
||||||
|
#define RPY_CLIENT_ACCESSES_BY_INDEX 10
|
||||||
|
#define RPY_MANUAL_LIST 11
|
||||||
|
#define RPY_ACTIVITY 12
|
||||||
|
#define RPY_SMOOTHING 13
|
||||||
|
#define RPY_SERVER_STATS 14
|
||||||
|
#define RPY_CLIENT_ACCESSES_BY_INDEX2 15
|
||||||
|
#define RPY_NTP_DATA 16
|
||||||
|
#define RPY_MANUAL_TIMESTAMP2 17
|
||||||
|
#define RPY_MANUAL_LIST2 18
|
||||||
|
#define RPY_NTP_SOURCE_NAME 19
|
||||||
|
#define RPY_AUTH_DATA 20
|
||||||
|
#define RPY_CLIENT_ACCESSES_BY_INDEX3 21
|
||||||
|
#define RPY_SERVER_STATS2 22
|
||||||
|
#define RPY_SELECT_DATA 23
|
||||||
|
#define RPY_SERVER_STATS3 24
|
||||||
|
#define RPY_SERVER_STATS4 25
|
||||||
|
#define RPY_NTP_DATA2 26
|
||||||
|
#define N_REPLY_TYPES 27
|
||||||
|
|
||||||
|
/* Status codes */
|
||||||
|
#define STT_SUCCESS 0
|
||||||
|
#define STT_FAILED 1
|
||||||
|
#define STT_UNAUTH 2
|
||||||
|
#define STT_INVALID 3
|
||||||
|
#define STT_NOSUCHSOURCE 4
|
||||||
|
#define STT_INVALIDTS 5
|
||||||
|
#define STT_NOTENABLED 6
|
||||||
|
#define STT_BADSUBNET 7
|
||||||
|
#define STT_ACCESSALLOWED 8
|
||||||
|
#define STT_ACCESSDENIED 9
|
||||||
|
#define STT_NOHOSTACCESS 10 /* Deprecated */
|
||||||
|
#define STT_SOURCEALREADYKNOWN 11
|
||||||
|
#define STT_TOOMANYSOURCES 12
|
||||||
|
#define STT_NORTC 13
|
||||||
|
#define STT_BADRTCFILE 14
|
||||||
|
#define STT_INACTIVE 15
|
||||||
|
#define STT_BADSAMPLE 16
|
||||||
|
#define STT_INVALIDAF 17
|
||||||
|
#define STT_BADPKTVERSION 18
|
||||||
|
#define STT_BADPKTLENGTH 19
|
||||||
|
#define STT_INVALIDNAME 21
|
||||||
|
|
||||||
|
typedef struct {
|
||||||
|
int32_t EOR;
|
||||||
|
} RPY_Null;
|
||||||
|
|
||||||
|
typedef struct {
|
||||||
|
uint32_t n_sources;
|
||||||
|
int32_t EOR;
|
||||||
|
} RPY_N_Sources;
|
||||||
|
|
||||||
|
#define RPY_SD_MD_CLIENT 0
|
||||||
|
#define RPY_SD_MD_PEER 1
|
||||||
|
#define RPY_SD_MD_REF 2
|
||||||
|
|
||||||
|
#define RPY_SD_ST_SELECTED 0
|
||||||
|
#define RPY_SD_ST_NONSELECTABLE 1
|
||||||
|
#define RPY_SD_ST_FALSETICKER 2
|
||||||
|
#define RPY_SD_ST_JITTERY 3
|
||||||
|
#define RPY_SD_ST_UNSELECTED 4
|
||||||
|
#define RPY_SD_ST_SELECTABLE 5
|
||||||
|
|
||||||
|
typedef struct {
|
||||||
|
IPAddr ip_addr;
|
||||||
|
int16_t poll;
|
||||||
|
uint16_t stratum;
|
||||||
|
uint16_t state;
|
||||||
|
uint16_t mode;
|
||||||
|
uint16_t flags;
|
||||||
|
uint16_t reachability;
|
||||||
|
uint32_t since_sample;
|
||||||
|
Float orig_latest_meas;
|
||||||
|
Float latest_meas;
|
||||||
|
Float latest_meas_err;
|
||||||
|
int32_t EOR;
|
||||||
|
} RPY_Source_Data;
|
||||||
|
|
||||||
|
typedef struct {
|
||||||
|
uint32_t ref_id;
|
||||||
|
IPAddr ip_addr;
|
||||||
|
uint16_t stratum;
|
||||||
|
uint16_t leap_status;
|
||||||
|
Timespec ref_time;
|
||||||
|
Float current_correction;
|
||||||
|
Float last_offset;
|
||||||
|
Float rms_offset;
|
||||||
|
Float freq_ppm;
|
||||||
|
Float resid_freq_ppm;
|
||||||
|
Float skew_ppm;
|
||||||
|
Float root_delay;
|
||||||
|
Float root_dispersion;
|
||||||
|
Float last_update_interval;
|
||||||
|
int32_t EOR;
|
||||||
|
} RPY_Tracking;
|
||||||
|
|
||||||
|
typedef struct {
|
||||||
|
uint32_t ref_id;
|
||||||
|
IPAddr ip_addr;
|
||||||
|
uint32_t n_samples;
|
||||||
|
uint32_t n_runs;
|
||||||
|
uint32_t span_seconds;
|
||||||
|
Float sd;
|
||||||
|
Float resid_freq_ppm;
|
||||||
|
Float skew_ppm;
|
||||||
|
Float est_offset;
|
||||||
|
Float est_offset_err;
|
||||||
|
int32_t EOR;
|
||||||
|
} RPY_Sourcestats;
|
||||||
|
|
||||||
|
typedef struct {
|
||||||
|
Timespec ref_time;
|
||||||
|
uint16_t n_samples;
|
||||||
|
uint16_t n_runs;
|
||||||
|
uint32_t span_seconds;
|
||||||
|
Float rtc_seconds_fast;
|
||||||
|
Float rtc_gain_rate_ppm;
|
||||||
|
int32_t EOR;
|
||||||
|
} RPY_Rtc;
|
||||||
|
|
||||||
|
typedef struct {
|
||||||
|
Float offset;
|
||||||
|
Float dfreq_ppm;
|
||||||
|
Float new_afreq_ppm;
|
||||||
|
int32_t EOR;
|
||||||
|
} RPY_ManualTimestamp;
|
||||||
|
|
||||||
|
typedef struct {
|
||||||
|
IPAddr ip;
|
||||||
|
uint32_t ntp_hits;
|
||||||
|
uint32_t nke_hits;
|
||||||
|
uint32_t cmd_hits;
|
||||||
|
uint32_t ntp_drops;
|
||||||
|
uint32_t nke_drops;
|
||||||
|
uint32_t cmd_drops;
|
||||||
|
int8_t ntp_interval;
|
||||||
|
int8_t nke_interval;
|
||||||
|
int8_t cmd_interval;
|
||||||
|
int8_t ntp_timeout_interval;
|
||||||
|
uint32_t last_ntp_hit_ago;
|
||||||
|
uint32_t last_nke_hit_ago;
|
||||||
|
uint32_t last_cmd_hit_ago;
|
||||||
|
} RPY_ClientAccesses_Client;
|
||||||
|
|
||||||
|
typedef struct {
|
||||||
|
uint32_t n_indices; /* how many indices there are in the server's table */
|
||||||
|
uint32_t next_index; /* the index 1 beyond those processed on this call */
|
||||||
|
uint32_t n_clients; /* the number of valid entries in the following array */
|
||||||
|
RPY_ClientAccesses_Client clients[MAX_CLIENT_ACCESSES];
|
||||||
|
int32_t EOR;
|
||||||
|
} RPY_ClientAccessesByIndex;
|
||||||
|
|
||||||
|
typedef struct {
|
||||||
|
Integer64 ntp_hits;
|
||||||
|
Integer64 nke_hits;
|
||||||
|
Integer64 cmd_hits;
|
||||||
|
Integer64 ntp_drops;
|
||||||
|
Integer64 nke_drops;
|
||||||
|
Integer64 cmd_drops;
|
||||||
|
Integer64 log_drops;
|
||||||
|
Integer64 ntp_auth_hits;
|
||||||
|
Integer64 ntp_interleaved_hits;
|
||||||
|
Integer64 ntp_timestamps;
|
||||||
|
Integer64 ntp_span_seconds;
|
||||||
|
Integer64 ntp_daemon_rx_timestamps;
|
||||||
|
Integer64 ntp_daemon_tx_timestamps;
|
||||||
|
Integer64 ntp_kernel_rx_timestamps;
|
||||||
|
Integer64 ntp_kernel_tx_timestamps;
|
||||||
|
Integer64 ntp_hw_rx_timestamps;
|
||||||
|
Integer64 ntp_hw_tx_timestamps;
|
||||||
|
Integer64 reserved[4];
|
||||||
|
int32_t EOR;
|
||||||
|
} RPY_ServerStats;
|
||||||
|
|
||||||
|
#define MAX_MANUAL_LIST_SAMPLES 16
|
||||||
|
|
||||||
|
typedef struct {
|
||||||
|
Timespec when;
|
||||||
|
Float slewed_offset;
|
||||||
|
Float orig_offset;
|
||||||
|
Float residual;
|
||||||
|
} RPY_ManualListSample;
|
||||||
|
|
||||||
|
typedef struct {
|
||||||
|
uint32_t n_samples;
|
||||||
|
RPY_ManualListSample samples[MAX_MANUAL_LIST_SAMPLES];
|
||||||
|
int32_t EOR;
|
||||||
|
} RPY_ManualList;
|
||||||
|
|
||||||
|
typedef struct {
|
||||||
|
int32_t online;
|
||||||
|
int32_t offline;
|
||||||
|
int32_t burst_online;
|
||||||
|
int32_t burst_offline;
|
||||||
|
int32_t unresolved;
|
||||||
|
int32_t EOR;
|
||||||
|
} RPY_Activity;
|
||||||
|
|
||||||
|
#define RPY_SMT_FLAG_ACTIVE 0x1
|
||||||
|
#define RPY_SMT_FLAG_LEAPONLY 0x2
|
||||||
|
|
||||||
|
typedef struct {
|
||||||
|
uint32_t flags;
|
||||||
|
Float offset;
|
||||||
|
Float freq_ppm;
|
||||||
|
Float wander_ppm;
|
||||||
|
Float last_update_ago;
|
||||||
|
Float remaining_time;
|
||||||
|
int32_t EOR;
|
||||||
|
} RPY_Smoothing;
|
||||||
|
|
||||||
|
#define RPY_NTP_FLAGS_TESTS 0x3ff
|
||||||
|
#define RPY_NTP_FLAG_INTERLEAVED 0x4000
|
||||||
|
#define RPY_NTP_FLAG_AUTHENTICATED 0x8000
|
||||||
|
|
||||||
|
typedef struct {
|
||||||
|
IPAddr remote_addr;
|
||||||
|
IPAddr local_addr;
|
||||||
|
uint16_t remote_port;
|
||||||
|
uint8_t leap;
|
||||||
|
uint8_t version;
|
||||||
|
uint8_t mode;
|
||||||
|
uint8_t stratum;
|
||||||
|
int8_t poll;
|
||||||
|
int8_t precision;
|
||||||
|
Float root_delay;
|
||||||
|
Float root_dispersion;
|
||||||
|
uint32_t ref_id;
|
||||||
|
Timespec ref_time;
|
||||||
|
Float offset;
|
||||||
|
Float peer_delay;
|
||||||
|
Float peer_dispersion;
|
||||||
|
Float response_time;
|
||||||
|
Float jitter_asymmetry;
|
||||||
|
uint16_t flags;
|
||||||
|
uint8_t tx_tss_char;
|
||||||
|
uint8_t rx_tss_char;
|
||||||
|
uint32_t total_tx_count;
|
||||||
|
uint32_t total_rx_count;
|
||||||
|
uint32_t total_valid_count;
|
||||||
|
uint32_t total_good_count;
|
||||||
|
uint32_t total_kernel_tx_ts;
|
||||||
|
uint32_t total_kernel_rx_ts;
|
||||||
|
uint32_t total_hw_tx_ts;
|
||||||
|
uint32_t total_hw_rx_ts;
|
||||||
|
uint32_t reserved[4];
|
||||||
|
int32_t EOR;
|
||||||
|
} RPY_NTPData;
|
||||||
|
|
||||||
|
typedef struct {
|
||||||
|
uint8_t name[256];
|
||||||
|
int32_t EOR;
|
||||||
|
} RPY_NTPSourceName;
|
||||||
|
|
||||||
|
#define RPY_AD_MD_NONE 0
|
||||||
|
#define RPY_AD_MD_SYMMETRIC 1
|
||||||
|
#define RPY_AD_MD_NTS 2
|
||||||
|
|
||||||
|
typedef struct {
|
||||||
|
uint16_t mode;
|
||||||
|
uint16_t key_type;
|
||||||
|
uint32_t key_id;
|
||||||
|
uint16_t key_length;
|
||||||
|
uint16_t ke_attempts;
|
||||||
|
uint32_t last_ke_ago;
|
||||||
|
uint16_t cookies;
|
||||||
|
uint16_t cookie_length;
|
||||||
|
uint16_t nak;
|
||||||
|
uint16_t pad;
|
||||||
|
int32_t EOR;
|
||||||
|
} RPY_AuthData;
|
||||||
|
|
||||||
|
#define RPY_SD_OPTION_NOSELECT 0x1
|
||||||
|
#define RPY_SD_OPTION_PREFER 0x2
|
||||||
|
#define RPY_SD_OPTION_TRUST 0x4
|
||||||
|
#define RPY_SD_OPTION_REQUIRE 0x8
|
||||||
|
|
||||||
|
typedef struct {
|
||||||
|
uint32_t ref_id;
|
||||||
|
IPAddr ip_addr;
|
||||||
|
uint8_t state_char;
|
||||||
|
uint8_t authentication;
|
||||||
|
uint8_t leap;
|
||||||
|
uint8_t pad;
|
||||||
|
uint16_t conf_options;
|
||||||
|
uint16_t eff_options;
|
||||||
|
uint32_t last_sample_ago;
|
||||||
|
Float score;
|
||||||
|
Float lo_limit;
|
||||||
|
Float hi_limit;
|
||||||
|
int32_t EOR;
|
||||||
|
} RPY_SelectData;
|
||||||
|
|
||||||
|
typedef struct {
|
||||||
|
uint8_t version;
|
||||||
|
uint8_t pkt_type;
|
||||||
|
uint8_t res1;
|
||||||
|
uint8_t res2;
|
||||||
|
uint16_t command; /* Which command is being replied to */
|
||||||
|
uint16_t reply; /* Which format of reply this is */
|
||||||
|
uint16_t status; /* Status of command processing */
|
||||||
|
uint16_t pad1; /* Padding for compatibility and 4 byte alignment */
|
||||||
|
uint16_t pad2;
|
||||||
|
uint16_t pad3;
|
||||||
|
uint32_t sequence; /* Echo of client's sequence number */
|
||||||
|
uint32_t pad4;
|
||||||
|
uint32_t pad5;
|
||||||
|
|
||||||
|
union {
|
||||||
|
RPY_Null null;
|
||||||
|
RPY_N_Sources n_sources;
|
||||||
|
RPY_Source_Data source_data;
|
||||||
|
RPY_ManualTimestamp manual_timestamp;
|
||||||
|
RPY_Tracking tracking;
|
||||||
|
RPY_Sourcestats sourcestats;
|
||||||
|
RPY_Rtc rtc;
|
||||||
|
RPY_ClientAccessesByIndex client_accesses_by_index;
|
||||||
|
RPY_ServerStats server_stats;
|
||||||
|
RPY_ManualList manual_list;
|
||||||
|
RPY_Activity activity;
|
||||||
|
RPY_Smoothing smoothing;
|
||||||
|
RPY_NTPData ntp_data;
|
||||||
|
RPY_NTPSourceName ntp_source_name;
|
||||||
|
RPY_AuthData auth_data;
|
||||||
|
RPY_SelectData select_data;
|
||||||
|
} data; /* Reply specific parameters */
|
||||||
|
|
||||||
|
} CMD_Reply;
|
||||||
|
|
||||||
|
/* ================================================== */
|
||||||
|
|
||||||
|
#endif /* GOT_CANDM_H */
|
1125
clientlog.c
Normal file
1125
clientlog.c
Normal file
File diff suppressed because it is too large
Load diff
73
clientlog.h
Normal file
73
clientlog.h
Normal file
|
@ -0,0 +1,73 @@
|
||||||
|
/*
|
||||||
|
chronyd/chronyc - Programs for keeping computer clocks accurate.
|
||||||
|
|
||||||
|
**********************************************************************
|
||||||
|
* Copyright (C) Richard P. Curnow 1997-2003
|
||||||
|
*
|
||||||
|
* This program is free software; you can redistribute it and/or modify
|
||||||
|
* it under the terms of version 2 of the GNU General Public License as
|
||||||
|
* published by the Free Software Foundation.
|
||||||
|
*
|
||||||
|
* This program is distributed in the hope that it will be useful, but
|
||||||
|
* WITHOUT ANY WARRANTY; without even the implied warranty of
|
||||||
|
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
|
||||||
|
* General Public License for more details.
|
||||||
|
*
|
||||||
|
* You should have received a copy of the GNU General Public License along
|
||||||
|
* with this program; if not, write to the Free Software Foundation, Inc.,
|
||||||
|
* 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA.
|
||||||
|
*
|
||||||
|
**********************************************************************
|
||||||
|
|
||||||
|
=======================================================================
|
||||||
|
|
||||||
|
This module contains facilities for logging access by clients.
|
||||||
|
|
||||||
|
*/
|
||||||
|
|
||||||
|
#ifndef GOT_CLIENTLOG_H
|
||||||
|
#define GOT_CLIENTLOG_H
|
||||||
|
|
||||||
|
#include "sysincl.h"
|
||||||
|
#include "reports.h"
|
||||||
|
|
||||||
|
typedef enum {
|
||||||
|
CLG_NTP = 0,
|
||||||
|
CLG_NTSKE,
|
||||||
|
CLG_CMDMON,
|
||||||
|
} CLG_Service;
|
||||||
|
|
||||||
|
typedef enum {
|
||||||
|
CLG_PASS = 0,
|
||||||
|
CLG_DROP,
|
||||||
|
CLG_KOD,
|
||||||
|
} CLG_Limit;
|
||||||
|
|
||||||
|
extern void CLG_Initialise(void);
|
||||||
|
extern void CLG_Finalise(void);
|
||||||
|
extern int CLG_GetClientIndex(IPAddr *client);
|
||||||
|
extern int CLG_LogServiceAccess(CLG_Service service, IPAddr *client, struct timespec *now);
|
||||||
|
extern CLG_Limit CLG_LimitServiceRate(CLG_Service service, int index);
|
||||||
|
extern void CLG_UpdateNtpStats(int auth, NTP_Timestamp_Source rx_ts_src,
|
||||||
|
NTP_Timestamp_Source tx_ts_src);
|
||||||
|
extern int CLG_GetNtpMinPoll(void);
|
||||||
|
|
||||||
|
/* Functions to save and retrieve timestamps for server interleaved mode */
|
||||||
|
extern void CLG_SaveNtpTimestamps(NTP_int64 *rx_ts, struct timespec *tx_ts,
|
||||||
|
NTP_Timestamp_Source tx_src);
|
||||||
|
extern void CLG_UndoNtpTxTimestampSlew(NTP_int64 *rx_ts, struct timespec *tx_ts);
|
||||||
|
extern void CLG_UpdateNtpTxTimestamp(NTP_int64 *rx_ts, struct timespec *tx_ts,
|
||||||
|
NTP_Timestamp_Source tx_src);
|
||||||
|
extern int CLG_GetNtpTxTimestamp(NTP_int64 *rx_ts, struct timespec *tx_ts,
|
||||||
|
NTP_Timestamp_Source *tx_src);
|
||||||
|
extern void CLG_DisableNtpTimestamps(NTP_int64 *rx_ts);
|
||||||
|
|
||||||
|
/* And some reporting functions, for use by chronyc. */
|
||||||
|
|
||||||
|
extern int CLG_GetNumberOfIndices(void);
|
||||||
|
extern int CLG_GetClientAccessReportByIndex(int index, int reset, uint32_t min_hits,
|
||||||
|
RPT_ClientAccessByIndex_Report *report,
|
||||||
|
struct timespec *now);
|
||||||
|
extern void CLG_GetServerStatsReport(RPT_ServerStatsReport *report);
|
||||||
|
|
||||||
|
#endif /* GOT_CLIENTLOG_H */
|
48
cmac.h
Normal file
48
cmac.h
Normal file
|
@ -0,0 +1,48 @@
|
||||||
|
/*
|
||||||
|
chronyd/chronyc - Programs for keeping computer clocks accurate.
|
||||||
|
|
||||||
|
**********************************************************************
|
||||||
|
* Copyright (C) Miroslav Lichvar 2019
|
||||||
|
*
|
||||||
|
* This program is free software; you can redistribute it and/or modify
|
||||||
|
* it under the terms of version 2 of the GNU General Public License as
|
||||||
|
* published by the Free Software Foundation.
|
||||||
|
*
|
||||||
|
* This program is distributed in the hope that it will be useful, but
|
||||||
|
* WITHOUT ANY WARRANTY; without even the implied warranty of
|
||||||
|
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
|
||||||
|
* General Public License for more details.
|
||||||
|
*
|
||||||
|
* You should have received a copy of the GNU General Public License along
|
||||||
|
* with this program; if not, write to the Free Software Foundation, Inc.,
|
||||||
|
* 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA.
|
||||||
|
*
|
||||||
|
**********************************************************************
|
||||||
|
|
||||||
|
=======================================================================
|
||||||
|
|
||||||
|
Header file for CMAC.
|
||||||
|
|
||||||
|
*/
|
||||||
|
|
||||||
|
#ifndef GOT_CMAC_H
|
||||||
|
#define GOT_CMAC_H
|
||||||
|
|
||||||
|
/* Avoid overlapping with the hash enumeration */
|
||||||
|
typedef enum {
|
||||||
|
CMC_INVALID = 0,
|
||||||
|
CMC_AES128 = 13,
|
||||||
|
CMC_AES256 = 14,
|
||||||
|
} CMC_Algorithm;
|
||||||
|
|
||||||
|
typedef struct CMC_Instance_Record *CMC_Instance;
|
||||||
|
|
||||||
|
extern int CMC_GetKeyLength(CMC_Algorithm algorithm);
|
||||||
|
extern CMC_Instance CMC_CreateInstance(CMC_Algorithm algorithm, const unsigned char *key,
|
||||||
|
int length);
|
||||||
|
extern int CMC_Hash(CMC_Instance inst, const void *in, int in_len,
|
||||||
|
unsigned char *out, int out_len);
|
||||||
|
extern void CMC_DestroyInstance(CMC_Instance inst);
|
||||||
|
|
||||||
|
#endif
|
||||||
|
|
189
cmac_gnutls.c
Normal file
189
cmac_gnutls.c
Normal file
|
@ -0,0 +1,189 @@
|
||||||
|
/*
|
||||||
|
chronyd/chronyc - Programs for keeping computer clocks accurate.
|
||||||
|
|
||||||
|
**********************************************************************
|
||||||
|
* Copyright (C) Miroslav Lichvar 2021
|
||||||
|
*
|
||||||
|
* This program is free software; you can redistribute it and/or modify
|
||||||
|
* it under the terms of version 2 of the GNU General Public License as
|
||||||
|
* published by the Free Software Foundation.
|
||||||
|
*
|
||||||
|
* This program is distributed in the hope that it will be useful, but
|
||||||
|
* WITHOUT ANY WARRANTY; without even the implied warranty of
|
||||||
|
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
|
||||||
|
* General Public License for more details.
|
||||||
|
*
|
||||||
|
* You should have received a copy of the GNU General Public License along
|
||||||
|
* with this program; if not, write to the Free Software Foundation, Inc.,
|
||||||
|
* 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA.
|
||||||
|
*
|
||||||
|
**********************************************************************
|
||||||
|
|
||||||
|
=======================================================================
|
||||||
|
|
||||||
|
CMAC using the GnuTLS library
|
||||||
|
*/
|
||||||
|
|
||||||
|
#include "config.h"
|
||||||
|
|
||||||
|
#include "sysincl.h"
|
||||||
|
|
||||||
|
#include <gnutls/crypto.h>
|
||||||
|
|
||||||
|
#include "cmac.h"
|
||||||
|
#include "hash.h"
|
||||||
|
#include "logging.h"
|
||||||
|
#include "memory.h"
|
||||||
|
|
||||||
|
struct CMC_Instance_Record {
|
||||||
|
gnutls_mac_algorithm_t algorithm;
|
||||||
|
gnutls_hmac_hd_t mac;
|
||||||
|
};
|
||||||
|
|
||||||
|
/* ================================================== */
|
||||||
|
|
||||||
|
static int instance_counter = 0;
|
||||||
|
static int gnutls_initialised = 0;
|
||||||
|
|
||||||
|
/* ================================================== */
|
||||||
|
|
||||||
|
static void
|
||||||
|
init_gnutls(void)
|
||||||
|
{
|
||||||
|
int r;
|
||||||
|
|
||||||
|
if (gnutls_initialised)
|
||||||
|
return;
|
||||||
|
|
||||||
|
r = gnutls_global_init();
|
||||||
|
if (r < 0)
|
||||||
|
LOG_FATAL("Could not initialise %s : %s", "gnutls", gnutls_strerror(r));
|
||||||
|
|
||||||
|
DEBUG_LOG("Initialised");
|
||||||
|
gnutls_initialised = 1;
|
||||||
|
}
|
||||||
|
|
||||||
|
/* ================================================== */
|
||||||
|
|
||||||
|
static void
|
||||||
|
deinit_gnutls(void)
|
||||||
|
{
|
||||||
|
assert(gnutls_initialised);
|
||||||
|
gnutls_global_deinit();
|
||||||
|
gnutls_initialised = 0;
|
||||||
|
DEBUG_LOG("Deinitialised");
|
||||||
|
}
|
||||||
|
|
||||||
|
/* ================================================== */
|
||||||
|
|
||||||
|
static gnutls_mac_algorithm_t
|
||||||
|
get_mac_algorithm(CMC_Algorithm algorithm)
|
||||||
|
{
|
||||||
|
switch (algorithm) {
|
||||||
|
case CMC_AES128:
|
||||||
|
return GNUTLS_MAC_AES_CMAC_128;
|
||||||
|
case CMC_AES256:
|
||||||
|
return GNUTLS_MAC_AES_CMAC_256;
|
||||||
|
default:
|
||||||
|
return GNUTLS_MAC_UNKNOWN;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/* ================================================== */
|
||||||
|
|
||||||
|
int
|
||||||
|
CMC_GetKeyLength(CMC_Algorithm algorithm)
|
||||||
|
{
|
||||||
|
gnutls_mac_algorithm_t malgo = get_mac_algorithm(algorithm);
|
||||||
|
int len;
|
||||||
|
|
||||||
|
if (malgo == GNUTLS_MAC_UNKNOWN)
|
||||||
|
return 0;
|
||||||
|
|
||||||
|
len = gnutls_hmac_get_key_size(malgo);
|
||||||
|
|
||||||
|
if (len < 0)
|
||||||
|
return 0;
|
||||||
|
|
||||||
|
return len;
|
||||||
|
}
|
||||||
|
|
||||||
|
/* ================================================== */
|
||||||
|
|
||||||
|
CMC_Instance
|
||||||
|
CMC_CreateInstance(CMC_Algorithm algorithm, const unsigned char *key, int length)
|
||||||
|
{
|
||||||
|
gnutls_hmac_hd_t handle;
|
||||||
|
CMC_Instance inst;
|
||||||
|
|
||||||
|
int r;
|
||||||
|
|
||||||
|
if (instance_counter == 0)
|
||||||
|
init_gnutls();
|
||||||
|
|
||||||
|
if (length <= 0 || length != CMC_GetKeyLength(algorithm))
|
||||||
|
goto error;
|
||||||
|
|
||||||
|
r = gnutls_hmac_init(&handle, get_mac_algorithm(algorithm), key, length);
|
||||||
|
if (r < 0) {
|
||||||
|
DEBUG_LOG("Could not initialise %s : %s", "mac", gnutls_strerror(r));
|
||||||
|
goto error;
|
||||||
|
}
|
||||||
|
|
||||||
|
inst = MallocNew(struct CMC_Instance_Record);
|
||||||
|
inst->algorithm = get_mac_algorithm(algorithm);
|
||||||
|
inst->mac = handle;
|
||||||
|
|
||||||
|
instance_counter++;
|
||||||
|
|
||||||
|
return inst;
|
||||||
|
|
||||||
|
error:
|
||||||
|
if (instance_counter == 0)
|
||||||
|
deinit_gnutls();
|
||||||
|
return NULL;
|
||||||
|
}
|
||||||
|
|
||||||
|
/* ================================================== */
|
||||||
|
|
||||||
|
int
|
||||||
|
CMC_Hash(CMC_Instance inst, const void *in, int in_len, unsigned char *out, int out_len)
|
||||||
|
{
|
||||||
|
unsigned char buf[MAX_HASH_LENGTH];
|
||||||
|
int hash_len;
|
||||||
|
|
||||||
|
if (in_len < 0 || out_len < 0)
|
||||||
|
return 0;
|
||||||
|
|
||||||
|
hash_len = gnutls_hmac_get_len(inst->algorithm);
|
||||||
|
|
||||||
|
if (out_len > hash_len)
|
||||||
|
out_len = hash_len;
|
||||||
|
|
||||||
|
if (hash_len > sizeof (buf))
|
||||||
|
return 0;
|
||||||
|
|
||||||
|
if (gnutls_hmac(inst->mac, in, in_len) < 0) {
|
||||||
|
/* Reset the state */
|
||||||
|
gnutls_hmac_output(inst->mac, buf);
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
gnutls_hmac_output(inst->mac, buf);
|
||||||
|
memcpy(out, buf, out_len);
|
||||||
|
|
||||||
|
return out_len;
|
||||||
|
}
|
||||||
|
|
||||||
|
/* ================================================== */
|
||||||
|
|
||||||
|
void
|
||||||
|
CMC_DestroyInstance(CMC_Instance inst)
|
||||||
|
{
|
||||||
|
gnutls_hmac_deinit(inst->mac, NULL);
|
||||||
|
Free(inst);
|
||||||
|
|
||||||
|
instance_counter--;
|
||||||
|
if (instance_counter == 0)
|
||||||
|
deinit_gnutls();
|
||||||
|
}
|
117
cmac_nettle.c
Normal file
117
cmac_nettle.c
Normal file
|
@ -0,0 +1,117 @@
|
||||||
|
/*
|
||||||
|
chronyd/chronyc - Programs for keeping computer clocks accurate.
|
||||||
|
|
||||||
|
**********************************************************************
|
||||||
|
* Copyright (C) Miroslav Lichvar 2019
|
||||||
|
*
|
||||||
|
* This program is free software; you can redistribute it and/or modify
|
||||||
|
* it under the terms of version 2 of the GNU General Public License as
|
||||||
|
* published by the Free Software Foundation.
|
||||||
|
*
|
||||||
|
* This program is distributed in the hope that it will be useful, but
|
||||||
|
* WITHOUT ANY WARRANTY; without even the implied warranty of
|
||||||
|
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
|
||||||
|
* General Public License for more details.
|
||||||
|
*
|
||||||
|
* You should have received a copy of the GNU General Public License along
|
||||||
|
* with this program; if not, write to the Free Software Foundation, Inc.,
|
||||||
|
* 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA.
|
||||||
|
*
|
||||||
|
**********************************************************************
|
||||||
|
|
||||||
|
=======================================================================
|
||||||
|
|
||||||
|
Support for AES128 and AES256 CMAC in Nettle.
|
||||||
|
|
||||||
|
*/
|
||||||
|
|
||||||
|
#include "config.h"
|
||||||
|
|
||||||
|
#include "sysincl.h"
|
||||||
|
|
||||||
|
#include <nettle/cmac.h>
|
||||||
|
|
||||||
|
#include "cmac.h"
|
||||||
|
#include "memory.h"
|
||||||
|
|
||||||
|
struct CMC_Instance_Record {
|
||||||
|
int key_length;
|
||||||
|
union {
|
||||||
|
struct cmac_aes128_ctx aes128;
|
||||||
|
struct cmac_aes256_ctx aes256;
|
||||||
|
} context;
|
||||||
|
};
|
||||||
|
|
||||||
|
/* ================================================== */
|
||||||
|
|
||||||
|
int
|
||||||
|
CMC_GetKeyLength(CMC_Algorithm algorithm)
|
||||||
|
{
|
||||||
|
if (algorithm == CMC_AES128)
|
||||||
|
return AES128_KEY_SIZE;
|
||||||
|
else if (algorithm == CMC_AES256)
|
||||||
|
return AES256_KEY_SIZE;
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
/* ================================================== */
|
||||||
|
|
||||||
|
CMC_Instance
|
||||||
|
CMC_CreateInstance(CMC_Algorithm algorithm, const unsigned char *key, int length)
|
||||||
|
{
|
||||||
|
CMC_Instance inst;
|
||||||
|
|
||||||
|
if (length <= 0 || length != CMC_GetKeyLength(algorithm))
|
||||||
|
return NULL;
|
||||||
|
|
||||||
|
inst = MallocNew(struct CMC_Instance_Record);
|
||||||
|
inst->key_length = length;
|
||||||
|
|
||||||
|
switch (length) {
|
||||||
|
case AES128_KEY_SIZE:
|
||||||
|
cmac_aes128_set_key(&inst->context.aes128, key);
|
||||||
|
break;
|
||||||
|
case AES256_KEY_SIZE:
|
||||||
|
cmac_aes256_set_key(&inst->context.aes256, key);
|
||||||
|
break;
|
||||||
|
default:
|
||||||
|
assert(0);
|
||||||
|
}
|
||||||
|
|
||||||
|
return inst;
|
||||||
|
}
|
||||||
|
|
||||||
|
/* ================================================== */
|
||||||
|
|
||||||
|
int
|
||||||
|
CMC_Hash(CMC_Instance inst, const void *in, int in_len, unsigned char *out, int out_len)
|
||||||
|
{
|
||||||
|
if (in_len < 0 || out_len < 0)
|
||||||
|
return 0;
|
||||||
|
|
||||||
|
if (out_len > CMAC128_DIGEST_SIZE)
|
||||||
|
out_len = CMAC128_DIGEST_SIZE;
|
||||||
|
|
||||||
|
switch (inst->key_length) {
|
||||||
|
case AES128_KEY_SIZE:
|
||||||
|
cmac_aes128_update(&inst->context.aes128, in_len, in);
|
||||||
|
cmac_aes128_digest(&inst->context.aes128, out_len, out);
|
||||||
|
break;
|
||||||
|
case AES256_KEY_SIZE:
|
||||||
|
cmac_aes256_update(&inst->context.aes256, in_len, in);
|
||||||
|
cmac_aes256_digest(&inst->context.aes256, out_len, out);
|
||||||
|
break;
|
||||||
|
default:
|
||||||
|
assert(0);
|
||||||
|
}
|
||||||
|
|
||||||
|
return out_len;
|
||||||
|
}
|
||||||
|
|
||||||
|
/* ================================================== */
|
||||||
|
|
||||||
|
void
|
||||||
|
CMC_DestroyInstance(CMC_Instance inst)
|
||||||
|
{
|
||||||
|
Free(inst);
|
||||||
|
}
|
40
cmdmon.h
Normal file
40
cmdmon.h
Normal file
|
@ -0,0 +1,40 @@
|
||||||
|
/*
|
||||||
|
chronyd/chronyc - Programs for keeping computer clocks accurate.
|
||||||
|
|
||||||
|
**********************************************************************
|
||||||
|
* Copyright (C) Richard P. Curnow 1997-2002
|
||||||
|
*
|
||||||
|
* This program is free software; you can redistribute it and/or modify
|
||||||
|
* it under the terms of version 2 of the GNU General Public License as
|
||||||
|
* published by the Free Software Foundation.
|
||||||
|
*
|
||||||
|
* This program is distributed in the hope that it will be useful, but
|
||||||
|
* WITHOUT ANY WARRANTY; without even the implied warranty of
|
||||||
|
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
|
||||||
|
* General Public License for more details.
|
||||||
|
*
|
||||||
|
* You should have received a copy of the GNU General Public License along
|
||||||
|
* with this program; if not, write to the Free Software Foundation, Inc.,
|
||||||
|
* 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA.
|
||||||
|
*
|
||||||
|
**********************************************************************
|
||||||
|
|
||||||
|
=======================================================================
|
||||||
|
|
||||||
|
Header file for the control and monitoring module in the software
|
||||||
|
*/
|
||||||
|
|
||||||
|
#ifndef GOT_CMDMON_H
|
||||||
|
#define GOT_CMDMON_H
|
||||||
|
|
||||||
|
#include "addressing.h"
|
||||||
|
|
||||||
|
extern void CAM_Initialise(void);
|
||||||
|
|
||||||
|
extern void CAM_Finalise(void);
|
||||||
|
|
||||||
|
extern void CAM_OpenUnixSocket(void);
|
||||||
|
extern int CAM_AddAccessRestriction(IPAddr *ip_addr, int subnet_bits, int allow, int all);
|
||||||
|
extern int CAM_CheckAccessRestriction(IPAddr *ip_addr);
|
||||||
|
|
||||||
|
#endif /* GOT_CMDMON_H */
|
437
cmdparse.c
Normal file
437
cmdparse.c
Normal file
|
@ -0,0 +1,437 @@
|
||||||
|
/*
|
||||||
|
chronyd/chronyc - Programs for keeping computer clocks accurate.
|
||||||
|
|
||||||
|
**********************************************************************
|
||||||
|
* Copyright (C) Richard P. Curnow 1997-2003
|
||||||
|
* Copyright (C) Miroslav Lichvar 2013-2014, 2016, 2021
|
||||||
|
*
|
||||||
|
* This program is free software; you can redistribute it and/or modify
|
||||||
|
* it under the terms of version 2 of the GNU General Public License as
|
||||||
|
* published by the Free Software Foundation.
|
||||||
|
*
|
||||||
|
* This program is distributed in the hope that it will be useful, but
|
||||||
|
* WITHOUT ANY WARRANTY; without even the implied warranty of
|
||||||
|
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
|
||||||
|
* General Public License for more details.
|
||||||
|
*
|
||||||
|
* You should have received a copy of the GNU General Public License along
|
||||||
|
* with this program; if not, write to the Free Software Foundation, Inc.,
|
||||||
|
* 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA.
|
||||||
|
*
|
||||||
|
**********************************************************************
|
||||||
|
|
||||||
|
=======================================================================
|
||||||
|
|
||||||
|
Module for parsing various forms of directive and command lines that
|
||||||
|
are common to the configuration file and to the command client.
|
||||||
|
|
||||||
|
*/
|
||||||
|
|
||||||
|
#include "config.h"
|
||||||
|
|
||||||
|
#include "sysincl.h"
|
||||||
|
|
||||||
|
#include "cmdparse.h"
|
||||||
|
#include "memory.h"
|
||||||
|
#include "nameserv.h"
|
||||||
|
#include "ntp.h"
|
||||||
|
#include "util.h"
|
||||||
|
|
||||||
|
/* ================================================== */
|
||||||
|
|
||||||
|
int
|
||||||
|
CPS_ParseNTPSourceAdd(char *line, CPS_NTP_Source *src)
|
||||||
|
{
|
||||||
|
char *hostname, *cmd;
|
||||||
|
uint32_t ef_type;
|
||||||
|
int n, sel_option;
|
||||||
|
|
||||||
|
src->family = IPADDR_UNSPEC;
|
||||||
|
src->port = SRC_DEFAULT_PORT;
|
||||||
|
src->params.minpoll = SRC_DEFAULT_MINPOLL;
|
||||||
|
src->params.maxpoll = SRC_DEFAULT_MAXPOLL;
|
||||||
|
src->params.connectivity = SRC_ONLINE;
|
||||||
|
src->params.auto_offline = 0;
|
||||||
|
src->params.presend_minpoll = SRC_DEFAULT_PRESEND_MINPOLL;
|
||||||
|
src->params.burst = 0;
|
||||||
|
src->params.iburst = 0;
|
||||||
|
src->params.min_stratum = SRC_DEFAULT_MINSTRATUM;
|
||||||
|
src->params.poll_target = SRC_DEFAULT_POLLTARGET;
|
||||||
|
src->params.version = 0;
|
||||||
|
src->params.max_sources = SRC_DEFAULT_MAXSOURCES;
|
||||||
|
src->params.min_samples = SRC_DEFAULT_MINSAMPLES;
|
||||||
|
src->params.max_samples = SRC_DEFAULT_MAXSAMPLES;
|
||||||
|
src->params.filter_length = 0;
|
||||||
|
src->params.interleaved = 0;
|
||||||
|
src->params.sel_options = 0;
|
||||||
|
src->params.nts = 0;
|
||||||
|
src->params.nts_port = SRC_DEFAULT_NTSPORT;
|
||||||
|
src->params.copy = 0;
|
||||||
|
src->params.ext_fields = 0;
|
||||||
|
src->params.authkey = INACTIVE_AUTHKEY;
|
||||||
|
src->params.cert_set = SRC_DEFAULT_CERTSET;
|
||||||
|
src->params.max_delay = SRC_DEFAULT_MAXDELAY;
|
||||||
|
src->params.max_delay_ratio = SRC_DEFAULT_MAXDELAYRATIO;
|
||||||
|
src->params.max_delay_dev_ratio = SRC_DEFAULT_MAXDELAYDEVRATIO;
|
||||||
|
src->params.max_delay_quant = 0.0;
|
||||||
|
src->params.min_delay = 0.0;
|
||||||
|
src->params.asymmetry = SRC_DEFAULT_ASYMMETRY;
|
||||||
|
src->params.offset = 0.0;
|
||||||
|
|
||||||
|
hostname = line;
|
||||||
|
line = CPS_SplitWord(line);
|
||||||
|
|
||||||
|
if (!*hostname)
|
||||||
|
return 0;
|
||||||
|
|
||||||
|
src->name = hostname;
|
||||||
|
|
||||||
|
/* Parse options */
|
||||||
|
for (; *line; line += n) {
|
||||||
|
cmd = line;
|
||||||
|
line = CPS_SplitWord(line);
|
||||||
|
n = 0;
|
||||||
|
|
||||||
|
if (!strcasecmp(cmd, "auto_offline")) {
|
||||||
|
src->params.auto_offline = 1;
|
||||||
|
} else if (!strcasecmp(cmd, "burst")) {
|
||||||
|
src->params.burst = 1;
|
||||||
|
} else if (!strcasecmp(cmd, "copy")) {
|
||||||
|
src->params.copy = 1;
|
||||||
|
} else if (!strcasecmp(cmd, "iburst")) {
|
||||||
|
src->params.iburst = 1;
|
||||||
|
} else if (!strcasecmp(cmd, "offline")) {
|
||||||
|
src->params.connectivity = SRC_OFFLINE;
|
||||||
|
} else if (!strcasecmp(cmd, "certset")) {
|
||||||
|
if (sscanf(line, "%"SCNu32"%n", &src->params.cert_set, &n) != 1)
|
||||||
|
return 0;
|
||||||
|
} else if (!strcasecmp(cmd, "key")) {
|
||||||
|
if (sscanf(line, "%"SCNu32"%n", &src->params.authkey, &n) != 1 ||
|
||||||
|
src->params.authkey == INACTIVE_AUTHKEY)
|
||||||
|
return 0;
|
||||||
|
} else if (!strcasecmp(cmd, "asymmetry")) {
|
||||||
|
if (sscanf(line, "%lf%n", &src->params.asymmetry, &n) != 1)
|
||||||
|
return 0;
|
||||||
|
} else if (!strcasecmp(cmd, "extfield")) {
|
||||||
|
if (sscanf(line, "%"SCNx32"%n", &ef_type, &n) != 1)
|
||||||
|
return 0;
|
||||||
|
switch (ef_type) {
|
||||||
|
case NTP_EF_EXP_MONO_ROOT:
|
||||||
|
src->params.ext_fields |= NTP_EF_FLAG_EXP_MONO_ROOT;
|
||||||
|
break;
|
||||||
|
case NTP_EF_EXP_NET_CORRECTION:
|
||||||
|
src->params.ext_fields |= NTP_EF_FLAG_EXP_NET_CORRECTION;
|
||||||
|
break;
|
||||||
|
default:
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
} else if (!strcasecmp(cmd, "filter")) {
|
||||||
|
if (sscanf(line, "%d%n", &src->params.filter_length, &n) != 1)
|
||||||
|
return 0;
|
||||||
|
} else if (!strcasecmp(cmd, "ipv4")) {
|
||||||
|
src->family = IPADDR_INET4;
|
||||||
|
} else if (!strcasecmp(cmd, "ipv6")) {
|
||||||
|
src->family = IPADDR_INET6;
|
||||||
|
} else if (!strcasecmp(cmd, "maxdelay")) {
|
||||||
|
if (sscanf(line, "%lf%n", &src->params.max_delay, &n) != 1)
|
||||||
|
return 0;
|
||||||
|
} else if (!strcasecmp(cmd, "maxdelayratio")) {
|
||||||
|
if (sscanf(line, "%lf%n", &src->params.max_delay_ratio, &n) != 1)
|
||||||
|
return 0;
|
||||||
|
} else if (!strcasecmp(cmd, "maxdelaydevratio")) {
|
||||||
|
if (sscanf(line, "%lf%n", &src->params.max_delay_dev_ratio, &n) != 1)
|
||||||
|
return 0;
|
||||||
|
} else if (!strcasecmp(cmd, "maxdelayquant")) {
|
||||||
|
if (sscanf(line, "%lf%n", &src->params.max_delay_quant, &n) != 1)
|
||||||
|
return 0;
|
||||||
|
} else if (!strcasecmp(cmd, "maxpoll")) {
|
||||||
|
if (sscanf(line, "%d%n", &src->params.maxpoll, &n) != 1)
|
||||||
|
return 0;
|
||||||
|
} else if (!strcasecmp(cmd, "maxsamples")) {
|
||||||
|
if (sscanf(line, "%d%n", &src->params.max_samples, &n) != 1)
|
||||||
|
return 0;
|
||||||
|
} else if (!strcasecmp(cmd, "maxsources")) {
|
||||||
|
if (sscanf(line, "%d%n", &src->params.max_sources, &n) != 1)
|
||||||
|
return 0;
|
||||||
|
} else if (!strcasecmp(cmd, "mindelay")) {
|
||||||
|
if (sscanf(line, "%lf%n", &src->params.min_delay, &n) != 1)
|
||||||
|
return 0;
|
||||||
|
} else if (!strcasecmp(cmd, "minpoll")) {
|
||||||
|
if (sscanf(line, "%d%n", &src->params.minpoll, &n) != 1)
|
||||||
|
return 0;
|
||||||
|
} else if (!strcasecmp(cmd, "minsamples")) {
|
||||||
|
if (sscanf(line, "%d%n", &src->params.min_samples, &n) != 1)
|
||||||
|
return 0;
|
||||||
|
} else if (!strcasecmp(cmd, "minstratum")) {
|
||||||
|
if (sscanf(line, "%d%n", &src->params.min_stratum, &n) != 1)
|
||||||
|
return 0;
|
||||||
|
} else if (!strcasecmp(cmd, "nts")) {
|
||||||
|
src->params.nts = 1;
|
||||||
|
} else if (!strcasecmp(cmd, "ntsport")) {
|
||||||
|
if (sscanf(line, "%d%n", &src->params.nts_port, &n) != 1)
|
||||||
|
return 0;
|
||||||
|
} else if (!strcasecmp(cmd, "offset")) {
|
||||||
|
if (sscanf(line, "%lf%n", &src->params.offset, &n) != 1)
|
||||||
|
return 0;
|
||||||
|
} else if (!strcasecmp(cmd, "port")) {
|
||||||
|
if (sscanf(line, "%d%n", &src->port, &n) != 1)
|
||||||
|
return 0;
|
||||||
|
} else if (!strcasecmp(cmd, "polltarget")) {
|
||||||
|
if (sscanf(line, "%d%n", &src->params.poll_target, &n) != 1)
|
||||||
|
return 0;
|
||||||
|
} else if (!strcasecmp(cmd, "presend")) {
|
||||||
|
if (sscanf(line, "%d%n", &src->params.presend_minpoll, &n) != 1)
|
||||||
|
return 0;
|
||||||
|
} else if (!strcasecmp(cmd, "version")) {
|
||||||
|
if (sscanf(line, "%d%n", &src->params.version, &n) != 1)
|
||||||
|
return 0;
|
||||||
|
} else if (!strcasecmp(cmd, "xleave")) {
|
||||||
|
src->params.interleaved = 1;
|
||||||
|
} else if ((sel_option = CPS_GetSelectOption(cmd)) != 0) {
|
||||||
|
src->params.sel_options |= sel_option;
|
||||||
|
} else {
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
return 1;
|
||||||
|
}
|
||||||
|
|
||||||
|
/* ================================================== */
|
||||||
|
|
||||||
|
int
|
||||||
|
CPS_GetSelectOption(char *option)
|
||||||
|
{
|
||||||
|
if (!strcasecmp(option, "noselect")) {
|
||||||
|
return SRC_SELECT_NOSELECT;
|
||||||
|
} else if (!strcasecmp(option, "prefer")) {
|
||||||
|
return SRC_SELECT_PREFER;
|
||||||
|
} else if (!strcasecmp(option, "require")) {
|
||||||
|
return SRC_SELECT_REQUIRE;
|
||||||
|
} else if (!strcasecmp(option, "trust")) {
|
||||||
|
return SRC_SELECT_TRUST;
|
||||||
|
}
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
/* ================================================== */
|
||||||
|
|
||||||
|
int
|
||||||
|
CPS_ParseAllowDeny(char *line, int *all, IPAddr *ip, int *subnet_bits)
|
||||||
|
{
|
||||||
|
char *p, *net, *slash;
|
||||||
|
uint32_t a, b, c;
|
||||||
|
int bits, len, n;
|
||||||
|
|
||||||
|
p = CPS_SplitWord(line);
|
||||||
|
|
||||||
|
if (strcmp(line, "all") == 0) {
|
||||||
|
*all = 1;
|
||||||
|
net = p;
|
||||||
|
p = CPS_SplitWord(p);
|
||||||
|
} else {
|
||||||
|
*all = 0;
|
||||||
|
net = line;
|
||||||
|
}
|
||||||
|
|
||||||
|
/* Make sure there are no other arguments */
|
||||||
|
if (*p)
|
||||||
|
return 0;
|
||||||
|
|
||||||
|
/* No specified address or network means all IPv4 and IPv6 addresses */
|
||||||
|
if (!*net) {
|
||||||
|
ip->family = IPADDR_UNSPEC;
|
||||||
|
*subnet_bits = 0;
|
||||||
|
return 1;
|
||||||
|
}
|
||||||
|
|
||||||
|
slash = strchr(net, '/');
|
||||||
|
if (slash) {
|
||||||
|
if (sscanf(slash + 1, "%d%n", &bits, &len) != 1 || slash[len + 1] || bits < 0)
|
||||||
|
return 0;
|
||||||
|
*slash = '\0';
|
||||||
|
} else {
|
||||||
|
bits = -1;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (UTI_StringToIP(net, ip)) {
|
||||||
|
if (bits >= 0)
|
||||||
|
*subnet_bits = bits;
|
||||||
|
else
|
||||||
|
*subnet_bits = ip->family == IPADDR_INET6 ? 128 : 32;
|
||||||
|
return 1;
|
||||||
|
}
|
||||||
|
|
||||||
|
/* Check for a shortened IPv4 network notation using only 1, 2, or 3 decimal
|
||||||
|
numbers. This is different than the numbers-and-dots notation accepted
|
||||||
|
by inet_aton()! */
|
||||||
|
|
||||||
|
a = b = c = 0;
|
||||||
|
n = sscanf(net, "%"PRIu32"%n.%"PRIu32"%n.%"PRIu32"%n", &a, &len, &b, &len, &c, &len);
|
||||||
|
|
||||||
|
if (n > 0 && !net[len]) {
|
||||||
|
if (a > 255 || b > 255 || c > 255)
|
||||||
|
return 0;
|
||||||
|
|
||||||
|
ip->family = IPADDR_INET4;
|
||||||
|
ip->addr.in4 = (a << 24) | (b << 16) | (c << 8);
|
||||||
|
|
||||||
|
if (bits >= 0)
|
||||||
|
*subnet_bits = bits;
|
||||||
|
else
|
||||||
|
*subnet_bits = n * 8;
|
||||||
|
|
||||||
|
return 1;
|
||||||
|
}
|
||||||
|
|
||||||
|
/* The last possibility is a hostname */
|
||||||
|
if (bits < 0 && DNS_Name2IPAddress(net, ip, 1) == DNS_Success) {
|
||||||
|
*subnet_bits = ip->family == IPADDR_INET6 ? 128 : 32;
|
||||||
|
return 1;
|
||||||
|
}
|
||||||
|
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
/* ================================================== */
|
||||||
|
|
||||||
|
int
|
||||||
|
CPS_ParseLocal(char *line, int *stratum, int *orphan, double *distance, double *activate)
|
||||||
|
{
|
||||||
|
int n;
|
||||||
|
char *cmd;
|
||||||
|
|
||||||
|
*stratum = 10;
|
||||||
|
*distance = 1.0;
|
||||||
|
*activate = 0.0;
|
||||||
|
*orphan = 0;
|
||||||
|
|
||||||
|
while (*line) {
|
||||||
|
cmd = line;
|
||||||
|
line = CPS_SplitWord(line);
|
||||||
|
|
||||||
|
if (!strcasecmp(cmd, "stratum")) {
|
||||||
|
if (sscanf(line, "%d%n", stratum, &n) != 1 ||
|
||||||
|
*stratum >= NTP_MAX_STRATUM || *stratum <= 0)
|
||||||
|
return 0;
|
||||||
|
} else if (!strcasecmp(cmd, "orphan")) {
|
||||||
|
*orphan = 1;
|
||||||
|
n = 0;
|
||||||
|
} else if (!strcasecmp(cmd, "distance")) {
|
||||||
|
if (sscanf(line, "%lf%n", distance, &n) != 1)
|
||||||
|
return 0;
|
||||||
|
} else if (!strcasecmp(cmd, "activate")) {
|
||||||
|
if (sscanf(line, "%lf%n", activate, &n) != 1)
|
||||||
|
return 0;
|
||||||
|
} else {
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
line += n;
|
||||||
|
}
|
||||||
|
|
||||||
|
return 1;
|
||||||
|
}
|
||||||
|
|
||||||
|
/* ================================================== */
|
||||||
|
|
||||||
|
void
|
||||||
|
CPS_NormalizeLine(char *line)
|
||||||
|
{
|
||||||
|
char *p, *q;
|
||||||
|
int space = 1, first = 1;
|
||||||
|
|
||||||
|
/* Remove white-space at beginning and replace white-spaces with space char */
|
||||||
|
for (p = q = line; *p; p++) {
|
||||||
|
if (isspace((unsigned char)*p)) {
|
||||||
|
if (!space)
|
||||||
|
*q++ = ' ';
|
||||||
|
space = 1;
|
||||||
|
continue;
|
||||||
|
}
|
||||||
|
|
||||||
|
/* Discard comment lines */
|
||||||
|
if (first && strchr("!;#%", *p))
|
||||||
|
break;
|
||||||
|
|
||||||
|
*q++ = *p;
|
||||||
|
space = first = 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
/* Strip trailing space */
|
||||||
|
if (q > line && q[-1] == ' ')
|
||||||
|
q--;
|
||||||
|
|
||||||
|
*q = '\0';
|
||||||
|
}
|
||||||
|
|
||||||
|
/* ================================================== */
|
||||||
|
|
||||||
|
char *
|
||||||
|
CPS_SplitWord(char *line)
|
||||||
|
{
|
||||||
|
char *p = line, *q = line;
|
||||||
|
|
||||||
|
/* Skip white-space before the word */
|
||||||
|
while (*q && isspace((unsigned char)*q))
|
||||||
|
q++;
|
||||||
|
|
||||||
|
/* Move the word to the beginning */
|
||||||
|
while (*q && !isspace((unsigned char)*q))
|
||||||
|
*p++ = *q++;
|
||||||
|
|
||||||
|
/* Find the next word */
|
||||||
|
while (*q && isspace((unsigned char)*q))
|
||||||
|
q++;
|
||||||
|
|
||||||
|
*p = '\0';
|
||||||
|
|
||||||
|
/* Return pointer to the next word or NUL */
|
||||||
|
return q;
|
||||||
|
}
|
||||||
|
|
||||||
|
/* ================================================== */
|
||||||
|
|
||||||
|
int
|
||||||
|
CPS_ParseKey(char *line, uint32_t *id, const char **type, char **key)
|
||||||
|
{
|
||||||
|
char *s1, *s2, *s3, *s4;
|
||||||
|
|
||||||
|
s1 = line;
|
||||||
|
s2 = CPS_SplitWord(s1);
|
||||||
|
s3 = CPS_SplitWord(s2);
|
||||||
|
s4 = CPS_SplitWord(s3);
|
||||||
|
|
||||||
|
/* Require two or three words */
|
||||||
|
if (!*s2 || *s4)
|
||||||
|
return 0;
|
||||||
|
|
||||||
|
if (sscanf(s1, "%"SCNu32, id) != 1)
|
||||||
|
return 0;
|
||||||
|
|
||||||
|
if (*s3) {
|
||||||
|
*type = s2;
|
||||||
|
*key = s3;
|
||||||
|
} else {
|
||||||
|
*type = "MD5";
|
||||||
|
*key = s2;
|
||||||
|
}
|
||||||
|
|
||||||
|
return 1;
|
||||||
|
}
|
||||||
|
|
||||||
|
/* ================================================== */
|
||||||
|
|
||||||
|
int
|
||||||
|
CPS_ParseRefid(char *line, uint32_t *ref_id)
|
||||||
|
{
|
||||||
|
int i;
|
||||||
|
|
||||||
|
for (i = *ref_id = 0; line[i] && !isspace((unsigned char)line[i]); i++) {
|
||||||
|
if (i >= 4)
|
||||||
|
return 0;
|
||||||
|
*ref_id |= (uint32_t)line[i] << (24 - i * 8);
|
||||||
|
}
|
||||||
|
|
||||||
|
return i;
|
||||||
|
}
|
64
cmdparse.h
Normal file
64
cmdparse.h
Normal file
|
@ -0,0 +1,64 @@
|
||||||
|
/*
|
||||||
|
chronyd/chronyc - Programs for keeping computer clocks accurate.
|
||||||
|
|
||||||
|
**********************************************************************
|
||||||
|
* Copyright (C) Richard P. Curnow 1997-2002
|
||||||
|
*
|
||||||
|
* This program is free software; you can redistribute it and/or modify
|
||||||
|
* it under the terms of version 2 of the GNU General Public License as
|
||||||
|
* published by the Free Software Foundation.
|
||||||
|
*
|
||||||
|
* This program is distributed in the hope that it will be useful, but
|
||||||
|
* WITHOUT ANY WARRANTY; without even the implied warranty of
|
||||||
|
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
|
||||||
|
* General Public License for more details.
|
||||||
|
*
|
||||||
|
* You should have received a copy of the GNU General Public License along
|
||||||
|
* with this program; if not, write to the Free Software Foundation, Inc.,
|
||||||
|
* 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA.
|
||||||
|
*
|
||||||
|
**********************************************************************
|
||||||
|
|
||||||
|
=======================================================================
|
||||||
|
|
||||||
|
Header file for the command parser
|
||||||
|
*/
|
||||||
|
|
||||||
|
#ifndef GOT_CMDPARSE_H
|
||||||
|
#define GOT_CMDPARSE_H
|
||||||
|
|
||||||
|
#include "srcparams.h"
|
||||||
|
#include "addressing.h"
|
||||||
|
|
||||||
|
typedef struct {
|
||||||
|
char *name;
|
||||||
|
int family;
|
||||||
|
int port;
|
||||||
|
SourceParameters params;
|
||||||
|
} CPS_NTP_Source;
|
||||||
|
|
||||||
|
/* Parse a command to add an NTP server or peer */
|
||||||
|
extern int CPS_ParseNTPSourceAdd(char *line, CPS_NTP_Source *src);
|
||||||
|
|
||||||
|
/* Get an NTP/refclock select option */
|
||||||
|
extern int CPS_GetSelectOption(char *option);
|
||||||
|
|
||||||
|
/* Parse a command to allow/deny access */
|
||||||
|
extern int CPS_ParseAllowDeny(char *line, int *all, IPAddr *ip, int *subnet_bits);
|
||||||
|
|
||||||
|
/* Parse a command to enable local reference */
|
||||||
|
extern int CPS_ParseLocal(char *line, int *stratum, int *orphan, double *distance, double *activate);
|
||||||
|
|
||||||
|
/* Remove extra white-space and comments */
|
||||||
|
extern void CPS_NormalizeLine(char *line);
|
||||||
|
|
||||||
|
/* Terminate first word and return pointer to the next word */
|
||||||
|
extern char *CPS_SplitWord(char *line);
|
||||||
|
|
||||||
|
/* Parse a key from keyfile */
|
||||||
|
extern int CPS_ParseKey(char *line, uint32_t *id, const char **type, char **key);
|
||||||
|
|
||||||
|
/* Parse a refclock reference ID (returns number of characters) */
|
||||||
|
extern int CPS_ParseRefid(char *line, uint32_t *ref_id);
|
||||||
|
|
||||||
|
#endif /* GOT_CMDPARSE_H */
|
180
conf.h
Normal file
180
conf.h
Normal file
|
@ -0,0 +1,180 @@
|
||||||
|
/*
|
||||||
|
chronyd/chronyc - Programs for keeping computer clocks accurate.
|
||||||
|
|
||||||
|
**********************************************************************
|
||||||
|
* Copyright (C) Richard P. Curnow 1997-2003
|
||||||
|
* Copyright (C) Miroslav Lichvar 2013-2014
|
||||||
|
*
|
||||||
|
* This program is free software; you can redistribute it and/or modify
|
||||||
|
* it under the terms of version 2 of the GNU General Public License as
|
||||||
|
* published by the Free Software Foundation.
|
||||||
|
*
|
||||||
|
* This program is distributed in the hope that it will be useful, but
|
||||||
|
* WITHOUT ANY WARRANTY; without even the implied warranty of
|
||||||
|
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
|
||||||
|
* General Public License for more details.
|
||||||
|
*
|
||||||
|
* You should have received a copy of the GNU General Public License along
|
||||||
|
* with this program; if not, write to the Free Software Foundation, Inc.,
|
||||||
|
* 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA.
|
||||||
|
*
|
||||||
|
**********************************************************************
|
||||||
|
|
||||||
|
=======================================================================
|
||||||
|
|
||||||
|
Header file for configuration module
|
||||||
|
*/
|
||||||
|
|
||||||
|
#ifndef GOT_CONF_H
|
||||||
|
#define GOT_CONF_H
|
||||||
|
|
||||||
|
#include "addressing.h"
|
||||||
|
#include "array.h"
|
||||||
|
#include "reference.h"
|
||||||
|
#include "sources.h"
|
||||||
|
|
||||||
|
extern void CNF_Initialise(int restarted, int client_only);
|
||||||
|
extern void CNF_Finalise(void);
|
||||||
|
|
||||||
|
extern void CNF_EnablePrint(void);
|
||||||
|
|
||||||
|
extern char *CNF_GetRtcDevice(void);
|
||||||
|
|
||||||
|
extern void CNF_ReadFile(const char *filename);
|
||||||
|
extern void CNF_ParseLine(const char *filename, int number, char *line);
|
||||||
|
|
||||||
|
extern void CNF_CreateDirs(uid_t uid, gid_t gid);
|
||||||
|
|
||||||
|
extern void CNF_CheckReadOnlyAccess(void);
|
||||||
|
|
||||||
|
extern void CNF_AddInitSources(void);
|
||||||
|
extern void CNF_AddSources(void);
|
||||||
|
extern void CNF_AddBroadcasts(void);
|
||||||
|
extern void CNF_AddRefclocks(void);
|
||||||
|
|
||||||
|
extern void CNF_ReloadSources(void);
|
||||||
|
|
||||||
|
extern int CNF_GetAcquisitionPort(void);
|
||||||
|
extern int CNF_GetNTPPort(void);
|
||||||
|
extern char *CNF_GetDriftFile(void);
|
||||||
|
extern char *CNF_GetLogDir(void);
|
||||||
|
extern char *CNF_GetDumpDir(void);
|
||||||
|
extern int CNF_GetLogBanner(void);
|
||||||
|
extern int CNF_GetLogMeasurements(int *raw);
|
||||||
|
extern int CNF_GetLogSelection(void);
|
||||||
|
extern int CNF_GetLogStatistics(void);
|
||||||
|
extern int CNF_GetLogTracking(void);
|
||||||
|
extern int CNF_GetLogRtc(void);
|
||||||
|
extern int CNF_GetLogRefclocks(void);
|
||||||
|
extern int CNF_GetLogTempComp(void);
|
||||||
|
extern char *CNF_GetKeysFile(void);
|
||||||
|
extern char *CNF_GetRtcFile(void);
|
||||||
|
extern int CNF_GetManualEnabled(void);
|
||||||
|
extern int CNF_GetCommandPort(void);
|
||||||
|
extern int CNF_GetRtcOnUtc(void);
|
||||||
|
extern int CNF_GetRtcSync(void);
|
||||||
|
extern void CNF_GetMakeStep(int *limit, double *threshold);
|
||||||
|
extern void CNF_GetMaxChange(int *delay, int *ignore, double *offset);
|
||||||
|
extern double CNF_GetLogChange(void);
|
||||||
|
extern void CNF_GetMailOnChange(int *enabled, double *threshold, char **user);
|
||||||
|
extern int CNF_GetNoClientLog(void);
|
||||||
|
extern unsigned long CNF_GetClientLogLimit(void);
|
||||||
|
extern void CNF_GetFallbackDrifts(int *min, int *max);
|
||||||
|
extern void CNF_GetBindAddress(int family, IPAddr *addr);
|
||||||
|
extern void CNF_GetBindAcquisitionAddress(int family, IPAddr *addr);
|
||||||
|
extern void CNF_GetBindCommandAddress(int family, IPAddr *addr);
|
||||||
|
extern char *CNF_GetBindNtpInterface(void);
|
||||||
|
extern char *CNF_GetBindAcquisitionInterface(void);
|
||||||
|
extern char *CNF_GetBindCommandInterface(void);
|
||||||
|
extern char *CNF_GetBindCommandPath(void);
|
||||||
|
extern int CNF_GetNtpDscp(void);
|
||||||
|
extern char *CNF_GetNtpSigndSocket(void);
|
||||||
|
extern char *CNF_GetPidFile(void);
|
||||||
|
extern REF_LeapMode CNF_GetLeapSecMode(void);
|
||||||
|
extern char *CNF_GetLeapSecTimezone(void);
|
||||||
|
extern char *CNF_GetLeapSecList(void);
|
||||||
|
|
||||||
|
/* Value returned in ppm, as read from file */
|
||||||
|
extern double CNF_GetMaxUpdateSkew(void);
|
||||||
|
extern double CNF_GetMaxClockError(void);
|
||||||
|
extern double CNF_GetMaxDrift(void);
|
||||||
|
extern double CNF_GetCorrectionTimeRatio(void);
|
||||||
|
extern double CNF_GetMaxSlewRate(void);
|
||||||
|
extern double CNF_GetClockPrecision(void);
|
||||||
|
|
||||||
|
extern SRC_AuthSelectMode CNF_GetAuthSelectMode(void);
|
||||||
|
extern double CNF_GetMaxDistance(void);
|
||||||
|
extern double CNF_GetMaxJitter(void);
|
||||||
|
extern double CNF_GetReselectDistance(void);
|
||||||
|
extern double CNF_GetStratumWeight(void);
|
||||||
|
extern double CNF_GetCombineLimit(void);
|
||||||
|
|
||||||
|
extern int CNF_AllowLocalReference(int *stratum, int *orphan, double *distance, double *activate);
|
||||||
|
|
||||||
|
extern void CNF_SetupAccessRestrictions(void);
|
||||||
|
|
||||||
|
extern int CNF_GetSchedPriority(void);
|
||||||
|
extern int CNF_GetLockMemory(void);
|
||||||
|
|
||||||
|
extern int CNF_GetNTPRateLimit(int *interval, int *burst, int *leak, int *kod);
|
||||||
|
extern int CNF_GetNtsRateLimit(int *interval, int *burst, int *leak);
|
||||||
|
extern int CNF_GetCommandRateLimit(int *interval, int *burst, int *leak);
|
||||||
|
extern void CNF_GetSmooth(double *max_freq, double *max_wander, int *leap_only);
|
||||||
|
extern void CNF_GetTempComp(char **file, double *interval, char **point_file, double *T0, double *k0, double *k1, double *k2);
|
||||||
|
|
||||||
|
extern char *CNF_GetUser(void);
|
||||||
|
|
||||||
|
extern int CNF_GetMaxSamples(void);
|
||||||
|
extern int CNF_GetMinSamples(void);
|
||||||
|
|
||||||
|
extern int CNF_GetMinSources(void);
|
||||||
|
|
||||||
|
extern double CNF_GetRtcAutotrim(void);
|
||||||
|
extern char *CNF_GetHwclockFile(void);
|
||||||
|
|
||||||
|
extern int CNF_GetInitSources(void);
|
||||||
|
extern double CNF_GetInitStepThreshold(void);
|
||||||
|
|
||||||
|
typedef enum {
|
||||||
|
CNF_HWTS_RXFILTER_ANY,
|
||||||
|
CNF_HWTS_RXFILTER_NONE,
|
||||||
|
CNF_HWTS_RXFILTER_NTP,
|
||||||
|
CNF_HWTS_RXFILTER_PTP,
|
||||||
|
CNF_HWTS_RXFILTER_ALL,
|
||||||
|
} CNF_HwTs_RxFilter;
|
||||||
|
|
||||||
|
typedef struct {
|
||||||
|
char *name;
|
||||||
|
int minpoll;
|
||||||
|
int maxpoll;
|
||||||
|
int min_samples;
|
||||||
|
int max_samples;
|
||||||
|
int nocrossts;
|
||||||
|
CNF_HwTs_RxFilter rxfilter;
|
||||||
|
double precision;
|
||||||
|
double tx_comp;
|
||||||
|
double rx_comp;
|
||||||
|
} CNF_HwTsInterface;
|
||||||
|
|
||||||
|
extern int CNF_GetHwTsInterface(unsigned int index, CNF_HwTsInterface **iface);
|
||||||
|
extern double CNF_GetHwTsTimeout(void);
|
||||||
|
|
||||||
|
extern int CNF_GetPtpPort(void);
|
||||||
|
extern int CNF_GetPtpDomain(void);
|
||||||
|
|
||||||
|
extern int CNF_GetRefresh(void);
|
||||||
|
|
||||||
|
extern ARR_Instance CNF_GetNtsAeads(void);
|
||||||
|
extern char *CNF_GetNtsDumpDir(void);
|
||||||
|
extern char *CNF_GetNtsNtpServer(void);
|
||||||
|
extern int CNF_GetNtsServerCertAndKeyFiles(const char ***certs, const char ***keys);
|
||||||
|
extern int CNF_GetNtsServerPort(void);
|
||||||
|
extern int CNF_GetNtsServerProcesses(void);
|
||||||
|
extern int CNF_GetNtsServerConnections(void);
|
||||||
|
extern int CNF_GetNtsRefresh(void);
|
||||||
|
extern int CNF_GetNtsRotate(void);
|
||||||
|
extern int CNF_GetNtsTrustedCertsPaths(const char ***paths, uint32_t **ids);
|
||||||
|
extern int CNF_GetNoSystemCert(void);
|
||||||
|
extern int CNF_GetNoCertTimeCheck(void);
|
||||||
|
|
||||||
|
#endif /* GOT_CONF_H */
|
114
contrib/andrew_bishop_1
Normal file
114
contrib/andrew_bishop_1
Normal file
|
@ -0,0 +1,114 @@
|
||||||
|
From amb@gedanken.demon.co.uk Tue Aug 17 22:14:00 1999
|
||||||
|
Date: Fri, 6 Aug 1999 19:00:24 +0100
|
||||||
|
From: Andrew M. Bishop <amb@gedanken.demon.co.uk>
|
||||||
|
To: richard@rrbcurnow.freeserve.co.uk
|
||||||
|
Subject: Re: Chrony and laptop configuration
|
||||||
|
|
||||||
|
Hi,
|
||||||
|
|
||||||
|
Attached is the apmd_proxy script from the apmd-3.0beta9 distribution.
|
||||||
|
|
||||||
|
The changes that I would make are the following:
|
||||||
|
|
||||||
|
Replace the update_clock function (line 122) with
|
||||||
|
|
||||||
|
update_clock () {
|
||||||
|
|
||||||
|
chronyd -f /etc/chrony.conf
|
||||||
|
|
||||||
|
}
|
||||||
|
|
||||||
|
Around line 171 (in the suspend actions section) I would kill chronyd.
|
||||||
|
|
||||||
|
begin 644 apmd_proxy.gz
|
||||||
|
M'XL("+L@JS<``V%P;61?<')O>'D`I5K[;]M&$OY9^BLV2JZQ6TF.`Q2XUDA1
|
||||||
|
MQU8<]>('_&A[.!R,-;D2MR&Y+)?TXZ[WO]\W,\N'%-E)[X08CJ2=V7G/-T,_
|
||||||
|
M?[9S8_,=GPR?*_S3119?%Z6[?U`3A=_+4F<JMK[059284BU<J?;/CE6L3>9R
|
||||||
|
M$!R4VB[5L2X_WNDRKM161!]D/Z:)T9FVZ73I%]$TUUY/E^YV6[W>Q>$'M?O=
|
||||||
|
M=]^!^E#?VEB]+=U=;M)4;<4W/Q;.5]-$E[=@-S5QO:V^4S_5N0DD(+I,K%<^
|
||||||
|
M(0(?E;:H%-Y'.DU-K&X>5)68GHAJBU3:5G>)R?DK7^G*@(M;*)T_J,+=0:U,
|
||||||
|
MYWII,I-7:E'G465!F&AP372^-/%4X5*CS+V.*F5N<<R#EZ[`IBKM<@D.Q)ID
|
||||||
|
ML/F267=VC$UA\E@11W?'7Z@[8NWRA5W6I8G)[!7>9X5-C:IL9J:LYR^V2BP)
|
||||||
|
M3>J*HJS`@Z],IG2<V=SZJM05?.(35Z>Q*NJ*M0(SJ!1[TK-4FA7RL(&-DN;H
|
||||||
|
MC5&%*>'/#&:K"XC'EH%".O>6"42,GBJ=G>\@F_*%B>S"1DJ7RSIKK0*-25Q<
|
||||||
|
M0.*RN<A0]`W9U$51799LU#F[KJ?@2Z]*XR&,MS<VM16\Z=3OM2G9K>""P(@1
|
||||||
|
M:(;4:ASM37EK(Z.V;JU6.Y`TVH',VQRKF<-9FY.:FG0:DT+P!MA6^J-(J`O0
|
||||||
|
M%*4E[<54HO@[T,/E69&:<?";1<R1!=2H9Q5?>_:P.&:D?JM]!?/B3L-ZMRYC
|
||||||
|
M-1/$-%D0$BP=1,/OAD'F8A-";<6[X)(A9^ZL3X@LN$W!2V;2^2`X.;50J\Y3
|
||||||
|
MIV.*Q;BTMZ:$U8D+$EG?<(B25+FI[ESYD60`1QW1W;^T:=**#(?4%",I[%..
|
||||||
|
M0T`\8@HYJW(R=SH:4UQ+T%%4@Z_SC8DI-+M(+`T)B9A@P\\7+,)'4Z(J2"ZS
|
||||||
|
M%B0-G:/LW)WNOJ(@2.R2ZM)6[<69),T(/$(&D+T6EM*OKK;'_5B.=(YKJ[K,
|
||||||
|
M$1#*E"6817``AXT$6^,7XH/DR&-4%\G^,8P6VP@A!5NBM%1)*`%TGCXC5ZYI
|
||||||
|
M^)N)*H[Z3^(J2$0^CN%-R&`78W6C81`E1?;LBG1M'$86O*7DP&?0&W;)$2FU
|
||||||
|
MV!4L6\$10A6%-=7%$(IDS;?STXM-HK'Q\7KV3)V<7L[HMY)JBW^YJT*VM`+7
|
||||||
|
M(B"41<`?FAL+0TK<^&>!U_GL\NK\1/V\_^%J]KUP5Z_06G*'^-9I\,#>$Z96
|
||||||
|
ME.TZBDQ1<:FDURXXB-0]9:D*!LJMXZN+2_5^_^<9ZWMX/O]Y=BXA<WJNWL^/
|
||||||
|
MWL_.MUG"]X:XEV:E?GN#DI-'QG,L=`I_OUX/<5]9J977I-][J-SQ&2GR?4)7
|
||||||
|
MJ"<(N<'5%8=7C.:X1KU2<'K4X0.ZMB0=/+F^7UTV\T'IK1#,Z3J?$..P0WNB
|
||||||
|
M(=GZ>OL34[#IOU"F<)ID4IN%XL!N3'-%;[Y8K<#\LQQZ0JQR"&6LN4(X7/05
|
||||||
|
MD9*X*%WVA"`-FW#/9]A\3IJ>EYY@L^ZI55:"9U!:JHIZZB=>*DWA*%I5"J@2
|
||||||
|
M3FWD(+!)/<%A_V"GN49H-O+AOO`DG]R0:E23Z6A=Q-2F/PF_P"[2A0[`X1%V
|
||||||
|
M`KNF#<'6-U('P'`U`ZD8HS;"&^BNTJZYVZ%.H-.T%8)(OV'26X!=-%>JW1LZ
|
||||||
|
MEZ\+$@"5%!`Y?(:?Z5H?$V$NYL=G'^;OYK-#=7!Z\FY^='6^?SD_/5$JX%_`
|
||||||
|
M'X``!C0$-OL(D=H:Y(::5>D8JAG+W8ERTTA%D>ZVP-?NCMX#:EL2W?>A4/HP
|
||||||
|
M)KFH#L>V(6L!=^5-NI"&<6$J=75YP)"JK%G_!U>CF:8N^DB5K&ED<HK@;>@J
|
||||||
|
M)`]Q!A.\<[!':>,8``27$HH-\L)HN=HQ5;0#9XK_=ICY=`B.;Q8Z]::5Y.+J
|
||||||
|
MXFQV<GA]>G*]SS+QUT&H%D+I6V?C34T'7$+;X8&!%<ET!!QN&/VZ/.=>24S`
|
||||||
|
MG],`ZKRESKW0=5IUL,&S+QCMDO*-&T).T!S5T`]7A'Y#5F1]S@Z.#^;[8%,&
|
||||||
|
M9AR0`G[A+;@+H)MCCE3*)=Z#3JS%5,`4:9[H6S0ZYB50'/RHI=^T90Y*D?--
|
||||||
|
MF:*!^AJ@#;5%JXN#BSGA*1`"0%5D%<1PAQY&`3V,`N15AN`DM]$^K`UQ*[I0
|
||||||
|
MPZ7+B^3!4['"C8'+GD)8PTYC05-<96A$`$C@'NH6"PZT!#E-<)\A'S&-JE1X
|
||||||
|
MC!KH-V:_,H!EP,0XB`<=\)&:.1V*B:]G/\T.+LG\P1-=4-&0.GM[=70T/SGB
|
||||||
|
M=U<Y\:=)L1FR@+9CB!D<?#R[?']Z"(CBVO^_YOH5FYMZN91PS^`]#)P^Z-)4
|
||||||
|
MJ=2A,JD36*8T<DL<BM_2Y$CXE'K:&,`?I8>2IN(RU6@GT0J,_U&B!&[.N7RV
|
||||||
|
M`ZP'5W((\*H3"`\NN%1196O5(;,)'F[%1*US-?E51NRN\$H):'6>J`]NZ5OP
|
||||||
|
MG5+:=,/AZESL\O1!KJ?YN??%BZ^I&W1L7S=LP<1"K[+.UUC!BCM55NSTY`);
|
||||||
|
MRN4H<>KEUVNOE^J''YXB&+UX!2%&?73V*`&WHPVOIV]X.5E[/272O8DV7/#Z
|
||||||
|
M40*4336Y9\?L4\5");#4K*@0)ZK$)!2J&<(HK6,I9IZ6$_33#MQ->2*.E:[J
|
||||||
|
ML`Y8-^;_\!H.A\SY&L+1\&ZVMM6_AP,IX//#&4_Y$LB^``@%``X#I".\GZ"1
|
||||||
|
M\GLB.>215L(RE&#(.=CQM-5*XD)C4)[\KB87:O>O:B<VM_A0#P:@Q/LW;]!_
|
||||||
|
MOU4816J@W.&`*N_>\#^->&'0UK!=(V&X[A$I_YQ`KUIY>C<_Y]'K>]4-:LT$
|
||||||
|
MCR88<,7KZ6M.4I[(I;(CU6/>*J"O#@4D74L#;B0/R"FP@"4S6NEU"Q4^/1R@
|
||||||
|
M4?Y#318;^ZWZYQXW@.&``G"Z\8Q\AT*=4D]]2(U\P&Q'+PX^G![\[?CT<#92
|
||||||
|
M;]31\67'<4"MG#L?GU_8X8!^F.X%`8<WH9.OR/#NP_[1F]%H.#!4KGL?3>J1
|
||||||
|
M,(`R]TJ,'W107WT5L,G$JQ=T7/WQATKN5C\C7Q"DX`D/U2;3L#_MEDQ<EX1@
|
||||||
|
MYDWRD/'".,\ESCM$:F9I&Q?2I[==!(>%I2%S')8QJX&",HVX+B<"F:0K]!%%
|
||||||
|
M#ZFTB*`9W\G&=S+,HL=UX.+Y<!B!':R_.T(0#8?DGO7Z$UX-?.*++B[W3P[?
|
||||||
|
M_OV1L\SF/8>=BFH,M)G]%REC$(_6=9OB9JYAT?O@Q#,'M$RRYJ%I-B<NH"Y1
|
||||||
|
M?LS@I;_]"HB'K>P6S*-*6@^T:S#&.`)QIZ3R*-P]^F,4)!IM#RDO]ION*2C:
|
||||||
|
M+-E/:W@Q^"HLMMK9:O`)6OS$_"W4%;[4]!0M<9#/"Z)G7;NE#'MYM+Y1;#8A
|
||||||
|
M"\IW>-W&*1<_%"CKJ+GOI]X%4\GRD9ID3N;4Z@.JVWUO)"'";@+9N$RC#LN+
|
||||||
|
MM)7UV12434:NPNPF-R=:O7B-=P'.=,D-(I=?Z^A:$-T/7/CR.DW;$P-S#SBS
|
||||||
|
M2^X\E\W.%N^@@MQA?FK5DRB(MT$92D4CV2[='V*N$X"YOY*2P%G=WS!5#T5`
|
||||||
|
MH;%C7+ITJM"^DI`HG,VKL;JI&3,3>0B.!@?"-K(9N\$(PY@'K?)/AU8`^V&9
|
||||||
|
M[:5DTU9^)6<"D`N'@QO#,HTBBR#=H+>YY"4M0>X`S@D+)S+,K:EFR@HECHC;
|
||||||
|
M>>.)^0!?06J,B#0>"`7'!TGG&YC?.`LW1V91DX#>+2IJ-Y-N'2H#"NO;0FN6
|
||||||
|
MCK:Q8>)@*4H!``CUE9.<"9$KY<%!+`L\,;4`\_/9Q=7Q##<=A/U>,S#T+#7B
|
||||||
|
M%2=C9:Y<88V)O*:5=3`W!IF:I^22[]3MR$\&)&,ANR::U.M=1B"]1>..8)AE
|
||||||
|
MY])Y--\,=NDE5M>LPDP30EA)$B&^'YE8$/4\?7<A/UAEQ`:G3.!.N?YM6]&Z
|
||||||
|
M=.J/.AL"&#-)J8O$1EZ&RM@961++:,5YDG=G>-/7Q8\UM$)7OR)B"3DA>$*A
|
||||||
|
ME/K-+N4-C6_6#H8>3#@,?655:YI6<N^H(M?9#7R\.PX/&L2XM:\Y'S`@FONJ
|
||||||
|
M.8P[KBBE1,)Q2$.D".^8*M?="J@,+(#K6-%?V]O0R9FQ>.RY:GU6^Y*?HT;)
|
||||||
|
M;=7YX+D:K'ZSNZ=\:DRA=NG+8&;'<R.9&K*%_@4`*,]PX%57TVZ&^EV`@D0T
|
||||||
|
M^W7_^.P#@.+I!=H&AK.8NR0Y(/AR)^P-&Z,3E:Q"`^.D9<A=6)Z+4:R5A@_4
|
||||||
|
MG:T"K5)GF,[YHEM=/M#BHTO,T)A'+(S+:1_`S7?P7(P`>*53>=C,)Q;H?WM[
|
||||||
|
M3\*1%I9("C]U;"B=@S6>3J>K.U6**^#JLS%@`CU(D&T_XSV!Q5-""*(J8`&5
|
||||||
|
ML;8ET%%IH:&K"1V%#(_OC,`P<D'1/N[^?`/H%B<K>Q[N(DTI;8!A;]]G^=9&
|
||||||
|
MV"[L>;%M2ZE?7-D^4\+6*E:ON7,+#QT$+?VQFO1_ER1+T]_C-:D)O7Y)>B17
|
||||||
|
M0L1^+E?"L2YEPJ'I>F`_&K3YE\<LX//YY9.GF,U!>*!.B<=/5A>V1-ODQT9^
|
||||||
|
MRD?F?6#?+!`9287V2!]E%-8(C&:F'C7XG[`O,4-@*_4%2.R3V;>9K@:K0[N`
|
||||||
|
MJ2^WQNG9TX<V6X/"W92$;_@))9TY##F1,3Y8.8JB,@X9<D//_=6HI\<H;"+9
|
||||||
|
M(*Z@1-^@Z]Z>>EHC=?!^_^1H1MSAXLNKBT=/TDW23^@N'L*06S;,KB.^G+ZA
|
||||||
|
M9+VS]`<VU(5VN'0U`\18Z1B@!,4\0UV(&_>CDE71EP'K#3JV";?N478I^50D
|
||||||
|
MI"5+$%`FGU0ZC/P1"#Y?J7@]PB!EH&U"ELIPI4;H"B.5`L^F:DLV'&9!H!EV
|
||||||
|
M5&:ZG-+>^=M7?Q%:P-"7%>'RC#;2\D<[%2/8VA.F[-_:/?T)%PLWLA_J]5VW
|
||||||
|
>Z]AB9"M2/=L.+(S7D<0S_V\81H;_`M>*^#$A)0``
|
||||||
|
`
|
||||||
|
end
|
||||||
|
|
||||||
|
--
|
||||||
|
Andrew.
|
||||||
|
----------------------------------------------------------------------
|
||||||
|
Andrew M. Bishop amb@gedanken.demon.co.uk
|
||||||
|
http://www.gedanken.demon.co.uk/
|
||||||
|
|
95
contrib/andrew_bishop_2
Normal file
95
contrib/andrew_bishop_2
Normal file
|
@ -0,0 +1,95 @@
|
||||||
|
From amb@gedanken.demon.co.uk Wed Sep 1 22:26:59 1999
|
||||||
|
Date: Thu, 19 Aug 1999 17:30:14 +0100
|
||||||
|
From: Andrew M. Bishop <amb@gedanken.demon.co.uk>
|
||||||
|
To: richard@rrbcurnow.freeserve.co.uk
|
||||||
|
Subject: [amb@gedanken.demon.co.uk: Chrony and laptop configuration]
|
||||||
|
|
||||||
|
Hi,
|
||||||
|
|
||||||
|
What you need to do is replace 10.0.0.0 with the network of the
|
||||||
|
freeserve nameservers in the two scripts below.
|
||||||
|
|
||||||
|
Other than that you can use it as is.
|
||||||
|
|
||||||
|
------- Start of forwarded message -------
|
||||||
|
From: "Andrew M. Bishop" <amb@gedanken.demon.co.uk>
|
||||||
|
To: richard@rrbcurnow.freeserve.co.uk
|
||||||
|
Subject: Chrony and laptop configuration
|
||||||
|
Date: Sat, 31 Jul 1999 11:02:04 +0100
|
||||||
|
|
||||||
|
Attached are the ip-up and ip-down files that I use for chrony.
|
||||||
|
(Actually because of the way that debian works they are separate file
|
||||||
|
in the /etc/ppp/ip-up.d directory that are run in a SysV init style).
|
||||||
|
|
||||||
|
They rely on the presence of an 'ipparam demon' or 'ipparam freeserve'
|
||||||
|
line in the PPP options file.
|
||||||
|
|
||||||
|
-------------------- /etc/ppp/ip-up --------------------
|
||||||
|
#!/bin/sh -f
|
||||||
|
#
|
||||||
|
# A script to start chrony
|
||||||
|
#
|
||||||
|
|
||||||
|
PPP_IPPARAM="$6"
|
||||||
|
|
||||||
|
if [ $PPP_IPPARAM = "demon" ]; then
|
||||||
|
|
||||||
|
/usr/local/bin/chronyc << EOF
|
||||||
|
password xxxxxxx
|
||||||
|
online 255.255.255.0/158.152.1.0
|
||||||
|
online 255.255.255.0/194.159.253.0
|
||||||
|
EOF
|
||||||
|
|
||||||
|
fi
|
||||||
|
|
||||||
|
if [ $PPP_IPPARAM = "freeserve" ]; then
|
||||||
|
|
||||||
|
/usr/local/bin/chronyc << EOF
|
||||||
|
password xxxxxxx
|
||||||
|
online 255.255.255.0/10.0.0.0
|
||||||
|
EOF
|
||||||
|
|
||||||
|
fi
|
||||||
|
-------------------- /etc/ppp/ip-up --------------------
|
||||||
|
|
||||||
|
-------------------- /etc/ppp/ip-down --------------------
|
||||||
|
#!/bin/sh -f
|
||||||
|
#
|
||||||
|
# A script to stop chrony
|
||||||
|
#
|
||||||
|
|
||||||
|
PPP_IPPARAM="$6"
|
||||||
|
|
||||||
|
if [ $PPP_IPPARAM = "demon" ]; then
|
||||||
|
|
||||||
|
/usr/local/bin/chronyc << EOF
|
||||||
|
password xxxxxxx
|
||||||
|
offline 255.255.255.0/158.152.1.0
|
||||||
|
offline 255.255.255.0/194.159.253.0
|
||||||
|
EOF
|
||||||
|
|
||||||
|
fi
|
||||||
|
|
||||||
|
if [ $PPP_IPPARAM = "freeserve" ]; then
|
||||||
|
|
||||||
|
/usr/local/bin/chronyc << EOF
|
||||||
|
password xxxxxxx
|
||||||
|
offline 255.255.255.0/10.0.0.0
|
||||||
|
EOF
|
||||||
|
|
||||||
|
fi
|
||||||
|
-------------------- /etc/ppp/ip-down --------------------
|
||||||
|
|
||||||
|
--
|
||||||
|
Andrew.
|
||||||
|
----------------------------------------------------------------------
|
||||||
|
Andrew M. Bishop amb@gedanken.demon.co.uk
|
||||||
|
http://www.gedanken.demon.co.uk/
|
||||||
|
------- End of forwarded message -------
|
||||||
|
|
||||||
|
--
|
||||||
|
Andrew.
|
||||||
|
----------------------------------------------------------------------
|
||||||
|
Andrew M. Bishop amb@gedanken.demon.co.uk
|
||||||
|
http://www.gedanken.demon.co.uk/
|
||||||
|
|
103
contrib/bryan_christianson_1/README.txt
Normal file
103
contrib/bryan_christianson_1/README.txt
Normal file
|
@ -0,0 +1,103 @@
|
||||||
|
Notes for installing chrony on macOS
|
||||||
|
Author: Bryan Christianson (bryan@whatroute.net)
|
||||||
|
------------------------------------------------
|
||||||
|
|
||||||
|
These files are for those admins/users who would prefer to install chrony
|
||||||
|
from the source distribution and are intended as guidelines rather than
|
||||||
|
being definitive. They can be edited with a plain text editor, such as
|
||||||
|
vi, emacs or your favourite IDE (Xcode)
|
||||||
|
|
||||||
|
It is assumed you are comfortable with installing software from the
|
||||||
|
terminal command line and know how to use sudo to acquire root access.
|
||||||
|
|
||||||
|
If you are not familiar with the macOS command line then
|
||||||
|
please consider using ChronyControl from http://whatroute.net/chronycontrol.html
|
||||||
|
|
||||||
|
ChronyControl provides a gui wrapper for installing these files and sets the
|
||||||
|
necessary permissions on each file.
|
||||||
|
|
||||||
|
|
||||||
|
Install the chrony software
|
||||||
|
---------------------------
|
||||||
|
|
||||||
|
You will need xcode and the commandline additions to build and install chrony.
|
||||||
|
These can be obtained from Apple's website via the App Store.
|
||||||
|
|
||||||
|
cd to the chrony directory
|
||||||
|
./configure
|
||||||
|
make
|
||||||
|
sudo make install
|
||||||
|
|
||||||
|
chrony is now installed in default locations (/usr/local/sbin/chronyd,
|
||||||
|
/usr/local/bin/chronyc)
|
||||||
|
|
||||||
|
Create a chrony.conf file - see the chrony website for details
|
||||||
|
|
||||||
|
The support files here assume the following directives are specified in the
|
||||||
|
chrony.conf file
|
||||||
|
|
||||||
|
keyfile /etc/chrony.d/chrony.keys
|
||||||
|
driftfile /var/db/chrony/chrony.drift
|
||||||
|
bindcmdaddress /var/db/chrony/chronyd.sock
|
||||||
|
logdir /var/log/chrony
|
||||||
|
dumpdir /var/db/chrony
|
||||||
|
|
||||||
|
Install this file as /etc/chrony.d/chrony.conf and create
|
||||||
|
the directories specified in the above directives if they don't exist.
|
||||||
|
You will need root permissions to create the directories.
|
||||||
|
|
||||||
|
|
||||||
|
Running chronyd
|
||||||
|
---------------
|
||||||
|
At this point chronyd *could* be run as a daemon. Apple discourage running
|
||||||
|
daemons and their preferred method uses the launchd facility. The
|
||||||
|
support files here provide a launchd configuration file for chronyd and also
|
||||||
|
a shell script and launchd configuration file to rotate the chronyd logs on a daily basis.
|
||||||
|
|
||||||
|
|
||||||
|
Support files
|
||||||
|
-------------
|
||||||
|
Dates and sizes may differ
|
||||||
|
-rw-r--r-- 1 yourname staff 2084 4 Aug 22:54 README.txt
|
||||||
|
-rwxr-xr-x 1 yourname staff 676 4 Aug 21:18 chronylogrotate.sh
|
||||||
|
-rw-r--r-- 1 yourname staff 543 18 Jul 20:10 org.chrony-project.chronyc.plist
|
||||||
|
-rw-r--r-- 1 yourname staff 511 19 Jun 18:30 org.chrony-project.chronyd.plist
|
||||||
|
|
||||||
|
If you have used chrony support directories other than those suggested, you
|
||||||
|
will need to edit each file and make the appropriate changes.
|
||||||
|
|
||||||
|
|
||||||
|
Installing the support files
|
||||||
|
----------------------------
|
||||||
|
|
||||||
|
1. chronylogrotate.sh
|
||||||
|
This is a simple shell script that deletes old log files. Unfortunately because
|
||||||
|
of the need to run chronyc, the standard macOS logrotation does not work with
|
||||||
|
chrony logs.
|
||||||
|
|
||||||
|
This script runs on a daily basis under control of launchd and should be
|
||||||
|
installed in the /usr/local/bin directory
|
||||||
|
|
||||||
|
sudo cp chronylogrotate.sh /usr/local/bin
|
||||||
|
sudo chmod +x /usr/local/bin/chronylogrotate.sh
|
||||||
|
sudo chown root:wheel /usr/local/bin/chronylogrotate.sh
|
||||||
|
|
||||||
|
|
||||||
|
2. org.chrony-project.chronyc.plist
|
||||||
|
This file is the launchd plist that runs logrotation each day. You may
|
||||||
|
wish to edit this file to change the time of day at which the rotation
|
||||||
|
will run, currently 04:05 am
|
||||||
|
|
||||||
|
sudo cp org.chrony-project.chronyc.plist /Library/LaunchDaemons
|
||||||
|
sudo chown root:wheel /Library/LaunchDaemons/org.chrony-project.chronyc.plist
|
||||||
|
sudo chmod 0644 /Library/LaunchDaemons/org.chrony-project.chronyc.plist
|
||||||
|
sudo launchctl load -w /Library/LaunchDaemons/org.chrony-project.chronyc.plist
|
||||||
|
|
||||||
|
|
||||||
|
3. org.chrony-project.chronyd.plist
|
||||||
|
This file is the launchd plist that runs chronyd when the Macintosh starts.
|
||||||
|
|
||||||
|
sudo cp org.chrony-project.chronyd.plist /Library/LaunchDaemons
|
||||||
|
sudo chown root:wheel /Library/LaunchDaemons/org.chrony-project.chronyd.plist
|
||||||
|
sudo chmod 0644 /Library/LaunchDaemons/org.chrony-project.chronyd.plist
|
||||||
|
sudo launchctl load -w /Library/LaunchDaemons/org.chrony-project.chronyd.plist
|
58
contrib/bryan_christianson_1/chronylogrotate.sh
Executable file
58
contrib/bryan_christianson_1/chronylogrotate.sh
Executable file
|
@ -0,0 +1,58 @@
|
||||||
|
#!/bin/sh
|
||||||
|
|
||||||
|
# chronyd/chronyc - Programs for keeping computer clocks accurate.
|
||||||
|
#
|
||||||
|
# **********************************************************************
|
||||||
|
# * Copyright (C) Bryan Christianson 2015
|
||||||
|
# *
|
||||||
|
# * This program is free software; you can redistribute it and/or modify
|
||||||
|
# * it under the terms of version 2 of the GNU General Public License as
|
||||||
|
# * published by the Free Software Foundation.
|
||||||
|
# *
|
||||||
|
# * This program is distributed in the hope that it will be useful, but
|
||||||
|
# * WITHOUT ANY WARRANTY; without even the implied warranty of
|
||||||
|
# * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
|
||||||
|
# * General Public License for more details.
|
||||||
|
# *
|
||||||
|
# * You should have received a copy of the GNU General Public License along
|
||||||
|
# * with this program; if not, write to the Free Software Foundation, Inc.,
|
||||||
|
# * 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA.
|
||||||
|
# *
|
||||||
|
# **********************************************************************
|
||||||
|
|
||||||
|
LOGDIR=/var/log/chrony
|
||||||
|
|
||||||
|
rotate () {
|
||||||
|
prefix=$1
|
||||||
|
|
||||||
|
rm -f $prefix.log.10
|
||||||
|
|
||||||
|
for (( count=9; count>= 0; count-- ))
|
||||||
|
do
|
||||||
|
next=$(( $count+1 ))
|
||||||
|
if [ -f $prefix.log.$count ]; then
|
||||||
|
mv $prefix.log.$count $prefix.log.$next
|
||||||
|
fi
|
||||||
|
done
|
||||||
|
|
||||||
|
if [ -f $prefix.log ]; then
|
||||||
|
mv $prefix.log $prefix.log.0
|
||||||
|
fi
|
||||||
|
}
|
||||||
|
|
||||||
|
if [ ! -e "$LOGDIR" ]; then
|
||||||
|
logger -s "missing directory: $LOGDIR"
|
||||||
|
exit 1
|
||||||
|
fi
|
||||||
|
|
||||||
|
cd $LOGDIR
|
||||||
|
|
||||||
|
rotate measurements
|
||||||
|
rotate statistics
|
||||||
|
rotate tracking
|
||||||
|
|
||||||
|
#
|
||||||
|
# signal chronyd via chronyc
|
||||||
|
/usr/local/bin/chronyc cyclelogs > /dev/null
|
||||||
|
|
||||||
|
exit $?
|
|
@ -0,0 +1,22 @@
|
||||||
|
<?xml version="1.0" encoding="UTF-8"?>
|
||||||
|
<!DOCTYPE plist PUBLIC "-//Apple//DTD PLIST 1.0//EN" "http://www.apple.com/DTDs/PropertyList-1.0.dtd">
|
||||||
|
<plist version="1.0">
|
||||||
|
<dict>
|
||||||
|
<key>Label</key>
|
||||||
|
<string>org.chrony-project.logrotate</string>
|
||||||
|
<key>KeepAlive</key>
|
||||||
|
<false/>
|
||||||
|
<key>ProgramArguments</key>
|
||||||
|
<array>
|
||||||
|
<string>/bin/sh</string>
|
||||||
|
<string>/usr/local/bin/chronylogrotate.sh</string>
|
||||||
|
</array>
|
||||||
|
<key>StartCalendarInterval</key>
|
||||||
|
<dict>
|
||||||
|
<key>Minute</key>
|
||||||
|
<integer>5</integer>
|
||||||
|
<key>Hour</key>
|
||||||
|
<integer>4</integer>
|
||||||
|
</dict>
|
||||||
|
</dict>
|
||||||
|
</plist>
|
|
@ -0,0 +1,19 @@
|
||||||
|
<?xml version="1.0" encoding="UTF-8"?>
|
||||||
|
<!DOCTYPE plist PUBLIC "-//Apple//DTD PLIST 1.0//EN" "http://www.apple.com/DTDs/PropertyList-1.0.dtd">
|
||||||
|
<plist version="1.0">
|
||||||
|
<dict>
|
||||||
|
<key>Label</key>
|
||||||
|
<string>org.chrony-project.chronyd</string>
|
||||||
|
<key>Program</key>
|
||||||
|
<string>/usr/local/sbin/chronyd</string>
|
||||||
|
<key>ProgramArguments</key>
|
||||||
|
<array>
|
||||||
|
<string>chronyd</string>
|
||||||
|
<string>-n</string>
|
||||||
|
<string>-f</string>
|
||||||
|
<string>/private/etc/chrony.d/chrony.conf</string>
|
||||||
|
</array>
|
||||||
|
<key>KeepAlive</key>
|
||||||
|
<true/>
|
||||||
|
</dict>
|
||||||
|
</plist>
|
65
contrib/erik_bryer_1
Normal file
65
contrib/erik_bryer_1
Normal file
|
@ -0,0 +1,65 @@
|
||||||
|
#!/bin/sh
|
||||||
|
#
|
||||||
|
# chrony Start time synchronization. This script
|
||||||
|
# starts chronyd.
|
||||||
|
#
|
||||||
|
# Hacked by: Erik Bryer <ebryer@spots.ab.ca> using inet as a template
|
||||||
|
#
|
||||||
|
# chkconfig: 2345 02 82
|
||||||
|
# description: chronyd helps keep the system time accurate by calculating \
|
||||||
|
# and applying correction factors to compensate for the drift \
|
||||||
|
# in the clock. chronyd can also correct the hardware clock \
|
||||||
|
# (RTC) on some systems.
|
||||||
|
# processname: chronyd
|
||||||
|
# config: /etc/chrony.conf
|
||||||
|
|
||||||
|
# Source function library.
|
||||||
|
. /etc/rc.d/init.d/functions
|
||||||
|
|
||||||
|
# Source networking configuration.
|
||||||
|
. /etc/sysconfig/network
|
||||||
|
|
||||||
|
# Set path to include chronyd in /usr/local/sbin
|
||||||
|
PATH="$PATH:/usr/local/sbin"
|
||||||
|
|
||||||
|
[ -f /usr/local/sbin/chronyd ] || exit 0
|
||||||
|
|
||||||
|
[ -f /etc/chrony.conf ] || exit 0
|
||||||
|
|
||||||
|
RETVAL=0
|
||||||
|
|
||||||
|
# See how we were called.
|
||||||
|
case "$1" in
|
||||||
|
start)
|
||||||
|
# Start daemons.
|
||||||
|
echo -n "Starting chronyd: "
|
||||||
|
daemon chronyd
|
||||||
|
RETVAL=$?
|
||||||
|
[ $RETVAL -eq 0 ] && touch /var/lock/subsys/chrony
|
||||||
|
echo
|
||||||
|
;;
|
||||||
|
stop)
|
||||||
|
# Stop daemons.
|
||||||
|
echo -n "Shutting down chronyd: "
|
||||||
|
# If not dead killproc automatically sleeps for 4.1 seconds then does
|
||||||
|
# kill -9. "chrony.txt" prefers a 5 second delay, but this should be ok.
|
||||||
|
killproc chronyd -15
|
||||||
|
RETVAL=$?
|
||||||
|
[ $RETVAL -eq 0 ] && rm -f /var/lock/subsys/chrony
|
||||||
|
echo
|
||||||
|
;;
|
||||||
|
status)
|
||||||
|
status chronyd
|
||||||
|
exit $?
|
||||||
|
;;
|
||||||
|
restart)
|
||||||
|
$0 stop
|
||||||
|
$0 start
|
||||||
|
;;
|
||||||
|
*)
|
||||||
|
echo "Usage: named {start|stop|status|restart}"
|
||||||
|
exit 1
|
||||||
|
esac
|
||||||
|
|
||||||
|
exit $RETVAL
|
||||||
|
|
100
contrib/ken_gillett_1
Normal file
100
contrib/ken_gillett_1
Normal file
|
@ -0,0 +1,100 @@
|
||||||
|
#!/bin/sh
|
||||||
|
#
|
||||||
|
# chronyd This shell script takes care of starting and stopping
|
||||||
|
# chronyd (NTP daemon).
|
||||||
|
#
|
||||||
|
# chkconfig: 45 80 20
|
||||||
|
# description: chronyd is the NTP daemon.
|
||||||
|
|
||||||
|
# Source function library.
|
||||||
|
. /etc/rc.d/init.d/functions
|
||||||
|
|
||||||
|
# Source networking configuration.
|
||||||
|
. /etc/sysconfig/network
|
||||||
|
|
||||||
|
# Check that networking is up.
|
||||||
|
[ ${NETWORKING} = "no" ] && exit 0
|
||||||
|
|
||||||
|
PREDIR="/usr/local"
|
||||||
|
CHRONYD=$PREDIR"/sbin/chronyd"
|
||||||
|
CHRONYC=$PREDIR"/bin/chronyc"
|
||||||
|
|
||||||
|
[ -x $CHRONYD -a -x $CHRONYC -a -f /etc/chrony.conf ] || exit 0
|
||||||
|
|
||||||
|
dochrony() {
|
||||||
|
if [ -z "$(pidofproc chronyd)" ]; then
|
||||||
|
echo -e "\n\tchronyd not running\n\n"
|
||||||
|
exit 2
|
||||||
|
fi
|
||||||
|
KEY=`awk '$1 == "commandkey" {print $2; exit}' /etc/chrony.conf`
|
||||||
|
PASSWORD=`awk '$1 == '$KEY' {print $2; exit}' /etc/chrony/keys`
|
||||||
|
|
||||||
|
$CHRONYC <<- EOF
|
||||||
|
password $PASSWORD
|
||||||
|
$@
|
||||||
|
quit
|
||||||
|
EOF
|
||||||
|
}
|
||||||
|
|
||||||
|
# make the first parameter' lower case
|
||||||
|
set - `echo $1 | awk '{print tolower($1)}';shift;echo "$@"`
|
||||||
|
|
||||||
|
# Expand any shortcuts.
|
||||||
|
case "$1" in
|
||||||
|
on|1)
|
||||||
|
set - "online"
|
||||||
|
;;
|
||||||
|
off|0)
|
||||||
|
set - "offline"
|
||||||
|
esac
|
||||||
|
|
||||||
|
# See how we were called.
|
||||||
|
case "$1" in
|
||||||
|
start)
|
||||||
|
# Start daemons.
|
||||||
|
echo -n "Starting chronyd: "
|
||||||
|
daemon $CHRONYD
|
||||||
|
if [ $? -eq 0 ]; then
|
||||||
|
echo $(pidofproc chronyd) > /var/run/chronyd.pid
|
||||||
|
touch /var/lock/subsys/chronyd
|
||||||
|
fi
|
||||||
|
echo
|
||||||
|
;;
|
||||||
|
stop)
|
||||||
|
# Stop daemons.
|
||||||
|
echo -n "Shutting down chronyd: "
|
||||||
|
killproc chronyd
|
||||||
|
echo
|
||||||
|
rm -f /var/lock/subsys/chronyd
|
||||||
|
;;
|
||||||
|
status)
|
||||||
|
status chronyd
|
||||||
|
;;
|
||||||
|
restart|reload)
|
||||||
|
$0 stop
|
||||||
|
$0 start
|
||||||
|
;;
|
||||||
|
condrestart)
|
||||||
|
if [ -f /var/lock/subsys/chronyd ]; then
|
||||||
|
$0 stop
|
||||||
|
$0 start
|
||||||
|
fi
|
||||||
|
;;
|
||||||
|
"")
|
||||||
|
echo "Usage: chronyd
|
||||||
|
{start|stop|restart|reload|condrestart|status|[on|off]line etc}"
|
||||||
|
exit 1
|
||||||
|
;;
|
||||||
|
|
||||||
|
accheck|cmdaccheck|clients|manual|rtcdata|sources|sourcestats|tracking|clients)
|
||||||
|
dochrony "$@"
|
||||||
|
;;
|
||||||
|
*)
|
||||||
|
echo -n "Chrony $1: "
|
||||||
|
dochrony "$@" > /dev/null
|
||||||
|
[ $? -eq 0 ] && echo_success || echo_failure
|
||||||
|
echo
|
||||||
|
esac
|
||||||
|
|
||||||
|
exit 0
|
||||||
|
|
162
contrib/stephan_boettcher_1
Normal file
162
contrib/stephan_boettcher_1
Normal file
|
@ -0,0 +1,162 @@
|
||||||
|
From stephan@nevis1.nevis.columbia.edu Mon Jun 7 20:51:57 1999
|
||||||
|
Date: 04 Jun 1999 00:17:25 -0400
|
||||||
|
From: Stephan I. Boettcher <stephan@nevis1.nevis.columbia.edu>
|
||||||
|
To: richard@rrbcurnow.freeserve.co.uk
|
||||||
|
Subject: chrony 1.1 sysV startup script for notebooks
|
||||||
|
|
||||||
|
|
||||||
|
Dear Richard,
|
||||||
|
|
||||||
|
I installed chrony on my notebook, running RedHat 5.1 Linux.
|
||||||
|
It looks like it works. No problems.
|
||||||
|
|
||||||
|
Thank you!
|
||||||
|
|
||||||
|
I like to donate my sysV startup script, appended below.
|
||||||
|
|
||||||
|
Special feature: the `online' command scans the config file to
|
||||||
|
selectively turn some servers online, depending on the pcmcia SCHEME.
|
||||||
|
|
||||||
|
booting: /etc/rc.d/init.d/chrony start
|
||||||
|
/etc/ppp/ip-up: /etc/rc.d/init.d/chrony online
|
||||||
|
/etc/ppp/ip-down: /etc/rc.d/init.d/chrony offline
|
||||||
|
logrotate cron: /etc/rc.d/init.d/chrony cyclelogs
|
||||||
|
a user: /etc/rc.d/init.d/chrony status
|
||||||
|
a sysadmin: /etc/rc.d/init.d/chrony restart
|
||||||
|
shutdown: /etc/rc.d/init.d/chrony stop
|
||||||
|
|
||||||
|
Best regards
|
||||||
|
Stephan
|
||||||
|
|
||||||
|
--
|
||||||
|
|
||||||
|
------------------------------------------------------------------------
|
||||||
|
Stephan Boettcher FAX: +1-914-591-4540
|
||||||
|
Columbia University, Nevis Labs Tel: +1-914-591-2863
|
||||||
|
P.O. Box 137, 136 South Broadway mailto:stephan@nevis1.columbia.edu
|
||||||
|
Irvington, NY 10533, USA http://www.nevis.columbia.edu/~stephan
|
||||||
|
------------------------------------------------------------------------
|
||||||
|
|
||||||
|
########################### cut here ###################################
|
||||||
|
#! /bin/bash
|
||||||
|
#
|
||||||
|
# /etc/rc.d/init.d/chrony
|
||||||
|
#
|
||||||
|
# SYS V startup script for
|
||||||
|
# chrony ntp daemon
|
||||||
|
# on Linux 2.0.3x notebooks with pcmcia scheme support
|
||||||
|
# $Id: stephan_boettcher_1,v 1.1 2000/04/24 21:36:04 richard Exp $
|
||||||
|
#
|
||||||
|
# 1999-06-02 SiB <stephan@nevis1.columbia.edu>
|
||||||
|
#
|
||||||
|
# For PCMCIA users:
|
||||||
|
# In /etc/chrony.conf, precede the server commands for each SCHEME
|
||||||
|
# with a comment line that contains the word SCHEME and the name of
|
||||||
|
# the scheme(s) that should use the servers, up to the next line that
|
||||||
|
# contains the word SCHEME. The servers must be `offline' and
|
||||||
|
# specified by their IP address. The hostname will not do.
|
||||||
|
#
|
||||||
|
# Like:
|
||||||
|
#
|
||||||
|
# # SCHEME nevisppp nevislan
|
||||||
|
# # stephanpc.nevis.columbia.edu
|
||||||
|
# server 192.12.82.222 offline
|
||||||
|
#
|
||||||
|
# # SCHEME desyppp desylan
|
||||||
|
#
|
||||||
|
# # dsygw2.desy.de
|
||||||
|
# server 131.169.30.15 offline
|
||||||
|
# # dscomsa.desy.de
|
||||||
|
# server 131.169.197.35 offline
|
||||||
|
|
||||||
|
CONF=/etc/chrony.conf
|
||||||
|
CHRONYD=/usr/local/sbin/chronyd
|
||||||
|
CHRONYC=/usr/local/bin/chronyc
|
||||||
|
KEYS=/etc/chrony.keys
|
||||||
|
|
||||||
|
# See if we got all we need:
|
||||||
|
|
||||||
|
[ -f $CHRONYD -a -f $CHRONYC -a -r $CONF ] || exit
|
||||||
|
|
||||||
|
|
||||||
|
[ -r $KEYS ] \
|
||||||
|
&& CMDKEY=`awk '/^commandkey/{print $2}' $CONF` \
|
||||||
|
&& PASSWORD=`awk -v KEY=$CMDKEY '$1==KEY{print $2}' $KEYS`
|
||||||
|
|
||||||
|
|
||||||
|
case "$1" in
|
||||||
|
|
||||||
|
start)
|
||||||
|
echo -n "Starting chronyd "
|
||||||
|
$CHRONYD -r -s -f $CONF
|
||||||
|
echo
|
||||||
|
;;
|
||||||
|
|
||||||
|
stop)
|
||||||
|
echo -n "Shutting down chronyd "
|
||||||
|
/usr/bin/killall chronyd
|
||||||
|
echo
|
||||||
|
;;
|
||||||
|
|
||||||
|
restart)
|
||||||
|
$0 stop
|
||||||
|
$0 start
|
||||||
|
;;
|
||||||
|
|
||||||
|
on*)
|
||||||
|
|
||||||
|
[ -f /var/run/pcmcia-scheme ] && SCHEME=`cat /var/run/pcmcia-scheme`
|
||||||
|
|
||||||
|
awk -v SCHEME=${SCHEME:-default} -v PASSWORD=$PASSWORD \
|
||||||
|
'
|
||||||
|
BEGIN {
|
||||||
|
SEL=1;
|
||||||
|
print "password", PASSWORD;
|
||||||
|
}
|
||||||
|
/SCHEME/ {
|
||||||
|
SEL=match($0, SCHEME);
|
||||||
|
}
|
||||||
|
SEL && /^server[ \t]*[0-9.]+[ \t].*offline/ {
|
||||||
|
print "online 255.255.255.255/" $2;
|
||||||
|
}
|
||||||
|
' \
|
||||||
|
$CONF \
|
||||||
|
| $CHRONYC
|
||||||
|
|
||||||
|
;;
|
||||||
|
|
||||||
|
off*)
|
||||||
|
cat <<-EOF | $CHRONYC
|
||||||
|
password $PASSWORD
|
||||||
|
offline
|
||||||
|
trimrtc
|
||||||
|
dump
|
||||||
|
EOF
|
||||||
|
;;
|
||||||
|
|
||||||
|
*log*)
|
||||||
|
cat <<-EOF | $CHRONYC
|
||||||
|
password $PASSWORD
|
||||||
|
cyclelogs
|
||||||
|
EOF
|
||||||
|
;;
|
||||||
|
|
||||||
|
stat*)
|
||||||
|
cat <<-EOF | $CHRONYC
|
||||||
|
sources
|
||||||
|
sourcestats
|
||||||
|
tracking
|
||||||
|
rtcdata
|
||||||
|
EOF
|
||||||
|
;;
|
||||||
|
|
||||||
|
*)
|
||||||
|
echo "Usage: chronyd {start|stop|restart|status|online|offline|cyclelogs}"
|
||||||
|
exit 1
|
||||||
|
;;
|
||||||
|
|
||||||
|
esac
|
||||||
|
|
||||||
|
exit 0
|
||||||
|
|
||||||
|
|
118
contrib/wolfgang_weisselberg1
Normal file
118
contrib/wolfgang_weisselberg1
Normal file
|
@ -0,0 +1,118 @@
|
||||||
|
|
||||||
|
> Is it possible to limit chronyc to only those commands that
|
||||||
|
> are readonly plus those necessary to bring a dialup connection up
|
||||||
|
> and down? That is: online offline dump writertc and password.
|
||||||
|
|
||||||
|
This is trivial on the same host and workable for non-local
|
||||||
|
hosts: use a wrapper program or script. An *untested*
|
||||||
|
sample follows. To use it, best create a special user (say
|
||||||
|
chronyc) and a special group (say chronyg). Make the script
|
||||||
|
chronyc:chronyg, and 4750 (suid, rwxr-x---). Add all users
|
||||||
|
who may run the script to the group chronyg.
|
||||||
|
|
||||||
|
Make a chrony password file e.g.
|
||||||
|
/usr/local/etc/chrony_password. It should be owned by chronyc
|
||||||
|
and readable only for the owner, containing only the chrony
|
||||||
|
password (and maybe a newline) in the first line.
|
||||||
|
|
||||||
|
In this way only the script (call it run_chrony, for example)
|
||||||
|
can read the password. It will allow only those commands you
|
||||||
|
explicitely allow. You can add a password check -- especially
|
||||||
|
if you add an internet port so you can access it over the
|
||||||
|
internet this is advisable. You really want to add logging
|
||||||
|
to this untested script as well.
|
||||||
|
|
||||||
|
|
||||||
|
BTW, if you use some sort of PPP, you probably can use
|
||||||
|
/etc/ppp/ip-up and /etc/ppp/ip-down to transparently set chrony
|
||||||
|
on- and offline as the ip connection goes up and comes down.
|
||||||
|
This is _far_ more user friendly, IMHO, and a DOS by switching
|
||||||
|
chrony offline all the time is avoided as well.
|
||||||
|
|
||||||
|
|
||||||
|
#! /usr/bin/perl -T
|
||||||
|
use v5.6.1;
|
||||||
|
use warnings;
|
||||||
|
use strict;
|
||||||
|
|
||||||
|
sub laundered_command();
|
||||||
|
sub order_chrony($$);
|
||||||
|
sub read_password();
|
||||||
|
sub usage($);
|
||||||
|
|
||||||
|
our $CHRONY = "/usr/local/bin/chronyc";
|
||||||
|
|
||||||
|
# NOTE: select the file system protection wisely for the
|
||||||
|
# PASSWORDFILE!
|
||||||
|
our $PASSWORDFILE = "/usr/local/etc/chrony_password";
|
||||||
|
|
||||||
|
our @ALLOWED_COMMANDS = (
|
||||||
|
'online', # switch online mode on
|
||||||
|
'offline', # switch online mode off
|
||||||
|
'dump', # save measurements to file
|
||||||
|
'writerc', # save RTC accumulated data
|
||||||
|
|
||||||
|
'clients', # which clients are served by us?
|
||||||
|
'rtcdata', # Quality of RTC measurements
|
||||||
|
'sources(?: -v)?', # Show our sources (verbose)
|
||||||
|
'sourcestats(?: -v)?', # How good are our sources (verbose)?
|
||||||
|
'tracking', # whom do we adjust to?
|
||||||
|
|
||||||
|
# 'burst \d+/\d+', # allow them to send bursts?
|
||||||
|
);
|
||||||
|
|
||||||
|
usage("No command given.") unless $ARGV[0];
|
||||||
|
|
||||||
|
%ENV = (); # nuke all environment variables. Rather
|
||||||
|
# drastic, but better safe than sorry!
|
||||||
|
# Add whatever you really need to get it
|
||||||
|
# working (again).
|
||||||
|
$ENV{'PATH'} = '/usr/local/bin:/bin:/usr/bin';
|
||||||
|
|
||||||
|
order_chrony(laundered_command(), read_password());
|
||||||
|
|
||||||
|
exit 0; # command succeeded
|
||||||
|
|
||||||
|
############################################################
|
||||||
|
|
||||||
|
sub usage($) {
|
||||||
|
print STDERR "Error: ", shift, "\n";
|
||||||
|
|
||||||
|
# OK, this eats the -v...
|
||||||
|
print STDERR "Legal commands are:\n\t", join "\n",
|
||||||
|
map { $_ =~ m:(\w+):; $1 } @ALLOWED_COMMANDS;
|
||||||
|
exit 1; # error
|
||||||
|
}
|
||||||
|
|
||||||
|
############################################################
|
||||||
|
|
||||||
|
sub laundered_command() {
|
||||||
|
my $regexp = "^(" . join ( "|", @ALLOWED_COMMANDS ) . ")\$";
|
||||||
|
my $parameters = join " ", @ARGV;
|
||||||
|
$parameters =~ m:$regexp: or usage("Command $parameters not allowed.");
|
||||||
|
|
||||||
|
return $1; # this value, then, is untainted.
|
||||||
|
};
|
||||||
|
|
||||||
|
############################################################
|
||||||
|
|
||||||
|
sub read_password() {
|
||||||
|
open PASS, $PASSWORDFILE
|
||||||
|
or die "Could not read protected password file: $!";
|
||||||
|
my $password = <PASS>;
|
||||||
|
chomp $password;
|
||||||
|
return $password;
|
||||||
|
};
|
||||||
|
|
||||||
|
############################################################
|
||||||
|
|
||||||
|
sub order_chrony($$) {
|
||||||
|
my ($clean_command, $password) = @_;
|
||||||
|
open CHRONY, "| $CHRONY &> /dev/null" or die "could not run $CHRONY: $!\n";
|
||||||
|
print CHRONY "password $password\n";
|
||||||
|
print CHRONY "$clean_command\n";
|
||||||
|
close CHRONY
|
||||||
|
or die "Error running command $clean_command\n", "\ton $CHRONY: $!\n";
|
||||||
|
}
|
||||||
|
|
||||||
|
############################################################
|
76
doc/Makefile.in
Normal file
76
doc/Makefile.in
Normal file
|
@ -0,0 +1,76 @@
|
||||||
|
ADOC = asciidoctor
|
||||||
|
ADOC_FLAGS =
|
||||||
|
SED = sed
|
||||||
|
HTML_TO_TXT = w3m -dump -T text/html
|
||||||
|
|
||||||
|
MAN_FILES = chrony.conf.man chronyc.man chronyd.man
|
||||||
|
TXT_FILES = faq.txt installation.txt
|
||||||
|
HTML_FILES = $(MAN_FILES:%.man=%.html) $(TXT_FILES:%.txt=%.html)
|
||||||
|
MAN_IN_FILES = $(MAN_FILES:%.man=%.man.in)
|
||||||
|
|
||||||
|
SYSCONFDIR = @SYSCONFDIR@
|
||||||
|
BINDIR = @BINDIR@
|
||||||
|
SBINDIR = @SBINDIR@
|
||||||
|
MANDIR = @MANDIR@
|
||||||
|
DOCDIR = @DOCDIR@
|
||||||
|
CHRONYRUNDIR = @CHRONYRUNDIR@
|
||||||
|
CHRONYVARDIR = @CHRONYVARDIR@
|
||||||
|
CHRONY_VERSION = @CHRONY_VERSION@
|
||||||
|
DEFAULT_USER = @DEFAULT_USER@
|
||||||
|
DEFAULT_HWCLOCK_FILE = @DEFAULT_HWCLOCK_FILE@
|
||||||
|
DEFAULT_PID_FILE = @DEFAULT_PID_FILE@
|
||||||
|
DEFAULT_RTC_DEVICE = @DEFAULT_RTC_DEVICE@
|
||||||
|
|
||||||
|
SED_COMMANDS = "s%\@SYSCONFDIR\@%$(SYSCONFDIR)%g;\
|
||||||
|
s%\@BINDIR\@%$(BINDIR)%g;\
|
||||||
|
s%\@SBINDIR\@%$(SBINDIR)%g;\
|
||||||
|
s%\@CHRONY_VERSION\@%$(CHRONY_VERSION)%g;\
|
||||||
|
s%\@DEFAULT_HWCLOCK_FILE\@%$(DEFAULT_HWCLOCK_FILE)%g;\
|
||||||
|
s%\@DEFAULT_PID_FILE\@%$(DEFAULT_PID_FILE)%g;\
|
||||||
|
s%\@DEFAULT_RTC_DEVICE\@%$(DEFAULT_RTC_DEVICE)%g;\
|
||||||
|
s%\@DEFAULT_USER\@%$(DEFAULT_USER)%g;\
|
||||||
|
s%\@CHRONYRUNDIR\@%$(CHRONYRUNDIR)%g;\
|
||||||
|
s%\@CHRONYVARDIR\@%$(CHRONYVARDIR)%g;"
|
||||||
|
|
||||||
|
man: $(MAN_FILES) $(MAN_IN_FILES)
|
||||||
|
html: $(HTML_FILES)
|
||||||
|
txt: $(TXT_FILES)
|
||||||
|
docs: man html
|
||||||
|
|
||||||
|
%.html: %.adoc
|
||||||
|
$(ADOC) $(ADOC_FLAGS) -b html -o - $< | $(SED) -e $(SED_COMMANDS) > $@
|
||||||
|
|
||||||
|
%.man.in: %.adoc
|
||||||
|
$(ADOC) $(ADOC_FLAGS) -b manpage -o $@ $<
|
||||||
|
|
||||||
|
%.man: %.man.in
|
||||||
|
$(SED) -e $(SED_COMMANDS) < $< > $@
|
||||||
|
|
||||||
|
%.txt: %.html
|
||||||
|
$(HTML_TO_TXT) < $< > $@
|
||||||
|
|
||||||
|
install: $(MAN_FILES)
|
||||||
|
[ -d $(DESTDIR)$(MANDIR)/man1 ] || mkdir -p $(DESTDIR)$(MANDIR)/man1
|
||||||
|
[ -d $(DESTDIR)$(MANDIR)/man5 ] || mkdir -p $(DESTDIR)$(MANDIR)/man5
|
||||||
|
[ -d $(DESTDIR)$(MANDIR)/man8 ] || mkdir -p $(DESTDIR)$(MANDIR)/man8
|
||||||
|
cp chronyc.man $(DESTDIR)$(MANDIR)/man1/chronyc.1
|
||||||
|
chmod 644 $(DESTDIR)$(MANDIR)/man1/chronyc.1
|
||||||
|
cp chronyd.man $(DESTDIR)$(MANDIR)/man8/chronyd.8
|
||||||
|
chmod 644 $(DESTDIR)$(MANDIR)/man8/chronyd.8
|
||||||
|
cp chrony.conf.man $(DESTDIR)$(MANDIR)/man5/chrony.conf.5
|
||||||
|
chmod 644 $(DESTDIR)$(MANDIR)/man5/chrony.conf.5
|
||||||
|
|
||||||
|
install-docs: $(HTML_FILES)
|
||||||
|
[ -d $(DESTDIR)$(DOCDIR) ] || mkdir -p $(DESTDIR)$(DOCDIR)
|
||||||
|
for f in $(HTML_FILES); do \
|
||||||
|
cp $$f $(DESTDIR)$(DOCDIR); \
|
||||||
|
chmod 644 $(DESTDIR)$(DOCDIR)/$$f; \
|
||||||
|
done
|
||||||
|
|
||||||
|
clean:
|
||||||
|
rm -f $(MAN_FILES) $(TXT_FILES) $(HTML_FILES)
|
||||||
|
rm -f $(MAN_IN_FILES)
|
||||||
|
|
||||||
|
distclean:
|
||||||
|
rm -f $(MAN_FILES) $(TXT_FILES) $(HTML_FILES)
|
||||||
|
rm -f Makefile
|
3281
doc/chrony.conf.adoc
Normal file
3281
doc/chrony.conf.adoc
Normal file
File diff suppressed because it is too large
Load diff
5235
doc/chrony.conf.man.in
Normal file
5235
doc/chrony.conf.man.in
Normal file
File diff suppressed because it is too large
Load diff
1582
doc/chronyc.adoc
Normal file
1582
doc/chronyc.adoc
Normal file
File diff suppressed because it is too large
Load diff
2793
doc/chronyc.man.in
Normal file
2793
doc/chronyc.man.in
Normal file
File diff suppressed because it is too large
Load diff
235
doc/chronyd.adoc
Normal file
235
doc/chronyd.adoc
Normal file
|
@ -0,0 +1,235 @@
|
||||||
|
// This file is part of chrony
|
||||||
|
//
|
||||||
|
// Copyright (C) Richard P. Curnow 1997-2003
|
||||||
|
// Copyright (C) Miroslav Lichvar 2009-2017
|
||||||
|
//
|
||||||
|
// This program is free software; you can redistribute it and/or modify
|
||||||
|
// it under the terms of version 2 of the GNU General Public License as
|
||||||
|
// published by the Free Software Foundation.
|
||||||
|
//
|
||||||
|
// This program is distributed in the hope that it will be useful, but
|
||||||
|
// WITHOUT ANY WARRANTY; without even the implied warranty of
|
||||||
|
// MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
|
||||||
|
// General Public License for more details.
|
||||||
|
//
|
||||||
|
// You should have received a copy of the GNU General Public License along
|
||||||
|
// with this program; if not, write to the Free Software Foundation, Inc.,
|
||||||
|
// 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA.
|
||||||
|
|
||||||
|
= chronyd(8)
|
||||||
|
:doctype: manpage
|
||||||
|
:man manual: System Administration
|
||||||
|
:man source: chrony @CHRONY_VERSION@
|
||||||
|
|
||||||
|
== NAME
|
||||||
|
|
||||||
|
chronyd - chrony daemon
|
||||||
|
|
||||||
|
== SYNOPSIS
|
||||||
|
|
||||||
|
*chronyd* [_OPTION_]... [_DIRECTIVE_]...
|
||||||
|
|
||||||
|
== DESCRIPTION
|
||||||
|
|
||||||
|
*chronyd* is a daemon for synchronisation of the system clock. It can
|
||||||
|
synchronise the clock with NTP servers, reference clocks (e.g. a GPS receiver),
|
||||||
|
and manual input using wristwatch and keyboard via *chronyc*. It can also
|
||||||
|
operate as an NTPv4 (RFC 5905) server and peer to provide a time service to
|
||||||
|
other computers in the network.
|
||||||
|
|
||||||
|
If no configuration directives are specified on the command line, *chronyd*
|
||||||
|
will read them from a configuration file. The compiled-in default location of
|
||||||
|
the file is _@SYSCONFDIR@/chrony.conf_.
|
||||||
|
|
||||||
|
Informational messages, warnings, and errors will be logged to syslog.
|
||||||
|
|
||||||
|
== OPTIONS
|
||||||
|
|
||||||
|
*-4*::
|
||||||
|
With this option hostnames will be resolved only to IPv4 addresses and only
|
||||||
|
IPv4 sockets will be created.
|
||||||
|
|
||||||
|
*-6*::
|
||||||
|
With this option hostnames will be resolved only to IPv6 addresses and only
|
||||||
|
IPv6 sockets will be created.
|
||||||
|
|
||||||
|
*-f* _file_::
|
||||||
|
This option can be used to specify an alternate location for the configuration
|
||||||
|
file. The compiled-in default value is _@SYSCONFDIR@/chrony.conf_.
|
||||||
|
|
||||||
|
*-n*::
|
||||||
|
When run in this mode, the program will not detach itself from the terminal.
|
||||||
|
|
||||||
|
*-d*::
|
||||||
|
When run in this mode, the program will not detach itself from the terminal,
|
||||||
|
and all messages will be written to the terminal instead of syslog. If
|
||||||
|
*chronyd* was compiled with enabled support for debugging, this option can be
|
||||||
|
used twice to enable debug messages.
|
||||||
|
|
||||||
|
*-l* _file_::
|
||||||
|
This option enables writing of log messages to a file instead of syslog or the
|
||||||
|
terminal.
|
||||||
|
|
||||||
|
*-L* _level_::
|
||||||
|
This option specifies the minimum severity level of messages to be written to
|
||||||
|
the log file, syslog, or terminal. The following levels can be specified: -1
|
||||||
|
(debug, if compiled with enabled support for debugging), 0 (informational), 1
|
||||||
|
(warning), 2 (non-fatal error), and 3 (fatal error). The default value is 0.
|
||||||
|
|
||||||
|
*-p*::
|
||||||
|
When run in this mode, *chronyd* will print the configuration and exit. It will
|
||||||
|
not detach from the terminal. This option can be used to verify the syntax of
|
||||||
|
the configuration and get the whole configuration, even if it is split into
|
||||||
|
multiple files and read by the *include* or *confdir* directive.
|
||||||
|
|
||||||
|
*-q*::
|
||||||
|
When run in this mode, *chronyd* will set the system clock once and exit. It
|
||||||
|
will not detach from the terminal.
|
||||||
|
|
||||||
|
*-Q*::
|
||||||
|
This option is similar to the *-q* option, except it only prints the offset
|
||||||
|
without making any corrections of the clock and disables server ports to allow
|
||||||
|
*chronyd* to be started without root privileges, assuming the configuration
|
||||||
|
does not have any directives which would require them (e.g. *refclock*,
|
||||||
|
*hwtimestamp*, *rtcfile*, etc).
|
||||||
|
|
||||||
|
*-r*::
|
||||||
|
This option will try to reload and then delete files containing sample
|
||||||
|
histories for each of the servers and reference clocks being used. The
|
||||||
|
files are expected to be in the directory specified by the
|
||||||
|
<<chrony.conf.adoc#dumpdir,*dumpdir*>>
|
||||||
|
directive in the configuration file. This option is useful if you want to stop
|
||||||
|
and restart *chronyd* briefly for any reason, e.g. to install a new version.
|
||||||
|
However, it should be used only on systems where the kernel can maintain clock
|
||||||
|
compensation whilst not under *chronyd*'s control (i.e. Linux, FreeBSD, NetBSD,
|
||||||
|
illumos, and macOS 10.13 or later).
|
||||||
|
|
||||||
|
*-R*::
|
||||||
|
When this option is used, the <<chrony.conf.adoc#initstepslew,*initstepslew*>>
|
||||||
|
directive and the <<chrony.conf.adoc#makestep,*makestep*>> directive used with
|
||||||
|
a positive limit will be ignored. This option is useful when restarting
|
||||||
|
*chronyd* and can be used in conjunction with the *-r* option.
|
||||||
|
|
||||||
|
*-s*::
|
||||||
|
This option will set the system clock from the computer's real-time clock (RTC)
|
||||||
|
or to the last modification time of the file specified by the
|
||||||
|
<<chrony.conf.adoc#driftfile,*driftfile*>> directive. Real-time clocks are
|
||||||
|
supported only on Linux.
|
||||||
|
+
|
||||||
|
If used in conjunction with the *-r* flag, *chronyd* will attempt to preserve
|
||||||
|
the old samples after setting the system clock from the RTC. This can be used
|
||||||
|
to allow *chronyd* to perform long term averaging of the gain or loss rate
|
||||||
|
across system reboots, and is useful for systems with intermittent access to
|
||||||
|
network that are shut down when not in use. For this to work well, it relies
|
||||||
|
on *chronyd* having been able to determine accurate statistics for the
|
||||||
|
difference between the RTC and system clock last time the computer was on.
|
||||||
|
+
|
||||||
|
If the last modification time of the drift file is later than both the current
|
||||||
|
time and the RTC time, the system time will be set to it to restore the time
|
||||||
|
when *chronyd* was previously stopped. This is useful on computers that have no
|
||||||
|
RTC or the RTC is broken (e.g. it has no battery).
|
||||||
|
|
||||||
|
*-t* _timeout_::
|
||||||
|
This option sets a timeout (in seconds) after which *chronyd* will exit. If the
|
||||||
|
clock is not synchronised, it will exit with a non-zero status. This is useful
|
||||||
|
with the *-q* or *-Q* option to shorten the maximum time waiting for
|
||||||
|
measurements, or with the *-r* option to limit the time when *chronyd* is
|
||||||
|
running, but still allow it to adjust the frequency of the system clock.
|
||||||
|
|
||||||
|
*-u* _user_::
|
||||||
|
This option sets the name of the system user to which *chronyd* will switch
|
||||||
|
after start in order to drop root privileges. It overrides the
|
||||||
|
<<chrony.conf.adoc#user,*user*>> directive. The compiled-in default value is
|
||||||
|
_@DEFAULT_USER@_.
|
||||||
|
+
|
||||||
|
On Linux, *chronyd* needs to be compiled with support for the *libcap* library.
|
||||||
|
On macOS, FreeBSD, NetBSD, and illumos *chronyd* forks into two processes.
|
||||||
|
The child process retains root privileges, but can only perform a very limited
|
||||||
|
range of privileged system calls on behalf of the parent.
|
||||||
|
|
||||||
|
*-U*::
|
||||||
|
This option disables a check for root privileges to allow *chronyd* to be
|
||||||
|
started under a non-root user, assuming the process will have all capabilities
|
||||||
|
(e.g. provided by the service manager) and access to all files, directories,
|
||||||
|
and devices, needed to operate correctly in the specified configuration. Note
|
||||||
|
that different capabilities might be needed with different configurations and
|
||||||
|
different Linux kernel versions. Starting *chronyd* under a non-root user is
|
||||||
|
not recommended when the configuration is not known, or at least limited to
|
||||||
|
specific directives.
|
||||||
|
|
||||||
|
*-F* _level_::
|
||||||
|
This option configures system call filters loaded by *chronyd* processes if it
|
||||||
|
was compiled with support for the Linux secure computing (seccomp) facility.
|
||||||
|
Three levels are defined: 0, 1, 2. The filters are disabled at level 0. At
|
||||||
|
levels 1 and 2, *chronyd* will be killed if it makes a system call which is
|
||||||
|
blocked by the filters. The level can be specified as a negative number to
|
||||||
|
trigger the SIGSYS signal instead of SIGKILL, which can be useful for
|
||||||
|
debugging. The default value is 0.
|
||||||
|
+
|
||||||
|
At level 1, the filters allow only selected system calls that are normally
|
||||||
|
expected to be made by *chronyd*. Other system calls are blocked. This level is
|
||||||
|
recommended only if it is known to work on the version of the system where
|
||||||
|
*chrony* is installed. The filters need to allow also system calls made by
|
||||||
|
libraries that *chronyd* is using (e.g. libc), but different versions or
|
||||||
|
implementations of the libraries might make different system calls. If the
|
||||||
|
filters are missing a system call, *chronyd* could be killed even in normal
|
||||||
|
operation.
|
||||||
|
+
|
||||||
|
At level 2, the filters block only a small number of specific system calls
|
||||||
|
(e.g. fork and exec). This approach should avoid false positives, but the
|
||||||
|
protection of the system against a compromised *chronyd* process is much more
|
||||||
|
limited.
|
||||||
|
+
|
||||||
|
The filters cannot be enabled with the *mailonchange* directive.
|
||||||
|
|
||||||
|
*-P* _priority_::
|
||||||
|
On Linux, FreeBSD, NetBSD, and illumos this option will select the SCHED_FIFO
|
||||||
|
real-time scheduler at the specified priority (which must be between 0 and
|
||||||
|
100). On macOS, this option must have either a value of 0 to disable the thread
|
||||||
|
time constraint policy or 1 for the policy to be enabled. Other systems do not
|
||||||
|
support this option. The default value is 0.
|
||||||
|
|
||||||
|
*-m*::
|
||||||
|
This option will lock *chronyd* into RAM so that it will never be paged out.
|
||||||
|
This mode is only supported on Linux, FreeBSD, NetBSD, and illumos.
|
||||||
|
|
||||||
|
*-x*::
|
||||||
|
This option disables the control of the system clock. *chronyd* will not try to
|
||||||
|
make any adjustments of the clock. It will assume the clock is free running and
|
||||||
|
still track its offset and frequency relative to the estimated true time. This
|
||||||
|
option allows *chronyd* to be started without the capability to adjust or set
|
||||||
|
the system clock (e.g. in some containers) to operate as an NTP server.
|
||||||
|
|
||||||
|
*-v*, *--version*::
|
||||||
|
With this option *chronyd* will print version number to the terminal and exit.
|
||||||
|
|
||||||
|
*-h*, *--help*::
|
||||||
|
With this option *chronyd* will print a help message to the terminal and exit.
|
||||||
|
|
||||||
|
== ENVIRONMENT VARIABLES
|
||||||
|
|
||||||
|
*LISTEN_FDS*::
|
||||||
|
On Linux systems, the systemd service manager may pass file descriptors for
|
||||||
|
pre-initialised sockets to *chronyd*. The service manager allocates and binds
|
||||||
|
the file descriptors, and passes a copy to each spawned instance of the
|
||||||
|
service. This allows for zero-downtime service restarts as the sockets buffer
|
||||||
|
client requests until the service is able to handle them. The service manager
|
||||||
|
sets the LISTEN_FDS environment variable to the number of passed file
|
||||||
|
descriptors.
|
||||||
|
|
||||||
|
== FILES
|
||||||
|
|
||||||
|
_@SYSCONFDIR@/chrony.conf_
|
||||||
|
|
||||||
|
== SEE ALSO
|
||||||
|
|
||||||
|
<<chronyc.adoc#,*chronyc(1)*>>, <<chrony.conf.adoc#,*chrony.conf(5)*>>
|
||||||
|
|
||||||
|
== BUGS
|
||||||
|
|
||||||
|
For instructions on how to report bugs, please visit
|
||||||
|
https://chrony-project.org/.
|
||||||
|
|
||||||
|
== AUTHORS
|
||||||
|
|
||||||
|
chrony was written by Richard Curnow, Miroslav Lichvar, and others.
|
278
doc/chronyd.man.in
Normal file
278
doc/chronyd.man.in
Normal file
|
@ -0,0 +1,278 @@
|
||||||
|
'\" t
|
||||||
|
.\" Title: chronyd
|
||||||
|
.\" Author: [see the "AUTHOR(S)" section]
|
||||||
|
.\" Generator: Asciidoctor 2.0.20
|
||||||
|
.\" Date: 2024-10-08
|
||||||
|
.\" Manual: System Administration
|
||||||
|
.\" Source: chrony @CHRONY_VERSION@
|
||||||
|
.\" Language: English
|
||||||
|
.\"
|
||||||
|
.TH "CHRONYD" "8" "2024-10-08" "chrony @CHRONY_VERSION@" "System Administration"
|
||||||
|
.ie \n(.g .ds Aq \(aq
|
||||||
|
.el .ds Aq '
|
||||||
|
.ss \n[.ss] 0
|
||||||
|
.nh
|
||||||
|
.ad l
|
||||||
|
.de URL
|
||||||
|
\fI\\$2\fP <\\$1>\\$3
|
||||||
|
..
|
||||||
|
.als MTO URL
|
||||||
|
.if \n[.g] \{\
|
||||||
|
. mso www.tmac
|
||||||
|
. am URL
|
||||||
|
. ad l
|
||||||
|
. .
|
||||||
|
. am MTO
|
||||||
|
. ad l
|
||||||
|
. .
|
||||||
|
. LINKSTYLE blue R < >
|
||||||
|
.\}
|
||||||
|
.SH "NAME"
|
||||||
|
chronyd \- chrony daemon
|
||||||
|
.SH "SYNOPSIS"
|
||||||
|
.sp
|
||||||
|
\fBchronyd\fP [\fIOPTION\fP]... [\fIDIRECTIVE\fP]...
|
||||||
|
.SH "DESCRIPTION"
|
||||||
|
.sp
|
||||||
|
\fBchronyd\fP is a daemon for synchronisation of the system clock. It can
|
||||||
|
synchronise the clock with NTP servers, reference clocks (e.g. a GPS receiver),
|
||||||
|
and manual input using wristwatch and keyboard via \fBchronyc\fP. It can also
|
||||||
|
operate as an NTPv4 (RFC 5905) server and peer to provide a time service to
|
||||||
|
other computers in the network.
|
||||||
|
.sp
|
||||||
|
If no configuration directives are specified on the command line, \fBchronyd\fP
|
||||||
|
will read them from a configuration file. The compiled\-in default location of
|
||||||
|
the file is \fI@SYSCONFDIR@/chrony.conf\fP.
|
||||||
|
.sp
|
||||||
|
Informational messages, warnings, and errors will be logged to syslog.
|
||||||
|
.SH "OPTIONS"
|
||||||
|
.sp
|
||||||
|
\fB\-4\fP
|
||||||
|
.RS 4
|
||||||
|
With this option hostnames will be resolved only to IPv4 addresses and only
|
||||||
|
IPv4 sockets will be created.
|
||||||
|
.RE
|
||||||
|
.sp
|
||||||
|
\fB\-6\fP
|
||||||
|
.RS 4
|
||||||
|
With this option hostnames will be resolved only to IPv6 addresses and only
|
||||||
|
IPv6 sockets will be created.
|
||||||
|
.RE
|
||||||
|
.sp
|
||||||
|
\fB\-f\fP \fIfile\fP
|
||||||
|
.RS 4
|
||||||
|
This option can be used to specify an alternate location for the configuration
|
||||||
|
file. The compiled\-in default value is \fI@SYSCONFDIR@/chrony.conf\fP.
|
||||||
|
.RE
|
||||||
|
.sp
|
||||||
|
\fB\-n\fP
|
||||||
|
.RS 4
|
||||||
|
When run in this mode, the program will not detach itself from the terminal.
|
||||||
|
.RE
|
||||||
|
.sp
|
||||||
|
\fB\-d\fP
|
||||||
|
.RS 4
|
||||||
|
When run in this mode, the program will not detach itself from the terminal,
|
||||||
|
and all messages will be written to the terminal instead of syslog. If
|
||||||
|
\fBchronyd\fP was compiled with enabled support for debugging, this option can be
|
||||||
|
used twice to enable debug messages.
|
||||||
|
.RE
|
||||||
|
.sp
|
||||||
|
\fB\-l\fP \fIfile\fP
|
||||||
|
.RS 4
|
||||||
|
This option enables writing of log messages to a file instead of syslog or the
|
||||||
|
terminal.
|
||||||
|
.RE
|
||||||
|
.sp
|
||||||
|
\fB\-L\fP \fIlevel\fP
|
||||||
|
.RS 4
|
||||||
|
This option specifies the minimum severity level of messages to be written to
|
||||||
|
the log file, syslog, or terminal. The following levels can be specified: \-1
|
||||||
|
(debug, if compiled with enabled support for debugging), 0 (informational), 1
|
||||||
|
(warning), 2 (non\-fatal error), and 3 (fatal error). The default value is 0.
|
||||||
|
.RE
|
||||||
|
.sp
|
||||||
|
\fB\-p\fP
|
||||||
|
.RS 4
|
||||||
|
When run in this mode, \fBchronyd\fP will print the configuration and exit. It will
|
||||||
|
not detach from the terminal. This option can be used to verify the syntax of
|
||||||
|
the configuration and get the whole configuration, even if it is split into
|
||||||
|
multiple files and read by the \fBinclude\fP or \fBconfdir\fP directive.
|
||||||
|
.RE
|
||||||
|
.sp
|
||||||
|
\fB\-q\fP
|
||||||
|
.RS 4
|
||||||
|
When run in this mode, \fBchronyd\fP will set the system clock once and exit. It
|
||||||
|
will not detach from the terminal.
|
||||||
|
.RE
|
||||||
|
.sp
|
||||||
|
\fB\-Q\fP
|
||||||
|
.RS 4
|
||||||
|
This option is similar to the \fB\-q\fP option, except it only prints the offset
|
||||||
|
without making any corrections of the clock and disables server ports to allow
|
||||||
|
\fBchronyd\fP to be started without root privileges, assuming the configuration
|
||||||
|
does not have any directives which would require them (e.g. \fBrefclock\fP,
|
||||||
|
\fBhwtimestamp\fP, \fBrtcfile\fP, etc).
|
||||||
|
.RE
|
||||||
|
.sp
|
||||||
|
\fB\-r\fP
|
||||||
|
.RS 4
|
||||||
|
This option will try to reload and then delete files containing sample
|
||||||
|
histories for each of the servers and reference clocks being used. The
|
||||||
|
files are expected to be in the directory specified by the
|
||||||
|
\fBdumpdir\fP
|
||||||
|
directive in the configuration file. This option is useful if you want to stop
|
||||||
|
and restart \fBchronyd\fP briefly for any reason, e.g. to install a new version.
|
||||||
|
However, it should be used only on systems where the kernel can maintain clock
|
||||||
|
compensation whilst not under \fBchronyd\fP\*(Aqs control (i.e. Linux, FreeBSD, NetBSD,
|
||||||
|
illumos, and macOS 10.13 or later).
|
||||||
|
.RE
|
||||||
|
.sp
|
||||||
|
\fB\-R\fP
|
||||||
|
.RS 4
|
||||||
|
When this option is used, the \fBinitstepslew\fP
|
||||||
|
directive and the \fBmakestep\fP directive used with
|
||||||
|
a positive limit will be ignored. This option is useful when restarting
|
||||||
|
\fBchronyd\fP and can be used in conjunction with the \fB\-r\fP option.
|
||||||
|
.RE
|
||||||
|
.sp
|
||||||
|
\fB\-s\fP
|
||||||
|
.RS 4
|
||||||
|
This option will set the system clock from the computer\(cqs real\-time clock (RTC)
|
||||||
|
or to the last modification time of the file specified by the
|
||||||
|
\fBdriftfile\fP directive. Real\-time clocks are
|
||||||
|
supported only on Linux.
|
||||||
|
.sp
|
||||||
|
If used in conjunction with the \fB\-r\fP flag, \fBchronyd\fP will attempt to preserve
|
||||||
|
the old samples after setting the system clock from the RTC. This can be used
|
||||||
|
to allow \fBchronyd\fP to perform long term averaging of the gain or loss rate
|
||||||
|
across system reboots, and is useful for systems with intermittent access to
|
||||||
|
network that are shut down when not in use. For this to work well, it relies
|
||||||
|
on \fBchronyd\fP having been able to determine accurate statistics for the
|
||||||
|
difference between the RTC and system clock last time the computer was on.
|
||||||
|
.sp
|
||||||
|
If the last modification time of the drift file is later than both the current
|
||||||
|
time and the RTC time, the system time will be set to it to restore the time
|
||||||
|
when \fBchronyd\fP was previously stopped. This is useful on computers that have no
|
||||||
|
RTC or the RTC is broken (e.g. it has no battery).
|
||||||
|
.RE
|
||||||
|
.sp
|
||||||
|
\fB\-t\fP \fItimeout\fP
|
||||||
|
.RS 4
|
||||||
|
This option sets a timeout (in seconds) after which \fBchronyd\fP will exit. If the
|
||||||
|
clock is not synchronised, it will exit with a non\-zero status. This is useful
|
||||||
|
with the \fB\-q\fP or \fB\-Q\fP option to shorten the maximum time waiting for
|
||||||
|
measurements, or with the \fB\-r\fP option to limit the time when \fBchronyd\fP is
|
||||||
|
running, but still allow it to adjust the frequency of the system clock.
|
||||||
|
.RE
|
||||||
|
.sp
|
||||||
|
\fB\-u\fP \fIuser\fP
|
||||||
|
.RS 4
|
||||||
|
This option sets the name of the system user to which \fBchronyd\fP will switch
|
||||||
|
after start in order to drop root privileges. It overrides the
|
||||||
|
\fBuser\fP directive. The compiled\-in default value is
|
||||||
|
\fI@DEFAULT_USER@\fP.
|
||||||
|
.sp
|
||||||
|
On Linux, \fBchronyd\fP needs to be compiled with support for the \fBlibcap\fP library.
|
||||||
|
On macOS, FreeBSD, NetBSD, and illumos \fBchronyd\fP forks into two processes.
|
||||||
|
The child process retains root privileges, but can only perform a very limited
|
||||||
|
range of privileged system calls on behalf of the parent.
|
||||||
|
.RE
|
||||||
|
.sp
|
||||||
|
\fB\-U\fP
|
||||||
|
.RS 4
|
||||||
|
This option disables a check for root privileges to allow \fBchronyd\fP to be
|
||||||
|
started under a non\-root user, assuming the process will have all capabilities
|
||||||
|
(e.g. provided by the service manager) and access to all files, directories,
|
||||||
|
and devices, needed to operate correctly in the specified configuration. Note
|
||||||
|
that different capabilities might be needed with different configurations and
|
||||||
|
different Linux kernel versions. Starting \fBchronyd\fP under a non\-root user is
|
||||||
|
not recommended when the configuration is not known, or at least limited to
|
||||||
|
specific directives.
|
||||||
|
.RE
|
||||||
|
.sp
|
||||||
|
\fB\-F\fP \fIlevel\fP
|
||||||
|
.RS 4
|
||||||
|
This option configures system call filters loaded by \fBchronyd\fP processes if it
|
||||||
|
was compiled with support for the Linux secure computing (seccomp) facility.
|
||||||
|
Three levels are defined: 0, 1, 2. The filters are disabled at level 0. At
|
||||||
|
levels 1 and 2, \fBchronyd\fP will be killed if it makes a system call which is
|
||||||
|
blocked by the filters. The level can be specified as a negative number to
|
||||||
|
trigger the SIGSYS signal instead of SIGKILL, which can be useful for
|
||||||
|
debugging. The default value is 0.
|
||||||
|
.sp
|
||||||
|
At level 1, the filters allow only selected system calls that are normally
|
||||||
|
expected to be made by \fBchronyd\fP. Other system calls are blocked. This level is
|
||||||
|
recommended only if it is known to work on the version of the system where
|
||||||
|
\fBchrony\fP is installed. The filters need to allow also system calls made by
|
||||||
|
libraries that \fBchronyd\fP is using (e.g. libc), but different versions or
|
||||||
|
implementations of the libraries might make different system calls. If the
|
||||||
|
filters are missing a system call, \fBchronyd\fP could be killed even in normal
|
||||||
|
operation.
|
||||||
|
.sp
|
||||||
|
At level 2, the filters block only a small number of specific system calls
|
||||||
|
(e.g. fork and exec). This approach should avoid false positives, but the
|
||||||
|
protection of the system against a compromised \fBchronyd\fP process is much more
|
||||||
|
limited.
|
||||||
|
.sp
|
||||||
|
The filters cannot be enabled with the \fBmailonchange\fP directive.
|
||||||
|
.RE
|
||||||
|
.sp
|
||||||
|
\fB\-P\fP \fIpriority\fP
|
||||||
|
.RS 4
|
||||||
|
On Linux, FreeBSD, NetBSD, and illumos this option will select the SCHED_FIFO
|
||||||
|
real\-time scheduler at the specified priority (which must be between 0 and
|
||||||
|
100). On macOS, this option must have either a value of 0 to disable the thread
|
||||||
|
time constraint policy or 1 for the policy to be enabled. Other systems do not
|
||||||
|
support this option. The default value is 0.
|
||||||
|
.RE
|
||||||
|
.sp
|
||||||
|
\fB\-m\fP
|
||||||
|
.RS 4
|
||||||
|
This option will lock \fBchronyd\fP into RAM so that it will never be paged out.
|
||||||
|
This mode is only supported on Linux, FreeBSD, NetBSD, and illumos.
|
||||||
|
.RE
|
||||||
|
.sp
|
||||||
|
\fB\-x\fP
|
||||||
|
.RS 4
|
||||||
|
This option disables the control of the system clock. \fBchronyd\fP will not try to
|
||||||
|
make any adjustments of the clock. It will assume the clock is free running and
|
||||||
|
still track its offset and frequency relative to the estimated true time. This
|
||||||
|
option allows \fBchronyd\fP to be started without the capability to adjust or set
|
||||||
|
the system clock (e.g. in some containers) to operate as an NTP server.
|
||||||
|
.RE
|
||||||
|
.sp
|
||||||
|
\fB\-v\fP, \fB\-\-version\fP
|
||||||
|
.RS 4
|
||||||
|
With this option \fBchronyd\fP will print version number to the terminal and exit.
|
||||||
|
.RE
|
||||||
|
.sp
|
||||||
|
\fB\-h\fP, \fB\-\-help\fP
|
||||||
|
.RS 4
|
||||||
|
With this option \fBchronyd\fP will print a help message to the terminal and exit.
|
||||||
|
.RE
|
||||||
|
.SH "ENVIRONMENT VARIABLES"
|
||||||
|
.sp
|
||||||
|
\fBLISTEN_FDS\fP
|
||||||
|
.RS 4
|
||||||
|
On Linux systems, the systemd service manager may pass file descriptors for
|
||||||
|
pre\-initialised sockets to \fBchronyd\fP. The service manager allocates and binds
|
||||||
|
the file descriptors, and passes a copy to each spawned instance of the
|
||||||
|
service. This allows for zero\-downtime service restarts as the sockets buffer
|
||||||
|
client requests until the service is able to handle them. The service manager
|
||||||
|
sets the LISTEN_FDS environment variable to the number of passed file
|
||||||
|
descriptors.
|
||||||
|
.RE
|
||||||
|
.SH "FILES"
|
||||||
|
.sp
|
||||||
|
\fI@SYSCONFDIR@/chrony.conf\fP
|
||||||
|
.SH "SEE ALSO"
|
||||||
|
.sp
|
||||||
|
\fBchronyc(1)\fP, \fBchrony.conf(5)\fP
|
||||||
|
.SH "BUGS"
|
||||||
|
.sp
|
||||||
|
For instructions on how to report bugs, please visit
|
||||||
|
.URL "https://chrony\-project.org/" "" "."
|
||||||
|
.SH "AUTHORS"
|
||||||
|
.sp
|
||||||
|
chrony was written by Richard Curnow, Miroslav Lichvar, and others.
|
74
doc/contributing.adoc
Normal file
74
doc/contributing.adoc
Normal file
|
@ -0,0 +1,74 @@
|
||||||
|
// This file is part of chrony
|
||||||
|
//
|
||||||
|
// Copyright (C) Miroslav Lichvar 2024
|
||||||
|
//
|
||||||
|
// This program is free software; you can redistribute it and/or modify
|
||||||
|
// it under the terms of version 2 of the GNU General Public License as
|
||||||
|
// published by the Free Software Foundation.
|
||||||
|
//
|
||||||
|
// This program is distributed in the hope that it will be useful, but
|
||||||
|
// WITHOUT ANY WARRANTY; without even the implied warranty of
|
||||||
|
// MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
|
||||||
|
// General Public License for more details.
|
||||||
|
//
|
||||||
|
// You should have received a copy of the GNU General Public License along
|
||||||
|
// with this program; if not, write to the Free Software Foundation, Inc.,
|
||||||
|
// 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA.
|
||||||
|
|
||||||
|
= Contributing
|
||||||
|
|
||||||
|
== Patches
|
||||||
|
|
||||||
|
The source code of `chrony` is maintained in a git repository at
|
||||||
|
https://gitlab.com/chrony/chrony. Patches can be submitted to the `chrony-dev`
|
||||||
|
mailing list, or as a merge request on gitlab. Before spending a lot of time
|
||||||
|
implementing a new major feature, it is recommended to ask on the mailing list
|
||||||
|
for comments about its design and whether such feature fits the goals of the
|
||||||
|
project.
|
||||||
|
|
||||||
|
Each commit should be a self-contained logical change, which does not break
|
||||||
|
the build or tests. New functionality and fixed bugs should be covered by a new
|
||||||
|
test or an extended existing test in the test suite. The test can be included
|
||||||
|
in the same commit or added as a separate commit. The same rule applies to
|
||||||
|
documentation. All command-line options, configuration directives, and
|
||||||
|
`chronyc` commands should be documented.
|
||||||
|
|
||||||
|
The most important tests can be executed by running `make check` or `make
|
||||||
|
quickcheck`. The unit and system tests run on all supported systems. The system
|
||||||
|
tests require root privileges. The simulation tests run only on Linux and
|
||||||
|
require https://gitlab.com/chrony/clknetsim[clknetsim] to be compiled in the
|
||||||
|
directory containing the tests, but they are executed with a merge request on
|
||||||
|
gitlab.
|
||||||
|
|
||||||
|
The commit message should explain any non-trivial changes, e.g. what problem is
|
||||||
|
the commit solving and how. The commit subject (first line of the message)
|
||||||
|
should be written in an imperative form, prefixed with the component name if it
|
||||||
|
is not a more general change, starting in lower case, and no period at the end.
|
||||||
|
See the git log for examples.
|
||||||
|
|
||||||
|
Simpler code is better. Less code is better. Security is a top priority.
|
||||||
|
|
||||||
|
Assertions should catch only bugs in the `chrony` code. Unexpected values in
|
||||||
|
external input (e.g. anything received from network) must be handled correctly
|
||||||
|
without crashing and memory corruption. Fuzzing support is available at
|
||||||
|
https://gitlab.com/chrony/chrony-fuzz. The fuzzing coverage is checked by the
|
||||||
|
project maintainer before each release.
|
||||||
|
|
||||||
|
The code should mostly be self-documenting. Comments should explain the
|
||||||
|
less obvious things.
|
||||||
|
|
||||||
|
== Coding style
|
||||||
|
|
||||||
|
The code uses two spaces for indentation. No tabs. The line length should
|
||||||
|
normally not exceed 95 characters. Too much indentation indicates the code will
|
||||||
|
not be very readable.
|
||||||
|
|
||||||
|
Function names are in an imperative form. Names of static functions use
|
||||||
|
lowercase characters and underscores. Public functions, structures, typedefs
|
||||||
|
are in CamelCase with a prefix specific to the module (e.g. LCL - local, NCR
|
||||||
|
- NTP core, NKS - NTS-KE server, SST - sourcestats).
|
||||||
|
|
||||||
|
Function names are not followed by space, but keywords of the language (e.g.
|
||||||
|
`if`, `for`, `while`, `sizeof`) are followed by space.
|
||||||
|
|
||||||
|
Have a look at the existing code to get a better idea what is expected.
|
1230
doc/faq.adoc
Normal file
1230
doc/faq.adoc
Normal file
File diff suppressed because it is too large
Load diff
200
doc/installation.adoc
Normal file
200
doc/installation.adoc
Normal file
|
@ -0,0 +1,200 @@
|
||||||
|
// This file is part of chrony
|
||||||
|
//
|
||||||
|
// Copyright (C) Richard P. Curnow 1997-2003
|
||||||
|
// Copyright (C) Miroslav Lichvar 2009-2016
|
||||||
|
//
|
||||||
|
// This program is free software; you can redistribute it and/or modify
|
||||||
|
// it under the terms of version 2 of the GNU General Public License as
|
||||||
|
// published by the Free Software Foundation.
|
||||||
|
//
|
||||||
|
// This program is distributed in the hope that it will be useful, but
|
||||||
|
// WITHOUT ANY WARRANTY; without even the implied warranty of
|
||||||
|
// MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
|
||||||
|
// General Public License for more details.
|
||||||
|
//
|
||||||
|
// You should have received a copy of the GNU General Public License along
|
||||||
|
// with this program; if not, write to the Free Software Foundation, Inc.,
|
||||||
|
// 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA.
|
||||||
|
|
||||||
|
= Installation
|
||||||
|
|
||||||
|
The software is distributed as source code which has to be compiled. The source
|
||||||
|
code is supplied in the form of a gzipped tar file, which unpacks to a
|
||||||
|
subdirectory identifying the name and version of the program.
|
||||||
|
|
||||||
|
A C compiler (e.g. `gcc` or `clang`) and GNU Make are needed to build `chrony`.
|
||||||
|
The following libraries with their development files, and programs, are needed
|
||||||
|
to enable optional features:
|
||||||
|
|
||||||
|
* pkg-config: detection of development libraries
|
||||||
|
* Nettle, GnuTLS, NSS, or LibTomCrypt: secure hash functions (`SECHASH`)
|
||||||
|
* libcap: dropping root privileges on Linux (`DROPROOT`)
|
||||||
|
* libseccomp: system call filter on Linux (`SCFILTER`)
|
||||||
|
* GnuTLS and Nettle: Network Time Security (`NTS`)
|
||||||
|
* Editline: line editing in `chronyc` (`READLINE`)
|
||||||
|
* timepps.h header: PPS reference clock
|
||||||
|
* Asciidoctor: documentation in HTML format
|
||||||
|
* Bash: test suite
|
||||||
|
|
||||||
|
The following programs are needed when building `chrony` from the git
|
||||||
|
repository instead of a released tar file:
|
||||||
|
|
||||||
|
* Asciidoctor: manual pages
|
||||||
|
* Bison: parser for chronyc settime command
|
||||||
|
|
||||||
|
After unpacking the source code, change directory into it, and type
|
||||||
|
|
||||||
|
----
|
||||||
|
./configure
|
||||||
|
----
|
||||||
|
|
||||||
|
This is a shell script that automatically determines the system type. There is
|
||||||
|
an optional parameter `--prefix`, which indicates the directory tree where the
|
||||||
|
software should be installed. For example,
|
||||||
|
|
||||||
|
----
|
||||||
|
./configure --prefix=/opt/free
|
||||||
|
----
|
||||||
|
|
||||||
|
will install the `chronyd` daemon into `/opt/free/sbin` and the `chronyc`
|
||||||
|
control program into `/opt/free/bin`. The default value for the prefix is
|
||||||
|
`/usr/local`.
|
||||||
|
|
||||||
|
The `configure` script assumes you want to use `gcc` as your compiler. If you
|
||||||
|
want to use a different compiler, you can configure this way:
|
||||||
|
|
||||||
|
----
|
||||||
|
CC=cc ./configure --prefix=/opt/free
|
||||||
|
----
|
||||||
|
|
||||||
|
for Bourne-family shells, or
|
||||||
|
|
||||||
|
----
|
||||||
|
setenv CC cc
|
||||||
|
setenv CFLAGS -O
|
||||||
|
./configure --prefix=/opt/free
|
||||||
|
----
|
||||||
|
|
||||||
|
for C-family shells.
|
||||||
|
|
||||||
|
If the software cannot (yet) be built on your system, an error message will be
|
||||||
|
shown. Otherwise, `Makefile` will be generated.
|
||||||
|
|
||||||
|
On Linux, if development files for the libcap library are available, `chronyd`
|
||||||
|
will be built with support for dropping root privileges. On other systems no
|
||||||
|
extra library is needed. The default user which `chronyd` should run as can be
|
||||||
|
specified with the `--with-user` option of the `configure` script.
|
||||||
|
|
||||||
|
If development files for the POSIX threads library are available, `chronyd`
|
||||||
|
will be built with support for asynchronous resolving of hostnames specified in
|
||||||
|
the `server`, `peer`, and `pool` directives. This allows `chronyd` operating as
|
||||||
|
a server to respond to client requests when resolving a hostname. If you don't
|
||||||
|
want to enable the support, specify the `--disable-asyncdns` flag to
|
||||||
|
`configure`.
|
||||||
|
|
||||||
|
If development files for the https://www.lysator.liu.se/~nisse/nettle/[Nettle],
|
||||||
|
https://developer.mozilla.org/en-US/docs/Mozilla/Projects/NSS[NSS], or
|
||||||
|
https://www.libtom.net/LibTomCrypt/[libtomcrypt] library are available,
|
||||||
|
`chronyd` will be built with support for other cryptographic hash functions
|
||||||
|
than MD5, which can be used for NTP authentication with a symmetric key. If you
|
||||||
|
don't want to enable the support, specify the `--disable-sechash` flag to
|
||||||
|
`configure`.
|
||||||
|
|
||||||
|
If development files for the editline library are available,
|
||||||
|
`chronyc` will be built with line editing support. If you don't want this,
|
||||||
|
specify the `--disable-readline` flag to `configure`.
|
||||||
|
|
||||||
|
If a `timepps.h` header is available (e.g. from the
|
||||||
|
http://linuxpps.org[LinuxPPS project]), `chronyd` will be built with PPS API
|
||||||
|
reference clock driver. If the header is installed in a location that isn't
|
||||||
|
normally searched by the compiler, you can add it to the searched locations by
|
||||||
|
setting the `CPPFLAGS` variable to `-I/path/to/timepps`.
|
||||||
|
|
||||||
|
The `--help` option can be specified to `configure` to print all options
|
||||||
|
supported by the script.
|
||||||
|
|
||||||
|
Now type
|
||||||
|
|
||||||
|
----
|
||||||
|
make
|
||||||
|
----
|
||||||
|
|
||||||
|
to build the programs.
|
||||||
|
|
||||||
|
If you want to build the manual in HTML, type
|
||||||
|
|
||||||
|
----
|
||||||
|
make docs
|
||||||
|
----
|
||||||
|
|
||||||
|
Once the programs have been successfully compiled, they need to be installed in
|
||||||
|
their target locations. This step normally needs to be performed by the
|
||||||
|
superuser, and requires the following command to be entered.
|
||||||
|
|
||||||
|
----
|
||||||
|
make install
|
||||||
|
----
|
||||||
|
|
||||||
|
This will install the binaries and man pages.
|
||||||
|
|
||||||
|
To install the HTML version of the manual, enter the command
|
||||||
|
|
||||||
|
----
|
||||||
|
make install-docs
|
||||||
|
----
|
||||||
|
|
||||||
|
Now that the software is successfully installed, the next step is to set up a
|
||||||
|
configuration file. The default location of the file is _/etc/chrony.conf_.
|
||||||
|
Several examples of configuration with comments are included in the examples
|
||||||
|
directory. Suppose you want to use public NTP servers from the pool.ntp.org
|
||||||
|
project as your time reference. A minimal useful configuration file could be
|
||||||
|
|
||||||
|
----
|
||||||
|
pool pool.ntp.org iburst
|
||||||
|
makestep 1.0 3
|
||||||
|
rtcsync
|
||||||
|
----
|
||||||
|
|
||||||
|
Then, `chronyd` can be run. For security reasons, it's recommended to create an
|
||||||
|
unprivileged user for `chronyd` and specify it with the `-u` command-line
|
||||||
|
option or the `user` directive in the configuration file, or set the default
|
||||||
|
user with the `--with-user` configure option before building.
|
||||||
|
|
||||||
|
== Support for system call filtering
|
||||||
|
|
||||||
|
`chronyd` can be built with support for the Linux secure computing (seccomp)
|
||||||
|
facility. This requires development files for the
|
||||||
|
https://github.com/seccomp/libseccomp[libseccomp] library and the
|
||||||
|
`--enable-scfilter` option specified to `configure`. The `-F` option of
|
||||||
|
`chronyd` will enable a system call filter, which should significantly reduce
|
||||||
|
the kernel attack surface and possibly prevent kernel exploits from `chronyd`
|
||||||
|
if it is compromised.
|
||||||
|
|
||||||
|
== Extra options for package builders
|
||||||
|
|
||||||
|
The `configure` and `make` procedures have some extra options that may be
|
||||||
|
useful if you are building a distribution package for `chrony`.
|
||||||
|
|
||||||
|
The `--mandir=DIR` option to `configure` specifies an installation directory
|
||||||
|
for the man pages. This overrides the `man` subdirectory of the argument to the
|
||||||
|
`--prefix` option.
|
||||||
|
|
||||||
|
----
|
||||||
|
./configure --prefix=/usr --mandir=/usr/share/man
|
||||||
|
----
|
||||||
|
|
||||||
|
to set both options together.
|
||||||
|
|
||||||
|
The final option is the `DESTDIR` option to the `make` command. For example,
|
||||||
|
you could use the commands
|
||||||
|
|
||||||
|
----
|
||||||
|
./configure --prefix=/usr --mandir=/usr/share/man
|
||||||
|
make all docs
|
||||||
|
make install DESTDIR=./tmp
|
||||||
|
cd tmp
|
||||||
|
tar cvf - . | gzip -9 > chrony.tar.gz
|
||||||
|
----
|
||||||
|
|
||||||
|
to build a package. When untarred within the root directory, this will install
|
||||||
|
the files to the intended final locations.
|
46
examples/chrony-wait.service
Normal file
46
examples/chrony-wait.service
Normal file
|
@ -0,0 +1,46 @@
|
||||||
|
[Unit]
|
||||||
|
Description=Wait for chrony to synchronize system clock
|
||||||
|
Documentation=man:chronyc(1)
|
||||||
|
After=chronyd.service
|
||||||
|
Requires=chronyd.service
|
||||||
|
Before=time-sync.target
|
||||||
|
Wants=time-sync.target
|
||||||
|
|
||||||
|
[Service]
|
||||||
|
Type=oneshot
|
||||||
|
# Wait for chronyd to update the clock and the remaining
|
||||||
|
# correction to be less than 0.1 seconds
|
||||||
|
ExecStart=/usr/bin/chronyc -h 127.0.0.1,::1 waitsync 0 0.1 0.0 1
|
||||||
|
# Wait for at most 3 minutes
|
||||||
|
TimeoutStartSec=180
|
||||||
|
RemainAfterExit=yes
|
||||||
|
StandardOutput=null
|
||||||
|
|
||||||
|
CapabilityBoundingSet=
|
||||||
|
DevicePolicy=closed
|
||||||
|
DynamicUser=yes
|
||||||
|
IPAddressAllow=localhost
|
||||||
|
IPAddressDeny=any
|
||||||
|
LockPersonality=yes
|
||||||
|
MemoryDenyWriteExecute=yes
|
||||||
|
PrivateDevices=yes
|
||||||
|
PrivateUsers=yes
|
||||||
|
ProtectClock=yes
|
||||||
|
ProtectControlGroups=yes
|
||||||
|
ProtectHome=yes
|
||||||
|
ProtectHostname=yes
|
||||||
|
ProtectKernelLogs=yes
|
||||||
|
ProtectKernelModules=yes
|
||||||
|
ProtectKernelTunables=yes
|
||||||
|
ProtectProc=invisible
|
||||||
|
ProtectSystem=strict
|
||||||
|
RestrictAddressFamilies=AF_INET AF_INET6
|
||||||
|
RestrictNamespaces=yes
|
||||||
|
RestrictRealtime=yes
|
||||||
|
SystemCallArchitectures=native
|
||||||
|
SystemCallFilter=@system-service
|
||||||
|
SystemCallFilter=~@privileged @resources
|
||||||
|
UMask=0777
|
||||||
|
|
||||||
|
[Install]
|
||||||
|
WantedBy=multi-user.target
|
12
examples/chrony.conf.example1
Normal file
12
examples/chrony.conf.example1
Normal file
|
@ -0,0 +1,12 @@
|
||||||
|
# Use public NTP servers from the pool.ntp.org project.
|
||||||
|
pool pool.ntp.org iburst
|
||||||
|
|
||||||
|
# Record the rate at which the system clock gains/losses time.
|
||||||
|
driftfile /var/lib/chrony/drift
|
||||||
|
|
||||||
|
# Allow the system clock to be stepped in the first three updates
|
||||||
|
# if its offset is larger than 1 second.
|
||||||
|
makestep 1.0 3
|
||||||
|
|
||||||
|
# Enable kernel synchronization of the real-time clock (RTC).
|
||||||
|
rtcsync
|
47
examples/chrony.conf.example2
Normal file
47
examples/chrony.conf.example2
Normal file
|
@ -0,0 +1,47 @@
|
||||||
|
# Use public servers from the pool.ntp.org project.
|
||||||
|
# Please consider joining the pool (https://www.pool.ntp.org/join.html).
|
||||||
|
pool pool.ntp.org iburst
|
||||||
|
|
||||||
|
# Record the rate at which the system clock gains/losses time.
|
||||||
|
driftfile /var/lib/chrony/drift
|
||||||
|
|
||||||
|
# Allow the system clock to be stepped in the first three updates
|
||||||
|
# if its offset is larger than 1 second.
|
||||||
|
makestep 1.0 3
|
||||||
|
|
||||||
|
# Enable kernel synchronization of the real-time clock (RTC).
|
||||||
|
rtcsync
|
||||||
|
|
||||||
|
# Enable hardware timestamping on all interfaces that support it.
|
||||||
|
#hwtimestamp *
|
||||||
|
|
||||||
|
# Increase the minimum number of selectable sources required to adjust
|
||||||
|
# the system clock.
|
||||||
|
#minsources 2
|
||||||
|
|
||||||
|
# Allow NTP client access from local network.
|
||||||
|
#allow 192.168.0.0/16
|
||||||
|
|
||||||
|
# Serve time even if not synchronized to a time source.
|
||||||
|
#local stratum 10
|
||||||
|
|
||||||
|
# Require authentication (nts or key option) for all NTP sources.
|
||||||
|
#authselectmode require
|
||||||
|
|
||||||
|
# Specify file containing keys for NTP authentication.
|
||||||
|
#keyfile /etc/chrony.keys
|
||||||
|
|
||||||
|
# Save NTS keys and cookies.
|
||||||
|
ntsdumpdir /var/lib/chrony
|
||||||
|
|
||||||
|
# Insert/delete leap seconds by slewing instead of stepping.
|
||||||
|
#leapsecmode slew
|
||||||
|
|
||||||
|
# Set the TAI-UTC offset of the system clock.
|
||||||
|
#leapseclist /usr/share/zoneinfo/leap-seconds.list
|
||||||
|
|
||||||
|
# Specify directory for log files.
|
||||||
|
logdir /var/log/chrony
|
||||||
|
|
||||||
|
# Select which information is logged.
|
||||||
|
#log measurements statistics tracking
|
334
examples/chrony.conf.example3
Normal file
334
examples/chrony.conf.example3
Normal file
|
@ -0,0 +1,334 @@
|
||||||
|
#######################################################################
|
||||||
|
#
|
||||||
|
# This is an example chrony configuration file. You should copy it to
|
||||||
|
# /etc/chrony.conf after uncommenting and editing the options that you
|
||||||
|
# want to enable. The more obscure options are not included. Refer
|
||||||
|
# to the documentation for these.
|
||||||
|
#
|
||||||
|
#######################################################################
|
||||||
|
### COMMENTS
|
||||||
|
# Any of the following lines are comments (you have a choice of
|
||||||
|
# comment start character):
|
||||||
|
# a comment
|
||||||
|
% a comment
|
||||||
|
! a comment
|
||||||
|
; a comment
|
||||||
|
#
|
||||||
|
# Below, the '!' form is used for lines that you might want to
|
||||||
|
# uncomment and edit to make your own chrony.conf file.
|
||||||
|
#
|
||||||
|
#######################################################################
|
||||||
|
#######################################################################
|
||||||
|
### SPECIFY YOUR NTP SERVERS
|
||||||
|
# Most computers using chrony will send measurement requests to one or
|
||||||
|
# more 'NTP servers'. You will probably find that your Internet Service
|
||||||
|
# Provider or company have one or more NTP servers that you can specify.
|
||||||
|
# Failing that, there are a lot of public NTP servers. There is a list
|
||||||
|
# you can access at http://support.ntp.org/bin/view/Servers/WebHome or
|
||||||
|
# you can use servers from the pool.ntp.org project.
|
||||||
|
|
||||||
|
! server ntp1.example.net iburst
|
||||||
|
! server ntp2.example.net iburst
|
||||||
|
! server ntp3.example.net iburst
|
||||||
|
|
||||||
|
! pool pool.ntp.org iburst
|
||||||
|
|
||||||
|
#######################################################################
|
||||||
|
### AVOIDING POTENTIALLY BOGUS CHANGES TO YOUR CLOCK
|
||||||
|
#
|
||||||
|
# To avoid changes being made to your computer's gain/loss compensation
|
||||||
|
# when the measurement history is too erratic, you might want to enable
|
||||||
|
# one of the following lines. The first seems good with servers on the
|
||||||
|
# Internet, the second seems OK for a LAN environment.
|
||||||
|
|
||||||
|
! maxupdateskew 100
|
||||||
|
! maxupdateskew 5
|
||||||
|
|
||||||
|
# If you want to increase the minimum number of selectable sources
|
||||||
|
# required to update the system clock in order to make the
|
||||||
|
# synchronisation more reliable, uncomment (and edit) the following
|
||||||
|
# line.
|
||||||
|
|
||||||
|
! minsources 2
|
||||||
|
|
||||||
|
# If your computer has a good stable clock (e.g. it is not a virtual
|
||||||
|
# machine), you might also want to reduce the maximum assumed drift
|
||||||
|
# (frequency error) of the clock (the value is specified in ppm).
|
||||||
|
|
||||||
|
! maxdrift 100
|
||||||
|
|
||||||
|
# By default, chronyd allows synchronisation to an unauthenticated NTP
|
||||||
|
# source (i.e. specified without the nts and key options) if it agrees with
|
||||||
|
# a majority of authenticated NTP sources, or if no authenticated source is
|
||||||
|
# specified. If you don't want chronyd to ever synchronise to an
|
||||||
|
# unauthenticated NTP source, uncomment the first from the following lines.
|
||||||
|
# If you don't want to synchronise to an unauthenticated NTP source only
|
||||||
|
# when an authenticated source is specified, uncomment the second line.
|
||||||
|
# If you want chronyd to ignore authentication in the source selection,
|
||||||
|
# uncomment the third line.
|
||||||
|
|
||||||
|
! authselectmode require
|
||||||
|
! authselectmode prefer
|
||||||
|
! authselectmode ignore
|
||||||
|
|
||||||
|
#######################################################################
|
||||||
|
### FILENAMES ETC
|
||||||
|
# Chrony likes to keep information about your computer's clock in files.
|
||||||
|
# The 'driftfile' stores the computer's clock gain/loss rate in parts
|
||||||
|
# per million. When chronyd starts, the system clock can be tuned
|
||||||
|
# immediately so that it doesn't gain or lose any more time. You
|
||||||
|
# generally want this, so it is uncommented.
|
||||||
|
|
||||||
|
driftfile /var/lib/chrony/drift
|
||||||
|
|
||||||
|
# If you want to enable NTP authentication with symmetric keys, you will need
|
||||||
|
# to uncomment the following line and edit the file to set up the keys.
|
||||||
|
|
||||||
|
! keyfile /etc/chrony.keys
|
||||||
|
|
||||||
|
# If you specify an NTP server with the nts option to enable authentication
|
||||||
|
# with the Network Time Security (NTS) mechanism, or enable server NTS with
|
||||||
|
# the ntsservercert and ntsserverkey directives below, the following line will
|
||||||
|
# allow the client/server to save the NTS keys and cookies in order to reduce
|
||||||
|
# the number of key establishments (NTS-KE sessions).
|
||||||
|
|
||||||
|
ntsdumpdir /var/lib/chrony
|
||||||
|
|
||||||
|
# If chronyd is configured to act as an NTP server and you want to enable NTS
|
||||||
|
# for its clients, you will need a TLS certificate and private key. Uncomment
|
||||||
|
# and edit the following lines to specify the locations of the certificate and
|
||||||
|
# key.
|
||||||
|
|
||||||
|
! ntsservercert /etc/.../nts-server.crt
|
||||||
|
! ntsserverkey /etc/.../nts-server.key
|
||||||
|
|
||||||
|
# chronyd can save the measurement history for the servers to files when
|
||||||
|
# it exits. This is useful in 2 situations:
|
||||||
|
#
|
||||||
|
# 1. If you stop chronyd and restart it with the '-r' option (e.g. after
|
||||||
|
# an upgrade), the old measurements will still be relevant when chronyd
|
||||||
|
# is restarted. This will reduce the time needed to get accurate
|
||||||
|
# gain/loss measurements.
|
||||||
|
#
|
||||||
|
# 2. On Linux, if you use the RTC support and start chronyd with
|
||||||
|
# '-r -s' on bootup, measurements from the last boot will still be
|
||||||
|
# useful (the real time clock is used to 'flywheel' chronyd between
|
||||||
|
# boots).
|
||||||
|
#
|
||||||
|
# Uncomment the following line to use this.
|
||||||
|
|
||||||
|
! dumpdir /var/lib/chrony
|
||||||
|
|
||||||
|
# chronyd writes its process ID to a file. If you try to start a second
|
||||||
|
# copy of chronyd, it will detect that the process named in the file is
|
||||||
|
# still running and bail out. If you want to change the path to the PID
|
||||||
|
# file, uncomment this line and edit it. The default path is shown.
|
||||||
|
|
||||||
|
! pidfile /var/run/chrony/chronyd.pid
|
||||||
|
|
||||||
|
# The system timezone database usually comes with a list of leap seconds and
|
||||||
|
# corresponding TAI-UTC offsets. chronyd can use it to set the offset of the
|
||||||
|
# system TAI clock and have an additional source of leap seconds.
|
||||||
|
|
||||||
|
! leapseclist /usr/share/zoneinfo/leap-seconds.list
|
||||||
|
|
||||||
|
#######################################################################
|
||||||
|
### INITIAL CLOCK CORRECTION
|
||||||
|
# This option is useful to quickly correct the clock on start if it's
|
||||||
|
# off by a large amount. The value '1.0' means that if the error is less
|
||||||
|
# than 1 second, it will be gradually removed by speeding up or slowing
|
||||||
|
# down your computer's clock until it is correct. If the error is above
|
||||||
|
# 1 second, an immediate time jump will be applied to correct it. The
|
||||||
|
# value '3' means the step is allowed only in the first three updates of
|
||||||
|
# the clock. Some software can get upset if the system clock jumps
|
||||||
|
# (especially backwards), so be careful!
|
||||||
|
|
||||||
|
! makestep 1.0 3
|
||||||
|
|
||||||
|
#######################################################################
|
||||||
|
### LEAP SECONDS
|
||||||
|
# A leap second is an occasional one-second correction of the UTC
|
||||||
|
# time scale. By default, chronyd tells the kernel to insert/delete
|
||||||
|
# the leap second, which makes a backward/forward step to correct the
|
||||||
|
# clock for it. As with the makestep directive, this jump can upset
|
||||||
|
# some applications. If you prefer chronyd to make a gradual
|
||||||
|
# correction, causing the clock to be off for a longer time, uncomment
|
||||||
|
# the following line.
|
||||||
|
|
||||||
|
! leapsecmode slew
|
||||||
|
|
||||||
|
#######################################################################
|
||||||
|
### LOGGING
|
||||||
|
# If you want to log information about the time measurements chronyd has
|
||||||
|
# gathered, you might want to enable the following lines. You probably
|
||||||
|
# only need this if you really enjoy looking at the logs, you want to
|
||||||
|
# produce some graphs of your system's timekeeping performance, or you
|
||||||
|
# need help in debugging a problem.
|
||||||
|
|
||||||
|
! logdir /var/log/chrony
|
||||||
|
! log measurements statistics tracking
|
||||||
|
|
||||||
|
# If you have real time clock support enabled (see below), you might want
|
||||||
|
# this line instead:
|
||||||
|
|
||||||
|
! log measurements statistics tracking rtc
|
||||||
|
|
||||||
|
#######################################################################
|
||||||
|
### ACTING AS AN NTP SERVER
|
||||||
|
# You might want the computer to be an NTP server for other computers.
|
||||||
|
#
|
||||||
|
# By default, chronyd does not allow any clients to access it. You need
|
||||||
|
# to explicitly enable access using 'allow' and 'deny' directives.
|
||||||
|
#
|
||||||
|
# e.g. to enable client access from the 192.168.*.* class B subnet,
|
||||||
|
|
||||||
|
! allow 192.168/16
|
||||||
|
|
||||||
|
# .. but disallow the 192.168.100.* subnet of that,
|
||||||
|
|
||||||
|
! deny 192.168.100/24
|
||||||
|
|
||||||
|
# You can have as many allow and deny directives as you need. The order
|
||||||
|
# is unimportant.
|
||||||
|
|
||||||
|
# If you want to present your computer's time for others to synchronise
|
||||||
|
# with, even if you don't seem to be synchronised to any NTP servers
|
||||||
|
# yourself, enable the following line. The value 10 may be varied
|
||||||
|
# between 1 and 15. You should avoid small values because you will look
|
||||||
|
# like a real NTP server. The value 10 means that you appear to be 10
|
||||||
|
# NTP 'hops' away from an authoritative source (atomic clock, GPS
|
||||||
|
# receiver, radio clock etc).
|
||||||
|
|
||||||
|
! local stratum 10
|
||||||
|
|
||||||
|
# Normally, chronyd will keep track of how many times each client
|
||||||
|
# machine accesses it. The information can be accessed by the 'clients'
|
||||||
|
# command of chronyc. You can disable this facility by uncommenting the
|
||||||
|
# following line. This will save a bit of memory if you have many
|
||||||
|
# clients and it will also disable support for the interleaved mode.
|
||||||
|
|
||||||
|
! noclientlog
|
||||||
|
|
||||||
|
# The clientlog size is limited to 512KB by default. If you have many
|
||||||
|
# clients, you might want to increase the limit.
|
||||||
|
|
||||||
|
! clientloglimit 4194304
|
||||||
|
|
||||||
|
# By default, chronyd tries to respond to all valid NTP requests from
|
||||||
|
# allowed addresses. If you want to limit the response rate for NTP
|
||||||
|
# clients that are sending requests too frequently, uncomment and edit
|
||||||
|
# the following line.
|
||||||
|
|
||||||
|
! ratelimit interval 3 burst 8
|
||||||
|
|
||||||
|
#######################################################################
|
||||||
|
### REPORTING BIG CLOCK CHANGES
|
||||||
|
# Perhaps you want to know if chronyd suddenly detects any large error
|
||||||
|
# in your computer's clock. This might indicate a fault or a problem
|
||||||
|
# with the server(s) you are using, for example.
|
||||||
|
#
|
||||||
|
# The next option causes a message to be written to syslog when chronyd
|
||||||
|
# has to correct an error above 0.5 seconds (you can use any amount you
|
||||||
|
# like).
|
||||||
|
|
||||||
|
! logchange 0.5
|
||||||
|
|
||||||
|
# The next option will send email to the named person when chronyd has
|
||||||
|
# to correct an error above 0.5 seconds. (If you need to send mail to
|
||||||
|
# several people, you need to set up a mailing list or sendmail alias
|
||||||
|
# for them and use the address of that.)
|
||||||
|
|
||||||
|
! mailonchange wibble@example.net 0.5
|
||||||
|
|
||||||
|
#######################################################################
|
||||||
|
### COMMAND ACCESS
|
||||||
|
# The program chronyc is used to show the current operation of chronyd
|
||||||
|
# and to change parts of its configuration whilst it is running.
|
||||||
|
|
||||||
|
# By default chronyd binds to the loopback interface. Uncomment the
|
||||||
|
# following lines to allow receiving command packets from remote hosts.
|
||||||
|
|
||||||
|
! bindcmdaddress 0.0.0.0
|
||||||
|
! bindcmdaddress ::
|
||||||
|
|
||||||
|
# Normally, chronyd will only allow connections from chronyc on the same
|
||||||
|
# machine as itself. This is for security. If you have a subnet
|
||||||
|
# 192.168.*.* and you want to be able to use chronyc from any machine on
|
||||||
|
# it, you could uncomment the following line. (Edit this to your own
|
||||||
|
# situation.)
|
||||||
|
|
||||||
|
! cmdallow 192.168/16
|
||||||
|
|
||||||
|
# You can add as many 'cmdallow' and 'cmddeny' lines as you like. The
|
||||||
|
# syntax and meaning is the same as for 'allow' and 'deny', except that
|
||||||
|
# 'cmdallow' and 'cmddeny' control access to the chronyd's command port.
|
||||||
|
|
||||||
|
# Rate limiting can be enabled also for command packets. (Note,
|
||||||
|
# commands from localhost are never limited.)
|
||||||
|
|
||||||
|
! cmdratelimit interval -4 burst 16
|
||||||
|
|
||||||
|
#######################################################################
|
||||||
|
### HARDWARE TIMESTAMPING
|
||||||
|
# On Linux, if the network interface controller and its driver support
|
||||||
|
# hardware timestamping, it can significantly improve the accuracy of
|
||||||
|
# synchronisation. It can be enabled on specified interfaces only, or it
|
||||||
|
# can be enabled on all interfaces that support it.
|
||||||
|
|
||||||
|
! hwtimestamp eth0
|
||||||
|
! hwtimestamp *
|
||||||
|
|
||||||
|
#######################################################################
|
||||||
|
### REAL TIME CLOCK
|
||||||
|
# chronyd can characterise the system's real-time clock. This is the
|
||||||
|
# clock that keeps running when the power is turned off, so that the
|
||||||
|
# machine knows the approximate time when it boots again. The error at
|
||||||
|
# a particular epoch and gain/loss rate can be written to a file and
|
||||||
|
# used later by chronyd when it is started with the '-s' option.
|
||||||
|
#
|
||||||
|
# You need to have 'enhanced RTC support' compiled into your Linux
|
||||||
|
# kernel. (Note, these options apply only to Linux.)
|
||||||
|
|
||||||
|
! rtcfile /var/lib/chrony/rtc
|
||||||
|
|
||||||
|
# Your RTC can be set to keep Universal Coordinated Time (UTC) or local
|
||||||
|
# time. (Local time means UTC +/- the effect of your timezone.) If you
|
||||||
|
# use UTC, chronyd will function correctly even if the computer is off
|
||||||
|
# at the epoch when you enter or leave summer time (aka daylight saving
|
||||||
|
# time). However, if you dual boot your system with Microsoft Windows,
|
||||||
|
# that will work better if your RTC maintains local time. You take your
|
||||||
|
# pick!
|
||||||
|
|
||||||
|
! rtconutc
|
||||||
|
|
||||||
|
# By default chronyd assumes that the enhanced RTC device is accessed as
|
||||||
|
# /dev/rtc. If it's accessed somewhere else on your system (e.g. you're
|
||||||
|
# using devfs), uncomment and edit the following line.
|
||||||
|
|
||||||
|
! rtcdevice /dev/misc/rtc
|
||||||
|
|
||||||
|
# Alternatively, if not using the -s option, this directive can be used
|
||||||
|
# to enable a mode in which the RTC is periodically set to the system
|
||||||
|
# time, with no tracking of its drift.
|
||||||
|
|
||||||
|
! rtcsync
|
||||||
|
|
||||||
|
#######################################################################
|
||||||
|
### REAL TIME SCHEDULER
|
||||||
|
# This directive tells chronyd to use the real-time FIFO scheduler with the
|
||||||
|
# specified priority (which must be between 0 and 100). This should result
|
||||||
|
# in reduced latency. You don't need it unless you really have a requirement
|
||||||
|
# for extreme clock stability. Works only on Linux. Note that the "-P"
|
||||||
|
# command-line switch will override this.
|
||||||
|
|
||||||
|
! sched_priority 1
|
||||||
|
|
||||||
|
#######################################################################
|
||||||
|
### LOCKING CHRONYD INTO RAM
|
||||||
|
# This directive tells chronyd to use the mlockall() syscall to lock itself
|
||||||
|
# into RAM so that it will never be paged out. This should result in reduced
|
||||||
|
# latency. You don't need it unless you really have a requirement
|
||||||
|
# for extreme clock stability. Works only on Linux. Note that the "-m"
|
||||||
|
# command-line switch will also enable this feature.
|
||||||
|
|
||||||
|
! lock_all
|
15
examples/chrony.keys.example
Normal file
15
examples/chrony.keys.example
Normal file
|
@ -0,0 +1,15 @@
|
||||||
|
# This is an example chrony keys file. It enables authentication of NTP
|
||||||
|
# packets with symmetric keys when its location is specified by the keyfile
|
||||||
|
# directive in chrony.conf(5). It should be readable only by root and the
|
||||||
|
# user under which chronyd is running.
|
||||||
|
#
|
||||||
|
# Don't use the example keys! It's recommended to generate random keys using
|
||||||
|
# the chronyc keygen command.
|
||||||
|
|
||||||
|
# Examples of valid keys:
|
||||||
|
|
||||||
|
#1 MD5 AVeryLongAndRandomPassword
|
||||||
|
#2 MD5 HEX:12114855C7931009B4049EF3EFC48A139C3F989F
|
||||||
|
#3 SHA1 HEX:B2159C05D6A219673A3B7E896B6DE07F6A440995
|
||||||
|
#4 AES128 HEX:2DA837C4B6573748CA692B8C828E4891
|
||||||
|
#5 AES256 HEX:2666B8099BFF2D5BA20876121788ED24D2BE59111B8FFB562F0F56AE6EC7246E
|
8
examples/chrony.logrotate
Normal file
8
examples/chrony.logrotate
Normal file
|
@ -0,0 +1,8 @@
|
||||||
|
/var/log/chrony/*.log {
|
||||||
|
missingok
|
||||||
|
nocreate
|
||||||
|
sharedscripts
|
||||||
|
postrotate
|
||||||
|
/usr/bin/chronyc cyclelogs > /dev/null 2>&1 || true
|
||||||
|
endscript
|
||||||
|
}
|
49
examples/chrony.nm-dispatcher.dhcp
Normal file
49
examples/chrony.nm-dispatcher.dhcp
Normal file
|
@ -0,0 +1,49 @@
|
||||||
|
#!/bin/sh
|
||||||
|
# This is a NetworkManager dispatcher script for chronyd to update
|
||||||
|
# its NTP sources with servers from DHCP options passed by NetworkManager
|
||||||
|
# in the DHCP4_NTP_SERVERS and DHCP6_DHCP6_NTP_SERVERS environment variables.
|
||||||
|
|
||||||
|
export LC_ALL=C
|
||||||
|
|
||||||
|
interface=$1
|
||||||
|
action=$2
|
||||||
|
|
||||||
|
chronyc=/usr/bin/chronyc
|
||||||
|
server_options=iburst
|
||||||
|
server_dir=/var/run/chrony-dhcp
|
||||||
|
|
||||||
|
dhcp_server_file=$server_dir/$interface.sources
|
||||||
|
dhcp_ntp_servers="$DHCP4_NTP_SERVERS $DHCP6_DHCP6_NTP_SERVERS"
|
||||||
|
|
||||||
|
add_servers_from_dhcp() {
|
||||||
|
rm -f "$dhcp_server_file"
|
||||||
|
for server in $dhcp_ntp_servers; do
|
||||||
|
# Check for invalid characters (from the DHCPv6 NTP FQDN suboption)
|
||||||
|
len1=$(printf '%s' "$server" | wc -c)
|
||||||
|
len2=$(printf '%s' "$server" | tr -d -c 'A-Za-z0-9:.-' | wc -c)
|
||||||
|
if [ "$len1" -ne "$len2" ] || [ "$len2" -lt 1 ] || [ "$len2" -gt 255 ]; then
|
||||||
|
continue
|
||||||
|
fi
|
||||||
|
|
||||||
|
printf 'server %s %s\n' "$server" "$server_options" >> "$dhcp_server_file"
|
||||||
|
done
|
||||||
|
$chronyc reload sources > /dev/null 2>&1 || :
|
||||||
|
}
|
||||||
|
|
||||||
|
clear_servers_from_dhcp() {
|
||||||
|
if [ -f "$dhcp_server_file" ]; then
|
||||||
|
rm -f "$dhcp_server_file"
|
||||||
|
$chronyc reload sources > /dev/null 2>&1 || :
|
||||||
|
fi
|
||||||
|
}
|
||||||
|
|
||||||
|
mkdir -p $server_dir
|
||||||
|
|
||||||
|
case "$action" in
|
||||||
|
up|dhcp4-change|dhcp6-change)
|
||||||
|
add_servers_from_dhcp;;
|
||||||
|
down)
|
||||||
|
clear_servers_from_dhcp;;
|
||||||
|
esac
|
||||||
|
|
||||||
|
exit 0
|
29
examples/chrony.nm-dispatcher.onoffline
Normal file
29
examples/chrony.nm-dispatcher.onoffline
Normal file
|
@ -0,0 +1,29 @@
|
||||||
|
#!/bin/sh
|
||||||
|
# This is a NetworkManager dispatcher / networkd-dispatcher script for
|
||||||
|
# chronyd to set its NTP sources online or offline when a network interface
|
||||||
|
# is configured or removed
|
||||||
|
|
||||||
|
export LC_ALL=C
|
||||||
|
|
||||||
|
chronyc=/usr/bin/chronyc
|
||||||
|
|
||||||
|
# For NetworkManager consider only selected events
|
||||||
|
if [ $# -ge 2 ]; then
|
||||||
|
case "$2" in
|
||||||
|
up|down|connectivity-change)
|
||||||
|
;;
|
||||||
|
dhcp4-change|dhcp6-change)
|
||||||
|
# Actions "up" and "connectivity-change" in some cases do not
|
||||||
|
# guarantee that the interface has a route (e.g. a bond).
|
||||||
|
# dhcp(x)-change handles at least cases that use DHCP.
|
||||||
|
;;
|
||||||
|
*)
|
||||||
|
exit 0;;
|
||||||
|
esac
|
||||||
|
fi
|
||||||
|
|
||||||
|
# Note: for networkd-dispatcher routable.d ~= on and off.d ~= off
|
||||||
|
|
||||||
|
$chronyc onoffline > /dev/null 2>&1
|
||||||
|
|
||||||
|
exit 0
|
58
examples/chronyd-restricted.service
Normal file
58
examples/chronyd-restricted.service
Normal file
|
@ -0,0 +1,58 @@
|
||||||
|
# This is a more restricted version of the chronyd service intended for
|
||||||
|
# minimal NTP/NTS client configurations. The daemon is started without root
|
||||||
|
# privileges and is allowed to write only to its own runtime, state, and log
|
||||||
|
# directories. It cannot bind to privileged ports in order to operate as an
|
||||||
|
# NTP server, or provide monitoring access over IPv4/IPv6. It cannot use
|
||||||
|
# reference clocks, HW timestamping, RTC tracking, and other features.
|
||||||
|
[Unit]
|
||||||
|
Description=NTP client (restricted)
|
||||||
|
Documentation=man:chronyd(8) man:chrony.conf(5)
|
||||||
|
After=chronyd.service ntpdate.service sntp.service ntpd.service
|
||||||
|
Conflicts=chronyd.service ntpd.service systemd-timesyncd.service
|
||||||
|
ConditionCapability=CAP_SYS_TIME
|
||||||
|
|
||||||
|
[Service]
|
||||||
|
Type=forking
|
||||||
|
PIDFile=/run/chrony/chronyd.pid
|
||||||
|
EnvironmentFile=-/etc/sysconfig/chronyd
|
||||||
|
ExecStart=/usr/sbin/chronyd -U $OPTIONS
|
||||||
|
|
||||||
|
User=chrony
|
||||||
|
LogsDirectory=chrony
|
||||||
|
LogsDirectoryMode=0750
|
||||||
|
RuntimeDirectory=chrony
|
||||||
|
RuntimeDirectoryMode=0750
|
||||||
|
RuntimeDirectoryPreserve=restart
|
||||||
|
StateDirectory=chrony
|
||||||
|
StateDirectoryMode=0750
|
||||||
|
|
||||||
|
AmbientCapabilities=CAP_SYS_TIME
|
||||||
|
CapabilityBoundingSet=CAP_SYS_TIME
|
||||||
|
DevicePolicy=closed
|
||||||
|
LockPersonality=yes
|
||||||
|
MemoryDenyWriteExecute=yes
|
||||||
|
NoNewPrivileges=yes
|
||||||
|
PrivateDevices=yes
|
||||||
|
PrivateTmp=yes
|
||||||
|
# This breaks adjtimex()
|
||||||
|
#PrivateUsers=yes
|
||||||
|
ProtectControlGroups=yes
|
||||||
|
ProtectHome=yes
|
||||||
|
ProtectHostname=yes
|
||||||
|
ProtectKernelLogs=yes
|
||||||
|
ProtectKernelModules=yes
|
||||||
|
ProtectKernelTunables=yes
|
||||||
|
ProtectProc=invisible
|
||||||
|
ProtectSystem=strict
|
||||||
|
RemoveIPC=yes
|
||||||
|
RestrictAddressFamilies=AF_INET AF_INET6 AF_UNIX
|
||||||
|
RestrictNamespaces=yes
|
||||||
|
RestrictRealtime=yes
|
||||||
|
RestrictSUIDSGID=yes
|
||||||
|
SystemCallArchitectures=native
|
||||||
|
SystemCallFilter=~@cpu-emulation @debug @module @mount @obsolete @raw-io
|
||||||
|
SystemCallFilter=~@reboot @resources @swap
|
||||||
|
UMask=0077
|
||||||
|
|
||||||
|
[Install]
|
||||||
|
WantedBy=multi-user.target
|
48
examples/chronyd.service
Normal file
48
examples/chronyd.service
Normal file
|
@ -0,0 +1,48 @@
|
||||||
|
[Unit]
|
||||||
|
Description=NTP client/server
|
||||||
|
Documentation=man:chronyd(8) man:chrony.conf(5)
|
||||||
|
After=ntpdate.service sntp.service ntpd.service
|
||||||
|
Conflicts=ntpd.service systemd-timesyncd.service
|
||||||
|
ConditionCapability=CAP_SYS_TIME
|
||||||
|
|
||||||
|
[Service]
|
||||||
|
Type=forking
|
||||||
|
PIDFile=/run/chrony/chronyd.pid
|
||||||
|
EnvironmentFile=-/etc/sysconfig/chronyd
|
||||||
|
ExecStart=/usr/sbin/chronyd $OPTIONS
|
||||||
|
|
||||||
|
CapabilityBoundingSet=~CAP_AUDIT_CONTROL CAP_AUDIT_READ CAP_AUDIT_WRITE
|
||||||
|
CapabilityBoundingSet=~CAP_BLOCK_SUSPEND CAP_KILL CAP_LEASE CAP_LINUX_IMMUTABLE
|
||||||
|
CapabilityBoundingSet=~CAP_MAC_ADMIN CAP_MAC_OVERRIDE CAP_MKNOD CAP_SYS_ADMIN
|
||||||
|
CapabilityBoundingSet=~CAP_SYS_BOOT CAP_SYS_CHROOT CAP_SYS_MODULE CAP_SYS_PACCT
|
||||||
|
CapabilityBoundingSet=~CAP_SYS_PTRACE CAP_SYS_RAWIO CAP_SYS_TTY_CONFIG CAP_WAKE_ALARM
|
||||||
|
DeviceAllow=char-pps rw
|
||||||
|
DeviceAllow=char-ptp rw
|
||||||
|
DeviceAllow=char-rtc rw
|
||||||
|
DevicePolicy=closed
|
||||||
|
LockPersonality=yes
|
||||||
|
MemoryDenyWriteExecute=yes
|
||||||
|
NoNewPrivileges=yes
|
||||||
|
PrivateTmp=yes
|
||||||
|
ProtectControlGroups=yes
|
||||||
|
ProtectHome=yes
|
||||||
|
ProtectHostname=yes
|
||||||
|
ProtectKernelLogs=yes
|
||||||
|
ProtectKernelModules=yes
|
||||||
|
ProtectKernelTunables=yes
|
||||||
|
ProtectProc=invisible
|
||||||
|
ProtectSystem=strict
|
||||||
|
ReadWritePaths=/run /var/lib/chrony -/var/log
|
||||||
|
RestrictAddressFamilies=AF_INET AF_INET6 AF_UNIX
|
||||||
|
RestrictNamespaces=yes
|
||||||
|
RestrictSUIDSGID=yes
|
||||||
|
SystemCallArchitectures=native
|
||||||
|
SystemCallFilter=~@cpu-emulation @debug @module @mount @obsolete @raw-io @reboot @swap
|
||||||
|
|
||||||
|
# Adjust restrictions for /usr/sbin/sendmail (mailonchange directive)
|
||||||
|
NoNewPrivileges=no
|
||||||
|
ReadWritePaths=-/var/spool
|
||||||
|
RestrictAddressFamilies=AF_NETLINK
|
||||||
|
|
||||||
|
[Install]
|
||||||
|
WantedBy=multi-user.target
|
28
getdate.h
Normal file
28
getdate.h
Normal file
|
@ -0,0 +1,28 @@
|
||||||
|
/* Copyright (C) 1995 Free Software Foundation, Inc.
|
||||||
|
|
||||||
|
This program is free software; you can redistribute it and/or modify
|
||||||
|
it under the terms of the GNU General Public License as published by
|
||||||
|
the Free Software Foundation; either version 2, or (at your option)
|
||||||
|
any later version.
|
||||||
|
|
||||||
|
This program is distributed in the hope that it will be useful,
|
||||||
|
but WITHOUT ANY WARRANTY; without even the implied warranty of
|
||||||
|
MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
||||||
|
GNU General Public License for more details.
|
||||||
|
|
||||||
|
You should have received a copy of the GNU General Public License
|
||||||
|
along with this program; if not, write to the Free Software Foundation,
|
||||||
|
Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA. */
|
||||||
|
|
||||||
|
/* Modified from the original to add stdlib.h and string.h */
|
||||||
|
|
||||||
|
#ifndef GOT_GETDATE_H
|
||||||
|
#define GOT_GETDATE_H
|
||||||
|
|
||||||
|
#include <stdlib.h>
|
||||||
|
#include <string.h>
|
||||||
|
#include <time.h>
|
||||||
|
|
||||||
|
time_t get_date (const char *p, const time_t *now);
|
||||||
|
|
||||||
|
#endif /* GOT_GETDATE_H */
|
57
hash.h
Normal file
57
hash.h
Normal file
|
@ -0,0 +1,57 @@
|
||||||
|
/*
|
||||||
|
chronyd/chronyc - Programs for keeping computer clocks accurate.
|
||||||
|
|
||||||
|
**********************************************************************
|
||||||
|
* Copyright (C) Miroslav Lichvar 2012
|
||||||
|
*
|
||||||
|
* This program is free software; you can redistribute it and/or modify
|
||||||
|
* it under the terms of version 2 of the GNU General Public License as
|
||||||
|
* published by the Free Software Foundation.
|
||||||
|
*
|
||||||
|
* This program is distributed in the hope that it will be useful, but
|
||||||
|
* WITHOUT ANY WARRANTY; without even the implied warranty of
|
||||||
|
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
|
||||||
|
* General Public License for more details.
|
||||||
|
*
|
||||||
|
* You should have received a copy of the GNU General Public License along
|
||||||
|
* with this program; if not, write to the Free Software Foundation, Inc.,
|
||||||
|
* 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA.
|
||||||
|
*
|
||||||
|
**********************************************************************
|
||||||
|
|
||||||
|
=======================================================================
|
||||||
|
|
||||||
|
Header file for crypto hashing.
|
||||||
|
|
||||||
|
*/
|
||||||
|
|
||||||
|
#ifndef GOT_HASH_H
|
||||||
|
#define GOT_HASH_H
|
||||||
|
|
||||||
|
/* length of hash values produced by SHA512 */
|
||||||
|
#define MAX_HASH_LENGTH 64
|
||||||
|
|
||||||
|
typedef enum {
|
||||||
|
HSH_INVALID = 0,
|
||||||
|
HSH_MD5 = 1,
|
||||||
|
HSH_SHA1 = 2,
|
||||||
|
HSH_SHA256 = 3,
|
||||||
|
HSH_SHA384 = 4,
|
||||||
|
HSH_SHA512 = 5,
|
||||||
|
HSH_SHA3_224 = 6,
|
||||||
|
HSH_SHA3_256 = 7,
|
||||||
|
HSH_SHA3_384 = 8,
|
||||||
|
HSH_SHA3_512 = 9,
|
||||||
|
HSH_TIGER = 10,
|
||||||
|
HSH_WHIRLPOOL = 11,
|
||||||
|
HSH_MD5_NONCRYPTO = 10000, /* For NTPv4 reference ID */
|
||||||
|
} HSH_Algorithm;
|
||||||
|
|
||||||
|
extern int HSH_GetHashId(HSH_Algorithm algorithm);
|
||||||
|
|
||||||
|
extern int HSH_Hash(int id, const void *in1, int in1_len, const void *in2, int in2_len,
|
||||||
|
unsigned char *out, int out_len);
|
||||||
|
|
||||||
|
extern void HSH_Finalise(void);
|
||||||
|
|
||||||
|
#endif
|
145
hash_gnutls.c
Normal file
145
hash_gnutls.c
Normal file
|
@ -0,0 +1,145 @@
|
||||||
|
/*
|
||||||
|
chronyd/chronyc - Programs for keeping computer clocks accurate.
|
||||||
|
|
||||||
|
**********************************************************************
|
||||||
|
* Copyright (C) Miroslav Lichvar 2021
|
||||||
|
*
|
||||||
|
* This program is free software; you can redistribute it and/or modify
|
||||||
|
* it under the terms of version 2 of the GNU General Public License as
|
||||||
|
* published by the Free Software Foundation.
|
||||||
|
*
|
||||||
|
* This program is distributed in the hope that it will be useful, but
|
||||||
|
* WITHOUT ANY WARRANTY; without even the implied warranty of
|
||||||
|
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
|
||||||
|
* General Public License for more details.
|
||||||
|
*
|
||||||
|
* You should have received a copy of the GNU General Public License along
|
||||||
|
* with this program; if not, write to the Free Software Foundation, Inc.,
|
||||||
|
* 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA.
|
||||||
|
*
|
||||||
|
**********************************************************************
|
||||||
|
|
||||||
|
=======================================================================
|
||||||
|
|
||||||
|
Crypto hashing using the GnuTLS library
|
||||||
|
*/
|
||||||
|
|
||||||
|
#include "config.h"
|
||||||
|
|
||||||
|
#include "sysincl.h"
|
||||||
|
|
||||||
|
#include <gnutls/crypto.h>
|
||||||
|
|
||||||
|
#include "hash.h"
|
||||||
|
#include "logging.h"
|
||||||
|
|
||||||
|
struct hash {
|
||||||
|
const HSH_Algorithm algorithm;
|
||||||
|
const gnutls_digest_algorithm_t type;
|
||||||
|
gnutls_hash_hd_t handle;
|
||||||
|
};
|
||||||
|
|
||||||
|
static struct hash hashes[] = {
|
||||||
|
{ HSH_MD5_NONCRYPTO, GNUTLS_DIG_MD5, NULL },
|
||||||
|
{ HSH_MD5, GNUTLS_DIG_MD5, NULL },
|
||||||
|
{ HSH_SHA1, GNUTLS_DIG_SHA1, NULL },
|
||||||
|
{ HSH_SHA256, GNUTLS_DIG_SHA256, NULL },
|
||||||
|
{ HSH_SHA384, GNUTLS_DIG_SHA384, NULL },
|
||||||
|
{ HSH_SHA512, GNUTLS_DIG_SHA512, NULL },
|
||||||
|
{ HSH_SHA3_224, GNUTLS_DIG_SHA3_224, NULL },
|
||||||
|
{ HSH_SHA3_256, GNUTLS_DIG_SHA3_256, NULL },
|
||||||
|
{ HSH_SHA3_384, GNUTLS_DIG_SHA3_384, NULL },
|
||||||
|
{ HSH_SHA3_512, GNUTLS_DIG_SHA3_512, NULL },
|
||||||
|
{ 0, 0, NULL }
|
||||||
|
};
|
||||||
|
|
||||||
|
static int gnutls_initialised = 0;
|
||||||
|
|
||||||
|
int
|
||||||
|
HSH_GetHashId(HSH_Algorithm algorithm)
|
||||||
|
{
|
||||||
|
int id, r;
|
||||||
|
|
||||||
|
if (!gnutls_initialised) {
|
||||||
|
r = gnutls_global_init();
|
||||||
|
if (r < 0)
|
||||||
|
LOG_FATAL("Could not initialise %s : %s", "gnutls", gnutls_strerror(r));
|
||||||
|
gnutls_initialised = 1;
|
||||||
|
}
|
||||||
|
|
||||||
|
for (id = 0; hashes[id].algorithm != 0; id++) {
|
||||||
|
if (hashes[id].algorithm == algorithm)
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (hashes[id].algorithm == 0)
|
||||||
|
return -1;
|
||||||
|
|
||||||
|
if (hashes[id].handle)
|
||||||
|
return id;
|
||||||
|
|
||||||
|
if (algorithm == HSH_MD5_NONCRYPTO)
|
||||||
|
GNUTLS_FIPS140_SET_LAX_MODE();
|
||||||
|
|
||||||
|
r = gnutls_hash_init(&hashes[id].handle, hashes[id].type);
|
||||||
|
|
||||||
|
if (algorithm == HSH_MD5_NONCRYPTO)
|
||||||
|
GNUTLS_FIPS140_SET_STRICT_MODE();
|
||||||
|
|
||||||
|
if (r < 0) {
|
||||||
|
DEBUG_LOG("Could not initialise %s : %s", "hash", gnutls_strerror(r));
|
||||||
|
hashes[id].handle = NULL;
|
||||||
|
return -1;
|
||||||
|
}
|
||||||
|
|
||||||
|
return id;
|
||||||
|
}
|
||||||
|
|
||||||
|
int
|
||||||
|
HSH_Hash(int id, const void *in1, int in1_len, const void *in2, int in2_len,
|
||||||
|
unsigned char *out, int out_len)
|
||||||
|
{
|
||||||
|
unsigned char buf[MAX_HASH_LENGTH];
|
||||||
|
gnutls_hash_hd_t handle;
|
||||||
|
int hash_len;
|
||||||
|
|
||||||
|
if (in1_len < 0 || in2_len < 0 || out_len < 0)
|
||||||
|
return 0;
|
||||||
|
|
||||||
|
handle = hashes[id].handle;
|
||||||
|
hash_len = gnutls_hash_get_len(hashes[id].type);
|
||||||
|
|
||||||
|
if (out_len > hash_len)
|
||||||
|
out_len = hash_len;
|
||||||
|
|
||||||
|
if (hash_len > sizeof (buf))
|
||||||
|
return 0;
|
||||||
|
|
||||||
|
if (gnutls_hash(handle, in1, in1_len) < 0 ||
|
||||||
|
(in2 && gnutls_hash(handle, in2, in2_len) < 0)) {
|
||||||
|
/* Reset the state */
|
||||||
|
gnutls_hash_output(handle, buf);
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
gnutls_hash_output(handle, buf);
|
||||||
|
memcpy(out, buf, out_len);
|
||||||
|
|
||||||
|
return out_len;
|
||||||
|
}
|
||||||
|
|
||||||
|
void
|
||||||
|
HSH_Finalise(void)
|
||||||
|
{
|
||||||
|
int i;
|
||||||
|
|
||||||
|
if (!gnutls_initialised)
|
||||||
|
return;
|
||||||
|
|
||||||
|
for (i = 0; hashes[i].algorithm != 0; i++) {
|
||||||
|
if (hashes[i].handle)
|
||||||
|
gnutls_hash_deinit(hashes[i].handle, NULL);
|
||||||
|
}
|
||||||
|
|
||||||
|
gnutls_global_deinit();
|
||||||
|
}
|
71
hash_intmd5.c
Normal file
71
hash_intmd5.c
Normal file
|
@ -0,0 +1,71 @@
|
||||||
|
/*
|
||||||
|
chronyd/chronyc - Programs for keeping computer clocks accurate.
|
||||||
|
|
||||||
|
**********************************************************************
|
||||||
|
* Copyright (C) Miroslav Lichvar 2012
|
||||||
|
*
|
||||||
|
* This program is free software; you can redistribute it and/or modify
|
||||||
|
* it under the terms of version 2 of the GNU General Public License as
|
||||||
|
* published by the Free Software Foundation.
|
||||||
|
*
|
||||||
|
* This program is distributed in the hope that it will be useful, but
|
||||||
|
* WITHOUT ANY WARRANTY; without even the implied warranty of
|
||||||
|
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
|
||||||
|
* General Public License for more details.
|
||||||
|
*
|
||||||
|
* You should have received a copy of the GNU General Public License along
|
||||||
|
* with this program; if not, write to the Free Software Foundation, Inc.,
|
||||||
|
* 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA.
|
||||||
|
*
|
||||||
|
**********************************************************************
|
||||||
|
|
||||||
|
=======================================================================
|
||||||
|
|
||||||
|
Routines implementing crypto hashing using internal MD5 implementation.
|
||||||
|
|
||||||
|
*/
|
||||||
|
|
||||||
|
#include "config.h"
|
||||||
|
#include "sysincl.h"
|
||||||
|
#include "hash.h"
|
||||||
|
#include "memory.h"
|
||||||
|
#include "util.h"
|
||||||
|
|
||||||
|
#include "md5.c"
|
||||||
|
|
||||||
|
static MD5_CTX ctx;
|
||||||
|
|
||||||
|
int
|
||||||
|
HSH_GetHashId(HSH_Algorithm algorithm)
|
||||||
|
{
|
||||||
|
/* only MD5 is supported */
|
||||||
|
if (algorithm != HSH_MD5 && algorithm != HSH_MD5_NONCRYPTO)
|
||||||
|
return -1;
|
||||||
|
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
int
|
||||||
|
HSH_Hash(int id, const void *in1, int in1_len, const void *in2, int in2_len,
|
||||||
|
unsigned char *out, int out_len)
|
||||||
|
{
|
||||||
|
if (in1_len < 0 || in2_len < 0 || out_len < 0)
|
||||||
|
return 0;
|
||||||
|
|
||||||
|
MD5Init(&ctx);
|
||||||
|
MD5Update(&ctx, in1, in1_len);
|
||||||
|
if (in2)
|
||||||
|
MD5Update(&ctx, in2, in2_len);
|
||||||
|
MD5Final(&ctx);
|
||||||
|
|
||||||
|
out_len = MIN(out_len, 16);
|
||||||
|
|
||||||
|
memcpy(out, ctx.digest, out_len);
|
||||||
|
|
||||||
|
return out_len;
|
||||||
|
}
|
||||||
|
|
||||||
|
void
|
||||||
|
HSH_Finalise(void)
|
||||||
|
{
|
||||||
|
}
|
124
hash_nettle.c
Normal file
124
hash_nettle.c
Normal file
|
@ -0,0 +1,124 @@
|
||||||
|
/*
|
||||||
|
chronyd/chronyc - Programs for keeping computer clocks accurate.
|
||||||
|
|
||||||
|
**********************************************************************
|
||||||
|
* Copyright (C) Miroslav Lichvar 2018
|
||||||
|
*
|
||||||
|
* This program is free software; you can redistribute it and/or modify
|
||||||
|
* it under the terms of version 2 of the GNU General Public License as
|
||||||
|
* published by the Free Software Foundation.
|
||||||
|
*
|
||||||
|
* This program is distributed in the hope that it will be useful, but
|
||||||
|
* WITHOUT ANY WARRANTY; without even the implied warranty of
|
||||||
|
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
|
||||||
|
* General Public License for more details.
|
||||||
|
*
|
||||||
|
* You should have received a copy of the GNU General Public License along
|
||||||
|
* with this program; if not, write to the Free Software Foundation, Inc.,
|
||||||
|
* 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA.
|
||||||
|
*
|
||||||
|
**********************************************************************
|
||||||
|
|
||||||
|
=======================================================================
|
||||||
|
|
||||||
|
Routines implementing crypto hashing using the nettle library.
|
||||||
|
|
||||||
|
*/
|
||||||
|
|
||||||
|
#include "config.h"
|
||||||
|
|
||||||
|
#include "sysincl.h"
|
||||||
|
|
||||||
|
#include <nettle/nettle-meta.h>
|
||||||
|
|
||||||
|
#include "hash.h"
|
||||||
|
#include "memory.h"
|
||||||
|
|
||||||
|
struct hash {
|
||||||
|
const HSH_Algorithm algorithm;
|
||||||
|
const char *int_name;
|
||||||
|
const struct nettle_hash *nettle_hash;
|
||||||
|
void *context;
|
||||||
|
};
|
||||||
|
|
||||||
|
static struct hash hashes[] = {
|
||||||
|
{ HSH_MD5, "md5", NULL, NULL },
|
||||||
|
{ HSH_SHA1, "sha1", NULL, NULL },
|
||||||
|
{ HSH_SHA256, "sha256", NULL, NULL },
|
||||||
|
{ HSH_SHA384, "sha384", NULL, NULL },
|
||||||
|
{ HSH_SHA512, "sha512", NULL, NULL },
|
||||||
|
{ HSH_SHA3_224, "sha3_224", NULL, NULL },
|
||||||
|
{ HSH_SHA3_256, "sha3_256", NULL, NULL },
|
||||||
|
{ HSH_SHA3_384, "sha3_384", NULL, NULL },
|
||||||
|
{ HSH_SHA3_512, "sha3_512", NULL, NULL },
|
||||||
|
{ 0, NULL, NULL, NULL }
|
||||||
|
};
|
||||||
|
|
||||||
|
int
|
||||||
|
HSH_GetHashId(HSH_Algorithm algorithm)
|
||||||
|
{
|
||||||
|
int id, nid;
|
||||||
|
|
||||||
|
if (algorithm == HSH_MD5_NONCRYPTO)
|
||||||
|
algorithm = HSH_MD5;
|
||||||
|
|
||||||
|
for (id = 0; hashes[id].algorithm != 0; id++) {
|
||||||
|
if (hashes[id].algorithm == algorithm)
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (hashes[id].algorithm == 0)
|
||||||
|
return -1;
|
||||||
|
|
||||||
|
if (hashes[id].context)
|
||||||
|
return id;
|
||||||
|
|
||||||
|
for (nid = 0; nettle_hashes[nid]; nid++) {
|
||||||
|
if (!strcmp(hashes[id].int_name, nettle_hashes[nid]->name))
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (!nettle_hashes[nid] || !nettle_hashes[nid]->context_size || !nettle_hashes[nid]->init)
|
||||||
|
return -1;
|
||||||
|
|
||||||
|
hashes[id].nettle_hash = nettle_hashes[nid];
|
||||||
|
hashes[id].context = Malloc(hashes[id].nettle_hash->context_size);
|
||||||
|
|
||||||
|
return id;
|
||||||
|
}
|
||||||
|
|
||||||
|
int
|
||||||
|
HSH_Hash(int id, const void *in1, int in1_len, const void *in2, int in2_len,
|
||||||
|
unsigned char *out, int out_len)
|
||||||
|
{
|
||||||
|
const struct nettle_hash *hash;
|
||||||
|
void *context;
|
||||||
|
|
||||||
|
if (in1_len < 0 || in2_len < 0 || out_len < 0)
|
||||||
|
return 0;
|
||||||
|
|
||||||
|
hash = hashes[id].nettle_hash;
|
||||||
|
context = hashes[id].context;
|
||||||
|
|
||||||
|
if (out_len > hash->digest_size)
|
||||||
|
out_len = hash->digest_size;
|
||||||
|
|
||||||
|
hash->init(context);
|
||||||
|
hash->update(context, in1_len, in1);
|
||||||
|
if (in2)
|
||||||
|
hash->update(context, in2_len, in2);
|
||||||
|
hash->digest(context, out_len, out);
|
||||||
|
|
||||||
|
return out_len;
|
||||||
|
}
|
||||||
|
|
||||||
|
void
|
||||||
|
HSH_Finalise(void)
|
||||||
|
{
|
||||||
|
int i;
|
||||||
|
|
||||||
|
for (i = 0; hashes[i].algorithm != 0; i++) {
|
||||||
|
if (hashes[i].context)
|
||||||
|
Free(hashes[i].context);
|
||||||
|
}
|
||||||
|
}
|
114
hash_nss.c
Normal file
114
hash_nss.c
Normal file
|
@ -0,0 +1,114 @@
|
||||||
|
/*
|
||||||
|
chronyd/chronyc - Programs for keeping computer clocks accurate.
|
||||||
|
|
||||||
|
**********************************************************************
|
||||||
|
* Copyright (C) Miroslav Lichvar 2012
|
||||||
|
*
|
||||||
|
* This program is free software; you can redistribute it and/or modify
|
||||||
|
* it under the terms of version 2 of the GNU General Public License as
|
||||||
|
* published by the Free Software Foundation.
|
||||||
|
*
|
||||||
|
* This program is distributed in the hope that it will be useful, but
|
||||||
|
* WITHOUT ANY WARRANTY; without even the implied warranty of
|
||||||
|
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
|
||||||
|
* General Public License for more details.
|
||||||
|
*
|
||||||
|
* You should have received a copy of the GNU General Public License along
|
||||||
|
* with this program; if not, write to the Free Software Foundation, Inc.,
|
||||||
|
* 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA.
|
||||||
|
*
|
||||||
|
**********************************************************************
|
||||||
|
|
||||||
|
=======================================================================
|
||||||
|
|
||||||
|
Routines implementing crypto hashing using NSSLOWHASH API of the NSS library.
|
||||||
|
|
||||||
|
*/
|
||||||
|
|
||||||
|
#include "config.h"
|
||||||
|
|
||||||
|
#include <nss.h>
|
||||||
|
#include <hasht.h>
|
||||||
|
#include <nsslowhash.h>
|
||||||
|
|
||||||
|
#include "hash.h"
|
||||||
|
#include "util.h"
|
||||||
|
|
||||||
|
static NSSLOWInitContext *ictx;
|
||||||
|
|
||||||
|
struct hash {
|
||||||
|
HASH_HashType type;
|
||||||
|
HSH_Algorithm algorithm;
|
||||||
|
NSSLOWHASHContext *context;
|
||||||
|
};
|
||||||
|
|
||||||
|
static struct hash hashes[] = {
|
||||||
|
{ HASH_AlgMD5, HSH_MD5, NULL },
|
||||||
|
{ HASH_AlgSHA1, HSH_SHA1, NULL },
|
||||||
|
{ HASH_AlgSHA256, HSH_SHA256, NULL },
|
||||||
|
{ HASH_AlgSHA384, HSH_SHA384, NULL },
|
||||||
|
{ HASH_AlgSHA512, HSH_SHA512, NULL },
|
||||||
|
{ 0, 0, NULL }
|
||||||
|
};
|
||||||
|
|
||||||
|
int
|
||||||
|
HSH_GetHashId(HSH_Algorithm algorithm)
|
||||||
|
{
|
||||||
|
int i;
|
||||||
|
|
||||||
|
if (algorithm == HSH_MD5_NONCRYPTO)
|
||||||
|
algorithm = HSH_MD5;
|
||||||
|
|
||||||
|
for (i = 0; hashes[i].algorithm != 0; i++) {
|
||||||
|
if (hashes[i].algorithm == algorithm)
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (hashes[i].algorithm == 0)
|
||||||
|
return -1; /* not found */
|
||||||
|
|
||||||
|
if (!ictx && !(ictx = NSSLOW_Init()))
|
||||||
|
return -1; /* couldn't init NSS */
|
||||||
|
|
||||||
|
if (!hashes[i].context &&
|
||||||
|
!(hashes[i].context = NSSLOWHASH_NewContext(ictx, hashes[i].type)))
|
||||||
|
return -1; /* couldn't init hash */
|
||||||
|
|
||||||
|
return i;
|
||||||
|
}
|
||||||
|
|
||||||
|
int
|
||||||
|
HSH_Hash(int id, const void *in1, int in1_len, const void *in2, int in2_len,
|
||||||
|
unsigned char *out, int out_len)
|
||||||
|
{
|
||||||
|
unsigned char buf[MAX_HASH_LENGTH];
|
||||||
|
unsigned int ret = 0;
|
||||||
|
|
||||||
|
if (in1_len < 0 || in2_len < 0 || out_len < 0)
|
||||||
|
return 0;
|
||||||
|
|
||||||
|
NSSLOWHASH_Begin(hashes[id].context);
|
||||||
|
NSSLOWHASH_Update(hashes[id].context, in1, in1_len);
|
||||||
|
if (in2)
|
||||||
|
NSSLOWHASH_Update(hashes[id].context, in2, in2_len);
|
||||||
|
NSSLOWHASH_End(hashes[id].context, buf, &ret, sizeof (buf));
|
||||||
|
|
||||||
|
ret = MIN(ret, out_len);
|
||||||
|
memcpy(out, buf, ret);
|
||||||
|
|
||||||
|
return ret;
|
||||||
|
}
|
||||||
|
|
||||||
|
void
|
||||||
|
HSH_Finalise(void)
|
||||||
|
{
|
||||||
|
int i;
|
||||||
|
|
||||||
|
for (i = 0; hashes[i].algorithm != 0; i++) {
|
||||||
|
if (hashes[i].context)
|
||||||
|
NSSLOWHASH_Destroy(hashes[i].context);
|
||||||
|
}
|
||||||
|
|
||||||
|
if (ictx)
|
||||||
|
NSSLOW_Shutdown(ictx);
|
||||||
|
}
|
126
hash_tomcrypt.c
Normal file
126
hash_tomcrypt.c
Normal file
|
@ -0,0 +1,126 @@
|
||||||
|
/*
|
||||||
|
chronyd/chronyc - Programs for keeping computer clocks accurate.
|
||||||
|
|
||||||
|
**********************************************************************
|
||||||
|
* Copyright (C) Miroslav Lichvar 2012, 2018
|
||||||
|
*
|
||||||
|
* This program is free software; you can redistribute it and/or modify
|
||||||
|
* it under the terms of version 2 of the GNU General Public License as
|
||||||
|
* published by the Free Software Foundation.
|
||||||
|
*
|
||||||
|
* This program is distributed in the hope that it will be useful, but
|
||||||
|
* WITHOUT ANY WARRANTY; without even the implied warranty of
|
||||||
|
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
|
||||||
|
* General Public License for more details.
|
||||||
|
*
|
||||||
|
* You should have received a copy of the GNU General Public License along
|
||||||
|
* with this program; if not, write to the Free Software Foundation, Inc.,
|
||||||
|
* 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA.
|
||||||
|
*
|
||||||
|
**********************************************************************
|
||||||
|
|
||||||
|
=======================================================================
|
||||||
|
|
||||||
|
Routines implementing crypto hashing using tomcrypt library.
|
||||||
|
|
||||||
|
*/
|
||||||
|
|
||||||
|
#include <tomcrypt.h>
|
||||||
|
|
||||||
|
#include "config.h"
|
||||||
|
#include "hash.h"
|
||||||
|
#include "util.h"
|
||||||
|
|
||||||
|
struct hash {
|
||||||
|
HSH_Algorithm algorithm;
|
||||||
|
const char *int_name;
|
||||||
|
const struct ltc_hash_descriptor *desc;
|
||||||
|
};
|
||||||
|
|
||||||
|
static const struct hash hashes[] = {
|
||||||
|
{ HSH_MD5, "md5", &md5_desc },
|
||||||
|
#ifdef LTC_SHA1
|
||||||
|
{ HSH_SHA1, "sha1", &sha1_desc },
|
||||||
|
#endif
|
||||||
|
#ifdef LTC_SHA256
|
||||||
|
{ HSH_SHA256, "sha256", &sha256_desc },
|
||||||
|
#endif
|
||||||
|
#ifdef LTC_SHA384
|
||||||
|
{ HSH_SHA384, "sha384", &sha384_desc },
|
||||||
|
#endif
|
||||||
|
#ifdef LTC_SHA512
|
||||||
|
{ HSH_SHA512, "sha512", &sha512_desc },
|
||||||
|
#endif
|
||||||
|
#ifdef LTC_SHA3
|
||||||
|
{ HSH_SHA3_224, "sha3-224", &sha3_224_desc },
|
||||||
|
{ HSH_SHA3_256, "sha3-256", &sha3_256_desc },
|
||||||
|
{ HSH_SHA3_384, "sha3-384", &sha3_384_desc },
|
||||||
|
{ HSH_SHA3_512, "sha3-512", &sha3_512_desc },
|
||||||
|
#endif
|
||||||
|
#ifdef LTC_TIGER
|
||||||
|
{ HSH_TIGER, "tiger", &tiger_desc },
|
||||||
|
#endif
|
||||||
|
#ifdef LTC_WHIRLPOOL
|
||||||
|
{ HSH_WHIRLPOOL, "whirlpool", &whirlpool_desc },
|
||||||
|
#endif
|
||||||
|
{ 0, NULL, NULL }
|
||||||
|
};
|
||||||
|
|
||||||
|
int
|
||||||
|
HSH_GetHashId(HSH_Algorithm algorithm)
|
||||||
|
{
|
||||||
|
int i, h;
|
||||||
|
|
||||||
|
if (algorithm == HSH_MD5_NONCRYPTO)
|
||||||
|
algorithm = HSH_MD5;
|
||||||
|
|
||||||
|
for (i = 0; hashes[i].algorithm != 0; i++) {
|
||||||
|
if (hashes[i].algorithm == algorithm)
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (hashes[i].algorithm == 0)
|
||||||
|
return -1; /* not found */
|
||||||
|
|
||||||
|
h = find_hash(hashes[i].int_name);
|
||||||
|
if (h >= 0)
|
||||||
|
return h; /* already registered */
|
||||||
|
|
||||||
|
/* register and try again */
|
||||||
|
register_hash(hashes[i].desc);
|
||||||
|
|
||||||
|
return find_hash(hashes[i].int_name);
|
||||||
|
}
|
||||||
|
|
||||||
|
int
|
||||||
|
HSH_Hash(int id, const void *in1, int in1_len, const void *in2, int in2_len,
|
||||||
|
unsigned char *out, int out_len)
|
||||||
|
{
|
||||||
|
unsigned char buf[MAX_HASH_LENGTH];
|
||||||
|
unsigned long len;
|
||||||
|
int r;
|
||||||
|
|
||||||
|
if (in1_len < 0 || in2_len < 0 || out_len < 0)
|
||||||
|
return 0;
|
||||||
|
|
||||||
|
len = sizeof (buf);
|
||||||
|
if (in2)
|
||||||
|
r = hash_memory_multi(id, buf, &len,
|
||||||
|
in1, (unsigned long)in1_len,
|
||||||
|
in2, (unsigned long)in2_len, NULL, 0);
|
||||||
|
else
|
||||||
|
r = hash_memory(id, in1, in1_len, buf, &len);
|
||||||
|
|
||||||
|
if (r != CRYPT_OK)
|
||||||
|
return 0;
|
||||||
|
|
||||||
|
len = MIN(len, out_len);
|
||||||
|
memcpy(out, buf, len);
|
||||||
|
|
||||||
|
return len;
|
||||||
|
}
|
||||||
|
|
||||||
|
void
|
||||||
|
HSH_Finalise(void)
|
||||||
|
{
|
||||||
|
}
|
334
hwclock.c
Normal file
334
hwclock.c
Normal file
|
@ -0,0 +1,334 @@
|
||||||
|
/*
|
||||||
|
chronyd/chronyc - Programs for keeping computer clocks accurate.
|
||||||
|
|
||||||
|
**********************************************************************
|
||||||
|
* Copyright (C) Miroslav Lichvar 2016-2018, 2022
|
||||||
|
*
|
||||||
|
* This program is free software; you can redistribute it and/or modify
|
||||||
|
* it under the terms of version 2 of the GNU General Public License as
|
||||||
|
* published by the Free Software Foundation.
|
||||||
|
*
|
||||||
|
* This program is distributed in the hope that it will be useful, but
|
||||||
|
* WITHOUT ANY WARRANTY; without even the implied warranty of
|
||||||
|
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
|
||||||
|
* General Public License for more details.
|
||||||
|
*
|
||||||
|
* You should have received a copy of the GNU General Public License along
|
||||||
|
* with this program; if not, write to the Free Software Foundation, Inc.,
|
||||||
|
* 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA.
|
||||||
|
*
|
||||||
|
**********************************************************************
|
||||||
|
|
||||||
|
=======================================================================
|
||||||
|
|
||||||
|
Tracking of hardware clocks (e.g. RTC, PHC)
|
||||||
|
*/
|
||||||
|
|
||||||
|
#include "config.h"
|
||||||
|
|
||||||
|
#include "sysincl.h"
|
||||||
|
|
||||||
|
#include "array.h"
|
||||||
|
#include "hwclock.h"
|
||||||
|
#include "local.h"
|
||||||
|
#include "logging.h"
|
||||||
|
#include "memory.h"
|
||||||
|
#include "quantiles.h"
|
||||||
|
#include "regress.h"
|
||||||
|
#include "util.h"
|
||||||
|
|
||||||
|
/* Minimum and maximum number of samples per clock */
|
||||||
|
#define MIN_SAMPLES 2
|
||||||
|
#define MAX_SAMPLES 64
|
||||||
|
|
||||||
|
/* Maximum acceptable frequency offset of the clock */
|
||||||
|
#define MAX_FREQ_OFFSET (2.0 / 3.0)
|
||||||
|
|
||||||
|
/* Quantiles for filtering readings by delay */
|
||||||
|
#define DELAY_QUANT_MIN_K 1
|
||||||
|
#define DELAY_QUANT_MAX_K 2
|
||||||
|
#define DELAY_QUANT_Q 10
|
||||||
|
#define DELAY_QUANT_REPEAT 7
|
||||||
|
#define DELAY_QUANT_MIN_STEP 1.0e-9
|
||||||
|
|
||||||
|
struct HCL_Instance_Record {
|
||||||
|
/* HW and local reference timestamp */
|
||||||
|
struct timespec hw_ref;
|
||||||
|
struct timespec local_ref;
|
||||||
|
|
||||||
|
/* Samples stored as intervals (uncorrected for frequency error)
|
||||||
|
relative to local_ref and hw_ref */
|
||||||
|
double *x_data;
|
||||||
|
double *y_data;
|
||||||
|
|
||||||
|
/* Minimum, maximum and current number of samples */
|
||||||
|
int min_samples;
|
||||||
|
int max_samples;
|
||||||
|
int n_samples;
|
||||||
|
|
||||||
|
/* Maximum error of the last sample */
|
||||||
|
double last_err;
|
||||||
|
|
||||||
|
/* Minimum interval between samples */
|
||||||
|
double min_separation;
|
||||||
|
|
||||||
|
/* Expected precision of readings */
|
||||||
|
double precision;
|
||||||
|
|
||||||
|
/* Flag indicating the offset and frequency values are valid */
|
||||||
|
int valid_coefs;
|
||||||
|
|
||||||
|
/* Estimated offset and frequency of HW clock relative to local clock */
|
||||||
|
double offset;
|
||||||
|
double frequency;
|
||||||
|
|
||||||
|
/* Estimated quantiles of reading delay */
|
||||||
|
QNT_Instance delay_quants;
|
||||||
|
};
|
||||||
|
|
||||||
|
/* ================================================== */
|
||||||
|
|
||||||
|
static void
|
||||||
|
handle_slew(struct timespec *raw, struct timespec *cooked, double dfreq,
|
||||||
|
double doffset, LCL_ChangeType change_type, void *anything)
|
||||||
|
{
|
||||||
|
HCL_Instance clock;
|
||||||
|
double delta;
|
||||||
|
|
||||||
|
clock = anything;
|
||||||
|
|
||||||
|
if (clock->n_samples)
|
||||||
|
UTI_AdjustTimespec(&clock->local_ref, cooked, &clock->local_ref, &delta, dfreq, doffset);
|
||||||
|
if (clock->valid_coefs)
|
||||||
|
clock->frequency /= 1.0 - dfreq;
|
||||||
|
}
|
||||||
|
|
||||||
|
/* ================================================== */
|
||||||
|
|
||||||
|
HCL_Instance
|
||||||
|
HCL_CreateInstance(int min_samples, int max_samples, double min_separation, double precision)
|
||||||
|
{
|
||||||
|
HCL_Instance clock;
|
||||||
|
|
||||||
|
min_samples = CLAMP(MIN_SAMPLES, min_samples, MAX_SAMPLES);
|
||||||
|
max_samples = CLAMP(MIN_SAMPLES, max_samples, MAX_SAMPLES);
|
||||||
|
max_samples = MAX(min_samples, max_samples);
|
||||||
|
|
||||||
|
clock = MallocNew(struct HCL_Instance_Record);
|
||||||
|
clock->x_data = MallocArray(double, max_samples);
|
||||||
|
clock->y_data = MallocArray(double, max_samples);
|
||||||
|
clock->x_data[max_samples - 1] = 0.0;
|
||||||
|
clock->y_data[max_samples - 1] = 0.0;
|
||||||
|
clock->min_samples = min_samples;
|
||||||
|
clock->max_samples = max_samples;
|
||||||
|
clock->n_samples = 0;
|
||||||
|
clock->valid_coefs = 0;
|
||||||
|
clock->min_separation = min_separation;
|
||||||
|
clock->precision = precision;
|
||||||
|
clock->delay_quants = QNT_CreateInstance(DELAY_QUANT_MIN_K, DELAY_QUANT_MAX_K,
|
||||||
|
DELAY_QUANT_Q, DELAY_QUANT_REPEAT,
|
||||||
|
DELAY_QUANT_MIN_STEP);
|
||||||
|
|
||||||
|
LCL_AddParameterChangeHandler(handle_slew, clock);
|
||||||
|
|
||||||
|
return clock;
|
||||||
|
}
|
||||||
|
|
||||||
|
/* ================================================== */
|
||||||
|
|
||||||
|
void HCL_DestroyInstance(HCL_Instance clock)
|
||||||
|
{
|
||||||
|
LCL_RemoveParameterChangeHandler(handle_slew, clock);
|
||||||
|
QNT_DestroyInstance(clock->delay_quants);
|
||||||
|
Free(clock->y_data);
|
||||||
|
Free(clock->x_data);
|
||||||
|
Free(clock);
|
||||||
|
}
|
||||||
|
|
||||||
|
/* ================================================== */
|
||||||
|
|
||||||
|
int
|
||||||
|
HCL_NeedsNewSample(HCL_Instance clock, struct timespec *now)
|
||||||
|
{
|
||||||
|
if (!clock->n_samples ||
|
||||||
|
fabs(UTI_DiffTimespecsToDouble(now, &clock->local_ref)) >= clock->min_separation)
|
||||||
|
return 1;
|
||||||
|
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
/* ================================================== */
|
||||||
|
|
||||||
|
int
|
||||||
|
HCL_ProcessReadings(HCL_Instance clock, int n_readings, struct timespec tss[][3],
|
||||||
|
struct timespec *hw_ts, struct timespec *local_ts, double *err)
|
||||||
|
{
|
||||||
|
double delay, raw_delay, min_delay, low_delay, high_delay, e, pred_err;
|
||||||
|
double delay_sum, hw_sum, local_sum, local_prec, freq;
|
||||||
|
int i, min_reading, combined;
|
||||||
|
struct timespec ts1, ts2;
|
||||||
|
|
||||||
|
if (n_readings < 1)
|
||||||
|
return 0;
|
||||||
|
|
||||||
|
/* Work out the current correction multiplier needed to get cooked delays */
|
||||||
|
LCL_CookTime(&tss[0][0], &ts1, NULL);
|
||||||
|
LCL_CookTime(&tss[n_readings - 1][2], &ts2, NULL);
|
||||||
|
if (UTI_CompareTimespecs(&tss[0][0], &tss[n_readings - 1][2]) < 0)
|
||||||
|
freq = UTI_DiffTimespecsToDouble(&ts1, &ts2) /
|
||||||
|
UTI_DiffTimespecsToDouble(&tss[0][0], &tss[n_readings - 1][2]);
|
||||||
|
else
|
||||||
|
freq = 1.0;
|
||||||
|
|
||||||
|
for (i = 0; i < n_readings; i++) {
|
||||||
|
delay = freq * UTI_DiffTimespecsToDouble(&tss[i][2], &tss[i][0]);
|
||||||
|
|
||||||
|
if (delay < 0.0) {
|
||||||
|
/* Step in the middle of a reading? */
|
||||||
|
DEBUG_LOG("Bad reading delay=%e", delay);
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (i == 0 || min_delay > delay) {
|
||||||
|
min_delay = delay;
|
||||||
|
min_reading = i;
|
||||||
|
}
|
||||||
|
|
||||||
|
QNT_Accumulate(clock->delay_quants, delay);
|
||||||
|
}
|
||||||
|
|
||||||
|
local_prec = LCL_GetSysPrecisionAsQuantum();
|
||||||
|
|
||||||
|
low_delay = QNT_GetQuantile(clock->delay_quants, DELAY_QUANT_MIN_K);
|
||||||
|
high_delay = QNT_GetQuantile(clock->delay_quants, DELAY_QUANT_MAX_K);
|
||||||
|
low_delay = MIN(low_delay, high_delay);
|
||||||
|
high_delay = MAX(high_delay, low_delay + local_prec);
|
||||||
|
|
||||||
|
/* Combine readings with delay in the expected interval */
|
||||||
|
for (i = combined = 0, delay_sum = hw_sum = local_sum = 0.0; i < n_readings; i++) {
|
||||||
|
raw_delay = UTI_DiffTimespecsToDouble(&tss[i][2], &tss[i][0]);
|
||||||
|
delay = freq * raw_delay;
|
||||||
|
|
||||||
|
if (delay < low_delay || delay > high_delay)
|
||||||
|
continue;
|
||||||
|
|
||||||
|
delay_sum += delay;
|
||||||
|
hw_sum += UTI_DiffTimespecsToDouble(&tss[i][1], &tss[0][1]);
|
||||||
|
local_sum += UTI_DiffTimespecsToDouble(&tss[i][0], &tss[0][0]) + raw_delay / 2.0;
|
||||||
|
combined++;
|
||||||
|
}
|
||||||
|
|
||||||
|
DEBUG_LOG("Combined %d readings lo=%e hi=%e", combined, low_delay, high_delay);
|
||||||
|
|
||||||
|
if (combined > 0) {
|
||||||
|
UTI_AddDoubleToTimespec(&tss[0][1], hw_sum / combined, hw_ts);
|
||||||
|
UTI_AddDoubleToTimespec(&tss[0][0], local_sum / combined, local_ts);
|
||||||
|
*err = MAX(delay_sum / combined / 2.0, clock->precision);
|
||||||
|
return 1;
|
||||||
|
}
|
||||||
|
|
||||||
|
/* Accept the reading with minimum delay if its interval does not contain
|
||||||
|
the current offset predicted from previous samples */
|
||||||
|
|
||||||
|
*hw_ts = tss[min_reading][1];
|
||||||
|
UTI_AddDoubleToTimespec(&tss[min_reading][0], min_delay / freq / 2.0, local_ts);
|
||||||
|
*err = MAX(min_delay / 2.0, clock->precision);
|
||||||
|
|
||||||
|
pred_err = 0.0;
|
||||||
|
LCL_CookTime(local_ts, &ts1, NULL);
|
||||||
|
if (!HCL_CookTime(clock, hw_ts, &ts2, &e) ||
|
||||||
|
((pred_err = UTI_DiffTimespecsToDouble(&ts1, &ts2)) > *err)) {
|
||||||
|
DEBUG_LOG("Accepted reading err=%e prerr=%e", *err, pred_err);
|
||||||
|
return 1;
|
||||||
|
}
|
||||||
|
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
/* ================================================== */
|
||||||
|
|
||||||
|
void
|
||||||
|
HCL_AccumulateSample(HCL_Instance clock, struct timespec *hw_ts,
|
||||||
|
struct timespec *local_ts, double err)
|
||||||
|
{
|
||||||
|
double hw_delta, local_delta, local_freq, raw_freq;
|
||||||
|
int i, n_runs, best_start;
|
||||||
|
|
||||||
|
local_freq = 1.0 - LCL_ReadAbsoluteFrequency() / 1.0e6;
|
||||||
|
|
||||||
|
/* Shift old samples */
|
||||||
|
if (clock->n_samples) {
|
||||||
|
if (clock->n_samples >= clock->max_samples)
|
||||||
|
clock->n_samples--;
|
||||||
|
|
||||||
|
hw_delta = UTI_DiffTimespecsToDouble(hw_ts, &clock->hw_ref);
|
||||||
|
local_delta = UTI_DiffTimespecsToDouble(local_ts, &clock->local_ref) / local_freq;
|
||||||
|
|
||||||
|
if (hw_delta <= 0.0 || local_delta < clock->min_separation / 2.0) {
|
||||||
|
clock->n_samples = 0;
|
||||||
|
DEBUG_LOG("HW clock reset interval=%f", local_delta);
|
||||||
|
}
|
||||||
|
|
||||||
|
for (i = clock->max_samples - clock->n_samples; i < clock->max_samples; i++) {
|
||||||
|
clock->y_data[i - 1] = clock->y_data[i] - hw_delta;
|
||||||
|
clock->x_data[i - 1] = clock->x_data[i] - local_delta;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
clock->n_samples++;
|
||||||
|
clock->hw_ref = *hw_ts;
|
||||||
|
clock->local_ref = *local_ts;
|
||||||
|
clock->last_err = err;
|
||||||
|
|
||||||
|
/* Get new coefficients */
|
||||||
|
clock->valid_coefs =
|
||||||
|
RGR_FindBestRobustRegression(clock->x_data + clock->max_samples - clock->n_samples,
|
||||||
|
clock->y_data + clock->max_samples - clock->n_samples,
|
||||||
|
clock->n_samples, 1.0e-10, &clock->offset, &raw_freq,
|
||||||
|
&n_runs, &best_start);
|
||||||
|
|
||||||
|
if (!clock->valid_coefs) {
|
||||||
|
DEBUG_LOG("HW clock needs more samples");
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
clock->frequency = raw_freq / local_freq;
|
||||||
|
|
||||||
|
/* Drop unneeded samples */
|
||||||
|
if (clock->n_samples > clock->min_samples)
|
||||||
|
clock->n_samples -= MIN(best_start, clock->n_samples - clock->min_samples);
|
||||||
|
|
||||||
|
/* If the fit doesn't cross the error interval of the last sample,
|
||||||
|
or the frequency is not sane, drop all samples and start again */
|
||||||
|
if (fabs(clock->offset) > err ||
|
||||||
|
fabs(clock->frequency - 1.0) > MAX_FREQ_OFFSET) {
|
||||||
|
DEBUG_LOG("HW clock reset");
|
||||||
|
clock->n_samples = 0;
|
||||||
|
clock->valid_coefs = 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
DEBUG_LOG("HW clock samples=%d offset=%e freq=%e raw_freq=%e err=%e ref_diff=%e",
|
||||||
|
clock->n_samples, clock->offset, clock->frequency - 1.0, raw_freq - 1.0, err,
|
||||||
|
UTI_DiffTimespecsToDouble(&clock->hw_ref, &clock->local_ref));
|
||||||
|
}
|
||||||
|
|
||||||
|
/* ================================================== */
|
||||||
|
|
||||||
|
int
|
||||||
|
HCL_CookTime(HCL_Instance clock, struct timespec *raw, struct timespec *cooked, double *err)
|
||||||
|
{
|
||||||
|
double offset, elapsed;
|
||||||
|
|
||||||
|
if (!clock->valid_coefs)
|
||||||
|
return 0;
|
||||||
|
|
||||||
|
elapsed = UTI_DiffTimespecsToDouble(raw, &clock->hw_ref);
|
||||||
|
offset = elapsed / clock->frequency - clock->offset;
|
||||||
|
UTI_AddDoubleToTimespec(&clock->local_ref, offset, cooked);
|
||||||
|
|
||||||
|
/* Fow now, just return the error of the last sample */
|
||||||
|
if (err)
|
||||||
|
*err = clock->last_err;
|
||||||
|
|
||||||
|
return 1;
|
||||||
|
}
|
54
hwclock.h
Normal file
54
hwclock.h
Normal file
|
@ -0,0 +1,54 @@
|
||||||
|
/*
|
||||||
|
chronyd/chronyc - Programs for keeping computer clocks accurate.
|
||||||
|
|
||||||
|
**********************************************************************
|
||||||
|
* Copyright (C) Miroslav Lichvar 2016
|
||||||
|
*
|
||||||
|
* This program is free software; you can redistribute it and/or modify
|
||||||
|
* it under the terms of version 2 of the GNU General Public License as
|
||||||
|
* published by the Free Software Foundation.
|
||||||
|
*
|
||||||
|
* This program is distributed in the hope that it will be useful, but
|
||||||
|
* WITHOUT ANY WARRANTY; without even the implied warranty of
|
||||||
|
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
|
||||||
|
* General Public License for more details.
|
||||||
|
*
|
||||||
|
* You should have received a copy of the GNU General Public License along
|
||||||
|
* with this program; if not, write to the Free Software Foundation, Inc.,
|
||||||
|
* 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA.
|
||||||
|
*
|
||||||
|
**********************************************************************
|
||||||
|
|
||||||
|
=======================================================================
|
||||||
|
|
||||||
|
Header for tracking of hardware clocks */
|
||||||
|
|
||||||
|
#ifndef GOT_HWCLOCK_H
|
||||||
|
#define GOT_HWCLOCK_H
|
||||||
|
|
||||||
|
typedef struct HCL_Instance_Record *HCL_Instance;
|
||||||
|
|
||||||
|
/* Create a new HW clock instance */
|
||||||
|
extern HCL_Instance HCL_CreateInstance(int min_samples, int max_samples,
|
||||||
|
double min_separation, double precision);
|
||||||
|
|
||||||
|
/* Destroy a HW clock instance */
|
||||||
|
extern void HCL_DestroyInstance(HCL_Instance clock);
|
||||||
|
|
||||||
|
/* Check if a new sample should be accumulated at this time */
|
||||||
|
extern int HCL_NeedsNewSample(HCL_Instance clock, struct timespec *now);
|
||||||
|
|
||||||
|
/* Process new readings of the HW clock in form of (sys, hw, sys) triplets and
|
||||||
|
produce a sample which can be accumulated */
|
||||||
|
extern int HCL_ProcessReadings(HCL_Instance clock, int n_readings, struct timespec tss[][3],
|
||||||
|
struct timespec *hw_ts, struct timespec *local_ts, double *err);
|
||||||
|
|
||||||
|
/* Accumulate a new sample */
|
||||||
|
extern void HCL_AccumulateSample(HCL_Instance clock, struct timespec *hw_ts,
|
||||||
|
struct timespec *local_ts, double err);
|
||||||
|
|
||||||
|
/* Convert raw hardware time to cooked local time */
|
||||||
|
extern int HCL_CookTime(HCL_Instance clock, struct timespec *raw, struct timespec *cooked,
|
||||||
|
double *err);
|
||||||
|
|
||||||
|
#endif
|
441
keys.c
Normal file
441
keys.c
Normal file
|
@ -0,0 +1,441 @@
|
||||||
|
/*
|
||||||
|
chronyd/chronyc - Programs for keeping computer clocks accurate.
|
||||||
|
|
||||||
|
**********************************************************************
|
||||||
|
* Copyright (C) Richard P. Curnow 1997-2003
|
||||||
|
* Copyright (C) Miroslav Lichvar 2012-2016, 2019-2020
|
||||||
|
*
|
||||||
|
* This program is free software; you can redistribute it and/or modify
|
||||||
|
* it under the terms of version 2 of the GNU General Public License as
|
||||||
|
* published by the Free Software Foundation.
|
||||||
|
*
|
||||||
|
* This program is distributed in the hope that it will be useful, but
|
||||||
|
* WITHOUT ANY WARRANTY; without even the implied warranty of
|
||||||
|
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
|
||||||
|
* General Public License for more details.
|
||||||
|
*
|
||||||
|
* You should have received a copy of the GNU General Public License along
|
||||||
|
* with this program; if not, write to the Free Software Foundation, Inc.,
|
||||||
|
* 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA.
|
||||||
|
*
|
||||||
|
**********************************************************************
|
||||||
|
|
||||||
|
=======================================================================
|
||||||
|
|
||||||
|
Module for managing keys used for authenticating NTP packets and commands
|
||||||
|
|
||||||
|
*/
|
||||||
|
|
||||||
|
#include "config.h"
|
||||||
|
|
||||||
|
#include "sysincl.h"
|
||||||
|
|
||||||
|
#include "array.h"
|
||||||
|
#include "keys.h"
|
||||||
|
#include "cmac.h"
|
||||||
|
#include "cmdparse.h"
|
||||||
|
#include "conf.h"
|
||||||
|
#include "memory.h"
|
||||||
|
#include "util.h"
|
||||||
|
#include "local.h"
|
||||||
|
#include "logging.h"
|
||||||
|
|
||||||
|
/* Consider 80 bits as the absolute minimum for a secure key */
|
||||||
|
#define MIN_SECURE_KEY_LENGTH 10
|
||||||
|
|
||||||
|
typedef enum {
|
||||||
|
NTP_MAC,
|
||||||
|
CMAC,
|
||||||
|
} KeyClass;
|
||||||
|
|
||||||
|
typedef struct {
|
||||||
|
uint32_t id;
|
||||||
|
int type;
|
||||||
|
int length;
|
||||||
|
KeyClass class;
|
||||||
|
union {
|
||||||
|
struct {
|
||||||
|
unsigned char *value;
|
||||||
|
int hash_id;
|
||||||
|
} ntp_mac;
|
||||||
|
CMC_Instance cmac;
|
||||||
|
} data;
|
||||||
|
} Key;
|
||||||
|
|
||||||
|
static ARR_Instance keys;
|
||||||
|
|
||||||
|
static int cache_valid;
|
||||||
|
static uint32_t cache_key_id;
|
||||||
|
static int cache_key_pos;
|
||||||
|
|
||||||
|
/* ================================================== */
|
||||||
|
|
||||||
|
static void
|
||||||
|
free_keys(void)
|
||||||
|
{
|
||||||
|
unsigned int i;
|
||||||
|
Key *key;
|
||||||
|
|
||||||
|
for (i = 0; i < ARR_GetSize(keys); i++) {
|
||||||
|
key = ARR_GetElement(keys, i);
|
||||||
|
switch (key->class) {
|
||||||
|
case NTP_MAC:
|
||||||
|
Free(key->data.ntp_mac.value);
|
||||||
|
break;
|
||||||
|
case CMAC:
|
||||||
|
CMC_DestroyInstance(key->data.cmac);
|
||||||
|
break;
|
||||||
|
default:
|
||||||
|
assert(0);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
ARR_SetSize(keys, 0);
|
||||||
|
cache_valid = 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
/* ================================================== */
|
||||||
|
|
||||||
|
void
|
||||||
|
KEY_Initialise(void)
|
||||||
|
{
|
||||||
|
keys = ARR_CreateInstance(sizeof (Key));
|
||||||
|
cache_valid = 0;
|
||||||
|
KEY_Reload();
|
||||||
|
}
|
||||||
|
|
||||||
|
/* ================================================== */
|
||||||
|
|
||||||
|
void
|
||||||
|
KEY_Finalise(void)
|
||||||
|
{
|
||||||
|
free_keys();
|
||||||
|
ARR_DestroyInstance(keys);
|
||||||
|
}
|
||||||
|
|
||||||
|
/* ================================================== */
|
||||||
|
|
||||||
|
static Key *
|
||||||
|
get_key(unsigned int index)
|
||||||
|
{
|
||||||
|
return ((Key *)ARR_GetElements(keys)) + index;
|
||||||
|
}
|
||||||
|
|
||||||
|
/* ================================================== */
|
||||||
|
/* Decode key encoded in ASCII or HEX */
|
||||||
|
|
||||||
|
static int
|
||||||
|
decode_key(char *key)
|
||||||
|
{
|
||||||
|
int len = strlen(key);
|
||||||
|
|
||||||
|
if (!strncmp(key, "ASCII:", 6)) {
|
||||||
|
memmove(key, key + 6, len - 6);
|
||||||
|
return len - 6;
|
||||||
|
} else if (!strncmp(key, "HEX:", 4)) {
|
||||||
|
return UTI_HexToBytes(key + 4, key, len);
|
||||||
|
} else {
|
||||||
|
/* assume ASCII */
|
||||||
|
return len;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/* ================================================== */
|
||||||
|
|
||||||
|
/* Compare two keys */
|
||||||
|
|
||||||
|
static int
|
||||||
|
compare_keys_by_id(const void *a, const void *b)
|
||||||
|
{
|
||||||
|
const Key *c = (const Key *) a;
|
||||||
|
const Key *d = (const Key *) b;
|
||||||
|
|
||||||
|
if (c->id < d->id) {
|
||||||
|
return -1;
|
||||||
|
} else if (c->id > d->id) {
|
||||||
|
return +1;
|
||||||
|
} else {
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
}
|
||||||
|
|
||||||
|
/* ================================================== */
|
||||||
|
|
||||||
|
void
|
||||||
|
KEY_Reload(void)
|
||||||
|
{
|
||||||
|
unsigned int i, line_number, key_length, cmac_key_length;
|
||||||
|
FILE *in;
|
||||||
|
char line[2048], *key_file, *key_value;
|
||||||
|
const char *key_type;
|
||||||
|
HSH_Algorithm hash_algorithm;
|
||||||
|
CMC_Algorithm cmac_algorithm;
|
||||||
|
int hash_id;
|
||||||
|
Key key;
|
||||||
|
|
||||||
|
free_keys();
|
||||||
|
|
||||||
|
key_file = CNF_GetKeysFile();
|
||||||
|
line_number = 0;
|
||||||
|
|
||||||
|
if (!key_file)
|
||||||
|
return;
|
||||||
|
|
||||||
|
if (!UTI_CheckFilePermissions(key_file, 0771))
|
||||||
|
;
|
||||||
|
|
||||||
|
in = UTI_OpenFile(NULL, key_file, NULL, 'r', 0);
|
||||||
|
if (!in) {
|
||||||
|
LOG(LOGS_WARN, "Could not open keyfile %s", key_file);
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
while (fgets(line, sizeof (line), in)) {
|
||||||
|
line_number++;
|
||||||
|
|
||||||
|
CPS_NormalizeLine(line);
|
||||||
|
if (!*line)
|
||||||
|
continue;
|
||||||
|
|
||||||
|
memset(&key, 0, sizeof (key));
|
||||||
|
|
||||||
|
if (!CPS_ParseKey(line, &key.id, &key_type, &key_value)) {
|
||||||
|
LOG(LOGS_WARN, "Could not parse key at line %u in file %s", line_number, key_file);
|
||||||
|
continue;
|
||||||
|
}
|
||||||
|
|
||||||
|
key_length = decode_key(key_value);
|
||||||
|
if (key_length == 0) {
|
||||||
|
LOG(LOGS_WARN, "Could not decode key %"PRIu32, key.id);
|
||||||
|
continue;
|
||||||
|
}
|
||||||
|
|
||||||
|
hash_algorithm = UTI_HashNameToAlgorithm(key_type);
|
||||||
|
cmac_algorithm = UTI_CmacNameToAlgorithm(key_type);
|
||||||
|
|
||||||
|
if (hash_algorithm != 0) {
|
||||||
|
hash_id = HSH_GetHashId(hash_algorithm);
|
||||||
|
if (hash_id < 0) {
|
||||||
|
LOG(LOGS_WARN, "Unsupported %s in key %"PRIu32, "hash function", key.id);
|
||||||
|
continue;
|
||||||
|
}
|
||||||
|
key.class = NTP_MAC;
|
||||||
|
key.type = hash_algorithm;
|
||||||
|
key.length = key_length;
|
||||||
|
key.data.ntp_mac.value = MallocArray(unsigned char, key_length);
|
||||||
|
memcpy(key.data.ntp_mac.value, key_value, key_length);
|
||||||
|
key.data.ntp_mac.hash_id = hash_id;
|
||||||
|
} else if (cmac_algorithm != 0) {
|
||||||
|
cmac_key_length = CMC_GetKeyLength(cmac_algorithm);
|
||||||
|
if (cmac_key_length == 0) {
|
||||||
|
LOG(LOGS_WARN, "Unsupported %s in key %"PRIu32, "cipher", key.id);
|
||||||
|
continue;
|
||||||
|
} else if (cmac_key_length != key_length) {
|
||||||
|
LOG(LOGS_WARN, "Invalid length of %s key %"PRIu32" (expected %u bits)",
|
||||||
|
key_type, key.id, 8 * cmac_key_length);
|
||||||
|
continue;
|
||||||
|
}
|
||||||
|
|
||||||
|
key.class = CMAC;
|
||||||
|
key.type = cmac_algorithm;
|
||||||
|
key.length = key_length;
|
||||||
|
key.data.cmac = CMC_CreateInstance(cmac_algorithm, (unsigned char *)key_value,
|
||||||
|
key_length);
|
||||||
|
assert(key.data.cmac);
|
||||||
|
} else {
|
||||||
|
LOG(LOGS_WARN, "Invalid type in key %"PRIu32, key.id);
|
||||||
|
continue;
|
||||||
|
}
|
||||||
|
|
||||||
|
ARR_AppendElement(keys, &key);
|
||||||
|
}
|
||||||
|
|
||||||
|
fclose(in);
|
||||||
|
|
||||||
|
/* Sort keys into order. Note, if there's a duplicate, it is
|
||||||
|
arbitrary which one we use later - the user should have been
|
||||||
|
more careful! */
|
||||||
|
qsort(ARR_GetElements(keys), ARR_GetSize(keys), sizeof (Key), compare_keys_by_id);
|
||||||
|
|
||||||
|
LOG(LOGS_INFO, "Loaded %u symmetric keys", ARR_GetSize(keys));
|
||||||
|
|
||||||
|
/* Check for duplicates */
|
||||||
|
for (i = 1; i < ARR_GetSize(keys); i++) {
|
||||||
|
if (get_key(i - 1)->id == get_key(i)->id)
|
||||||
|
LOG(LOGS_WARN, "Detected duplicate key %"PRIu32, get_key(i - 1)->id);
|
||||||
|
}
|
||||||
|
|
||||||
|
/* Erase any passwords from stack */
|
||||||
|
memset(line, 0, sizeof (line));
|
||||||
|
}
|
||||||
|
|
||||||
|
/* ================================================== */
|
||||||
|
|
||||||
|
static int
|
||||||
|
lookup_key(uint32_t id)
|
||||||
|
{
|
||||||
|
Key specimen, *where, *keys_ptr;
|
||||||
|
int pos;
|
||||||
|
|
||||||
|
keys_ptr = ARR_GetElements(keys);
|
||||||
|
specimen.id = id;
|
||||||
|
where = (Key *)bsearch((void *)&specimen, keys_ptr, ARR_GetSize(keys),
|
||||||
|
sizeof (Key), compare_keys_by_id);
|
||||||
|
if (!where) {
|
||||||
|
return -1;
|
||||||
|
} else {
|
||||||
|
pos = where - keys_ptr;
|
||||||
|
return pos;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/* ================================================== */
|
||||||
|
|
||||||
|
static Key *
|
||||||
|
get_key_by_id(uint32_t key_id)
|
||||||
|
{
|
||||||
|
int position;
|
||||||
|
|
||||||
|
if (cache_valid && key_id == cache_key_id)
|
||||||
|
return get_key(cache_key_pos);
|
||||||
|
|
||||||
|
position = lookup_key(key_id);
|
||||||
|
|
||||||
|
if (position >= 0) {
|
||||||
|
cache_valid = 1;
|
||||||
|
cache_key_pos = position;
|
||||||
|
cache_key_id = key_id;
|
||||||
|
|
||||||
|
return get_key(position);
|
||||||
|
}
|
||||||
|
|
||||||
|
return NULL;
|
||||||
|
}
|
||||||
|
|
||||||
|
/* ================================================== */
|
||||||
|
|
||||||
|
int
|
||||||
|
KEY_KeyKnown(uint32_t key_id)
|
||||||
|
{
|
||||||
|
return get_key_by_id(key_id) != NULL;
|
||||||
|
}
|
||||||
|
|
||||||
|
/* ================================================== */
|
||||||
|
|
||||||
|
int
|
||||||
|
KEY_GetAuthLength(uint32_t key_id)
|
||||||
|
{
|
||||||
|
unsigned char buf[MAX_HASH_LENGTH];
|
||||||
|
Key *key;
|
||||||
|
|
||||||
|
key = get_key_by_id(key_id);
|
||||||
|
|
||||||
|
if (!key)
|
||||||
|
return 0;
|
||||||
|
|
||||||
|
switch (key->class) {
|
||||||
|
case NTP_MAC:
|
||||||
|
return HSH_Hash(key->data.ntp_mac.hash_id, buf, 0, buf, 0, buf, sizeof (buf));
|
||||||
|
case CMAC:
|
||||||
|
return CMC_Hash(key->data.cmac, buf, 0, buf, sizeof (buf));
|
||||||
|
default:
|
||||||
|
assert(0);
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/* ================================================== */
|
||||||
|
|
||||||
|
int
|
||||||
|
KEY_CheckKeyLength(uint32_t key_id)
|
||||||
|
{
|
||||||
|
Key *key;
|
||||||
|
|
||||||
|
key = get_key_by_id(key_id);
|
||||||
|
|
||||||
|
if (!key)
|
||||||
|
return 0;
|
||||||
|
|
||||||
|
return key->length >= MIN_SECURE_KEY_LENGTH;
|
||||||
|
}
|
||||||
|
|
||||||
|
/* ================================================== */
|
||||||
|
|
||||||
|
int
|
||||||
|
KEY_GetKeyInfo(uint32_t key_id, int *type, int *bits)
|
||||||
|
{
|
||||||
|
Key *key;
|
||||||
|
|
||||||
|
key = get_key_by_id(key_id);
|
||||||
|
|
||||||
|
if (!key)
|
||||||
|
return 0;
|
||||||
|
|
||||||
|
*type = key->type;
|
||||||
|
*bits = 8 * key->length;
|
||||||
|
|
||||||
|
return 1;
|
||||||
|
}
|
||||||
|
|
||||||
|
/* ================================================== */
|
||||||
|
|
||||||
|
static int
|
||||||
|
generate_auth(Key *key, const void *data, int data_len, unsigned char *auth, int auth_len)
|
||||||
|
{
|
||||||
|
switch (key->class) {
|
||||||
|
case NTP_MAC:
|
||||||
|
return HSH_Hash(key->data.ntp_mac.hash_id, key->data.ntp_mac.value,
|
||||||
|
key->length, data, data_len, auth, auth_len);
|
||||||
|
case CMAC:
|
||||||
|
return CMC_Hash(key->data.cmac, data, data_len, auth, auth_len);
|
||||||
|
default:
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/* ================================================== */
|
||||||
|
|
||||||
|
static int
|
||||||
|
check_auth(Key *key, const void *data, int data_len,
|
||||||
|
const unsigned char *auth, int auth_len, int trunc_len)
|
||||||
|
{
|
||||||
|
unsigned char buf[MAX_HASH_LENGTH];
|
||||||
|
int hash_len;
|
||||||
|
|
||||||
|
hash_len = generate_auth(key, data, data_len, buf, sizeof (buf));
|
||||||
|
|
||||||
|
return MIN(hash_len, trunc_len) == auth_len && !memcmp(buf, auth, auth_len);
|
||||||
|
}
|
||||||
|
|
||||||
|
/* ================================================== */
|
||||||
|
|
||||||
|
int
|
||||||
|
KEY_GenerateAuth(uint32_t key_id, const void *data, int data_len,
|
||||||
|
unsigned char *auth, int auth_len)
|
||||||
|
{
|
||||||
|
Key *key;
|
||||||
|
|
||||||
|
key = get_key_by_id(key_id);
|
||||||
|
|
||||||
|
if (!key)
|
||||||
|
return 0;
|
||||||
|
|
||||||
|
return generate_auth(key, data, data_len, auth, auth_len);
|
||||||
|
}
|
||||||
|
|
||||||
|
/* ================================================== */
|
||||||
|
|
||||||
|
int
|
||||||
|
KEY_CheckAuth(uint32_t key_id, const void *data, int data_len,
|
||||||
|
const unsigned char *auth, int auth_len, int trunc_len)
|
||||||
|
{
|
||||||
|
Key *key;
|
||||||
|
|
||||||
|
key = get_key_by_id(key_id);
|
||||||
|
|
||||||
|
if (!key)
|
||||||
|
return 0;
|
||||||
|
|
||||||
|
return check_auth(key, data, data_len, auth, auth_len, trunc_len);
|
||||||
|
}
|
47
keys.h
Normal file
47
keys.h
Normal file
|
@ -0,0 +1,47 @@
|
||||||
|
/*
|
||||||
|
chronyd/chronyc - Programs for keeping computer clocks accurate.
|
||||||
|
|
||||||
|
**********************************************************************
|
||||||
|
* Copyright (C) Richard P. Curnow 1997-2002
|
||||||
|
*
|
||||||
|
* This program is free software; you can redistribute it and/or modify
|
||||||
|
* it under the terms of version 2 of the GNU General Public License as
|
||||||
|
* published by the Free Software Foundation.
|
||||||
|
*
|
||||||
|
* This program is distributed in the hope that it will be useful, but
|
||||||
|
* WITHOUT ANY WARRANTY; without even the implied warranty of
|
||||||
|
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
|
||||||
|
* General Public License for more details.
|
||||||
|
*
|
||||||
|
* You should have received a copy of the GNU General Public License along
|
||||||
|
* with this program; if not, write to the Free Software Foundation, Inc.,
|
||||||
|
* 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA.
|
||||||
|
*
|
||||||
|
**********************************************************************
|
||||||
|
|
||||||
|
=======================================================================
|
||||||
|
|
||||||
|
Header for key management module
|
||||||
|
*/
|
||||||
|
|
||||||
|
#ifndef GOT_KEYS_H
|
||||||
|
#define GOT_KEYS_H
|
||||||
|
|
||||||
|
#include "sysincl.h"
|
||||||
|
|
||||||
|
extern void KEY_Initialise(void);
|
||||||
|
extern void KEY_Finalise(void);
|
||||||
|
|
||||||
|
extern void KEY_Reload(void);
|
||||||
|
|
||||||
|
extern int KEY_KeyKnown(uint32_t key_id);
|
||||||
|
extern int KEY_GetAuthLength(uint32_t key_id);
|
||||||
|
extern int KEY_CheckKeyLength(uint32_t key_id);
|
||||||
|
extern int KEY_GetKeyInfo(uint32_t key_id, int *type, int *bits);
|
||||||
|
|
||||||
|
extern int KEY_GenerateAuth(uint32_t key_id, const void *data, int data_len,
|
||||||
|
unsigned char *auth, int auth_len);
|
||||||
|
extern int KEY_CheckAuth(uint32_t key_id, const void *data, int data_len,
|
||||||
|
const unsigned char *auth, int auth_len, int trunc_len);
|
||||||
|
|
||||||
|
#endif /* GOT_KEYS_H */
|
272
leapdb.c
Normal file
272
leapdb.c
Normal file
|
@ -0,0 +1,272 @@
|
||||||
|
/*
|
||||||
|
chronyd/chronyc - Programs for keeping computer clocks accurate.
|
||||||
|
|
||||||
|
**********************************************************************
|
||||||
|
* Copyright (C) Miroslav Lichvar 2009-2018, 2020, 2022
|
||||||
|
* Copyright (C) Patrick Oppenlander 2023, 2024
|
||||||
|
*
|
||||||
|
* This program is free software; you can redistribute it and/or modify
|
||||||
|
* it under the terms of version 2 of the GNU General Public License as
|
||||||
|
* published by the Free Software Foundation.
|
||||||
|
*
|
||||||
|
* This program is distributed in the hope that it will be useful, but
|
||||||
|
* WITHOUT ANY WARRANTY; without even the implied warranty of
|
||||||
|
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
|
||||||
|
* General Public License for more details.
|
||||||
|
*
|
||||||
|
* You should have received a copy of the GNU General Public License along
|
||||||
|
* with this program; if not, write to the Free Software Foundation, Inc.,
|
||||||
|
* 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA.
|
||||||
|
*
|
||||||
|
**********************************************************************
|
||||||
|
|
||||||
|
=======================================================================
|
||||||
|
|
||||||
|
This module provides leap second information. */
|
||||||
|
|
||||||
|
#include "config.h"
|
||||||
|
|
||||||
|
#include "sysincl.h"
|
||||||
|
|
||||||
|
#include "conf.h"
|
||||||
|
#include "leapdb.h"
|
||||||
|
#include "logging.h"
|
||||||
|
#include "util.h"
|
||||||
|
|
||||||
|
/* ================================================== */
|
||||||
|
|
||||||
|
/* Source of leap second data */
|
||||||
|
enum {
|
||||||
|
SRC_NONE,
|
||||||
|
SRC_TIMEZONE,
|
||||||
|
SRC_LIST,
|
||||||
|
} leap_src;
|
||||||
|
|
||||||
|
/* Offset between leap-seconds.list timestamp epoch and Unix epoch.
|
||||||
|
leap-seconds.list epoch is 1 Jan 1900, 00:00:00 */
|
||||||
|
#define LEAP_SEC_LIST_OFFSET 2208988800
|
||||||
|
|
||||||
|
/* ================================================== */
|
||||||
|
|
||||||
|
static NTP_Leap
|
||||||
|
get_tz_leap(time_t when, int *tai_offset)
|
||||||
|
{
|
||||||
|
struct tm stm, *tm;
|
||||||
|
time_t t;
|
||||||
|
char *tz_env, tz_orig[128];
|
||||||
|
NTP_Leap tz_leap = LEAP_Normal;
|
||||||
|
|
||||||
|
tm = gmtime(&when);
|
||||||
|
if (!tm)
|
||||||
|
return tz_leap;
|
||||||
|
|
||||||
|
stm = *tm;
|
||||||
|
|
||||||
|
/* Temporarily switch to the timezone containing leap seconds */
|
||||||
|
tz_env = getenv("TZ");
|
||||||
|
if (tz_env) {
|
||||||
|
if (strlen(tz_env) >= sizeof (tz_orig))
|
||||||
|
return tz_leap;
|
||||||
|
strcpy(tz_orig, tz_env);
|
||||||
|
}
|
||||||
|
setenv("TZ", CNF_GetLeapSecTimezone(), 1);
|
||||||
|
tzset();
|
||||||
|
|
||||||
|
/* Get the TAI-UTC offset, which started at the epoch at 10 seconds */
|
||||||
|
t = mktime(&stm);
|
||||||
|
if (t != -1)
|
||||||
|
*tai_offset = t - when + 10;
|
||||||
|
|
||||||
|
/* Set the time to 23:59:60 and see how it overflows in mktime() */
|
||||||
|
stm.tm_sec = 60;
|
||||||
|
stm.tm_min = 59;
|
||||||
|
stm.tm_hour = 23;
|
||||||
|
|
||||||
|
t = mktime(&stm);
|
||||||
|
|
||||||
|
if (tz_env)
|
||||||
|
setenv("TZ", tz_orig, 1);
|
||||||
|
else
|
||||||
|
unsetenv("TZ");
|
||||||
|
tzset();
|
||||||
|
|
||||||
|
if (t == -1)
|
||||||
|
return tz_leap;
|
||||||
|
|
||||||
|
if (stm.tm_sec == 60)
|
||||||
|
tz_leap = LEAP_InsertSecond;
|
||||||
|
else if (stm.tm_sec == 1)
|
||||||
|
tz_leap = LEAP_DeleteSecond;
|
||||||
|
|
||||||
|
return tz_leap;
|
||||||
|
}
|
||||||
|
|
||||||
|
/* ================================================== */
|
||||||
|
|
||||||
|
static NTP_Leap
|
||||||
|
get_list_leap(time_t when, int *tai_offset)
|
||||||
|
{
|
||||||
|
FILE *f;
|
||||||
|
char line[1024];
|
||||||
|
NTP_Leap ret_leap = LEAP_Normal;
|
||||||
|
int ret_tai_offset = 0, prev_lsl_tai_offset = 10;
|
||||||
|
int64_t when1900, lsl_updated = 0, lsl_expiry = 0;
|
||||||
|
const char *leap_sec_list = CNF_GetLeapSecList();
|
||||||
|
|
||||||
|
if (!(f = UTI_OpenFile(NULL, leap_sec_list, NULL, 'r', 0))) {
|
||||||
|
LOG(LOGS_ERR, "Failed to open leap seconds list %s", leap_sec_list);
|
||||||
|
goto out;
|
||||||
|
}
|
||||||
|
|
||||||
|
/* Leap second happens at midnight */
|
||||||
|
when = (when / (24 * 3600) + 1) * (24 * 3600);
|
||||||
|
|
||||||
|
/* leap-seconds.list timestamps are relative to 1 Jan 1900, 00:00:00 */
|
||||||
|
when1900 = (int64_t)when + LEAP_SEC_LIST_OFFSET;
|
||||||
|
|
||||||
|
while (fgets(line, sizeof line, f) > 0) {
|
||||||
|
int64_t lsl_when;
|
||||||
|
int lsl_tai_offset;
|
||||||
|
char *p;
|
||||||
|
|
||||||
|
/* Ignore blank lines */
|
||||||
|
for (p = line; *p && isspace(*p); ++p)
|
||||||
|
;
|
||||||
|
if (!*p)
|
||||||
|
continue;
|
||||||
|
|
||||||
|
if (*line == '#') {
|
||||||
|
/* Update time line starts with #$ */
|
||||||
|
if (line[1] == '$' && sscanf(line + 2, "%"SCNd64, &lsl_updated) != 1)
|
||||||
|
goto error;
|
||||||
|
/* Expiration time line starts with #@ */
|
||||||
|
if (line[1] == '@' && sscanf(line + 2, "%"SCNd64, &lsl_expiry) != 1)
|
||||||
|
goto error;
|
||||||
|
/* Comment or a special comment we don't care about */
|
||||||
|
continue;
|
||||||
|
}
|
||||||
|
|
||||||
|
/* Leap entry */
|
||||||
|
if (sscanf(line, "%"SCNd64" %d", &lsl_when, &lsl_tai_offset) != 2)
|
||||||
|
goto error;
|
||||||
|
|
||||||
|
if (when1900 == lsl_when) {
|
||||||
|
if (lsl_tai_offset > prev_lsl_tai_offset)
|
||||||
|
ret_leap = LEAP_InsertSecond;
|
||||||
|
else if (lsl_tai_offset < prev_lsl_tai_offset)
|
||||||
|
ret_leap = LEAP_DeleteSecond;
|
||||||
|
/* When is rounded to the end of the day, so offset hasn't changed yet! */
|
||||||
|
ret_tai_offset = prev_lsl_tai_offset;
|
||||||
|
} else if (when1900 > lsl_when) {
|
||||||
|
ret_tai_offset = lsl_tai_offset;
|
||||||
|
}
|
||||||
|
|
||||||
|
prev_lsl_tai_offset = lsl_tai_offset;
|
||||||
|
}
|
||||||
|
|
||||||
|
/* Make sure the file looks sensible */
|
||||||
|
if (!feof(f) || !lsl_updated || !lsl_expiry)
|
||||||
|
goto error;
|
||||||
|
|
||||||
|
if (when1900 >= lsl_expiry)
|
||||||
|
LOG(LOGS_WARN, "Leap second list %s needs update", leap_sec_list);
|
||||||
|
|
||||||
|
goto out;
|
||||||
|
|
||||||
|
error:
|
||||||
|
if (f)
|
||||||
|
fclose(f);
|
||||||
|
LOG(LOGS_ERR, "Failed to parse leap seconds list %s", leap_sec_list);
|
||||||
|
return LEAP_Normal;
|
||||||
|
|
||||||
|
out:
|
||||||
|
if (f)
|
||||||
|
fclose(f);
|
||||||
|
*tai_offset = ret_tai_offset;
|
||||||
|
return ret_leap;
|
||||||
|
}
|
||||||
|
|
||||||
|
/* ================================================== */
|
||||||
|
|
||||||
|
static int
|
||||||
|
check_leap_source(NTP_Leap (*src)(time_t when, int *tai_offset))
|
||||||
|
{
|
||||||
|
int tai_offset = 0;
|
||||||
|
|
||||||
|
/* Check that the leap second source has good data for Jun 30 2012 and Dec 31 2012 */
|
||||||
|
if (src(1341014400, &tai_offset) == LEAP_InsertSecond && tai_offset == 34 &&
|
||||||
|
src(1356912000, &tai_offset) == LEAP_Normal && tai_offset == 35)
|
||||||
|
return 1;
|
||||||
|
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
/* ================================================== */
|
||||||
|
|
||||||
|
void
|
||||||
|
LDB_Initialise(void)
|
||||||
|
{
|
||||||
|
const char *leap_tzname, *leap_sec_list;
|
||||||
|
|
||||||
|
leap_tzname = CNF_GetLeapSecTimezone();
|
||||||
|
if (leap_tzname && !check_leap_source(get_tz_leap)) {
|
||||||
|
LOG(LOGS_WARN, "Timezone %s failed leap second check, ignoring", leap_tzname);
|
||||||
|
leap_tzname = NULL;
|
||||||
|
}
|
||||||
|
|
||||||
|
leap_sec_list = CNF_GetLeapSecList();
|
||||||
|
if (leap_sec_list && !check_leap_source(get_list_leap)) {
|
||||||
|
LOG(LOGS_WARN, "Leap second list %s failed check, ignoring", leap_sec_list);
|
||||||
|
leap_sec_list = NULL;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (leap_sec_list) {
|
||||||
|
LOG(LOGS_INFO, "Using leap second list %s", leap_sec_list);
|
||||||
|
leap_src = SRC_LIST;
|
||||||
|
} else if (leap_tzname) {
|
||||||
|
LOG(LOGS_INFO, "Using %s timezone to obtain leap second data", leap_tzname);
|
||||||
|
leap_src = SRC_TIMEZONE;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/* ================================================== */
|
||||||
|
|
||||||
|
NTP_Leap
|
||||||
|
LDB_GetLeap(time_t when, int *tai_offset)
|
||||||
|
{
|
||||||
|
static time_t last_ldb_leap_check;
|
||||||
|
static NTP_Leap ldb_leap;
|
||||||
|
static int ldb_tai_offset;
|
||||||
|
|
||||||
|
/* Do this check at most twice a day */
|
||||||
|
when = when / (12 * 3600) * (12 * 3600);
|
||||||
|
if (last_ldb_leap_check == when)
|
||||||
|
goto out;
|
||||||
|
|
||||||
|
last_ldb_leap_check = when;
|
||||||
|
ldb_leap = LEAP_Normal;
|
||||||
|
ldb_tai_offset = 0;
|
||||||
|
|
||||||
|
switch (leap_src) {
|
||||||
|
case SRC_NONE:
|
||||||
|
break;
|
||||||
|
case SRC_TIMEZONE:
|
||||||
|
ldb_leap = get_tz_leap(when, &ldb_tai_offset);
|
||||||
|
break;
|
||||||
|
case SRC_LIST:
|
||||||
|
ldb_leap = get_list_leap(when, &ldb_tai_offset);
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
|
||||||
|
out:
|
||||||
|
*tai_offset = ldb_tai_offset;
|
||||||
|
return ldb_leap;
|
||||||
|
}
|
||||||
|
|
||||||
|
/* ================================================== */
|
||||||
|
|
||||||
|
void
|
||||||
|
LDB_Finalise(void)
|
||||||
|
{
|
||||||
|
/* Nothing to do */
|
||||||
|
}
|
37
leapdb.h
Normal file
37
leapdb.h
Normal file
|
@ -0,0 +1,37 @@
|
||||||
|
/*
|
||||||
|
chronyd/chronyc - Programs for keeping computer clocks accurate.
|
||||||
|
|
||||||
|
**********************************************************************
|
||||||
|
* Copyright (C) Patrick Oppenlander 2024
|
||||||
|
*
|
||||||
|
* This program is free software; you can redistribute it and/or modify
|
||||||
|
* it under the terms of version 2 of the GNU General Public License as
|
||||||
|
* published by the Free Software Foundation.
|
||||||
|
*
|
||||||
|
* This program is distributed in the hope that it will be useful, but
|
||||||
|
* WITHOUT ANY WARRANTY; without even the implied warranty of
|
||||||
|
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
|
||||||
|
* General Public License for more details.
|
||||||
|
*
|
||||||
|
* You should have received a copy of the GNU General Public License along
|
||||||
|
* with this program; if not, write to the Free Software Foundation, Inc.,
|
||||||
|
* 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA.
|
||||||
|
*
|
||||||
|
**********************************************************************
|
||||||
|
|
||||||
|
=======================================================================
|
||||||
|
|
||||||
|
This module provides leap second information.
|
||||||
|
|
||||||
|
*/
|
||||||
|
|
||||||
|
#ifndef GOT_LEAPDB_H
|
||||||
|
#define GOT_LEAPDB_H
|
||||||
|
|
||||||
|
#include "ntp.h"
|
||||||
|
|
||||||
|
extern void LDB_Initialise(void);
|
||||||
|
extern NTP_Leap LDB_GetLeap(time_t when, int *tai_offset);
|
||||||
|
extern void LDB_Finalise(void);
|
||||||
|
|
||||||
|
#endif /* GOT_LEAPDB_H */
|
781
local.c
Normal file
781
local.c
Normal file
|
@ -0,0 +1,781 @@
|
||||||
|
/*
|
||||||
|
chronyd/chronyc - Programs for keeping computer clocks accurate.
|
||||||
|
|
||||||
|
**********************************************************************
|
||||||
|
* Copyright (C) Richard P. Curnow 1997-2003
|
||||||
|
* Copyright (C) Miroslav Lichvar 2011, 2014-2015
|
||||||
|
*
|
||||||
|
* This program is free software; you can redistribute it and/or modify
|
||||||
|
* it under the terms of version 2 of the GNU General Public License as
|
||||||
|
* published by the Free Software Foundation.
|
||||||
|
*
|
||||||
|
* This program is distributed in the hope that it will be useful, but
|
||||||
|
* WITHOUT ANY WARRANTY; without even the implied warranty of
|
||||||
|
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
|
||||||
|
* General Public License for more details.
|
||||||
|
*
|
||||||
|
* You should have received a copy of the GNU General Public License along
|
||||||
|
* with this program; if not, write to the Free Software Foundation, Inc.,
|
||||||
|
* 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA.
|
||||||
|
*
|
||||||
|
**********************************************************************
|
||||||
|
|
||||||
|
=======================================================================
|
||||||
|
|
||||||
|
The routines in this file present a common local (system) clock
|
||||||
|
interface to the rest of the software.
|
||||||
|
|
||||||
|
They interface with the system specific driver files in sys_*.c
|
||||||
|
*/
|
||||||
|
|
||||||
|
#include "config.h"
|
||||||
|
|
||||||
|
#include "sysincl.h"
|
||||||
|
|
||||||
|
#include "conf.h"
|
||||||
|
#include "local.h"
|
||||||
|
#include "localp.h"
|
||||||
|
#include "memory.h"
|
||||||
|
#include "smooth.h"
|
||||||
|
#include "util.h"
|
||||||
|
#include "logging.h"
|
||||||
|
|
||||||
|
/* ================================================== */
|
||||||
|
|
||||||
|
/* Variable to store the current frequency, in ppm */
|
||||||
|
static double current_freq_ppm;
|
||||||
|
|
||||||
|
/* Maximum allowed frequency, in ppm */
|
||||||
|
static double max_freq_ppm;
|
||||||
|
|
||||||
|
/* Temperature compensation, in ppm */
|
||||||
|
static double temp_comp_ppm;
|
||||||
|
|
||||||
|
/* ================================================== */
|
||||||
|
/* Store the system dependent drivers */
|
||||||
|
|
||||||
|
static lcl_ReadFrequencyDriver drv_read_freq;
|
||||||
|
static lcl_SetFrequencyDriver drv_set_freq;
|
||||||
|
static lcl_AccrueOffsetDriver drv_accrue_offset;
|
||||||
|
static lcl_ApplyStepOffsetDriver drv_apply_step_offset;
|
||||||
|
static lcl_OffsetCorrectionDriver drv_offset_convert;
|
||||||
|
static lcl_SetLeapDriver drv_set_leap;
|
||||||
|
static lcl_SetSyncStatusDriver drv_set_sync_status;
|
||||||
|
|
||||||
|
/* ================================================== */
|
||||||
|
|
||||||
|
/* Types and variables associated with handling the parameter change
|
||||||
|
list */
|
||||||
|
|
||||||
|
typedef struct _ChangeListEntry {
|
||||||
|
struct _ChangeListEntry *next;
|
||||||
|
struct _ChangeListEntry *prev;
|
||||||
|
LCL_ParameterChangeHandler handler;
|
||||||
|
void *anything;
|
||||||
|
} ChangeListEntry;
|
||||||
|
|
||||||
|
static ChangeListEntry change_list;
|
||||||
|
|
||||||
|
/* ================================================== */
|
||||||
|
|
||||||
|
/* Types and variables associated with handling the parameter change
|
||||||
|
list */
|
||||||
|
|
||||||
|
typedef struct _DispersionNotifyListEntry {
|
||||||
|
struct _DispersionNotifyListEntry *next;
|
||||||
|
struct _DispersionNotifyListEntry *prev;
|
||||||
|
LCL_DispersionNotifyHandler handler;
|
||||||
|
void *anything;
|
||||||
|
} DispersionNotifyListEntry;
|
||||||
|
|
||||||
|
static DispersionNotifyListEntry dispersion_notify_list;
|
||||||
|
|
||||||
|
/* ================================================== */
|
||||||
|
|
||||||
|
static int precision_log;
|
||||||
|
static double precision_quantum;
|
||||||
|
|
||||||
|
static double max_clock_error;
|
||||||
|
|
||||||
|
/* ================================================== */
|
||||||
|
|
||||||
|
/* Define the number of increments of the system clock that we want
|
||||||
|
to see to be fairly sure that we've got something approaching
|
||||||
|
the minimum increment. Even on a crummy implementation that can't
|
||||||
|
interpolate between 10ms ticks, we should get this done in
|
||||||
|
under 1s of busy waiting. */
|
||||||
|
#define NITERS 100
|
||||||
|
|
||||||
|
#define NSEC_PER_SEC 1000000000
|
||||||
|
|
||||||
|
static double
|
||||||
|
measure_clock_precision(void)
|
||||||
|
{
|
||||||
|
struct timespec ts, old_ts;
|
||||||
|
int iters, diff, best;
|
||||||
|
|
||||||
|
LCL_ReadRawTime(&old_ts);
|
||||||
|
|
||||||
|
/* Assume we must be better than a second */
|
||||||
|
best = NSEC_PER_SEC;
|
||||||
|
iters = 0;
|
||||||
|
|
||||||
|
do {
|
||||||
|
LCL_ReadRawTime(&ts);
|
||||||
|
|
||||||
|
diff = NSEC_PER_SEC * (ts.tv_sec - old_ts.tv_sec) + (ts.tv_nsec - old_ts.tv_nsec);
|
||||||
|
|
||||||
|
old_ts = ts;
|
||||||
|
if (diff > 0) {
|
||||||
|
if (diff < best)
|
||||||
|
best = diff;
|
||||||
|
iters++;
|
||||||
|
}
|
||||||
|
} while (iters < NITERS);
|
||||||
|
|
||||||
|
assert(best > 0);
|
||||||
|
|
||||||
|
return 1.0e-9 * best;
|
||||||
|
}
|
||||||
|
|
||||||
|
/* ================================================== */
|
||||||
|
|
||||||
|
void
|
||||||
|
LCL_Initialise(void)
|
||||||
|
{
|
||||||
|
change_list.next = change_list.prev = &change_list;
|
||||||
|
|
||||||
|
dispersion_notify_list.next = dispersion_notify_list.prev = &dispersion_notify_list;
|
||||||
|
|
||||||
|
/* Null out the system drivers, so that we die
|
||||||
|
if they never get defined before use */
|
||||||
|
|
||||||
|
drv_read_freq = NULL;
|
||||||
|
drv_set_freq = NULL;
|
||||||
|
drv_accrue_offset = NULL;
|
||||||
|
drv_offset_convert = NULL;
|
||||||
|
|
||||||
|
/* This ought to be set from the system driver layer */
|
||||||
|
current_freq_ppm = 0.0;
|
||||||
|
temp_comp_ppm = 0.0;
|
||||||
|
|
||||||
|
precision_quantum = CNF_GetClockPrecision();
|
||||||
|
if (precision_quantum <= 0.0)
|
||||||
|
precision_quantum = measure_clock_precision();
|
||||||
|
|
||||||
|
precision_quantum = CLAMP(1.0e-9, precision_quantum, 1.0);
|
||||||
|
precision_log = round(log(precision_quantum) / log(2.0));
|
||||||
|
/* NTP code doesn't support smaller log than -30 */
|
||||||
|
assert(precision_log >= -30);
|
||||||
|
|
||||||
|
DEBUG_LOG("Clock precision %.9f (%d)", precision_quantum, precision_log);
|
||||||
|
|
||||||
|
/* This is the maximum allowed frequency offset in ppm, the time must
|
||||||
|
never stop or run backwards */
|
||||||
|
max_freq_ppm = CNF_GetMaxDrift();
|
||||||
|
max_freq_ppm = CLAMP(0.0, max_freq_ppm, 500000.0);
|
||||||
|
|
||||||
|
max_clock_error = CNF_GetMaxClockError() * 1e-6;
|
||||||
|
}
|
||||||
|
|
||||||
|
/* ================================================== */
|
||||||
|
|
||||||
|
void
|
||||||
|
LCL_Finalise(void)
|
||||||
|
{
|
||||||
|
/* Make sure all handlers have been removed */
|
||||||
|
if (change_list.next != &change_list)
|
||||||
|
assert(0);
|
||||||
|
if (dispersion_notify_list.next != &dispersion_notify_list)
|
||||||
|
assert(0);
|
||||||
|
}
|
||||||
|
|
||||||
|
/* ================================================== */
|
||||||
|
|
||||||
|
/* Routine to read the system precision as a log to base 2 value. */
|
||||||
|
int
|
||||||
|
LCL_GetSysPrecisionAsLog(void)
|
||||||
|
{
|
||||||
|
return precision_log;
|
||||||
|
}
|
||||||
|
|
||||||
|
/* ================================================== */
|
||||||
|
/* Routine to read the system precision in terms of the actual time step */
|
||||||
|
|
||||||
|
double
|
||||||
|
LCL_GetSysPrecisionAsQuantum(void)
|
||||||
|
{
|
||||||
|
return precision_quantum;
|
||||||
|
}
|
||||||
|
|
||||||
|
/* ================================================== */
|
||||||
|
|
||||||
|
double
|
||||||
|
LCL_GetMaxClockError(void)
|
||||||
|
{
|
||||||
|
return max_clock_error;
|
||||||
|
}
|
||||||
|
|
||||||
|
/* ================================================== */
|
||||||
|
|
||||||
|
void
|
||||||
|
LCL_AddParameterChangeHandler(LCL_ParameterChangeHandler handler, void *anything)
|
||||||
|
{
|
||||||
|
ChangeListEntry *ptr, *new_entry;
|
||||||
|
|
||||||
|
/* Check that the handler is not already registered */
|
||||||
|
for (ptr = change_list.next; ptr != &change_list; ptr = ptr->next) {
|
||||||
|
if (!(ptr->handler != handler || ptr->anything != anything)) {
|
||||||
|
assert(0);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
new_entry = MallocNew(ChangeListEntry);
|
||||||
|
|
||||||
|
new_entry->handler = handler;
|
||||||
|
new_entry->anything = anything;
|
||||||
|
|
||||||
|
/* Chain it into the list */
|
||||||
|
new_entry->next = &change_list;
|
||||||
|
new_entry->prev = change_list.prev;
|
||||||
|
change_list.prev->next = new_entry;
|
||||||
|
change_list.prev = new_entry;
|
||||||
|
}
|
||||||
|
|
||||||
|
/* ================================================== */
|
||||||
|
|
||||||
|
/* Remove a handler */
|
||||||
|
void LCL_RemoveParameterChangeHandler(LCL_ParameterChangeHandler handler, void *anything)
|
||||||
|
{
|
||||||
|
|
||||||
|
ChangeListEntry *ptr;
|
||||||
|
int ok;
|
||||||
|
|
||||||
|
ptr = NULL;
|
||||||
|
ok = 0;
|
||||||
|
|
||||||
|
for (ptr = change_list.next; ptr != &change_list; ptr = ptr->next) {
|
||||||
|
if (ptr->handler == handler && ptr->anything == anything) {
|
||||||
|
ok = 1;
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
assert(ok);
|
||||||
|
|
||||||
|
/* Unlink entry from the list */
|
||||||
|
ptr->next->prev = ptr->prev;
|
||||||
|
ptr->prev->next = ptr->next;
|
||||||
|
|
||||||
|
Free(ptr);
|
||||||
|
}
|
||||||
|
|
||||||
|
/* ================================================== */
|
||||||
|
|
||||||
|
int
|
||||||
|
LCL_IsFirstParameterChangeHandler(LCL_ParameterChangeHandler handler)
|
||||||
|
{
|
||||||
|
return change_list.next->handler == handler;
|
||||||
|
}
|
||||||
|
|
||||||
|
/* ================================================== */
|
||||||
|
|
||||||
|
static void
|
||||||
|
invoke_parameter_change_handlers(struct timespec *raw, struct timespec *cooked,
|
||||||
|
double dfreq, double doffset,
|
||||||
|
LCL_ChangeType change_type)
|
||||||
|
{
|
||||||
|
ChangeListEntry *ptr;
|
||||||
|
|
||||||
|
for (ptr = change_list.next; ptr != &change_list; ptr = ptr->next) {
|
||||||
|
(ptr->handler)(raw, cooked, dfreq, doffset, change_type, ptr->anything);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/* ================================================== */
|
||||||
|
|
||||||
|
void
|
||||||
|
LCL_AddDispersionNotifyHandler(LCL_DispersionNotifyHandler handler, void *anything)
|
||||||
|
{
|
||||||
|
DispersionNotifyListEntry *ptr, *new_entry;
|
||||||
|
|
||||||
|
/* Check that the handler is not already registered */
|
||||||
|
for (ptr = dispersion_notify_list.next; ptr != &dispersion_notify_list; ptr = ptr->next) {
|
||||||
|
if (!(ptr->handler != handler || ptr->anything != anything)) {
|
||||||
|
assert(0);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
new_entry = MallocNew(DispersionNotifyListEntry);
|
||||||
|
|
||||||
|
new_entry->handler = handler;
|
||||||
|
new_entry->anything = anything;
|
||||||
|
|
||||||
|
/* Chain it into the list */
|
||||||
|
new_entry->next = &dispersion_notify_list;
|
||||||
|
new_entry->prev = dispersion_notify_list.prev;
|
||||||
|
dispersion_notify_list.prev->next = new_entry;
|
||||||
|
dispersion_notify_list.prev = new_entry;
|
||||||
|
}
|
||||||
|
|
||||||
|
/* ================================================== */
|
||||||
|
|
||||||
|
/* Remove a handler */
|
||||||
|
extern
|
||||||
|
void LCL_RemoveDispersionNotifyHandler(LCL_DispersionNotifyHandler handler, void *anything)
|
||||||
|
{
|
||||||
|
|
||||||
|
DispersionNotifyListEntry *ptr;
|
||||||
|
int ok;
|
||||||
|
|
||||||
|
ptr = NULL;
|
||||||
|
ok = 0;
|
||||||
|
|
||||||
|
for (ptr = dispersion_notify_list.next; ptr != &dispersion_notify_list; ptr = ptr->next) {
|
||||||
|
if (ptr->handler == handler && ptr->anything == anything) {
|
||||||
|
ok = 1;
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
assert(ok);
|
||||||
|
|
||||||
|
/* Unlink entry from the list */
|
||||||
|
ptr->next->prev = ptr->prev;
|
||||||
|
ptr->prev->next = ptr->next;
|
||||||
|
|
||||||
|
Free(ptr);
|
||||||
|
}
|
||||||
|
|
||||||
|
/* ================================================== */
|
||||||
|
|
||||||
|
void
|
||||||
|
LCL_ReadRawTime(struct timespec *ts)
|
||||||
|
{
|
||||||
|
#if HAVE_CLOCK_GETTIME
|
||||||
|
if (clock_gettime(CLOCK_REALTIME, ts) < 0)
|
||||||
|
LOG_FATAL("clock_gettime() failed : %s", strerror(errno));
|
||||||
|
#else
|
||||||
|
struct timeval tv;
|
||||||
|
|
||||||
|
if (gettimeofday(&tv, NULL) < 0)
|
||||||
|
LOG_FATAL("gettimeofday() failed : %s", strerror(errno));
|
||||||
|
|
||||||
|
UTI_TimevalToTimespec(&tv, ts);
|
||||||
|
#endif
|
||||||
|
}
|
||||||
|
|
||||||
|
/* ================================================== */
|
||||||
|
|
||||||
|
void
|
||||||
|
LCL_ReadCookedTime(struct timespec *result, double *err)
|
||||||
|
{
|
||||||
|
struct timespec raw;
|
||||||
|
|
||||||
|
LCL_ReadRawTime(&raw);
|
||||||
|
LCL_CookTime(&raw, result, err);
|
||||||
|
}
|
||||||
|
|
||||||
|
/* ================================================== */
|
||||||
|
|
||||||
|
void
|
||||||
|
LCL_CookTime(struct timespec *raw, struct timespec *cooked, double *err)
|
||||||
|
{
|
||||||
|
double correction;
|
||||||
|
|
||||||
|
LCL_GetOffsetCorrection(raw, &correction, err);
|
||||||
|
UTI_AddDoubleToTimespec(raw, correction, cooked);
|
||||||
|
}
|
||||||
|
|
||||||
|
/* ================================================== */
|
||||||
|
|
||||||
|
void
|
||||||
|
LCL_GetOffsetCorrection(struct timespec *raw, double *correction, double *err)
|
||||||
|
{
|
||||||
|
/* Call system specific driver to get correction */
|
||||||
|
(*drv_offset_convert)(raw, correction, err);
|
||||||
|
}
|
||||||
|
|
||||||
|
/* ================================================== */
|
||||||
|
/* Return current frequency */
|
||||||
|
|
||||||
|
double
|
||||||
|
LCL_ReadAbsoluteFrequency(void)
|
||||||
|
{
|
||||||
|
double freq;
|
||||||
|
|
||||||
|
freq = current_freq_ppm;
|
||||||
|
|
||||||
|
/* Undo temperature compensation */
|
||||||
|
if (temp_comp_ppm != 0.0) {
|
||||||
|
freq = (freq + temp_comp_ppm) / (1.0 - 1.0e-6 * temp_comp_ppm);
|
||||||
|
}
|
||||||
|
|
||||||
|
return freq;
|
||||||
|
}
|
||||||
|
|
||||||
|
/* ================================================== */
|
||||||
|
|
||||||
|
static double
|
||||||
|
clamp_freq(double freq)
|
||||||
|
{
|
||||||
|
if (freq <= max_freq_ppm && freq >= -max_freq_ppm)
|
||||||
|
return freq;
|
||||||
|
|
||||||
|
LOG(LOGS_WARN, "Frequency %.1f ppm exceeds allowed maximum", freq);
|
||||||
|
|
||||||
|
return CLAMP(-max_freq_ppm, freq, max_freq_ppm);
|
||||||
|
}
|
||||||
|
|
||||||
|
/* ================================================== */
|
||||||
|
|
||||||
|
static int
|
||||||
|
check_offset(struct timespec *now, double offset)
|
||||||
|
{
|
||||||
|
/* Check if the time will be still sane with accumulated offset */
|
||||||
|
if (UTI_IsTimeOffsetSane(now, -offset))
|
||||||
|
return 1;
|
||||||
|
|
||||||
|
LOG(LOGS_WARN, "Adjustment of %.1f seconds is invalid", -offset);
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
/* ================================================== */
|
||||||
|
|
||||||
|
/* This involves both setting the absolute frequency with the
|
||||||
|
system-specific driver, as well as calling all notify handlers */
|
||||||
|
|
||||||
|
void
|
||||||
|
LCL_SetAbsoluteFrequency(double afreq_ppm)
|
||||||
|
{
|
||||||
|
struct timespec raw, cooked;
|
||||||
|
double dfreq;
|
||||||
|
|
||||||
|
afreq_ppm = clamp_freq(afreq_ppm);
|
||||||
|
|
||||||
|
/* Apply temperature compensation */
|
||||||
|
if (temp_comp_ppm != 0.0) {
|
||||||
|
afreq_ppm = afreq_ppm * (1.0 - 1.0e-6 * temp_comp_ppm) - temp_comp_ppm;
|
||||||
|
}
|
||||||
|
|
||||||
|
/* Call the system-specific driver for setting the frequency */
|
||||||
|
|
||||||
|
afreq_ppm = (*drv_set_freq)(afreq_ppm);
|
||||||
|
|
||||||
|
dfreq = (afreq_ppm - current_freq_ppm) / (1.0e6 - current_freq_ppm);
|
||||||
|
|
||||||
|
LCL_ReadRawTime(&raw);
|
||||||
|
LCL_CookTime(&raw, &cooked, NULL);
|
||||||
|
|
||||||
|
/* Dispatch to all handlers */
|
||||||
|
invoke_parameter_change_handlers(&raw, &cooked, dfreq, 0.0, LCL_ChangeAdjust);
|
||||||
|
|
||||||
|
current_freq_ppm = afreq_ppm;
|
||||||
|
|
||||||
|
}
|
||||||
|
|
||||||
|
/* ================================================== */
|
||||||
|
|
||||||
|
void
|
||||||
|
LCL_AccumulateDeltaFrequency(double dfreq)
|
||||||
|
{
|
||||||
|
struct timespec raw, cooked;
|
||||||
|
double old_freq_ppm;
|
||||||
|
|
||||||
|
old_freq_ppm = current_freq_ppm;
|
||||||
|
|
||||||
|
/* Work out new absolute frequency. Note that absolute frequencies
|
||||||
|
are handled in units of ppm, whereas the 'dfreq' argument is in
|
||||||
|
terms of the gradient of the (offset) v (local time) function. */
|
||||||
|
|
||||||
|
current_freq_ppm += dfreq * (1.0e6 - current_freq_ppm);
|
||||||
|
|
||||||
|
current_freq_ppm = clamp_freq(current_freq_ppm);
|
||||||
|
|
||||||
|
/* Call the system-specific driver for setting the frequency */
|
||||||
|
current_freq_ppm = (*drv_set_freq)(current_freq_ppm);
|
||||||
|
dfreq = (current_freq_ppm - old_freq_ppm) / (1.0e6 - old_freq_ppm);
|
||||||
|
|
||||||
|
LCL_ReadRawTime(&raw);
|
||||||
|
LCL_CookTime(&raw, &cooked, NULL);
|
||||||
|
|
||||||
|
/* Dispatch to all handlers */
|
||||||
|
invoke_parameter_change_handlers(&raw, &cooked, dfreq, 0.0, LCL_ChangeAdjust);
|
||||||
|
}
|
||||||
|
|
||||||
|
/* ================================================== */
|
||||||
|
|
||||||
|
int
|
||||||
|
LCL_AccumulateOffset(double offset, double corr_rate)
|
||||||
|
{
|
||||||
|
struct timespec raw, cooked;
|
||||||
|
|
||||||
|
/* In this case, the cooked time to be passed to the notify clients
|
||||||
|
has to be the cooked time BEFORE the change was made */
|
||||||
|
|
||||||
|
LCL_ReadRawTime(&raw);
|
||||||
|
LCL_CookTime(&raw, &cooked, NULL);
|
||||||
|
|
||||||
|
if (!check_offset(&cooked, offset))
|
||||||
|
return 0;
|
||||||
|
|
||||||
|
(*drv_accrue_offset)(offset, corr_rate);
|
||||||
|
|
||||||
|
/* Dispatch to all handlers */
|
||||||
|
invoke_parameter_change_handlers(&raw, &cooked, 0.0, offset, LCL_ChangeAdjust);
|
||||||
|
|
||||||
|
return 1;
|
||||||
|
}
|
||||||
|
|
||||||
|
/* ================================================== */
|
||||||
|
|
||||||
|
int
|
||||||
|
LCL_ApplyStepOffset(double offset)
|
||||||
|
{
|
||||||
|
struct timespec raw, cooked;
|
||||||
|
|
||||||
|
/* In this case, the cooked time to be passed to the notify clients
|
||||||
|
has to be the cooked time BEFORE the change was made */
|
||||||
|
|
||||||
|
LCL_ReadRawTime(&raw);
|
||||||
|
LCL_CookTime(&raw, &cooked, NULL);
|
||||||
|
|
||||||
|
if (!check_offset(&raw, offset))
|
||||||
|
return 0;
|
||||||
|
|
||||||
|
if (!(*drv_apply_step_offset)(offset)) {
|
||||||
|
LOG(LOGS_ERR, "Could not step system clock");
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
/* Reset smoothing on all clock steps */
|
||||||
|
SMT_Reset(&cooked);
|
||||||
|
|
||||||
|
/* Dispatch to all handlers */
|
||||||
|
invoke_parameter_change_handlers(&raw, &cooked, 0.0, offset, LCL_ChangeStep);
|
||||||
|
|
||||||
|
return 1;
|
||||||
|
}
|
||||||
|
|
||||||
|
/* ================================================== */
|
||||||
|
|
||||||
|
void
|
||||||
|
LCL_NotifyExternalTimeStep(struct timespec *raw, struct timespec *cooked,
|
||||||
|
double offset, double dispersion)
|
||||||
|
{
|
||||||
|
LCL_CancelOffsetCorrection();
|
||||||
|
|
||||||
|
/* Dispatch to all handlers */
|
||||||
|
invoke_parameter_change_handlers(raw, cooked, 0.0, offset, LCL_ChangeUnknownStep);
|
||||||
|
|
||||||
|
lcl_InvokeDispersionNotifyHandlers(dispersion);
|
||||||
|
}
|
||||||
|
|
||||||
|
/* ================================================== */
|
||||||
|
|
||||||
|
void
|
||||||
|
LCL_NotifyLeap(int leap)
|
||||||
|
{
|
||||||
|
struct timespec raw, cooked;
|
||||||
|
|
||||||
|
LCL_ReadRawTime(&raw);
|
||||||
|
LCL_CookTime(&raw, &cooked, NULL);
|
||||||
|
|
||||||
|
/* Smooth the leap second out */
|
||||||
|
SMT_Leap(&cooked, leap);
|
||||||
|
|
||||||
|
/* Dispatch to all handlers as if the clock was stepped */
|
||||||
|
invoke_parameter_change_handlers(&raw, &cooked, 0.0, -leap, LCL_ChangeStep);
|
||||||
|
}
|
||||||
|
|
||||||
|
/* ================================================== */
|
||||||
|
|
||||||
|
int
|
||||||
|
LCL_AccumulateFrequencyAndOffset(double dfreq, double doffset, double corr_rate)
|
||||||
|
{
|
||||||
|
struct timespec raw, cooked;
|
||||||
|
double old_freq_ppm;
|
||||||
|
|
||||||
|
LCL_ReadRawTime(&raw);
|
||||||
|
/* Due to modifying the offset, this has to be the cooked time prior
|
||||||
|
to the change we are about to make */
|
||||||
|
LCL_CookTime(&raw, &cooked, NULL);
|
||||||
|
|
||||||
|
if (!check_offset(&cooked, doffset))
|
||||||
|
return 0;
|
||||||
|
|
||||||
|
old_freq_ppm = current_freq_ppm;
|
||||||
|
|
||||||
|
/* Work out new absolute frequency. Note that absolute frequencies
|
||||||
|
are handled in units of ppm, whereas the 'dfreq' argument is in
|
||||||
|
terms of the gradient of the (offset) v (local time) function. */
|
||||||
|
current_freq_ppm += dfreq * (1.0e6 - current_freq_ppm);
|
||||||
|
|
||||||
|
current_freq_ppm = clamp_freq(current_freq_ppm);
|
||||||
|
|
||||||
|
DEBUG_LOG("old_freq=%.3fppm new_freq=%.3fppm offset=%.6fsec",
|
||||||
|
old_freq_ppm, current_freq_ppm, doffset);
|
||||||
|
|
||||||
|
/* Call the system-specific driver for setting the frequency */
|
||||||
|
current_freq_ppm = (*drv_set_freq)(current_freq_ppm);
|
||||||
|
dfreq = (current_freq_ppm - old_freq_ppm) / (1.0e6 - old_freq_ppm);
|
||||||
|
|
||||||
|
(*drv_accrue_offset)(doffset, corr_rate);
|
||||||
|
|
||||||
|
/* Dispatch to all handlers */
|
||||||
|
invoke_parameter_change_handlers(&raw, &cooked, dfreq, doffset, LCL_ChangeAdjust);
|
||||||
|
|
||||||
|
return 1;
|
||||||
|
}
|
||||||
|
|
||||||
|
/* ================================================== */
|
||||||
|
|
||||||
|
int
|
||||||
|
LCL_AccumulateFrequencyAndOffsetNoHandlers(double dfreq, double doffset, double corr_rate)
|
||||||
|
{
|
||||||
|
ChangeListEntry *first_handler;
|
||||||
|
int r;
|
||||||
|
|
||||||
|
first_handler = change_list.next;
|
||||||
|
change_list.next = &change_list;
|
||||||
|
|
||||||
|
r = LCL_AccumulateFrequencyAndOffset(dfreq, doffset, corr_rate);
|
||||||
|
|
||||||
|
change_list.next = first_handler;
|
||||||
|
|
||||||
|
return r;
|
||||||
|
}
|
||||||
|
|
||||||
|
/* ================================================== */
|
||||||
|
|
||||||
|
void
|
||||||
|
lcl_InvokeDispersionNotifyHandlers(double dispersion)
|
||||||
|
{
|
||||||
|
DispersionNotifyListEntry *ptr;
|
||||||
|
|
||||||
|
for (ptr = dispersion_notify_list.next; ptr != &dispersion_notify_list; ptr = ptr->next) {
|
||||||
|
(ptr->handler)(dispersion, ptr->anything);
|
||||||
|
}
|
||||||
|
|
||||||
|
}
|
||||||
|
|
||||||
|
/* ================================================== */
|
||||||
|
|
||||||
|
void
|
||||||
|
lcl_RegisterSystemDrivers(lcl_ReadFrequencyDriver read_freq,
|
||||||
|
lcl_SetFrequencyDriver set_freq,
|
||||||
|
lcl_AccrueOffsetDriver accrue_offset,
|
||||||
|
lcl_ApplyStepOffsetDriver apply_step_offset,
|
||||||
|
lcl_OffsetCorrectionDriver offset_convert,
|
||||||
|
lcl_SetLeapDriver set_leap,
|
||||||
|
lcl_SetSyncStatusDriver set_sync_status)
|
||||||
|
{
|
||||||
|
drv_read_freq = read_freq;
|
||||||
|
drv_set_freq = set_freq;
|
||||||
|
drv_accrue_offset = accrue_offset;
|
||||||
|
drv_apply_step_offset = apply_step_offset;
|
||||||
|
drv_offset_convert = offset_convert;
|
||||||
|
drv_set_leap = set_leap;
|
||||||
|
drv_set_sync_status = set_sync_status;
|
||||||
|
|
||||||
|
current_freq_ppm = (*drv_read_freq)();
|
||||||
|
|
||||||
|
DEBUG_LOG("Local freq=%.3fppm", current_freq_ppm);
|
||||||
|
}
|
||||||
|
|
||||||
|
/* ================================================== */
|
||||||
|
/* Look at the current difference between the system time and the NTP
|
||||||
|
time, and make a step to cancel it. */
|
||||||
|
|
||||||
|
int
|
||||||
|
LCL_MakeStep(void)
|
||||||
|
{
|
||||||
|
struct timespec raw;
|
||||||
|
double correction;
|
||||||
|
|
||||||
|
LCL_ReadRawTime(&raw);
|
||||||
|
LCL_GetOffsetCorrection(&raw, &correction, NULL);
|
||||||
|
|
||||||
|
if (!check_offset(&raw, -correction))
|
||||||
|
return 0;
|
||||||
|
|
||||||
|
/* Cancel remaining slew and make the step */
|
||||||
|
LCL_AccumulateOffset(correction, 0.0);
|
||||||
|
if (!LCL_ApplyStepOffset(-correction))
|
||||||
|
return 0;
|
||||||
|
|
||||||
|
LOG(LOGS_WARN, "System clock was stepped by %.6f seconds", correction);
|
||||||
|
|
||||||
|
return 1;
|
||||||
|
}
|
||||||
|
|
||||||
|
/* ================================================== */
|
||||||
|
|
||||||
|
void
|
||||||
|
LCL_CancelOffsetCorrection(void)
|
||||||
|
{
|
||||||
|
struct timespec raw;
|
||||||
|
double correction;
|
||||||
|
|
||||||
|
LCL_ReadRawTime(&raw);
|
||||||
|
LCL_GetOffsetCorrection(&raw, &correction, NULL);
|
||||||
|
LCL_AccumulateOffset(correction, 0.0);
|
||||||
|
}
|
||||||
|
|
||||||
|
/* ================================================== */
|
||||||
|
|
||||||
|
int
|
||||||
|
LCL_CanSystemLeap(void)
|
||||||
|
{
|
||||||
|
return drv_set_leap ? 1 : 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
/* ================================================== */
|
||||||
|
|
||||||
|
void
|
||||||
|
LCL_SetSystemLeap(int leap, int tai_offset)
|
||||||
|
{
|
||||||
|
if (drv_set_leap) {
|
||||||
|
(drv_set_leap)(leap, tai_offset);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/* ================================================== */
|
||||||
|
|
||||||
|
double
|
||||||
|
LCL_SetTempComp(double comp)
|
||||||
|
{
|
||||||
|
double uncomp_freq_ppm;
|
||||||
|
|
||||||
|
if (temp_comp_ppm == comp)
|
||||||
|
return comp;
|
||||||
|
|
||||||
|
/* Undo previous compensation */
|
||||||
|
current_freq_ppm = (current_freq_ppm + temp_comp_ppm) /
|
||||||
|
(1.0 - 1.0e-6 * temp_comp_ppm);
|
||||||
|
|
||||||
|
uncomp_freq_ppm = current_freq_ppm;
|
||||||
|
|
||||||
|
/* Apply new compensation */
|
||||||
|
current_freq_ppm = current_freq_ppm * (1.0 - 1.0e-6 * comp) - comp;
|
||||||
|
|
||||||
|
/* Call the system-specific driver for setting the frequency */
|
||||||
|
current_freq_ppm = (*drv_set_freq)(current_freq_ppm);
|
||||||
|
|
||||||
|
temp_comp_ppm = (uncomp_freq_ppm - current_freq_ppm) /
|
||||||
|
(1.0e-6 * uncomp_freq_ppm + 1.0);
|
||||||
|
|
||||||
|
return temp_comp_ppm;
|
||||||
|
}
|
||||||
|
|
||||||
|
/* ================================================== */
|
||||||
|
|
||||||
|
void
|
||||||
|
LCL_SetSyncStatus(int synchronised, double est_error, double max_error)
|
||||||
|
{
|
||||||
|
if (drv_set_sync_status) {
|
||||||
|
(drv_set_sync_status)(synchronised, est_error, max_error);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/* ================================================== */
|
229
local.h
Normal file
229
local.h
Normal file
|
@ -0,0 +1,229 @@
|
||||||
|
/*
|
||||||
|
chronyd/chronyc - Programs for keeping computer clocks accurate.
|
||||||
|
|
||||||
|
**********************************************************************
|
||||||
|
* Copyright (C) Richard P. Curnow 1997-2002
|
||||||
|
*
|
||||||
|
* This program is free software; you can redistribute it and/or modify
|
||||||
|
* it under the terms of version 2 of the GNU General Public License as
|
||||||
|
* published by the Free Software Foundation.
|
||||||
|
*
|
||||||
|
* This program is distributed in the hope that it will be useful, but
|
||||||
|
* WITHOUT ANY WARRANTY; without even the implied warranty of
|
||||||
|
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
|
||||||
|
* General Public License for more details.
|
||||||
|
*
|
||||||
|
* You should have received a copy of the GNU General Public License along
|
||||||
|
* with this program; if not, write to the Free Software Foundation, Inc.,
|
||||||
|
* 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA.
|
||||||
|
*
|
||||||
|
**********************************************************************
|
||||||
|
|
||||||
|
=======================================================================
|
||||||
|
|
||||||
|
This module provides an interface to the system time, and
|
||||||
|
insulates the rest of the program from the different way
|
||||||
|
that interface has to be done on various operating systems.
|
||||||
|
*/
|
||||||
|
|
||||||
|
#ifndef GOT_LOCAL_H
|
||||||
|
#define GOT_LOCAL_H
|
||||||
|
|
||||||
|
#include "sysincl.h"
|
||||||
|
|
||||||
|
/* Read the system clock */
|
||||||
|
extern void LCL_ReadRawTime(struct timespec *ts);
|
||||||
|
|
||||||
|
/* Read the system clock, corrected according to all accumulated
|
||||||
|
drifts and uncompensated offsets.
|
||||||
|
|
||||||
|
In a kernel implementation with vernier frequency control (like
|
||||||
|
Linux), and if we were to apply offsets by stepping the clock, this
|
||||||
|
would be identical to raw time. In any other case (use of
|
||||||
|
adjtime()-like interface to correct offsets, and to adjust the
|
||||||
|
frequency), we must correct the raw time to get this value */
|
||||||
|
|
||||||
|
extern void LCL_ReadCookedTime(struct timespec *ts, double *err);
|
||||||
|
|
||||||
|
/* Convert raw time to cooked. */
|
||||||
|
extern void LCL_CookTime(struct timespec *raw, struct timespec *cooked, double *err);
|
||||||
|
|
||||||
|
/* Read the current offset between the system clock and true time
|
||||||
|
(i.e. 'cooked' - 'raw') (in seconds). */
|
||||||
|
|
||||||
|
extern void LCL_GetOffsetCorrection(struct timespec *raw, double *correction, double *err);
|
||||||
|
|
||||||
|
/* Type of routines that may be invoked as callbacks when there is a
|
||||||
|
change to the frequency or offset.
|
||||||
|
|
||||||
|
raw : raw local clock time at which change occurred
|
||||||
|
|
||||||
|
cooked : cooked local time at which change occurred
|
||||||
|
|
||||||
|
dfreq : delta frequency relative to previous value (in terms of
|
||||||
|
seconds gained by system clock per unit system clock time)
|
||||||
|
|
||||||
|
doffset : delta offset applied (positive => make local system fast
|
||||||
|
by that amount, negative => make it slow by that amount)
|
||||||
|
|
||||||
|
change_type : what type of change is being applied
|
||||||
|
|
||||||
|
anything : Passthrough argument from call to registration routine */
|
||||||
|
|
||||||
|
|
||||||
|
typedef enum {
|
||||||
|
LCL_ChangeAdjust,
|
||||||
|
LCL_ChangeStep,
|
||||||
|
LCL_ChangeUnknownStep
|
||||||
|
} LCL_ChangeType;
|
||||||
|
|
||||||
|
typedef void (*LCL_ParameterChangeHandler)
|
||||||
|
(struct timespec *raw, struct timespec *cooked,
|
||||||
|
double dfreq,
|
||||||
|
double doffset,
|
||||||
|
LCL_ChangeType change_type,
|
||||||
|
void *anything
|
||||||
|
);
|
||||||
|
|
||||||
|
/* Add a handler. Then handler MUST NOT deregister itself!!! */
|
||||||
|
extern void LCL_AddParameterChangeHandler(LCL_ParameterChangeHandler handler, void *anything);
|
||||||
|
|
||||||
|
/* Remove a handler */
|
||||||
|
extern void LCL_RemoveParameterChangeHandler(LCL_ParameterChangeHandler, void *anything);
|
||||||
|
|
||||||
|
/* Check if a handler is invoked first when dispatching */
|
||||||
|
extern int LCL_IsFirstParameterChangeHandler(LCL_ParameterChangeHandler handler);
|
||||||
|
|
||||||
|
/* Function type for handlers to be called back when an indeterminate
|
||||||
|
offset is introduced into the local time. This situation occurs
|
||||||
|
when the frequency must be adjusted to effect a clock slew and
|
||||||
|
there is doubt about one of the endpoints of the interval over
|
||||||
|
which the frequency change was applied.It is expected that such
|
||||||
|
handlers will add extra dispersion to any existing samples stored
|
||||||
|
in their registers.
|
||||||
|
|
||||||
|
dispersion : The bound on how much error has been introduced in the
|
||||||
|
local clock, in seconds.
|
||||||
|
|
||||||
|
anything : passthrough from the registration routine
|
||||||
|
|
||||||
|
*/
|
||||||
|
|
||||||
|
typedef void (*LCL_DispersionNotifyHandler)(double dispersion, void *anything);
|
||||||
|
|
||||||
|
/* Register a handler for being notified of dispersion being added to
|
||||||
|
the local clock. The handler MUST NOT unregister itself!!! */
|
||||||
|
|
||||||
|
extern void LCL_AddDispersionNotifyHandler(LCL_DispersionNotifyHandler handler, void *anything);
|
||||||
|
|
||||||
|
/* Delete a handler */
|
||||||
|
|
||||||
|
extern void LCL_RemoveDispersionNotifyHandler(LCL_DispersionNotifyHandler handler, void *anything);
|
||||||
|
|
||||||
|
|
||||||
|
/* Read the absolute system frequency, relative to the uncompensated
|
||||||
|
system. Returned in units of parts per million. Thus the result of
|
||||||
|
this is how many seconds fast the uncompensated system would be after
|
||||||
|
its own time has reached 1 million seconds from the start of the
|
||||||
|
measurement. */
|
||||||
|
extern double LCL_ReadAbsoluteFrequency(void);
|
||||||
|
|
||||||
|
/* Routine to set the absolute frequency. Only expected to be used
|
||||||
|
when either (i) reading the drift from a file at the start of a
|
||||||
|
run, or (ii) responsing to a user parameter 'poke'. This is
|
||||||
|
defined in ppm, as for the absolute frequency reading routine. */
|
||||||
|
|
||||||
|
extern void LCL_SetAbsoluteFrequency(double afreq);
|
||||||
|
|
||||||
|
/* Routine to apply a change of frequency to the local clock. The
|
||||||
|
argument is the estimated gain (positive) or loss (negative) of the
|
||||||
|
local clock relative to true time, per unit time of the PREVIOUS
|
||||||
|
frequency setting of the local clock. This is assumed to be based
|
||||||
|
on a regression of y=offset v x=cooked local time. */
|
||||||
|
|
||||||
|
extern void LCL_AccumulateDeltaFrequency(double dfreq);
|
||||||
|
|
||||||
|
/* Routine to apply an offset (in seconds) to the local clock. The
|
||||||
|
argument should be positive to move the clock backwards (i.e. the
|
||||||
|
local clock is currently fast of true time), or negative to move it
|
||||||
|
forwards (i.e. it is currently slow of true time). Provided is also
|
||||||
|
a suggested correction rate (correction time * offset). */
|
||||||
|
|
||||||
|
extern int LCL_AccumulateOffset(double offset, double corr_rate);
|
||||||
|
|
||||||
|
/* Routine to apply an immediate offset by doing a sudden step if
|
||||||
|
possible. (Intended for use after an initial estimate of offset has
|
||||||
|
been obtained, so that we don't end up using adjtime to achieve a
|
||||||
|
slew of an hour or something like that). A positive argument means
|
||||||
|
the system clock is fast on true time, i.e. it needs to be stepped
|
||||||
|
backwards. (Same convention as for AccumulateOffset routine). */
|
||||||
|
|
||||||
|
extern int LCL_ApplyStepOffset(double offset);
|
||||||
|
|
||||||
|
/* Routine to invoke notify handlers on an unexpected time jump
|
||||||
|
in system clock */
|
||||||
|
extern void LCL_NotifyExternalTimeStep(struct timespec *raw, struct timespec *cooked,
|
||||||
|
double offset, double dispersion);
|
||||||
|
|
||||||
|
/* Routine to invoke notify handlers on leap second when the system clock
|
||||||
|
doesn't correct itself */
|
||||||
|
extern void LCL_NotifyLeap(int leap);
|
||||||
|
|
||||||
|
/* Perform the combination of modifying the frequency and applying
|
||||||
|
a slew, in one easy step */
|
||||||
|
extern int LCL_AccumulateFrequencyAndOffset(double dfreq, double doffset, double corr_rate);
|
||||||
|
|
||||||
|
/* Same as the routine above, except it does not call the registered
|
||||||
|
parameter change handlers */
|
||||||
|
extern int LCL_AccumulateFrequencyAndOffsetNoHandlers(double dfreq, double doffset,
|
||||||
|
double corr_rate);
|
||||||
|
|
||||||
|
/* Routine to read the system precision as a log to base 2 value. */
|
||||||
|
extern int LCL_GetSysPrecisionAsLog(void);
|
||||||
|
|
||||||
|
/* Routine to read the system precision in terms of the actual time step */
|
||||||
|
extern double LCL_GetSysPrecisionAsQuantum(void);
|
||||||
|
|
||||||
|
/* Routine to read the maximum frequency error of the local clock. This
|
||||||
|
is a frequency stability, not an absolute error. */
|
||||||
|
extern double LCL_GetMaxClockError(void);
|
||||||
|
|
||||||
|
/* Routine to initialise the module (to be called once at program
|
||||||
|
start-up) */
|
||||||
|
|
||||||
|
extern void LCL_Initialise(void);
|
||||||
|
|
||||||
|
/* Routine to finalise the module (to be called once at end of
|
||||||
|
run). */
|
||||||
|
extern void LCL_Finalise(void);
|
||||||
|
|
||||||
|
/* Routine to convert the outstanding system clock error to a step and
|
||||||
|
apply it, e.g. if the system clock has ended up an hour wrong due
|
||||||
|
to a timezone problem. */
|
||||||
|
extern int LCL_MakeStep(void);
|
||||||
|
|
||||||
|
/* Routine to cancel the outstanding system clock correction */
|
||||||
|
extern void LCL_CancelOffsetCorrection(void);
|
||||||
|
|
||||||
|
/* Check if the system driver supports leap seconds, i.e. LCL_SetSystemLeap
|
||||||
|
does something */
|
||||||
|
extern int LCL_CanSystemLeap(void);
|
||||||
|
|
||||||
|
/* Routine to set the system clock to correct itself for a leap second and also
|
||||||
|
set its TAI-UTC offset. If supported, leap second will be inserted at the
|
||||||
|
end of the day if the argument is positive, deleted if negative, and zero
|
||||||
|
resets the setting. */
|
||||||
|
extern void LCL_SetSystemLeap(int leap, int tai_offset);
|
||||||
|
|
||||||
|
/* Routine to set a frequency correction (in ppm) that should be applied
|
||||||
|
to local clock to compensate for temperature changes. A positive
|
||||||
|
argument means that the clock frequency should be increased. Return the
|
||||||
|
actual compensation (may be different from the requested compensation
|
||||||
|
due to clamping or rounding). */
|
||||||
|
extern double LCL_SetTempComp(double comp);
|
||||||
|
|
||||||
|
/* Routine to update the synchronisation status in the kernel to allow other
|
||||||
|
applications to know if the system clock is synchronised and error bounds */
|
||||||
|
extern void LCL_SetSyncStatus(int synchronised, double est_error, double max_error);
|
||||||
|
|
||||||
|
#endif /* GOT_LOCAL_H */
|
74
localp.h
Normal file
74
localp.h
Normal file
|
@ -0,0 +1,74 @@
|
||||||
|
/*
|
||||||
|
chronyd/chronyc - Programs for keeping computer clocks accurate.
|
||||||
|
|
||||||
|
**********************************************************************
|
||||||
|
* Copyright (C) Richard P. Curnow 1997-2002
|
||||||
|
*
|
||||||
|
* This program is free software; you can redistribute it and/or modify
|
||||||
|
* it under the terms of version 2 of the GNU General Public License as
|
||||||
|
* published by the Free Software Foundation.
|
||||||
|
*
|
||||||
|
* This program is distributed in the hope that it will be useful, but
|
||||||
|
* WITHOUT ANY WARRANTY; without even the implied warranty of
|
||||||
|
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
|
||||||
|
* General Public License for more details.
|
||||||
|
*
|
||||||
|
* You should have received a copy of the GNU General Public License along
|
||||||
|
* with this program; if not, write to the Free Software Foundation, Inc.,
|
||||||
|
* 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA.
|
||||||
|
*
|
||||||
|
**********************************************************************
|
||||||
|
|
||||||
|
=======================================================================
|
||||||
|
|
||||||
|
Private include file for local.c and all system dependent
|
||||||
|
driver modules.
|
||||||
|
*/
|
||||||
|
|
||||||
|
|
||||||
|
#ifndef GOT_LOCALP_H
|
||||||
|
#define GOT_LOCALP_H
|
||||||
|
|
||||||
|
/* System driver to read the current local frequency, in ppm relative
|
||||||
|
to nominal. A positive value indicates that the local clock runs
|
||||||
|
fast when uncompensated. */
|
||||||
|
typedef double (*lcl_ReadFrequencyDriver)(void);
|
||||||
|
|
||||||
|
/* System driver to set the current local frequency, in ppm relative
|
||||||
|
to nominal. A positive value indicates that the local clock runs
|
||||||
|
fast when uncompensated. Return actual frequency (may be different
|
||||||
|
from the requested frequency due to clamping or rounding). */
|
||||||
|
typedef double (*lcl_SetFrequencyDriver)(double freq_ppm);
|
||||||
|
|
||||||
|
/* System driver to accrue an offset. A positive argument means slew
|
||||||
|
the clock forwards. The suggested correction rate of time to correct the
|
||||||
|
offset is given in 'corr_rate'. */
|
||||||
|
typedef void (*lcl_AccrueOffsetDriver)(double offset, double corr_rate);
|
||||||
|
|
||||||
|
/* System driver to apply a step offset. A positive argument means step
|
||||||
|
the clock forwards. */
|
||||||
|
typedef int (*lcl_ApplyStepOffsetDriver)(double offset);
|
||||||
|
|
||||||
|
/* System driver to convert a raw time to an adjusted (cooked) time.
|
||||||
|
The number of seconds returned in 'corr' have to be added to the
|
||||||
|
raw time to get the corrected time */
|
||||||
|
typedef void (*lcl_OffsetCorrectionDriver)(struct timespec *raw, double *corr, double *err);
|
||||||
|
|
||||||
|
/* System driver to schedule leap seconds and set TAI-UTC offset */
|
||||||
|
typedef void (*lcl_SetLeapDriver)(int leap, int tai_offset);
|
||||||
|
|
||||||
|
/* System driver to set the synchronisation status */
|
||||||
|
typedef void (*lcl_SetSyncStatusDriver)(int synchronised, double est_error, double max_error);
|
||||||
|
|
||||||
|
extern void lcl_InvokeDispersionNotifyHandlers(double dispersion);
|
||||||
|
|
||||||
|
extern void
|
||||||
|
lcl_RegisterSystemDrivers(lcl_ReadFrequencyDriver read_freq,
|
||||||
|
lcl_SetFrequencyDriver set_freq,
|
||||||
|
lcl_AccrueOffsetDriver accrue_offset,
|
||||||
|
lcl_ApplyStepOffsetDriver apply_step_offset,
|
||||||
|
lcl_OffsetCorrectionDriver offset_convert,
|
||||||
|
lcl_SetLeapDriver set_leap,
|
||||||
|
lcl_SetSyncStatusDriver set_sync_status);
|
||||||
|
|
||||||
|
#endif /* GOT_LOCALP_H */
|
398
logging.c
Normal file
398
logging.c
Normal file
|
@ -0,0 +1,398 @@
|
||||||
|
/*
|
||||||
|
chronyd/chronyc - Programs for keeping computer clocks accurate.
|
||||||
|
|
||||||
|
**********************************************************************
|
||||||
|
* Copyright (C) Richard P. Curnow 1997-2003
|
||||||
|
* Copyright (C) Miroslav Lichvar 2011-2014, 2018-2020
|
||||||
|
*
|
||||||
|
* This program is free software; you can redistribute it and/or modify
|
||||||
|
* it under the terms of version 2 of the GNU General Public License as
|
||||||
|
* published by the Free Software Foundation.
|
||||||
|
*
|
||||||
|
* This program is distributed in the hope that it will be useful, but
|
||||||
|
* WITHOUT ANY WARRANTY; without even the implied warranty of
|
||||||
|
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
|
||||||
|
* General Public License for more details.
|
||||||
|
*
|
||||||
|
* You should have received a copy of the GNU General Public License along
|
||||||
|
* with this program; if not, write to the Free Software Foundation, Inc.,
|
||||||
|
* 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA.
|
||||||
|
*
|
||||||
|
**********************************************************************
|
||||||
|
|
||||||
|
=======================================================================
|
||||||
|
|
||||||
|
Module to handle logging of diagnostic information
|
||||||
|
*/
|
||||||
|
|
||||||
|
#include "config.h"
|
||||||
|
|
||||||
|
#include "sysincl.h"
|
||||||
|
|
||||||
|
#include <syslog.h>
|
||||||
|
|
||||||
|
#include "conf.h"
|
||||||
|
#include "logging.h"
|
||||||
|
#include "memory.h"
|
||||||
|
#include "util.h"
|
||||||
|
|
||||||
|
/* This is used by DEBUG_LOG macro */
|
||||||
|
LOG_Severity log_min_severity = LOGS_INFO;
|
||||||
|
|
||||||
|
/* Current logging contexts */
|
||||||
|
static LOG_Context log_contexts;
|
||||||
|
|
||||||
|
/* ================================================== */
|
||||||
|
/* Flag indicating we have initialised */
|
||||||
|
static int initialised = 0;
|
||||||
|
|
||||||
|
static FILE *file_log = NULL;
|
||||||
|
static int system_log = 0;
|
||||||
|
|
||||||
|
static int parent_fd = 0;
|
||||||
|
|
||||||
|
struct LogFile {
|
||||||
|
const char *name;
|
||||||
|
const char *banner;
|
||||||
|
FILE *file;
|
||||||
|
unsigned long writes;
|
||||||
|
};
|
||||||
|
|
||||||
|
static int n_filelogs = 0;
|
||||||
|
|
||||||
|
/* Increase this when adding a new logfile */
|
||||||
|
#define MAX_FILELOGS 6
|
||||||
|
|
||||||
|
static struct LogFile logfiles[MAX_FILELOGS];
|
||||||
|
|
||||||
|
/* Global prefix for debug messages */
|
||||||
|
static char *debug_prefix;
|
||||||
|
|
||||||
|
/* ================================================== */
|
||||||
|
/* Init function */
|
||||||
|
|
||||||
|
void
|
||||||
|
LOG_Initialise(void)
|
||||||
|
{
|
||||||
|
debug_prefix = Strdup("");
|
||||||
|
log_contexts = 0;
|
||||||
|
|
||||||
|
initialised = 1;
|
||||||
|
LOG_OpenFileLog(NULL);
|
||||||
|
}
|
||||||
|
|
||||||
|
/* ================================================== */
|
||||||
|
/* Fini function */
|
||||||
|
|
||||||
|
void
|
||||||
|
LOG_Finalise(void)
|
||||||
|
{
|
||||||
|
if (system_log)
|
||||||
|
closelog();
|
||||||
|
|
||||||
|
if (file_log)
|
||||||
|
fclose(file_log);
|
||||||
|
|
||||||
|
LOG_CycleLogFiles();
|
||||||
|
|
||||||
|
Free(debug_prefix);
|
||||||
|
|
||||||
|
initialised = 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
/* ================================================== */
|
||||||
|
|
||||||
|
static void log_message(int fatal, LOG_Severity severity, const char *message)
|
||||||
|
{
|
||||||
|
if (system_log) {
|
||||||
|
int priority;
|
||||||
|
switch (severity) {
|
||||||
|
case LOGS_DEBUG:
|
||||||
|
priority = LOG_DEBUG;
|
||||||
|
break;
|
||||||
|
case LOGS_INFO:
|
||||||
|
priority = LOG_INFO;
|
||||||
|
break;
|
||||||
|
case LOGS_WARN:
|
||||||
|
priority = LOG_WARNING;
|
||||||
|
break;
|
||||||
|
case LOGS_ERR:
|
||||||
|
priority = LOG_ERR;
|
||||||
|
break;
|
||||||
|
case LOGS_FATAL:
|
||||||
|
priority = LOG_CRIT;
|
||||||
|
break;
|
||||||
|
default:
|
||||||
|
assert(0);
|
||||||
|
}
|
||||||
|
syslog(priority, fatal ? "Fatal error : %s" : "%s", message);
|
||||||
|
} else if (file_log) {
|
||||||
|
fprintf(file_log, fatal ? "Fatal error : %s\n" : "%s\n", message);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/* ================================================== */
|
||||||
|
|
||||||
|
void LOG_Message(LOG_Severity severity,
|
||||||
|
#if DEBUG > 0
|
||||||
|
int line_number, const char *filename, const char *function_name,
|
||||||
|
#endif
|
||||||
|
const char *format, ...)
|
||||||
|
{
|
||||||
|
char buf[2048];
|
||||||
|
va_list other_args;
|
||||||
|
time_t t;
|
||||||
|
struct tm *tm;
|
||||||
|
|
||||||
|
assert(initialised);
|
||||||
|
severity = CLAMP(LOGS_DEBUG, severity, LOGS_FATAL);
|
||||||
|
|
||||||
|
if (!system_log && file_log && severity >= log_min_severity) {
|
||||||
|
/* Don't clutter up syslog with timestamps and internal debugging info */
|
||||||
|
time(&t);
|
||||||
|
tm = gmtime(&t);
|
||||||
|
if (tm) {
|
||||||
|
strftime(buf, sizeof (buf), "%Y-%m-%dT%H:%M:%SZ", tm);
|
||||||
|
fprintf(file_log, "%s ", buf);
|
||||||
|
}
|
||||||
|
#if DEBUG > 0
|
||||||
|
if (log_min_severity <= LOGS_DEBUG) {
|
||||||
|
/* Log severity to character mapping (debug, info, warn, err, fatal) */
|
||||||
|
const char severity_chars[LOGS_FATAL - LOGS_DEBUG + 1] = {'D', 'I', 'W', 'E', 'F'};
|
||||||
|
|
||||||
|
fprintf(file_log, "%c:%s%s:%d:(%s) ", severity_chars[severity - LOGS_DEBUG],
|
||||||
|
debug_prefix, filename, line_number, function_name);
|
||||||
|
}
|
||||||
|
#endif
|
||||||
|
}
|
||||||
|
|
||||||
|
va_start(other_args, format);
|
||||||
|
vsnprintf(buf, sizeof(buf), format, other_args);
|
||||||
|
va_end(other_args);
|
||||||
|
|
||||||
|
switch (severity) {
|
||||||
|
case LOGS_DEBUG:
|
||||||
|
case LOGS_INFO:
|
||||||
|
case LOGS_WARN:
|
||||||
|
case LOGS_ERR:
|
||||||
|
if (severity >= log_min_severity)
|
||||||
|
log_message(0, severity, buf);
|
||||||
|
break;
|
||||||
|
case LOGS_FATAL:
|
||||||
|
if (severity >= log_min_severity)
|
||||||
|
log_message(1, severity, buf);
|
||||||
|
|
||||||
|
/* Send the message also to the foreground process if it is
|
||||||
|
still running, or stderr if it is still open */
|
||||||
|
if (parent_fd > 0) {
|
||||||
|
if (!LOG_NotifyParent(buf))
|
||||||
|
; /* Not much we can do here */
|
||||||
|
} else if (system_log && parent_fd == 0) {
|
||||||
|
system_log = 0;
|
||||||
|
log_message(1, severity, buf);
|
||||||
|
}
|
||||||
|
exit(1);
|
||||||
|
break;
|
||||||
|
default:
|
||||||
|
assert(0);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/* ================================================== */
|
||||||
|
|
||||||
|
void
|
||||||
|
LOG_OpenFileLog(const char *log_file)
|
||||||
|
{
|
||||||
|
FILE *f;
|
||||||
|
|
||||||
|
if (log_file) {
|
||||||
|
f = UTI_OpenFile(NULL, log_file, NULL, 'A', 0640);
|
||||||
|
} else {
|
||||||
|
f = stderr;
|
||||||
|
}
|
||||||
|
|
||||||
|
/* Enable line buffering */
|
||||||
|
setvbuf(f, NULL, _IOLBF, BUFSIZ);
|
||||||
|
|
||||||
|
if (file_log && file_log != stderr)
|
||||||
|
fclose(file_log);
|
||||||
|
|
||||||
|
file_log = f;
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
/* ================================================== */
|
||||||
|
|
||||||
|
void
|
||||||
|
LOG_OpenSystemLog(void)
|
||||||
|
{
|
||||||
|
system_log = 1;
|
||||||
|
openlog("chronyd", LOG_PID, LOG_DAEMON);
|
||||||
|
}
|
||||||
|
|
||||||
|
/* ================================================== */
|
||||||
|
|
||||||
|
void LOG_SetMinSeverity(LOG_Severity severity)
|
||||||
|
{
|
||||||
|
/* Don't print any debug messages in a non-debug build */
|
||||||
|
log_min_severity = CLAMP(DEBUG > 0 ? LOGS_DEBUG : LOGS_INFO, severity, LOGS_FATAL);
|
||||||
|
}
|
||||||
|
|
||||||
|
/* ================================================== */
|
||||||
|
|
||||||
|
LOG_Severity
|
||||||
|
LOG_GetMinSeverity(void)
|
||||||
|
{
|
||||||
|
return log_min_severity;
|
||||||
|
}
|
||||||
|
|
||||||
|
/* ================================================== */
|
||||||
|
|
||||||
|
void
|
||||||
|
LOG_SetContext(LOG_Context context)
|
||||||
|
{
|
||||||
|
log_contexts |= context;
|
||||||
|
}
|
||||||
|
|
||||||
|
/* ================================================== */
|
||||||
|
|
||||||
|
void
|
||||||
|
LOG_UnsetContext(LOG_Context context)
|
||||||
|
{
|
||||||
|
log_contexts &= ~context;
|
||||||
|
}
|
||||||
|
|
||||||
|
/* ================================================== */
|
||||||
|
|
||||||
|
LOG_Severity
|
||||||
|
LOG_GetContextSeverity(LOG_Context contexts)
|
||||||
|
{
|
||||||
|
return log_contexts & contexts ? LOGS_INFO : LOGS_DEBUG;
|
||||||
|
}
|
||||||
|
|
||||||
|
/* ================================================== */
|
||||||
|
|
||||||
|
void
|
||||||
|
LOG_SetDebugPrefix(const char *prefix)
|
||||||
|
{
|
||||||
|
Free(debug_prefix);
|
||||||
|
debug_prefix = Strdup(prefix);
|
||||||
|
}
|
||||||
|
|
||||||
|
/* ================================================== */
|
||||||
|
|
||||||
|
void
|
||||||
|
LOG_SetParentFd(int fd)
|
||||||
|
{
|
||||||
|
parent_fd = fd;
|
||||||
|
if (file_log == stderr)
|
||||||
|
file_log = NULL;
|
||||||
|
}
|
||||||
|
|
||||||
|
/* ================================================== */
|
||||||
|
|
||||||
|
int
|
||||||
|
LOG_NotifyParent(const char *message)
|
||||||
|
{
|
||||||
|
if (parent_fd <= 0)
|
||||||
|
return 1;
|
||||||
|
|
||||||
|
return write(parent_fd, message, strlen(message) + 1) > 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
/* ================================================== */
|
||||||
|
|
||||||
|
void
|
||||||
|
LOG_CloseParentFd()
|
||||||
|
{
|
||||||
|
if (parent_fd > 0)
|
||||||
|
close(parent_fd);
|
||||||
|
parent_fd = -1;
|
||||||
|
}
|
||||||
|
|
||||||
|
/* ================================================== */
|
||||||
|
|
||||||
|
LOG_FileID
|
||||||
|
LOG_FileOpen(const char *name, const char *banner)
|
||||||
|
{
|
||||||
|
if (n_filelogs >= MAX_FILELOGS) {
|
||||||
|
assert(0);
|
||||||
|
return -1;
|
||||||
|
}
|
||||||
|
|
||||||
|
logfiles[n_filelogs].name = name;
|
||||||
|
logfiles[n_filelogs].banner = banner;
|
||||||
|
logfiles[n_filelogs].file = NULL;
|
||||||
|
logfiles[n_filelogs].writes = 0;
|
||||||
|
|
||||||
|
return n_filelogs++;
|
||||||
|
}
|
||||||
|
|
||||||
|
/* ================================================== */
|
||||||
|
|
||||||
|
void
|
||||||
|
LOG_FileWrite(LOG_FileID id, const char *format, ...)
|
||||||
|
{
|
||||||
|
va_list other_args;
|
||||||
|
int banner;
|
||||||
|
|
||||||
|
if (id < 0 || id >= n_filelogs || !logfiles[id].name)
|
||||||
|
return;
|
||||||
|
|
||||||
|
if (!logfiles[id].file) {
|
||||||
|
char *logdir = CNF_GetLogDir();
|
||||||
|
|
||||||
|
if (!logdir) {
|
||||||
|
LOG(LOGS_WARN, "logdir not specified");
|
||||||
|
logfiles[id].name = NULL;
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
logfiles[id].file = UTI_OpenFile(logdir, logfiles[id].name, ".log", 'a', 0644);
|
||||||
|
if (!logfiles[id].file) {
|
||||||
|
/* Disable the log */
|
||||||
|
logfiles[id].name = NULL;
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
banner = CNF_GetLogBanner();
|
||||||
|
if (banner && logfiles[id].writes++ % banner == 0) {
|
||||||
|
char bannerline[256];
|
||||||
|
int i, bannerlen;
|
||||||
|
|
||||||
|
bannerlen = MIN(strlen(logfiles[id].banner), sizeof (bannerline) - 1);
|
||||||
|
|
||||||
|
for (i = 0; i < bannerlen; i++)
|
||||||
|
bannerline[i] = '=';
|
||||||
|
bannerline[i] = '\0';
|
||||||
|
|
||||||
|
fprintf(logfiles[id].file, "%s\n", bannerline);
|
||||||
|
fprintf(logfiles[id].file, "%s\n", logfiles[id].banner);
|
||||||
|
fprintf(logfiles[id].file, "%s\n", bannerline);
|
||||||
|
}
|
||||||
|
|
||||||
|
va_start(other_args, format);
|
||||||
|
vfprintf(logfiles[id].file, format, other_args);
|
||||||
|
va_end(other_args);
|
||||||
|
fprintf(logfiles[id].file, "\n");
|
||||||
|
|
||||||
|
fflush(logfiles[id].file);
|
||||||
|
}
|
||||||
|
|
||||||
|
/* ================================================== */
|
||||||
|
|
||||||
|
void
|
||||||
|
LOG_CycleLogFiles(void)
|
||||||
|
{
|
||||||
|
LOG_FileID i;
|
||||||
|
|
||||||
|
for (i = 0; i < n_filelogs; i++) {
|
||||||
|
if (logfiles[i].file)
|
||||||
|
fclose(logfiles[i].file);
|
||||||
|
logfiles[i].file = NULL;
|
||||||
|
logfiles[i].writes = 0;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/* ================================================== */
|
146
logging.h
Normal file
146
logging.h
Normal file
|
@ -0,0 +1,146 @@
|
||||||
|
/*
|
||||||
|
chronyd/chronyc - Programs for keeping computer clocks accurate.
|
||||||
|
|
||||||
|
**********************************************************************
|
||||||
|
* Copyright (C) Richard P. Curnow 1997-2002
|
||||||
|
* Copyright (C) Miroslav Lichvar 2013-2015
|
||||||
|
*
|
||||||
|
* This program is free software; you can redistribute it and/or modify
|
||||||
|
* it under the terms of version 2 of the GNU General Public License as
|
||||||
|
* published by the Free Software Foundation.
|
||||||
|
*
|
||||||
|
* This program is distributed in the hope that it will be useful, but
|
||||||
|
* WITHOUT ANY WARRANTY; without even the implied warranty of
|
||||||
|
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
|
||||||
|
* General Public License for more details.
|
||||||
|
*
|
||||||
|
* You should have received a copy of the GNU General Public License along
|
||||||
|
* with this program; if not, write to the Free Software Foundation, Inc.,
|
||||||
|
* 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA.
|
||||||
|
*
|
||||||
|
**********************************************************************
|
||||||
|
|
||||||
|
=======================================================================
|
||||||
|
|
||||||
|
Header file for diagnostic logging module
|
||||||
|
|
||||||
|
*/
|
||||||
|
|
||||||
|
#ifndef GOT_LOGGING_H
|
||||||
|
#define GOT_LOGGING_H
|
||||||
|
|
||||||
|
#include "sysincl.h"
|
||||||
|
|
||||||
|
/* Line logging macros. If the compiler is GNU C, we take advantage of
|
||||||
|
being able to get the function name also. */
|
||||||
|
|
||||||
|
#ifdef __GNUC__
|
||||||
|
#define FUNCTION_NAME __FUNCTION__
|
||||||
|
#define FORMAT_ATTRIBUTE_PRINTF(str, first) __attribute__ ((format (printf, str, first)))
|
||||||
|
#else
|
||||||
|
#define FUNCTION_NAME ""
|
||||||
|
#define FORMAT_ATTRIBUTE_PRINTF(str, first)
|
||||||
|
#endif
|
||||||
|
|
||||||
|
#if DEBUG > 0
|
||||||
|
#define LOG_MESSAGE(severity, ...) \
|
||||||
|
LOG_Message(severity, __LINE__, __FILE__, FUNCTION_NAME, __VA_ARGS__)
|
||||||
|
#else
|
||||||
|
#define LOG_MESSAGE(severity, ...) \
|
||||||
|
LOG_Message(severity, __VA_ARGS__)
|
||||||
|
#endif
|
||||||
|
|
||||||
|
#define DEBUG_LOG(...) \
|
||||||
|
do { \
|
||||||
|
if (DEBUG && log_min_severity == LOGS_DEBUG) \
|
||||||
|
LOG_MESSAGE(LOGS_DEBUG, __VA_ARGS__); \
|
||||||
|
} while (0)
|
||||||
|
|
||||||
|
#define LOG_FATAL(...) \
|
||||||
|
do { \
|
||||||
|
LOG_MESSAGE(LOGS_FATAL, __VA_ARGS__); \
|
||||||
|
exit(1); \
|
||||||
|
} while (0)
|
||||||
|
|
||||||
|
#define LOG(severity, ...) LOG_MESSAGE(severity, __VA_ARGS__)
|
||||||
|
|
||||||
|
/* Definition of severity */
|
||||||
|
typedef enum {
|
||||||
|
LOGS_DEBUG = -1,
|
||||||
|
LOGS_INFO = 0,
|
||||||
|
LOGS_WARN,
|
||||||
|
LOGS_ERR,
|
||||||
|
LOGS_FATAL,
|
||||||
|
} LOG_Severity;
|
||||||
|
|
||||||
|
/* Minimum severity of messages to be logged */
|
||||||
|
extern LOG_Severity log_min_severity;
|
||||||
|
|
||||||
|
/* Init function */
|
||||||
|
extern void LOG_Initialise(void);
|
||||||
|
|
||||||
|
/* Fini function */
|
||||||
|
extern void LOG_Finalise(void);
|
||||||
|
|
||||||
|
/* Line logging function */
|
||||||
|
#if DEBUG > 0
|
||||||
|
FORMAT_ATTRIBUTE_PRINTF(5, 6)
|
||||||
|
extern void LOG_Message(LOG_Severity severity, int line_number, const char *filename,
|
||||||
|
const char *function_name, const char *format, ...);
|
||||||
|
#else
|
||||||
|
FORMAT_ATTRIBUTE_PRINTF(2, 3)
|
||||||
|
extern void LOG_Message(LOG_Severity severity, const char *format, ...);
|
||||||
|
#endif
|
||||||
|
|
||||||
|
/* Set the minimum severity of a message to be logged or printed to terminal.
|
||||||
|
If the severity is LOGS_DEBUG and DEBUG is enabled, all messages will be
|
||||||
|
prefixed with the filename, line number, and function name. */
|
||||||
|
extern void LOG_SetMinSeverity(LOG_Severity severity);
|
||||||
|
|
||||||
|
/* Get the minimum severity */
|
||||||
|
extern LOG_Severity LOG_GetMinSeverity(void);
|
||||||
|
|
||||||
|
/* Flags for info messages that should be logged only in specific contexts */
|
||||||
|
typedef enum {
|
||||||
|
LOGC_Command = 1,
|
||||||
|
LOGC_SourceFile = 2,
|
||||||
|
} LOG_Context;
|
||||||
|
|
||||||
|
/* Modify current contexts */
|
||||||
|
extern void LOG_SetContext(LOG_Context context);
|
||||||
|
extern void LOG_UnsetContext(LOG_Context context);
|
||||||
|
|
||||||
|
/* Get severity depending on the current active contexts: INFO if they contain
|
||||||
|
at least one of the specified contexts, DEBUG otherwise */
|
||||||
|
extern LOG_Severity LOG_GetContextSeverity(LOG_Context contexts);
|
||||||
|
|
||||||
|
/* Set a prefix for debug messages */
|
||||||
|
extern void LOG_SetDebugPrefix(const char *prefix);
|
||||||
|
|
||||||
|
/* Log messages to a file instead of stderr, or stderr again if NULL */
|
||||||
|
extern void LOG_OpenFileLog(const char *log_file);
|
||||||
|
|
||||||
|
/* Log messages to syslog instead of stderr */
|
||||||
|
extern void LOG_OpenSystemLog(void);
|
||||||
|
|
||||||
|
/* Stop using stderr and send fatal message to the foreground process */
|
||||||
|
extern void LOG_SetParentFd(int fd);
|
||||||
|
|
||||||
|
/* Send a message to the foreground process */
|
||||||
|
extern int LOG_NotifyParent(const char *message);
|
||||||
|
|
||||||
|
/* Close the pipe to the foreground process */
|
||||||
|
extern void LOG_CloseParentFd(void);
|
||||||
|
|
||||||
|
/* File logging functions */
|
||||||
|
|
||||||
|
typedef int LOG_FileID;
|
||||||
|
|
||||||
|
extern LOG_FileID LOG_FileOpen(const char *name, const char *banner);
|
||||||
|
|
||||||
|
FORMAT_ATTRIBUTE_PRINTF(2, 3)
|
||||||
|
extern void LOG_FileWrite(LOG_FileID id, const char *format, ...);
|
||||||
|
|
||||||
|
extern void LOG_CycleLogFiles(void);
|
||||||
|
|
||||||
|
#endif /* GOT_LOGGING_H */
|
712
main.c
Normal file
712
main.c
Normal file
|
@ -0,0 +1,712 @@
|
||||||
|
/*
|
||||||
|
chronyd/chronyc - Programs for keeping computer clocks accurate.
|
||||||
|
|
||||||
|
**********************************************************************
|
||||||
|
* Copyright (C) Richard P. Curnow 1997-2003
|
||||||
|
* Copyright (C) John G. Hasler 2009
|
||||||
|
* Copyright (C) Miroslav Lichvar 2012-2020
|
||||||
|
*
|
||||||
|
* This program is free software; you can redistribute it and/or modify
|
||||||
|
* it under the terms of version 2 of the GNU General Public License as
|
||||||
|
* published by the Free Software Foundation.
|
||||||
|
*
|
||||||
|
* This program is distributed in the hope that it will be useful, but
|
||||||
|
* WITHOUT ANY WARRANTY; without even the implied warranty of
|
||||||
|
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
|
||||||
|
* General Public License for more details.
|
||||||
|
*
|
||||||
|
* You should have received a copy of the GNU General Public License along
|
||||||
|
* with this program; if not, write to the Free Software Foundation, Inc.,
|
||||||
|
* 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA.
|
||||||
|
*
|
||||||
|
**********************************************************************
|
||||||
|
|
||||||
|
=======================================================================
|
||||||
|
|
||||||
|
The main program
|
||||||
|
*/
|
||||||
|
|
||||||
|
#include "config.h"
|
||||||
|
|
||||||
|
#include "sysincl.h"
|
||||||
|
|
||||||
|
#include "main.h"
|
||||||
|
#include "sched.h"
|
||||||
|
#include "leapdb.h"
|
||||||
|
#include "local.h"
|
||||||
|
#include "sys.h"
|
||||||
|
#include "ntp_io.h"
|
||||||
|
#include "ntp_signd.h"
|
||||||
|
#include "ntp_sources.h"
|
||||||
|
#include "ntp_core.h"
|
||||||
|
#include "nts_ke_server.h"
|
||||||
|
#include "nts_ntp_server.h"
|
||||||
|
#include "socket.h"
|
||||||
|
#include "sources.h"
|
||||||
|
#include "sourcestats.h"
|
||||||
|
#include "reference.h"
|
||||||
|
#include "logging.h"
|
||||||
|
#include "conf.h"
|
||||||
|
#include "cmdmon.h"
|
||||||
|
#include "keys.h"
|
||||||
|
#include "manual.h"
|
||||||
|
#include "rtc.h"
|
||||||
|
#include "refclock.h"
|
||||||
|
#include "clientlog.h"
|
||||||
|
#include "nameserv.h"
|
||||||
|
#include "privops.h"
|
||||||
|
#include "smooth.h"
|
||||||
|
#include "tempcomp.h"
|
||||||
|
#include "util.h"
|
||||||
|
|
||||||
|
/* ================================================== */
|
||||||
|
|
||||||
|
/* Set when the initialisation chain has been completed. Prevents finalisation
|
||||||
|
* chain being run if a fatal error happened early. */
|
||||||
|
|
||||||
|
static int initialised = 0;
|
||||||
|
|
||||||
|
static int exit_status = 0;
|
||||||
|
|
||||||
|
static int reload = 0;
|
||||||
|
|
||||||
|
static REF_Mode ref_mode = REF_ModeNormal;
|
||||||
|
|
||||||
|
/* ================================================== */
|
||||||
|
|
||||||
|
static void
|
||||||
|
do_platform_checks(void)
|
||||||
|
{
|
||||||
|
struct timespec ts;
|
||||||
|
|
||||||
|
/* Require at least 32-bit integers, two's complement representation and
|
||||||
|
the usual implementation of conversion of unsigned integers */
|
||||||
|
assert(sizeof (int) >= 4);
|
||||||
|
assert(-1 == ~0);
|
||||||
|
assert((int32_t)4294967295U == (int32_t)-1);
|
||||||
|
|
||||||
|
/* Require time_t and tv_nsec in timespec to be signed */
|
||||||
|
ts.tv_sec = -1;
|
||||||
|
ts.tv_nsec = -1;
|
||||||
|
assert(ts.tv_sec < 0 && ts.tv_nsec < 0);
|
||||||
|
}
|
||||||
|
|
||||||
|
/* ================================================== */
|
||||||
|
|
||||||
|
static void
|
||||||
|
delete_pidfile(void)
|
||||||
|
{
|
||||||
|
const char *pidfile = CNF_GetPidFile();
|
||||||
|
|
||||||
|
if (!pidfile)
|
||||||
|
return;
|
||||||
|
|
||||||
|
if (!UTI_RemoveFile(NULL, pidfile, NULL))
|
||||||
|
;
|
||||||
|
}
|
||||||
|
|
||||||
|
/* ================================================== */
|
||||||
|
|
||||||
|
void
|
||||||
|
MAI_CleanupAndExit(void)
|
||||||
|
{
|
||||||
|
if (!initialised) exit(exit_status);
|
||||||
|
|
||||||
|
LCL_CancelOffsetCorrection();
|
||||||
|
SRC_DumpSources();
|
||||||
|
|
||||||
|
/* Don't update clock when removing sources */
|
||||||
|
REF_SetMode(REF_ModeIgnore);
|
||||||
|
|
||||||
|
SMT_Finalise();
|
||||||
|
TMC_Finalise();
|
||||||
|
MNL_Finalise();
|
||||||
|
CLG_Finalise();
|
||||||
|
NKS_Finalise();
|
||||||
|
NNS_Finalise();
|
||||||
|
NSD_Finalise();
|
||||||
|
NSR_Finalise();
|
||||||
|
SST_Finalise();
|
||||||
|
NCR_Finalise();
|
||||||
|
NIO_Finalise();
|
||||||
|
CAM_Finalise();
|
||||||
|
|
||||||
|
KEY_Finalise();
|
||||||
|
RCL_Finalise();
|
||||||
|
SRC_Finalise();
|
||||||
|
REF_Finalise();
|
||||||
|
LDB_Finalise();
|
||||||
|
RTC_Finalise();
|
||||||
|
SYS_Finalise();
|
||||||
|
|
||||||
|
SCK_Finalise();
|
||||||
|
SCH_Finalise();
|
||||||
|
LCL_Finalise();
|
||||||
|
PRV_Finalise();
|
||||||
|
|
||||||
|
delete_pidfile();
|
||||||
|
|
||||||
|
CNF_Finalise();
|
||||||
|
HSH_Finalise();
|
||||||
|
LOG_Finalise();
|
||||||
|
|
||||||
|
UTI_ResetGetRandomFunctions();
|
||||||
|
|
||||||
|
exit(exit_status);
|
||||||
|
}
|
||||||
|
|
||||||
|
/* ================================================== */
|
||||||
|
|
||||||
|
static void
|
||||||
|
signal_cleanup(int x)
|
||||||
|
{
|
||||||
|
SCH_QuitProgram();
|
||||||
|
}
|
||||||
|
|
||||||
|
/* ================================================== */
|
||||||
|
|
||||||
|
static void
|
||||||
|
quit_timeout(void *arg)
|
||||||
|
{
|
||||||
|
LOG(LOGS_INFO, "Timeout reached");
|
||||||
|
|
||||||
|
/* Return with non-zero status if the clock is not synchronised */
|
||||||
|
exit_status = REF_GetOurStratum() >= NTP_MAX_STRATUM;
|
||||||
|
SCH_QuitProgram();
|
||||||
|
}
|
||||||
|
|
||||||
|
/* ================================================== */
|
||||||
|
|
||||||
|
static void
|
||||||
|
ntp_source_resolving_end(void)
|
||||||
|
{
|
||||||
|
NSR_SetSourceResolvingEndHandler(NULL);
|
||||||
|
|
||||||
|
if (reload) {
|
||||||
|
/* Note, we want reload to come well after the initialisation from
|
||||||
|
the real time clock - this gives us a fighting chance that the
|
||||||
|
system-clock scale for the reloaded samples still has a
|
||||||
|
semblence of validity about it. */
|
||||||
|
SRC_ReloadSources();
|
||||||
|
}
|
||||||
|
|
||||||
|
SRC_RemoveDumpFiles();
|
||||||
|
RTC_StartMeasurements();
|
||||||
|
RCL_StartRefclocks();
|
||||||
|
NSR_StartSources();
|
||||||
|
NSR_AutoStartSources();
|
||||||
|
|
||||||
|
/* Special modes can end only when sources update their reachability.
|
||||||
|
Give up immediately if there are no active sources. */
|
||||||
|
if (ref_mode != REF_ModeNormal && !SRC_ActiveSources()) {
|
||||||
|
REF_SetUnsynchronised();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/* ================================================== */
|
||||||
|
|
||||||
|
static void
|
||||||
|
post_init_ntp_hook(void *anything)
|
||||||
|
{
|
||||||
|
if (ref_mode == REF_ModeInitStepSlew) {
|
||||||
|
/* Remove the initstepslew sources and set normal mode */
|
||||||
|
NSR_RemoveAllSources();
|
||||||
|
ref_mode = REF_ModeNormal;
|
||||||
|
REF_SetMode(ref_mode);
|
||||||
|
}
|
||||||
|
|
||||||
|
/* Send an empty message to the foreground process so it can exit.
|
||||||
|
If that fails, indicating the process was killed, exit too. */
|
||||||
|
if (!LOG_NotifyParent(""))
|
||||||
|
SCH_QuitProgram();
|
||||||
|
LOG_CloseParentFd();
|
||||||
|
|
||||||
|
CNF_AddSources();
|
||||||
|
CNF_AddBroadcasts();
|
||||||
|
|
||||||
|
NSR_SetSourceResolvingEndHandler(ntp_source_resolving_end);
|
||||||
|
NSR_ResolveSources();
|
||||||
|
}
|
||||||
|
|
||||||
|
/* ================================================== */
|
||||||
|
|
||||||
|
static void
|
||||||
|
reference_mode_end(int result)
|
||||||
|
{
|
||||||
|
switch (ref_mode) {
|
||||||
|
case REF_ModeNormal:
|
||||||
|
case REF_ModeUpdateOnce:
|
||||||
|
case REF_ModePrintOnce:
|
||||||
|
exit_status = !result;
|
||||||
|
SCH_QuitProgram();
|
||||||
|
break;
|
||||||
|
case REF_ModeInitStepSlew:
|
||||||
|
/* Switch to the normal mode, the delay is used to prevent polling
|
||||||
|
interval shorter than the burst interval if some configured servers
|
||||||
|
were used also for initstepslew */
|
||||||
|
SCH_AddTimeoutByDelay(2.0, post_init_ntp_hook, NULL);
|
||||||
|
break;
|
||||||
|
default:
|
||||||
|
assert(0);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/* ================================================== */
|
||||||
|
|
||||||
|
static void
|
||||||
|
post_init_rtc_hook(void *anything)
|
||||||
|
{
|
||||||
|
if (CNF_GetInitSources() > 0) {
|
||||||
|
CNF_AddInitSources();
|
||||||
|
NSR_StartSources();
|
||||||
|
assert(REF_GetMode() != REF_ModeNormal);
|
||||||
|
/* Wait for mode end notification */
|
||||||
|
} else {
|
||||||
|
(post_init_ntp_hook)(NULL);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/* ================================================== */
|
||||||
|
|
||||||
|
static void
|
||||||
|
check_pidfile(void)
|
||||||
|
{
|
||||||
|
const char *pidfile = CNF_GetPidFile();
|
||||||
|
FILE *in;
|
||||||
|
int pid, count;
|
||||||
|
|
||||||
|
if (!pidfile)
|
||||||
|
return;
|
||||||
|
|
||||||
|
in = UTI_OpenFile(NULL, pidfile, NULL, 'r', 0);
|
||||||
|
if (!in)
|
||||||
|
return;
|
||||||
|
|
||||||
|
count = fscanf(in, "%d", &pid);
|
||||||
|
fclose(in);
|
||||||
|
|
||||||
|
if (count != 1)
|
||||||
|
return;
|
||||||
|
|
||||||
|
if (getsid(pid) < 0)
|
||||||
|
return;
|
||||||
|
|
||||||
|
LOG_FATAL("Another chronyd may already be running (pid=%d), check %s",
|
||||||
|
pid, pidfile);
|
||||||
|
}
|
||||||
|
|
||||||
|
/* ================================================== */
|
||||||
|
|
||||||
|
static void
|
||||||
|
write_pidfile(void)
|
||||||
|
{
|
||||||
|
const char *pidfile = CNF_GetPidFile();
|
||||||
|
FILE *out;
|
||||||
|
|
||||||
|
if (!pidfile)
|
||||||
|
return;
|
||||||
|
|
||||||
|
out = UTI_OpenFile(NULL, pidfile, NULL, 'W', 0644);
|
||||||
|
fprintf(out, "%d\n", (int)getpid());
|
||||||
|
fclose(out);
|
||||||
|
}
|
||||||
|
|
||||||
|
/* ================================================== */
|
||||||
|
|
||||||
|
#define DEV_NULL "/dev/null"
|
||||||
|
|
||||||
|
static void
|
||||||
|
go_daemon(void)
|
||||||
|
{
|
||||||
|
int pid, fd, pipefd[2];
|
||||||
|
|
||||||
|
/* Create pipe which will the daemon use to notify the grandparent
|
||||||
|
when it's initialised or send an error message */
|
||||||
|
if (pipe(pipefd)) {
|
||||||
|
LOG_FATAL("pipe() failed : %s", strerror(errno));
|
||||||
|
}
|
||||||
|
|
||||||
|
/* Does this preserve existing signal handlers? */
|
||||||
|
pid = fork();
|
||||||
|
|
||||||
|
if (pid < 0) {
|
||||||
|
LOG_FATAL("fork() failed : %s", strerror(errno));
|
||||||
|
} else if (pid > 0) {
|
||||||
|
/* In the 'grandparent' */
|
||||||
|
char message[1024];
|
||||||
|
int r;
|
||||||
|
|
||||||
|
/* Don't exit before the 'parent' */
|
||||||
|
waitpid(pid, NULL, 0);
|
||||||
|
|
||||||
|
close(pipefd[1]);
|
||||||
|
r = read(pipefd[0], message, sizeof (message));
|
||||||
|
if (r != 1 || message[0] != '\0') {
|
||||||
|
if (r > 1) {
|
||||||
|
/* Print the error message from the child */
|
||||||
|
message[sizeof (message) - 1] = '\0';
|
||||||
|
fprintf(stderr, "%s\n", message);
|
||||||
|
}
|
||||||
|
exit(1);
|
||||||
|
} else
|
||||||
|
exit(0);
|
||||||
|
} else {
|
||||||
|
close(pipefd[0]);
|
||||||
|
|
||||||
|
setsid();
|
||||||
|
|
||||||
|
/* Do 2nd fork, as-per recommended practice for launching daemons. */
|
||||||
|
pid = fork();
|
||||||
|
|
||||||
|
if (pid < 0) {
|
||||||
|
LOG_FATAL("fork() failed : %s", strerror(errno));
|
||||||
|
} else if (pid > 0) {
|
||||||
|
/* In the 'parent' */
|
||||||
|
close(pipefd[1]);
|
||||||
|
exit(0);
|
||||||
|
} else {
|
||||||
|
/* In the child we want to leave running as the daemon */
|
||||||
|
|
||||||
|
/* Change current directory to / */
|
||||||
|
if (chdir("/") < 0) {
|
||||||
|
LOG_FATAL("chdir() failed : %s", strerror(errno));
|
||||||
|
}
|
||||||
|
|
||||||
|
/* Don't keep stdin/out/err from before. But don't close
|
||||||
|
the parent pipe yet, or reusable file descriptors. */
|
||||||
|
for (fd=0; fd<1024; fd++) {
|
||||||
|
if (fd != pipefd[1] && !SCK_IsReusable(fd))
|
||||||
|
close(fd);
|
||||||
|
}
|
||||||
|
|
||||||
|
LOG_SetParentFd(pipefd[1]);
|
||||||
|
|
||||||
|
/* Open /dev/null as new stdin/out/err */
|
||||||
|
errno = 0;
|
||||||
|
if (open(DEV_NULL, O_RDONLY) != STDIN_FILENO ||
|
||||||
|
open(DEV_NULL, O_WRONLY) != STDOUT_FILENO ||
|
||||||
|
open(DEV_NULL, O_RDWR) != STDERR_FILENO)
|
||||||
|
LOG_FATAL("Could not open %s : %s", DEV_NULL, strerror(errno));
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/* ================================================== */
|
||||||
|
|
||||||
|
static void
|
||||||
|
print_help(const char *progname)
|
||||||
|
{
|
||||||
|
printf("Usage: %s [OPTION]... [DIRECTIVE]...\n\n"
|
||||||
|
"Options:\n"
|
||||||
|
" -4\t\tUse IPv4 addresses only\n"
|
||||||
|
" -6\t\tUse IPv6 addresses only\n"
|
||||||
|
" -f FILE\tSpecify configuration file (%s)\n"
|
||||||
|
" -n\t\tDon't run as daemon\n"
|
||||||
|
" -d\t\tDon't run as daemon and log to stderr\n"
|
||||||
|
#if DEBUG > 0
|
||||||
|
" -d -d\t\tEnable debug messages\n"
|
||||||
|
#endif
|
||||||
|
" -l FILE\tLog to file\n"
|
||||||
|
" -L LEVEL\tSet logging threshold (0)\n"
|
||||||
|
" -p\t\tPrint configuration and exit\n"
|
||||||
|
" -q\t\tSet clock and exit\n"
|
||||||
|
" -Q\t\tLog offset and exit\n"
|
||||||
|
" -r\t\tReload dump files\n"
|
||||||
|
" -R\t\tAdapt configuration for restart\n"
|
||||||
|
" -s\t\tSet clock from RTC\n"
|
||||||
|
" -t SECONDS\tExit after elapsed time\n"
|
||||||
|
" -u USER\tSpecify user (%s)\n"
|
||||||
|
" -U\t\tDon't check for root\n"
|
||||||
|
" -F LEVEL\tSet system call filter level (0)\n"
|
||||||
|
" -P PRIORITY\tSet process priority (0)\n"
|
||||||
|
" -m\t\tLock memory\n"
|
||||||
|
" -x\t\tDon't control clock\n"
|
||||||
|
" -v, --version\tPrint version and exit\n"
|
||||||
|
" -h, --help\tPrint usage and exit\n",
|
||||||
|
progname, DEFAULT_CONF_FILE, DEFAULT_USER);
|
||||||
|
}
|
||||||
|
|
||||||
|
/* ================================================== */
|
||||||
|
|
||||||
|
static void
|
||||||
|
print_version(void)
|
||||||
|
{
|
||||||
|
printf("chronyd (chrony) version %s (%s)\n", CHRONY_VERSION, CHRONYD_FEATURES);
|
||||||
|
}
|
||||||
|
|
||||||
|
/* ================================================== */
|
||||||
|
|
||||||
|
static int
|
||||||
|
parse_int_arg(const char *arg)
|
||||||
|
{
|
||||||
|
int i;
|
||||||
|
|
||||||
|
if (sscanf(arg, "%d", &i) != 1)
|
||||||
|
LOG_FATAL("Invalid argument %s", arg);
|
||||||
|
return i;
|
||||||
|
}
|
||||||
|
|
||||||
|
/* ================================================== */
|
||||||
|
|
||||||
|
int main
|
||||||
|
(int argc, char **argv)
|
||||||
|
{
|
||||||
|
const char *conf_file = DEFAULT_CONF_FILE;
|
||||||
|
const char *progname = argv[0];
|
||||||
|
char *user = NULL, *log_file = NULL;
|
||||||
|
struct passwd *pw;
|
||||||
|
int opt, debug = 0, nofork = 0, address_family = IPADDR_UNSPEC;
|
||||||
|
int do_init_rtc = 0, restarted = 0, client_only = 0, timeout = -1;
|
||||||
|
int scfilter_level = 0, lock_memory = 0, sched_priority = 0;
|
||||||
|
int clock_control = 1, system_log = 1, log_severity = LOGS_INFO;
|
||||||
|
int user_check = 1, config_args = 0, print_config = 0;
|
||||||
|
|
||||||
|
do_platform_checks();
|
||||||
|
|
||||||
|
LOG_Initialise();
|
||||||
|
|
||||||
|
/* Parse long command-line options */
|
||||||
|
for (optind = 1; optind < argc; optind++) {
|
||||||
|
if (!strcmp("--help", argv[optind])) {
|
||||||
|
print_help(progname);
|
||||||
|
return 0;
|
||||||
|
} else if (!strcmp("--version", argv[optind])) {
|
||||||
|
print_version();
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
optind = 1;
|
||||||
|
|
||||||
|
/* Parse short command-line options */
|
||||||
|
while ((opt = getopt(argc, argv, "46df:F:hl:L:mnpP:qQrRst:u:Uvx")) != -1) {
|
||||||
|
switch (opt) {
|
||||||
|
case '4':
|
||||||
|
case '6':
|
||||||
|
address_family = opt == '4' ? IPADDR_INET4 : IPADDR_INET6;
|
||||||
|
break;
|
||||||
|
case 'd':
|
||||||
|
debug++;
|
||||||
|
nofork = 1;
|
||||||
|
system_log = 0;
|
||||||
|
break;
|
||||||
|
case 'f':
|
||||||
|
conf_file = optarg;
|
||||||
|
break;
|
||||||
|
case 'F':
|
||||||
|
scfilter_level = parse_int_arg(optarg);
|
||||||
|
break;
|
||||||
|
case 'l':
|
||||||
|
log_file = optarg;
|
||||||
|
break;
|
||||||
|
case 'L':
|
||||||
|
log_severity = parse_int_arg(optarg);
|
||||||
|
break;
|
||||||
|
case 'm':
|
||||||
|
lock_memory = 1;
|
||||||
|
break;
|
||||||
|
case 'n':
|
||||||
|
nofork = 1;
|
||||||
|
break;
|
||||||
|
case 'p':
|
||||||
|
print_config = 1;
|
||||||
|
user_check = 0;
|
||||||
|
nofork = 1;
|
||||||
|
system_log = 0;
|
||||||
|
log_severity = LOGS_WARN;
|
||||||
|
break;
|
||||||
|
case 'P':
|
||||||
|
sched_priority = parse_int_arg(optarg);
|
||||||
|
break;
|
||||||
|
case 'q':
|
||||||
|
ref_mode = REF_ModeUpdateOnce;
|
||||||
|
nofork = 1;
|
||||||
|
client_only = 0;
|
||||||
|
system_log = 0;
|
||||||
|
break;
|
||||||
|
case 'Q':
|
||||||
|
ref_mode = REF_ModePrintOnce;
|
||||||
|
nofork = 1;
|
||||||
|
client_only = 1;
|
||||||
|
user_check = 0;
|
||||||
|
clock_control = 0;
|
||||||
|
system_log = 0;
|
||||||
|
break;
|
||||||
|
case 'r':
|
||||||
|
reload = 1;
|
||||||
|
break;
|
||||||
|
case 'R':
|
||||||
|
restarted = 1;
|
||||||
|
break;
|
||||||
|
case 's':
|
||||||
|
do_init_rtc = 1;
|
||||||
|
break;
|
||||||
|
case 't':
|
||||||
|
timeout = parse_int_arg(optarg);
|
||||||
|
break;
|
||||||
|
case 'u':
|
||||||
|
user = optarg;
|
||||||
|
break;
|
||||||
|
case 'U':
|
||||||
|
user_check = 0;
|
||||||
|
break;
|
||||||
|
case 'v':
|
||||||
|
print_version();
|
||||||
|
return 0;
|
||||||
|
case 'x':
|
||||||
|
clock_control = 0;
|
||||||
|
break;
|
||||||
|
default:
|
||||||
|
print_help(progname);
|
||||||
|
return opt != 'h';
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
if (user_check && getuid() != 0)
|
||||||
|
LOG_FATAL("Not superuser");
|
||||||
|
|
||||||
|
/* Initialise reusable file descriptors before fork */
|
||||||
|
SCK_PreInitialise();
|
||||||
|
|
||||||
|
/* Turn into a daemon */
|
||||||
|
if (!nofork) {
|
||||||
|
go_daemon();
|
||||||
|
}
|
||||||
|
|
||||||
|
if (log_file) {
|
||||||
|
LOG_OpenFileLog(log_file);
|
||||||
|
} else if (system_log) {
|
||||||
|
LOG_OpenSystemLog();
|
||||||
|
}
|
||||||
|
|
||||||
|
LOG_SetMinSeverity(debug >= 2 ? LOGS_DEBUG : log_severity);
|
||||||
|
|
||||||
|
LOG(LOGS_INFO, "chronyd version %s starting (%s)", CHRONY_VERSION, CHRONYD_FEATURES);
|
||||||
|
|
||||||
|
DNS_SetAddressFamily(address_family);
|
||||||
|
|
||||||
|
CNF_Initialise(restarted, client_only);
|
||||||
|
if (print_config)
|
||||||
|
CNF_EnablePrint();
|
||||||
|
|
||||||
|
/* Parse the config file or the remaining command line arguments */
|
||||||
|
config_args = argc - optind;
|
||||||
|
if (!config_args) {
|
||||||
|
CNF_ReadFile(conf_file);
|
||||||
|
} else {
|
||||||
|
for (; optind < argc; optind++)
|
||||||
|
CNF_ParseLine(NULL, config_args + optind - argc + 1, argv[optind]);
|
||||||
|
}
|
||||||
|
|
||||||
|
if (print_config)
|
||||||
|
return 0;
|
||||||
|
|
||||||
|
/* Check whether another chronyd may already be running */
|
||||||
|
check_pidfile();
|
||||||
|
|
||||||
|
if (!user)
|
||||||
|
user = CNF_GetUser();
|
||||||
|
|
||||||
|
pw = getpwnam(user);
|
||||||
|
if (!pw)
|
||||||
|
LOG_FATAL("Could not get user/group ID of %s", user);
|
||||||
|
|
||||||
|
/* Create directories for sockets, log files, and dump files */
|
||||||
|
CNF_CreateDirs(pw->pw_uid, pw->pw_gid);
|
||||||
|
|
||||||
|
/* Write our pidfile to prevent other instances from running */
|
||||||
|
write_pidfile();
|
||||||
|
|
||||||
|
PRV_Initialise();
|
||||||
|
LCL_Initialise();
|
||||||
|
SCH_Initialise();
|
||||||
|
SCK_Initialise(address_family);
|
||||||
|
|
||||||
|
/* Start helper processes if needed */
|
||||||
|
NKS_PreInitialise(pw->pw_uid, pw->pw_gid, scfilter_level);
|
||||||
|
|
||||||
|
SYS_Initialise(clock_control);
|
||||||
|
RTC_Initialise(do_init_rtc);
|
||||||
|
SRC_Initialise();
|
||||||
|
RCL_Initialise();
|
||||||
|
KEY_Initialise();
|
||||||
|
|
||||||
|
/* Open privileged ports before dropping root */
|
||||||
|
CAM_Initialise();
|
||||||
|
NIO_Initialise();
|
||||||
|
NCR_Initialise();
|
||||||
|
CNF_SetupAccessRestrictions();
|
||||||
|
|
||||||
|
/* Command-line switch must have priority */
|
||||||
|
if (!sched_priority) {
|
||||||
|
sched_priority = CNF_GetSchedPriority();
|
||||||
|
}
|
||||||
|
if (sched_priority) {
|
||||||
|
SYS_SetScheduler(sched_priority);
|
||||||
|
}
|
||||||
|
|
||||||
|
if (lock_memory || CNF_GetLockMemory()) {
|
||||||
|
SYS_LockMemory();
|
||||||
|
}
|
||||||
|
|
||||||
|
/* Drop root privileges if the specified user has a non-zero UID */
|
||||||
|
if (!geteuid() && (pw->pw_uid || pw->pw_gid)) {
|
||||||
|
SYS_DropRoot(pw->pw_uid, pw->pw_gid, SYS_MAIN_PROCESS);
|
||||||
|
|
||||||
|
/* Warn if missing read access or having write access to keys */
|
||||||
|
CNF_CheckReadOnlyAccess();
|
||||||
|
}
|
||||||
|
|
||||||
|
if (!geteuid())
|
||||||
|
LOG(LOGS_WARN, "Running with root privileges");
|
||||||
|
|
||||||
|
LDB_Initialise();
|
||||||
|
REF_Initialise();
|
||||||
|
SST_Initialise();
|
||||||
|
NSR_Initialise();
|
||||||
|
NSD_Initialise();
|
||||||
|
NNS_Initialise();
|
||||||
|
NKS_Initialise();
|
||||||
|
CLG_Initialise();
|
||||||
|
MNL_Initialise();
|
||||||
|
TMC_Initialise();
|
||||||
|
SMT_Initialise();
|
||||||
|
|
||||||
|
/* From now on, it is safe to do finalisation on exit */
|
||||||
|
initialised = 1;
|
||||||
|
|
||||||
|
UTI_SetQuitSignalsHandler(signal_cleanup, 1);
|
||||||
|
|
||||||
|
CAM_OpenUnixSocket();
|
||||||
|
|
||||||
|
if (scfilter_level)
|
||||||
|
SYS_EnableSystemCallFilter(scfilter_level, SYS_MAIN_PROCESS);
|
||||||
|
|
||||||
|
if (ref_mode == REF_ModeNormal && CNF_GetInitSources() > 0) {
|
||||||
|
ref_mode = REF_ModeInitStepSlew;
|
||||||
|
}
|
||||||
|
|
||||||
|
REF_SetModeEndHandler(reference_mode_end);
|
||||||
|
REF_SetMode(ref_mode);
|
||||||
|
|
||||||
|
if (timeout >= 0)
|
||||||
|
SCH_AddTimeoutByDelay(timeout, quit_timeout, NULL);
|
||||||
|
|
||||||
|
if (do_init_rtc) {
|
||||||
|
RTC_TimeInit(post_init_rtc_hook, NULL);
|
||||||
|
} else {
|
||||||
|
post_init_rtc_hook(NULL);
|
||||||
|
}
|
||||||
|
|
||||||
|
/* The program normally runs under control of the main loop in
|
||||||
|
the scheduler. */
|
||||||
|
SCH_MainLoop();
|
||||||
|
|
||||||
|
LOG(LOGS_INFO, "chronyd exiting");
|
||||||
|
|
||||||
|
MAI_CleanupAndExit();
|
||||||
|
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
/* ================================================== */
|
35
main.h
Normal file
35
main.h
Normal file
|
@ -0,0 +1,35 @@
|
||||||
|
/*
|
||||||
|
chronyd/chronyc - Programs for keeping computer clocks accurate.
|
||||||
|
|
||||||
|
**********************************************************************
|
||||||
|
* Copyright (C) Richard P. Curnow 1997-2002
|
||||||
|
*
|
||||||
|
* This program is free software; you can redistribute it and/or modify
|
||||||
|
* it under the terms of version 2 of the GNU General Public License as
|
||||||
|
* published by the Free Software Foundation.
|
||||||
|
*
|
||||||
|
* This program is distributed in the hope that it will be useful, but
|
||||||
|
* WITHOUT ANY WARRANTY; without even the implied warranty of
|
||||||
|
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
|
||||||
|
* General Public License for more details.
|
||||||
|
*
|
||||||
|
* You should have received a copy of the GNU General Public License along
|
||||||
|
* with this program; if not, write to the Free Software Foundation, Inc.,
|
||||||
|
* 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA.
|
||||||
|
*
|
||||||
|
**********************************************************************
|
||||||
|
|
||||||
|
=======================================================================
|
||||||
|
|
||||||
|
Header file for main routine
|
||||||
|
*/
|
||||||
|
|
||||||
|
#ifndef GOT_MAIN_H
|
||||||
|
#define GOT_MAIN_H
|
||||||
|
|
||||||
|
/* Function to clean up at end of run */
|
||||||
|
extern void MAI_CleanupAndExit(void);
|
||||||
|
|
||||||
|
#endif /* GOT_MAIN_H */
|
||||||
|
|
||||||
|
|
332
manual.c
Normal file
332
manual.c
Normal file
|
@ -0,0 +1,332 @@
|
||||||
|
/*
|
||||||
|
chronyd/chronyc - Programs for keeping computer clocks accurate.
|
||||||
|
|
||||||
|
**********************************************************************
|
||||||
|
* Copyright (C) Richard P. Curnow 1997-2003
|
||||||
|
*
|
||||||
|
* This program is free software; you can redistribute it and/or modify
|
||||||
|
* it under the terms of version 2 of the GNU General Public License as
|
||||||
|
* published by the Free Software Foundation.
|
||||||
|
*
|
||||||
|
* This program is distributed in the hope that it will be useful, but
|
||||||
|
* WITHOUT ANY WARRANTY; without even the implied warranty of
|
||||||
|
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
|
||||||
|
* General Public License for more details.
|
||||||
|
*
|
||||||
|
* You should have received a copy of the GNU General Public License along
|
||||||
|
* with this program; if not, write to the Free Software Foundation, Inc.,
|
||||||
|
* 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA.
|
||||||
|
*
|
||||||
|
**********************************************************************
|
||||||
|
|
||||||
|
=======================================================================
|
||||||
|
|
||||||
|
Routines for implementing manual input of real time.
|
||||||
|
|
||||||
|
The daemon accepts manual time input over the control connection,
|
||||||
|
and adjusts the system time to match. Besides this, though, it can
|
||||||
|
determine the average rate of time loss or gain of the local system
|
||||||
|
and adjust the frequency accordingly.
|
||||||
|
|
||||||
|
*/
|
||||||
|
|
||||||
|
#include "config.h"
|
||||||
|
|
||||||
|
#include "sysincl.h"
|
||||||
|
|
||||||
|
#include "manual.h"
|
||||||
|
#include "logging.h"
|
||||||
|
#include "local.h"
|
||||||
|
#include "conf.h"
|
||||||
|
#include "util.h"
|
||||||
|
#include "ntp.h"
|
||||||
|
#include "reference.h"
|
||||||
|
#include "regress.h"
|
||||||
|
|
||||||
|
static int enabled = 0;
|
||||||
|
|
||||||
|
/* More recent samples at highest indices */
|
||||||
|
typedef struct {
|
||||||
|
struct timespec when; /* This is our 'cooked' time */
|
||||||
|
double orig_offset; /*+ Not modified by slew samples */
|
||||||
|
double offset; /*+ if we are fast of the supplied reference */
|
||||||
|
double residual; /*+ regression residual (sign convention given by
|
||||||
|
(measured-predicted)) */
|
||||||
|
} Sample;
|
||||||
|
|
||||||
|
#define MIN_SAMPLE_SEPARATION 1.0
|
||||||
|
|
||||||
|
#define MAX_SAMPLES 16
|
||||||
|
|
||||||
|
static Sample samples[16];
|
||||||
|
static int n_samples;
|
||||||
|
|
||||||
|
/* ================================================== */
|
||||||
|
|
||||||
|
static void
|
||||||
|
slew_samples(struct timespec *raw,
|
||||||
|
struct timespec *cooked,
|
||||||
|
double dfreq,
|
||||||
|
double doffset,
|
||||||
|
LCL_ChangeType change_type,
|
||||||
|
void *not_used);
|
||||||
|
|
||||||
|
/* ================================================== */
|
||||||
|
|
||||||
|
void
|
||||||
|
MNL_Initialise(void)
|
||||||
|
{
|
||||||
|
if (CNF_GetManualEnabled()) {
|
||||||
|
enabled = 1;
|
||||||
|
} else {
|
||||||
|
enabled = 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
n_samples = 0;
|
||||||
|
|
||||||
|
LCL_AddParameterChangeHandler(slew_samples, NULL);
|
||||||
|
}
|
||||||
|
|
||||||
|
/* ================================================== */
|
||||||
|
|
||||||
|
void
|
||||||
|
MNL_Finalise(void)
|
||||||
|
{
|
||||||
|
LCL_RemoveParameterChangeHandler(slew_samples, NULL);
|
||||||
|
}
|
||||||
|
|
||||||
|
/* ================================================== */
|
||||||
|
|
||||||
|
static void
|
||||||
|
estimate_and_set_system(struct timespec *now, int offset_provided, double offset,
|
||||||
|
double *reg_offset, double *dfreq_ppm, double *new_afreq_ppm)
|
||||||
|
{
|
||||||
|
double agos[MAX_SAMPLES], offsets[MAX_SAMPLES];
|
||||||
|
double b0, b1;
|
||||||
|
int n_runs, best_start; /* Unused results from regression analyser */
|
||||||
|
int i;
|
||||||
|
double freq = 0.0;
|
||||||
|
double skew = 0.099999999; /* All 9's when printed to log file */
|
||||||
|
int found_freq;
|
||||||
|
double slew_by;
|
||||||
|
|
||||||
|
b0 = offset_provided ? offset : 0.0;
|
||||||
|
b1 = freq = 0.0;
|
||||||
|
found_freq = 0;
|
||||||
|
|
||||||
|
if (n_samples > 1) {
|
||||||
|
for (i=0; i<n_samples; i++) {
|
||||||
|
agos[i] = UTI_DiffTimespecsToDouble(&samples[n_samples - 1].when, &samples[i].when);
|
||||||
|
offsets[i] = samples[i].offset;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (RGR_FindBestRobustRegression(agos, offsets, n_samples, 1.0e-8,
|
||||||
|
&b0, &b1, &n_runs, &best_start)) {
|
||||||
|
/* Ignore b0 from regression; treat offset as being the most
|
||||||
|
recently entered value. (If the administrator knows he's put
|
||||||
|
an outlier in, he will rerun the settime operation.) However,
|
||||||
|
the frequency estimate comes from the regression. */
|
||||||
|
freq = -b1;
|
||||||
|
found_freq = 1;
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
agos[0] = 0.0;
|
||||||
|
offsets[0] = b0;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (offset_provided) {
|
||||||
|
slew_by = offset;
|
||||||
|
} else {
|
||||||
|
slew_by = b0;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (found_freq) {
|
||||||
|
LOG(LOGS_INFO, "Making a frequency change of %.3f ppm and a slew of %.6f",
|
||||||
|
1.0e6 * freq, slew_by);
|
||||||
|
|
||||||
|
REF_SetManualReference(now,
|
||||||
|
slew_by,
|
||||||
|
freq, skew);
|
||||||
|
} else {
|
||||||
|
LOG(LOGS_INFO, "Making a slew of %.6f", slew_by);
|
||||||
|
REF_SetManualReference(now,
|
||||||
|
slew_by,
|
||||||
|
0.0, skew);
|
||||||
|
}
|
||||||
|
|
||||||
|
if (reg_offset) *reg_offset = b0;
|
||||||
|
if (dfreq_ppm) *dfreq_ppm = 1.0e6 * freq;
|
||||||
|
if (new_afreq_ppm) *new_afreq_ppm = LCL_ReadAbsoluteFrequency();
|
||||||
|
|
||||||
|
/* Calculate residuals to store them */
|
||||||
|
for (i=0; i<n_samples; i++) {
|
||||||
|
samples[i].residual = offsets[i] - (b0 + agos[i] * b1);
|
||||||
|
}
|
||||||
|
|
||||||
|
}
|
||||||
|
|
||||||
|
/* ================================================== */
|
||||||
|
|
||||||
|
int
|
||||||
|
MNL_AcceptTimestamp(struct timespec *ts, double *reg_offset, double *dfreq_ppm, double *new_afreq_ppm)
|
||||||
|
{
|
||||||
|
struct timespec now;
|
||||||
|
double offset, diff;
|
||||||
|
int i;
|
||||||
|
|
||||||
|
if (enabled) {
|
||||||
|
LCL_ReadCookedTime(&now, NULL);
|
||||||
|
|
||||||
|
/* Make sure the provided timestamp is sane and the sample
|
||||||
|
is not too close to the last one */
|
||||||
|
|
||||||
|
if (!UTI_IsTimeOffsetSane(ts, 0.0))
|
||||||
|
return 0;
|
||||||
|
|
||||||
|
if (n_samples) {
|
||||||
|
diff = UTI_DiffTimespecsToDouble(&now, &samples[n_samples - 1].when);
|
||||||
|
if (diff < MIN_SAMPLE_SEPARATION)
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
offset = UTI_DiffTimespecsToDouble(&now, ts);
|
||||||
|
|
||||||
|
/* Check if buffer full up */
|
||||||
|
if (n_samples == MAX_SAMPLES) {
|
||||||
|
/* Shift samples down */
|
||||||
|
for (i=1; i<n_samples; i++) {
|
||||||
|
samples[i-1] = samples[i];
|
||||||
|
}
|
||||||
|
--n_samples;
|
||||||
|
}
|
||||||
|
|
||||||
|
samples[n_samples].when = now;
|
||||||
|
samples[n_samples].offset = offset;
|
||||||
|
samples[n_samples].orig_offset = offset;
|
||||||
|
++n_samples;
|
||||||
|
|
||||||
|
estimate_and_set_system(&now, 1, offset, reg_offset, dfreq_ppm, new_afreq_ppm);
|
||||||
|
|
||||||
|
return 1;
|
||||||
|
|
||||||
|
} else {
|
||||||
|
|
||||||
|
return 0;
|
||||||
|
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/* ================================================== */
|
||||||
|
|
||||||
|
static void
|
||||||
|
slew_samples(struct timespec *raw,
|
||||||
|
struct timespec *cooked,
|
||||||
|
double dfreq,
|
||||||
|
double doffset,
|
||||||
|
LCL_ChangeType change_type,
|
||||||
|
void *not_used)
|
||||||
|
{
|
||||||
|
double delta_time;
|
||||||
|
int i;
|
||||||
|
|
||||||
|
if (change_type == LCL_ChangeUnknownStep) {
|
||||||
|
MNL_Reset();
|
||||||
|
}
|
||||||
|
|
||||||
|
for (i=0; i<n_samples; i++) {
|
||||||
|
UTI_AdjustTimespec(&samples[i].when, cooked, &samples[i].when, &delta_time,
|
||||||
|
dfreq, doffset);
|
||||||
|
samples[i].offset += delta_time;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/* ================================================== */
|
||||||
|
|
||||||
|
void
|
||||||
|
MNL_Enable(void)
|
||||||
|
{
|
||||||
|
enabled = 1;
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
/* ================================================== */
|
||||||
|
|
||||||
|
void
|
||||||
|
MNL_Disable(void)
|
||||||
|
{
|
||||||
|
enabled = 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
/* ================================================== */
|
||||||
|
|
||||||
|
void
|
||||||
|
MNL_Reset(void)
|
||||||
|
{
|
||||||
|
n_samples = 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
/* ================================================== */
|
||||||
|
|
||||||
|
int
|
||||||
|
MNL_IsEnabled(void)
|
||||||
|
{
|
||||||
|
return enabled;
|
||||||
|
}
|
||||||
|
|
||||||
|
/* ================================================== */
|
||||||
|
/* Generate report data for the REQ_MANUAL_LIST command/monitoring
|
||||||
|
protocol */
|
||||||
|
|
||||||
|
void
|
||||||
|
MNL_ReportSamples(RPT_ManualSamplesReport *report, int max, int *n)
|
||||||
|
{
|
||||||
|
int i;
|
||||||
|
|
||||||
|
if (n_samples > max) {
|
||||||
|
*n = max;
|
||||||
|
} else {
|
||||||
|
*n = n_samples;
|
||||||
|
}
|
||||||
|
|
||||||
|
for (i=0; i<n_samples && i<max; i++) {
|
||||||
|
report[i].when = samples[i].when;
|
||||||
|
report[i].slewed_offset = samples[i].offset;
|
||||||
|
report[i].orig_offset = samples[i].orig_offset;
|
||||||
|
report[i].residual = samples[i].residual;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/* ================================================== */
|
||||||
|
/* Delete a sample if it's within range, re-estimate the error and
|
||||||
|
drift and apply it to the system clock. */
|
||||||
|
|
||||||
|
int
|
||||||
|
MNL_DeleteSample(int index)
|
||||||
|
{
|
||||||
|
int i;
|
||||||
|
struct timespec now;
|
||||||
|
|
||||||
|
if ((index < 0) || (index >= n_samples)) {
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
/* Crunch the samples down onto the one being deleted */
|
||||||
|
|
||||||
|
for (i=index; i<(n_samples-1); i++) {
|
||||||
|
samples[i] = samples[i+1];
|
||||||
|
}
|
||||||
|
|
||||||
|
n_samples -= 1;
|
||||||
|
|
||||||
|
/* Now re-estimate. NULLs because we don't want the parameters back
|
||||||
|
in this case. */
|
||||||
|
LCL_ReadCookedTime(&now, NULL);
|
||||||
|
estimate_and_set_system(&now, 0, 0.0, NULL, NULL, NULL);
|
||||||
|
|
||||||
|
return 1;
|
||||||
|
|
||||||
|
}
|
||||||
|
|
||||||
|
/* ================================================== */
|
||||||
|
|
||||||
|
|
46
manual.h
Normal file
46
manual.h
Normal file
|
@ -0,0 +1,46 @@
|
||||||
|
/*
|
||||||
|
chronyd/chronyc - Programs for keeping computer clocks accurate.
|
||||||
|
|
||||||
|
**********************************************************************
|
||||||
|
* Copyright (C) Richard P. Curnow 1997-2002
|
||||||
|
*
|
||||||
|
* This program is free software; you can redistribute it and/or modify
|
||||||
|
* it under the terms of version 2 of the GNU General Public License as
|
||||||
|
* published by the Free Software Foundation.
|
||||||
|
*
|
||||||
|
* This program is distributed in the hope that it will be useful, but
|
||||||
|
* WITHOUT ANY WARRANTY; without even the implied warranty of
|
||||||
|
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
|
||||||
|
* General Public License for more details.
|
||||||
|
*
|
||||||
|
* You should have received a copy of the GNU General Public License along
|
||||||
|
* with this program; if not, write to the Free Software Foundation, Inc.,
|
||||||
|
* 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA.
|
||||||
|
*
|
||||||
|
**********************************************************************
|
||||||
|
|
||||||
|
=======================================================================
|
||||||
|
|
||||||
|
Header file for manual time input module.
|
||||||
|
|
||||||
|
*/
|
||||||
|
|
||||||
|
#ifndef GOT_MANUAL_H
|
||||||
|
#define GOT_MANUAL_H
|
||||||
|
|
||||||
|
#include "sysincl.h"
|
||||||
|
#include "reports.h"
|
||||||
|
|
||||||
|
extern void MNL_Initialise(void);
|
||||||
|
extern void MNL_Finalise(void);
|
||||||
|
extern int MNL_AcceptTimestamp(struct timespec *ts, double *reg_offset, double *dfreq_ppm, double *new_afreq_ppm);
|
||||||
|
|
||||||
|
extern void MNL_Enable(void);
|
||||||
|
extern void MNL_Disable(void);
|
||||||
|
extern void MNL_Reset(void);
|
||||||
|
extern int MNL_IsEnabled(void);
|
||||||
|
|
||||||
|
extern void MNL_ReportSamples(RPT_ManualSamplesReport *report, int max, int *n);
|
||||||
|
extern int MNL_DeleteSample(int index);
|
||||||
|
|
||||||
|
#endif /* GOT_MANUAL_H */
|
315
md5.c
Normal file
315
md5.c
Normal file
|
@ -0,0 +1,315 @@
|
||||||
|
/*
|
||||||
|
***********************************************************************
|
||||||
|
** md5.c -- the source code for MD5 routines **
|
||||||
|
** RSA Data Security, Inc. MD5 Message-Digest Algorithm **
|
||||||
|
** Created: 2/17/90 RLR **
|
||||||
|
** Revised: 1/91 SRD,AJ,BSK,JT Reference C Version **
|
||||||
|
** Revised (for MD5): RLR 4/27/91 **
|
||||||
|
** -- G modified to have y&~z instead of y&z **
|
||||||
|
** -- FF, GG, HH modified to add in last register done **
|
||||||
|
** -- Access pattern: round 2 works mod 5, round 3 works mod 3 **
|
||||||
|
** -- distinct additive constant for each step **
|
||||||
|
** -- round 4 added, working mod 7 **
|
||||||
|
***********************************************************************
|
||||||
|
*/
|
||||||
|
|
||||||
|
/*
|
||||||
|
***********************************************************************
|
||||||
|
** Copyright (C) 1990, RSA Data Security, Inc. All rights reserved. **
|
||||||
|
** **
|
||||||
|
** License to copy and use this software is granted provided that **
|
||||||
|
** it is identified as the "RSA Data Security, Inc. MD5 Message- **
|
||||||
|
** Digest Algorithm" in all material mentioning or referencing this **
|
||||||
|
** software or this function. **
|
||||||
|
** **
|
||||||
|
** License is also granted to make and use derivative works **
|
||||||
|
** provided that such works are identified as "derived from the RSA **
|
||||||
|
** Data Security, Inc. MD5 Message-Digest Algorithm" in all **
|
||||||
|
** material mentioning or referencing the derived work. **
|
||||||
|
** **
|
||||||
|
** RSA Data Security, Inc. makes no representations concerning **
|
||||||
|
** either the merchantability of this software or the suitability **
|
||||||
|
** of this software for any particular purpose. It is provided "as **
|
||||||
|
** is" without express or implied warranty of any kind. **
|
||||||
|
** **
|
||||||
|
** These notices must be retained in any copies of any part of this **
|
||||||
|
** documentation and/or software. **
|
||||||
|
***********************************************************************
|
||||||
|
*/
|
||||||
|
|
||||||
|
#include "md5.h"
|
||||||
|
|
||||||
|
/*
|
||||||
|
***********************************************************************
|
||||||
|
** Message-digest routines: **
|
||||||
|
** To form the message digest for a message M **
|
||||||
|
** (1) Initialize a context buffer mdContext using MD5Init **
|
||||||
|
** (2) Call MD5Update on mdContext and M **
|
||||||
|
** (3) Call MD5Final on mdContext **
|
||||||
|
** The message digest is now in mdContext->digest[0...15] **
|
||||||
|
***********************************************************************
|
||||||
|
*/
|
||||||
|
|
||||||
|
/* forward declaration */
|
||||||
|
static void Transform (UINT4 *, UINT4 *);
|
||||||
|
|
||||||
|
#ifdef __STDC__
|
||||||
|
static const
|
||||||
|
#else
|
||||||
|
static
|
||||||
|
#endif
|
||||||
|
unsigned char PADDING[64] = {
|
||||||
|
0x80, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
|
||||||
|
0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
|
||||||
|
0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
|
||||||
|
0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
|
||||||
|
0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
|
||||||
|
0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
|
||||||
|
0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
|
||||||
|
0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00
|
||||||
|
};
|
||||||
|
|
||||||
|
/* F, G, H and I are basic MD5 functions */
|
||||||
|
#define F(x, y, z) (((x) & (y)) | ((~x) & (z)))
|
||||||
|
#define G(x, y, z) (((x) & (z)) | ((y) & (~z)))
|
||||||
|
#define H(x, y, z) ((x) ^ (y) ^ (z))
|
||||||
|
#define I(x, y, z) ((y) ^ ((x) | (~z)))
|
||||||
|
|
||||||
|
/* ROTATE_LEFT rotates x left n bits */
|
||||||
|
#if defined(FAST_MD5) && defined(__GNUC__) && defined(mc68000)
|
||||||
|
/*
|
||||||
|
* If we're on a 68000 based CPU and using a GNU C compiler with
|
||||||
|
* inline assembly code, we can speed this up a bit.
|
||||||
|
*/
|
||||||
|
inline UINT4 ROTATE_LEFT(UINT4 x, int n)
|
||||||
|
{
|
||||||
|
asm("roll %2,%0" : "=d" (x) : "0" (x), "Ir" (n));
|
||||||
|
return x;
|
||||||
|
}
|
||||||
|
#else
|
||||||
|
#define ROTATE_LEFT(x, n) (((x) << (n)) | ((x) >> (32-(n))))
|
||||||
|
#endif
|
||||||
|
|
||||||
|
|
||||||
|
/* FF, GG, HH, and II transformations for rounds 1, 2, 3, and 4 */
|
||||||
|
/* Rotation is separate from addition to prevent recomputation */
|
||||||
|
#define FF(a, b, c, d, x, s, ac) \
|
||||||
|
{(a) += F ((b), (c), (d)) + (x) + (UINT4)(ac); \
|
||||||
|
(a) = ROTATE_LEFT ((a), (s)); \
|
||||||
|
(a) += (b); \
|
||||||
|
}
|
||||||
|
#define GG(a, b, c, d, x, s, ac) \
|
||||||
|
{(a) += G ((b), (c), (d)) + (x) + (UINT4)(ac); \
|
||||||
|
(a) = ROTATE_LEFT ((a), (s)); \
|
||||||
|
(a) += (b); \
|
||||||
|
}
|
||||||
|
#define HH(a, b, c, d, x, s, ac) \
|
||||||
|
{(a) += H ((b), (c), (d)) + (x) + (UINT4)(ac); \
|
||||||
|
(a) = ROTATE_LEFT ((a), (s)); \
|
||||||
|
(a) += (b); \
|
||||||
|
}
|
||||||
|
#define II(a, b, c, d, x, s, ac) \
|
||||||
|
{(a) += I ((b), (c), (d)) + (x) + (UINT4)(ac); \
|
||||||
|
(a) = ROTATE_LEFT ((a), (s)); \
|
||||||
|
(a) += (b); \
|
||||||
|
}
|
||||||
|
|
||||||
|
/* The routine MD5Init initializes the message-digest context
|
||||||
|
mdContext. All fields are set to zero.
|
||||||
|
*/
|
||||||
|
void MD5Init (MD5_CTX *mdContext)
|
||||||
|
{
|
||||||
|
mdContext->i[0] = mdContext->i[1] = (UINT4)0;
|
||||||
|
|
||||||
|
/* Load magic initialization constants.
|
||||||
|
*/
|
||||||
|
mdContext->buf[0] = (UINT4)0x67452301;
|
||||||
|
mdContext->buf[1] = (UINT4)0xefcdab89;
|
||||||
|
mdContext->buf[2] = (UINT4)0x98badcfe;
|
||||||
|
mdContext->buf[3] = (UINT4)0x10325476;
|
||||||
|
}
|
||||||
|
|
||||||
|
/* The routine MD5Update updates the message-digest context to
|
||||||
|
account for the presence of each of the characters inBuf[0..inLen-1]
|
||||||
|
in the message whose digest is being computed.
|
||||||
|
*/
|
||||||
|
void MD5Update (MD5_CTX *mdContext, unsigned const char *inBuf, unsigned int inLen)
|
||||||
|
{
|
||||||
|
UINT4 in[16];
|
||||||
|
int mdi;
|
||||||
|
unsigned int i, ii;
|
||||||
|
|
||||||
|
/* compute number of bytes mod 64 */
|
||||||
|
mdi = (int)((mdContext->i[0] >> 3) & 0x3F);
|
||||||
|
|
||||||
|
/* update number of bits */
|
||||||
|
if ((mdContext->i[0] + ((UINT4)inLen << 3)) < mdContext->i[0])
|
||||||
|
mdContext->i[1]++;
|
||||||
|
mdContext->i[0] += ((UINT4)inLen << 3);
|
||||||
|
mdContext->i[1] += ((UINT4)inLen >> 29);
|
||||||
|
|
||||||
|
while (inLen--) {
|
||||||
|
/* add new character to buffer, increment mdi */
|
||||||
|
mdContext->in[mdi++] = *inBuf++;
|
||||||
|
|
||||||
|
/* transform if necessary */
|
||||||
|
if (mdi == 0x40) {
|
||||||
|
for (i = 0, ii = 0; i < 16; i++, ii += 4)
|
||||||
|
in[i] = (((UINT4)mdContext->in[ii+3]) << 24) |
|
||||||
|
(((UINT4)mdContext->in[ii+2]) << 16) |
|
||||||
|
(((UINT4)mdContext->in[ii+1]) << 8) |
|
||||||
|
((UINT4)mdContext->in[ii]);
|
||||||
|
Transform (mdContext->buf, in);
|
||||||
|
mdi = 0;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/* The routine MD5Final terminates the message-digest computation and
|
||||||
|
ends with the desired message digest in mdContext->digest[0...15].
|
||||||
|
*/
|
||||||
|
|
||||||
|
void MD5Final (MD5_CTX *mdContext)
|
||||||
|
{
|
||||||
|
UINT4 in[16];
|
||||||
|
int mdi;
|
||||||
|
unsigned int i, ii;
|
||||||
|
unsigned int padLen;
|
||||||
|
|
||||||
|
/* save number of bits */
|
||||||
|
in[14] = mdContext->i[0];
|
||||||
|
in[15] = mdContext->i[1];
|
||||||
|
|
||||||
|
/* compute number of bytes mod 64 */
|
||||||
|
mdi = (int)((mdContext->i[0] >> 3) & 0x3F);
|
||||||
|
|
||||||
|
/* pad out to 56 mod 64 */
|
||||||
|
padLen = (mdi < 56) ? (56 - mdi) : (120 - mdi);
|
||||||
|
MD5Update (mdContext, PADDING, padLen);
|
||||||
|
|
||||||
|
/* append length in bits and transform */
|
||||||
|
for (i = 0, ii = 0; i < 14; i++, ii += 4)
|
||||||
|
in[i] = (((UINT4)mdContext->in[ii+3]) << 24) |
|
||||||
|
(((UINT4)mdContext->in[ii+2]) << 16) |
|
||||||
|
(((UINT4)mdContext->in[ii+1]) << 8) |
|
||||||
|
((UINT4)mdContext->in[ii]);
|
||||||
|
Transform (mdContext->buf, in);
|
||||||
|
|
||||||
|
/* store buffer in digest */
|
||||||
|
for (i = 0, ii = 0; i < 4; i++, ii += 4) {
|
||||||
|
mdContext->digest[ii] = (unsigned char)(mdContext->buf[i] & 0xFF);
|
||||||
|
mdContext->digest[ii+1] =
|
||||||
|
(unsigned char)((mdContext->buf[i] >> 8) & 0xFF);
|
||||||
|
mdContext->digest[ii+2] =
|
||||||
|
(unsigned char)((mdContext->buf[i] >> 16) & 0xFF);
|
||||||
|
mdContext->digest[ii+3] =
|
||||||
|
(unsigned char)((mdContext->buf[i] >> 24) & 0xFF);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/* Basic MD5 step. Transforms buf based on in.
|
||||||
|
*/
|
||||||
|
static void Transform (UINT4 *buf, UINT4 *in)
|
||||||
|
{
|
||||||
|
UINT4 a = buf[0], b = buf[1], c = buf[2], d = buf[3];
|
||||||
|
|
||||||
|
/* Round 1 */
|
||||||
|
#define S11 7
|
||||||
|
#define S12 12
|
||||||
|
#define S13 17
|
||||||
|
#define S14 22
|
||||||
|
|
||||||
|
FF ( a, b, c, d, in[ 0], S11, 0xd76aa478); /* 1 */
|
||||||
|
FF ( d, a, b, c, in[ 1], S12, 0xe8c7b756); /* 2 */
|
||||||
|
FF ( c, d, a, b, in[ 2], S13, 0x242070db); /* 3 */
|
||||||
|
FF ( b, c, d, a, in[ 3], S14, 0xc1bdceee); /* 4 */
|
||||||
|
FF ( a, b, c, d, in[ 4], S11, 0xf57c0faf); /* 5 */
|
||||||
|
FF ( d, a, b, c, in[ 5], S12, 0x4787c62a); /* 6 */
|
||||||
|
FF ( c, d, a, b, in[ 6], S13, 0xa8304613); /* 7 */
|
||||||
|
FF ( b, c, d, a, in[ 7], S14, 0xfd469501); /* 8 */
|
||||||
|
FF ( a, b, c, d, in[ 8], S11, 0x698098d8); /* 9 */
|
||||||
|
FF ( d, a, b, c, in[ 9], S12, 0x8b44f7af); /* 10 */
|
||||||
|
FF ( c, d, a, b, in[10], S13, 0xffff5bb1); /* 11 */
|
||||||
|
FF ( b, c, d, a, in[11], S14, 0x895cd7be); /* 12 */
|
||||||
|
FF ( a, b, c, d, in[12], S11, 0x6b901122); /* 13 */
|
||||||
|
FF ( d, a, b, c, in[13], S12, 0xfd987193); /* 14 */
|
||||||
|
FF ( c, d, a, b, in[14], S13, 0xa679438e); /* 15 */
|
||||||
|
FF ( b, c, d, a, in[15], S14, 0x49b40821); /* 16 */
|
||||||
|
|
||||||
|
/* Round 2 */
|
||||||
|
#define S21 5
|
||||||
|
#define S22 9
|
||||||
|
#define S23 14
|
||||||
|
#define S24 20
|
||||||
|
GG ( a, b, c, d, in[ 1], S21, 0xf61e2562); /* 17 */
|
||||||
|
GG ( d, a, b, c, in[ 6], S22, 0xc040b340); /* 18 */
|
||||||
|
GG ( c, d, a, b, in[11], S23, 0x265e5a51); /* 19 */
|
||||||
|
GG ( b, c, d, a, in[ 0], S24, 0xe9b6c7aa); /* 20 */
|
||||||
|
GG ( a, b, c, d, in[ 5], S21, 0xd62f105d); /* 21 */
|
||||||
|
GG ( d, a, b, c, in[10], S22, 0x2441453); /* 22 */
|
||||||
|
GG ( c, d, a, b, in[15], S23, 0xd8a1e681); /* 23 */
|
||||||
|
GG ( b, c, d, a, in[ 4], S24, 0xe7d3fbc8); /* 24 */
|
||||||
|
GG ( a, b, c, d, in[ 9], S21, 0x21e1cde6); /* 25 */
|
||||||
|
GG ( d, a, b, c, in[14], S22, 0xc33707d6); /* 26 */
|
||||||
|
GG ( c, d, a, b, in[ 3], S23, 0xf4d50d87); /* 27 */
|
||||||
|
GG ( b, c, d, a, in[ 8], S24, 0x455a14ed); /* 28 */
|
||||||
|
GG ( a, b, c, d, in[13], S21, 0xa9e3e905); /* 29 */
|
||||||
|
GG ( d, a, b, c, in[ 2], S22, 0xfcefa3f8); /* 30 */
|
||||||
|
GG ( c, d, a, b, in[ 7], S23, 0x676f02d9); /* 31 */
|
||||||
|
GG ( b, c, d, a, in[12], S24, 0x8d2a4c8a); /* 32 */
|
||||||
|
|
||||||
|
/* Round 3 */
|
||||||
|
#define S31 4
|
||||||
|
#define S32 11
|
||||||
|
#define S33 16
|
||||||
|
#define S34 23
|
||||||
|
HH ( a, b, c, d, in[ 5], S31, 0xfffa3942); /* 33 */
|
||||||
|
HH ( d, a, b, c, in[ 8], S32, 0x8771f681); /* 34 */
|
||||||
|
HH ( c, d, a, b, in[11], S33, 0x6d9d6122); /* 35 */
|
||||||
|
HH ( b, c, d, a, in[14], S34, 0xfde5380c); /* 36 */
|
||||||
|
HH ( a, b, c, d, in[ 1], S31, 0xa4beea44); /* 37 */
|
||||||
|
HH ( d, a, b, c, in[ 4], S32, 0x4bdecfa9); /* 38 */
|
||||||
|
HH ( c, d, a, b, in[ 7], S33, 0xf6bb4b60); /* 39 */
|
||||||
|
HH ( b, c, d, a, in[10], S34, 0xbebfbc70); /* 40 */
|
||||||
|
HH ( a, b, c, d, in[13], S31, 0x289b7ec6); /* 41 */
|
||||||
|
HH ( d, a, b, c, in[ 0], S32, 0xeaa127fa); /* 42 */
|
||||||
|
HH ( c, d, a, b, in[ 3], S33, 0xd4ef3085); /* 43 */
|
||||||
|
HH ( b, c, d, a, in[ 6], S34, 0x4881d05); /* 44 */
|
||||||
|
HH ( a, b, c, d, in[ 9], S31, 0xd9d4d039); /* 45 */
|
||||||
|
HH ( d, a, b, c, in[12], S32, 0xe6db99e5); /* 46 */
|
||||||
|
HH ( c, d, a, b, in[15], S33, 0x1fa27cf8); /* 47 */
|
||||||
|
HH ( b, c, d, a, in[ 2], S34, 0xc4ac5665); /* 48 */
|
||||||
|
|
||||||
|
/* Round 4 */
|
||||||
|
#define S41 6
|
||||||
|
#define S42 10
|
||||||
|
#define S43 15
|
||||||
|
#define S44 21
|
||||||
|
II ( a, b, c, d, in[ 0], S41, 0xf4292244); /* 49 */
|
||||||
|
II ( d, a, b, c, in[ 7], S42, 0x432aff97); /* 50 */
|
||||||
|
II ( c, d, a, b, in[14], S43, 0xab9423a7); /* 51 */
|
||||||
|
II ( b, c, d, a, in[ 5], S44, 0xfc93a039); /* 52 */
|
||||||
|
II ( a, b, c, d, in[12], S41, 0x655b59c3); /* 53 */
|
||||||
|
II ( d, a, b, c, in[ 3], S42, 0x8f0ccc92); /* 54 */
|
||||||
|
II ( c, d, a, b, in[10], S43, 0xffeff47d); /* 55 */
|
||||||
|
II ( b, c, d, a, in[ 1], S44, 0x85845dd1); /* 56 */
|
||||||
|
II ( a, b, c, d, in[ 8], S41, 0x6fa87e4f); /* 57 */
|
||||||
|
II ( d, a, b, c, in[15], S42, 0xfe2ce6e0); /* 58 */
|
||||||
|
II ( c, d, a, b, in[ 6], S43, 0xa3014314); /* 59 */
|
||||||
|
II ( b, c, d, a, in[13], S44, 0x4e0811a1); /* 60 */
|
||||||
|
II ( a, b, c, d, in[ 4], S41, 0xf7537e82); /* 61 */
|
||||||
|
II ( d, a, b, c, in[11], S42, 0xbd3af235); /* 62 */
|
||||||
|
II ( c, d, a, b, in[ 2], S43, 0x2ad7d2bb); /* 63 */
|
||||||
|
II ( b, c, d, a, in[ 9], S44, 0xeb86d391); /* 64 */
|
||||||
|
|
||||||
|
buf[0] += a;
|
||||||
|
buf[1] += b;
|
||||||
|
buf[2] += c;
|
||||||
|
buf[3] += d;
|
||||||
|
}
|
||||||
|
|
||||||
|
/*
|
||||||
|
***********************************************************************
|
||||||
|
** End of md5.c **
|
||||||
|
******************************** (cut) ********************************
|
||||||
|
*/
|
56
md5.h
Normal file
56
md5.h
Normal file
|
@ -0,0 +1,56 @@
|
||||||
|
/*
|
||||||
|
***********************************************************************
|
||||||
|
** md5.h -- header file for implementation of MD5 **
|
||||||
|
** RSA Data Security, Inc. MD5 Message-Digest Algorithm **
|
||||||
|
** Created: 2/17/90 RLR **
|
||||||
|
** Revised: 12/27/90 SRD,AJ,BSK,JT Reference C version **
|
||||||
|
** Revised (for MD5): RLR 4/27/91 **
|
||||||
|
***********************************************************************
|
||||||
|
*/
|
||||||
|
|
||||||
|
/*
|
||||||
|
***********************************************************************
|
||||||
|
** Copyright (C) 1990, RSA Data Security, Inc. All rights reserved. **
|
||||||
|
** **
|
||||||
|
** License to copy and use this software is granted provided that **
|
||||||
|
** it is identified as the "RSA Data Security, Inc. MD5 Message- **
|
||||||
|
** Digest Algorithm" in all material mentioning or referencing this **
|
||||||
|
** software or this function. **
|
||||||
|
** **
|
||||||
|
** License is also granted to make and use derivative works **
|
||||||
|
** provided that such works are identified as "derived from the RSA **
|
||||||
|
** Data Security, Inc. MD5 Message-Digest Algorithm" in all **
|
||||||
|
** material mentioning or referencing the derived work. **
|
||||||
|
** **
|
||||||
|
** RSA Data Security, Inc. makes no representations concerning **
|
||||||
|
** either the merchantability of this software or the suitability **
|
||||||
|
** of this software for any particular purpose. It is provided "as **
|
||||||
|
** is" without express or implied warranty of any kind. **
|
||||||
|
** **
|
||||||
|
** These notices must be retained in any copies of any part of this **
|
||||||
|
** documentation and/or software. **
|
||||||
|
***********************************************************************
|
||||||
|
*/
|
||||||
|
|
||||||
|
#include "sysincl.h"
|
||||||
|
|
||||||
|
/* typedef a 32-bit type */
|
||||||
|
typedef uint32_t UINT4;
|
||||||
|
|
||||||
|
/* Data structure for MD5 (Message-Digest) computation */
|
||||||
|
typedef struct {
|
||||||
|
UINT4 i[2]; /* number of _bits_ handled mod 2^64 */
|
||||||
|
UINT4 buf[4]; /* scratch buffer */
|
||||||
|
unsigned char in[64]; /* input buffer */
|
||||||
|
unsigned char digest[16]; /* actual digest after MD5Final call */
|
||||||
|
} MD5_CTX;
|
||||||
|
|
||||||
|
void MD5Init (MD5_CTX *mdContext);
|
||||||
|
void MD5Update (MD5_CTX *, unsigned const char *, unsigned int);
|
||||||
|
void MD5Final (MD5_CTX *);
|
||||||
|
|
||||||
|
/*
|
||||||
|
***********************************************************************
|
||||||
|
** End of md5.h **
|
||||||
|
******************************** (cut) ********************************
|
||||||
|
*/
|
98
memory.c
Normal file
98
memory.c
Normal file
|
@ -0,0 +1,98 @@
|
||||||
|
/*
|
||||||
|
chronyd/chronyc - Programs for keeping computer clocks accurate.
|
||||||
|
|
||||||
|
**********************************************************************
|
||||||
|
* Copyright (C) Miroslav Lichvar 2014, 2017
|
||||||
|
*
|
||||||
|
* This program is free software; you can redistribute it and/or modify
|
||||||
|
* it under the terms of version 2 of the GNU General Public License as
|
||||||
|
* published by the Free Software Foundation.
|
||||||
|
*
|
||||||
|
* This program is distributed in the hope that it will be useful, but
|
||||||
|
* WITHOUT ANY WARRANTY; without even the implied warranty of
|
||||||
|
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
|
||||||
|
* General Public License for more details.
|
||||||
|
*
|
||||||
|
* You should have received a copy of the GNU General Public License along
|
||||||
|
* with this program; if not, write to the Free Software Foundation, Inc.,
|
||||||
|
* 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA.
|
||||||
|
*
|
||||||
|
**********************************************************************
|
||||||
|
|
||||||
|
=======================================================================
|
||||||
|
|
||||||
|
Utility functions for memory allocation.
|
||||||
|
|
||||||
|
*/
|
||||||
|
|
||||||
|
#include "config.h"
|
||||||
|
|
||||||
|
#include "logging.h"
|
||||||
|
#include "memory.h"
|
||||||
|
|
||||||
|
void *
|
||||||
|
Malloc(size_t size)
|
||||||
|
{
|
||||||
|
void *r;
|
||||||
|
|
||||||
|
r = malloc(size);
|
||||||
|
if (!r && size)
|
||||||
|
LOG_FATAL("Could not allocate memory");
|
||||||
|
|
||||||
|
return r;
|
||||||
|
}
|
||||||
|
|
||||||
|
void *
|
||||||
|
Realloc(void *ptr, size_t size)
|
||||||
|
{
|
||||||
|
void *r;
|
||||||
|
|
||||||
|
if (size == 0) {
|
||||||
|
Free(ptr);
|
||||||
|
return NULL;
|
||||||
|
}
|
||||||
|
|
||||||
|
r = realloc(ptr, size);
|
||||||
|
if (!r)
|
||||||
|
LOG_FATAL("Could not allocate memory");
|
||||||
|
|
||||||
|
return r;
|
||||||
|
}
|
||||||
|
|
||||||
|
static size_t
|
||||||
|
get_array_size(size_t nmemb, size_t size)
|
||||||
|
{
|
||||||
|
size_t array_size;
|
||||||
|
|
||||||
|
array_size = nmemb * size;
|
||||||
|
|
||||||
|
/* Check for overflow */
|
||||||
|
if (nmemb > 0 && array_size / nmemb != size)
|
||||||
|
LOG_FATAL("Could not allocate memory");
|
||||||
|
|
||||||
|
return array_size;
|
||||||
|
}
|
||||||
|
|
||||||
|
void *
|
||||||
|
Malloc2(size_t nmemb, size_t size)
|
||||||
|
{
|
||||||
|
return Malloc(get_array_size(nmemb, size));
|
||||||
|
}
|
||||||
|
|
||||||
|
void *
|
||||||
|
Realloc2(void *ptr, size_t nmemb, size_t size)
|
||||||
|
{
|
||||||
|
return Realloc(ptr, get_array_size(nmemb, size));
|
||||||
|
}
|
||||||
|
|
||||||
|
char *
|
||||||
|
Strdup(const char *s)
|
||||||
|
{
|
||||||
|
void *r;
|
||||||
|
|
||||||
|
r = strdup(s);
|
||||||
|
if (!r)
|
||||||
|
LOG_FATAL("Could not allocate memory");
|
||||||
|
|
||||||
|
return r;
|
||||||
|
}
|
45
memory.h
Normal file
45
memory.h
Normal file
|
@ -0,0 +1,45 @@
|
||||||
|
/*
|
||||||
|
chronyd/chronyc - Programs for keeping computer clocks accurate.
|
||||||
|
|
||||||
|
**********************************************************************
|
||||||
|
* Copyright (C) Richard P. Curnow 1997-2002
|
||||||
|
*
|
||||||
|
* This program is free software; you can redistribute it and/or modify
|
||||||
|
* it under the terms of version 2 of the GNU General Public License as
|
||||||
|
* published by the Free Software Foundation.
|
||||||
|
*
|
||||||
|
* This program is distributed in the hope that it will be useful, but
|
||||||
|
* WITHOUT ANY WARRANTY; without even the implied warranty of
|
||||||
|
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
|
||||||
|
* General Public License for more details.
|
||||||
|
*
|
||||||
|
* You should have received a copy of the GNU General Public License along
|
||||||
|
* with this program; if not, write to the Free Software Foundation, Inc.,
|
||||||
|
* 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA.
|
||||||
|
*
|
||||||
|
**********************************************************************
|
||||||
|
|
||||||
|
=======================================================================
|
||||||
|
|
||||||
|
Header file for memory functions
|
||||||
|
*/
|
||||||
|
|
||||||
|
#ifndef GOT_MEMORY_H
|
||||||
|
#define GOT_MEMORY_H
|
||||||
|
|
||||||
|
#include "sysincl.h"
|
||||||
|
|
||||||
|
/* Wrappers checking for errors */
|
||||||
|
extern void *Malloc(size_t size);
|
||||||
|
extern void *Realloc(void *ptr, size_t size);
|
||||||
|
extern void *Malloc2(size_t nmemb, size_t size);
|
||||||
|
extern void *Realloc2(void *ptr, size_t nmemb, size_t size);
|
||||||
|
extern char *Strdup(const char *s);
|
||||||
|
|
||||||
|
/* Convenient macros */
|
||||||
|
#define MallocNew(T) ((T *) Malloc(sizeof(T)))
|
||||||
|
#define MallocArray(T, n) ((T *) Malloc2(n, sizeof(T)))
|
||||||
|
#define ReallocArray(T, n, x) ((T *) Realloc2((void *)(x), n, sizeof(T)))
|
||||||
|
#define Free(x) free(x)
|
||||||
|
|
||||||
|
#endif /* GOT_MEMORY_H */
|
166
nameserv.c
Normal file
166
nameserv.c
Normal file
|
@ -0,0 +1,166 @@
|
||||||
|
/*
|
||||||
|
chronyd/chronyc - Programs for keeping computer clocks accurate.
|
||||||
|
|
||||||
|
**********************************************************************
|
||||||
|
* Copyright (C) Richard P. Curnow 1997-2003
|
||||||
|
* Copyright (C) Miroslav Lichvar 2009-2011
|
||||||
|
*
|
||||||
|
* This program is free software; you can redistribute it and/or modify
|
||||||
|
* it under the terms of version 2 of the GNU General Public License as
|
||||||
|
* published by the Free Software Foundation.
|
||||||
|
*
|
||||||
|
* This program is distributed in the hope that it will be useful, but
|
||||||
|
* WITHOUT ANY WARRANTY; without even the implied warranty of
|
||||||
|
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
|
||||||
|
* General Public License for more details.
|
||||||
|
*
|
||||||
|
* You should have received a copy of the GNU General Public License along
|
||||||
|
* with this program; if not, write to the Free Software Foundation, Inc.,
|
||||||
|
* 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA.
|
||||||
|
*
|
||||||
|
**********************************************************************
|
||||||
|
|
||||||
|
=======================================================================
|
||||||
|
|
||||||
|
Functions to do name to IP address conversion
|
||||||
|
|
||||||
|
*/
|
||||||
|
|
||||||
|
#include "config.h"
|
||||||
|
|
||||||
|
#include "sysincl.h"
|
||||||
|
|
||||||
|
#include <netdb.h>
|
||||||
|
#include <resolv.h>
|
||||||
|
|
||||||
|
#include "nameserv.h"
|
||||||
|
#include "socket.h"
|
||||||
|
#include "util.h"
|
||||||
|
|
||||||
|
/* ================================================== */
|
||||||
|
|
||||||
|
static int address_family = IPADDR_UNSPEC;
|
||||||
|
|
||||||
|
void
|
||||||
|
DNS_SetAddressFamily(int family)
|
||||||
|
{
|
||||||
|
address_family = family;
|
||||||
|
}
|
||||||
|
|
||||||
|
DNS_Status
|
||||||
|
DNS_Name2IPAddress(const char *name, IPAddr *ip_addrs, int max_addrs)
|
||||||
|
{
|
||||||
|
struct addrinfo hints, *res, *ai;
|
||||||
|
int i, result;
|
||||||
|
IPAddr ip;
|
||||||
|
|
||||||
|
max_addrs = MIN(max_addrs, DNS_MAX_ADDRESSES);
|
||||||
|
|
||||||
|
for (i = 0; i < max_addrs; i++)
|
||||||
|
ip_addrs[i].family = IPADDR_UNSPEC;
|
||||||
|
|
||||||
|
/* Avoid calling getaddrinfo() if the name is an IP address */
|
||||||
|
if (UTI_StringToIP(name, &ip)) {
|
||||||
|
if (address_family != IPADDR_UNSPEC && ip.family != address_family)
|
||||||
|
return DNS_Failure;
|
||||||
|
if (max_addrs >= 1)
|
||||||
|
ip_addrs[0] = ip;
|
||||||
|
return DNS_Success;
|
||||||
|
}
|
||||||
|
|
||||||
|
memset(&hints, 0, sizeof (hints));
|
||||||
|
|
||||||
|
switch (address_family) {
|
||||||
|
case IPADDR_INET4:
|
||||||
|
hints.ai_family = AF_INET;
|
||||||
|
break;
|
||||||
|
#ifdef FEAT_IPV6
|
||||||
|
case IPADDR_INET6:
|
||||||
|
hints.ai_family = AF_INET6;
|
||||||
|
break;
|
||||||
|
#endif
|
||||||
|
default:
|
||||||
|
hints.ai_family = AF_UNSPEC;
|
||||||
|
}
|
||||||
|
hints.ai_socktype = SOCK_DGRAM;
|
||||||
|
|
||||||
|
result = getaddrinfo(name, NULL, &hints, &res);
|
||||||
|
|
||||||
|
if (result) {
|
||||||
|
#ifdef FORCE_DNSRETRY
|
||||||
|
return DNS_TryAgain;
|
||||||
|
#else
|
||||||
|
return result == EAI_AGAIN ? DNS_TryAgain : DNS_Failure;
|
||||||
|
#endif
|
||||||
|
}
|
||||||
|
|
||||||
|
for (ai = res, i = 0; i < max_addrs && ai != NULL; ai = ai->ai_next) {
|
||||||
|
switch (ai->ai_family) {
|
||||||
|
case AF_INET:
|
||||||
|
if (address_family != IPADDR_UNSPEC && address_family != IPADDR_INET4)
|
||||||
|
continue;
|
||||||
|
ip_addrs[i].family = IPADDR_INET4;
|
||||||
|
ip_addrs[i].addr.in4 = ntohl(((struct sockaddr_in *)ai->ai_addr)->sin_addr.s_addr);
|
||||||
|
i++;
|
||||||
|
break;
|
||||||
|
#ifdef FEAT_IPV6
|
||||||
|
case AF_INET6:
|
||||||
|
if (address_family != IPADDR_UNSPEC && address_family != IPADDR_INET6)
|
||||||
|
continue;
|
||||||
|
/* Don't return an address that would lose a scope ID */
|
||||||
|
if (((struct sockaddr_in6 *)ai->ai_addr)->sin6_scope_id != 0)
|
||||||
|
continue;
|
||||||
|
ip_addrs[i].family = IPADDR_INET6;
|
||||||
|
memcpy(&ip_addrs[i].addr.in6, &((struct sockaddr_in6 *)ai->ai_addr)->sin6_addr.s6_addr,
|
||||||
|
sizeof (ip_addrs->addr.in6));
|
||||||
|
i++;
|
||||||
|
break;
|
||||||
|
#endif
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
freeaddrinfo(res);
|
||||||
|
|
||||||
|
return !max_addrs || ip_addrs[0].family != IPADDR_UNSPEC ? DNS_Success : DNS_Failure;
|
||||||
|
}
|
||||||
|
|
||||||
|
/* ================================================== */
|
||||||
|
|
||||||
|
int
|
||||||
|
DNS_IPAddress2Name(IPAddr *ip_addr, char *name, int len)
|
||||||
|
{
|
||||||
|
char *result = NULL;
|
||||||
|
#ifdef FEAT_IPV6
|
||||||
|
struct sockaddr_in6 saddr;
|
||||||
|
#else
|
||||||
|
struct sockaddr_in saddr;
|
||||||
|
#endif
|
||||||
|
IPSockAddr ip_saddr;
|
||||||
|
socklen_t slen;
|
||||||
|
char hbuf[NI_MAXHOST];
|
||||||
|
|
||||||
|
ip_saddr.ip_addr = *ip_addr;
|
||||||
|
ip_saddr.port = 0;
|
||||||
|
|
||||||
|
slen = SCK_IPSockAddrToSockaddr(&ip_saddr, (struct sockaddr *)&saddr, sizeof (saddr));
|
||||||
|
if (!getnameinfo((struct sockaddr *)&saddr, slen, hbuf, sizeof (hbuf), NULL, 0, 0))
|
||||||
|
result = hbuf;
|
||||||
|
|
||||||
|
if (result == NULL)
|
||||||
|
result = UTI_IPToString(ip_addr);
|
||||||
|
if (snprintf(name, len, "%s", result) >= len)
|
||||||
|
return 0;
|
||||||
|
|
||||||
|
return 1;
|
||||||
|
}
|
||||||
|
|
||||||
|
/* ================================================== */
|
||||||
|
|
||||||
|
void
|
||||||
|
DNS_Reload(void)
|
||||||
|
{
|
||||||
|
res_init();
|
||||||
|
}
|
||||||
|
|
||||||
|
/* ================================================== */
|
||||||
|
|
52
nameserv.h
Normal file
52
nameserv.h
Normal file
|
@ -0,0 +1,52 @@
|
||||||
|
/*
|
||||||
|
chronyd/chronyc - Programs for keeping computer clocks accurate.
|
||||||
|
|
||||||
|
**********************************************************************
|
||||||
|
* Copyright (C) Richard P. Curnow 1997-2002
|
||||||
|
*
|
||||||
|
* This program is free software; you can redistribute it and/or modify
|
||||||
|
* it under the terms of version 2 of the GNU General Public License as
|
||||||
|
* published by the Free Software Foundation.
|
||||||
|
*
|
||||||
|
* This program is distributed in the hope that it will be useful, but
|
||||||
|
* WITHOUT ANY WARRANTY; without even the implied warranty of
|
||||||
|
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
|
||||||
|
* General Public License for more details.
|
||||||
|
*
|
||||||
|
* You should have received a copy of the GNU General Public License along
|
||||||
|
* with this program; if not, write to the Free Software Foundation, Inc.,
|
||||||
|
* 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA.
|
||||||
|
*
|
||||||
|
**********************************************************************
|
||||||
|
|
||||||
|
=======================================================================
|
||||||
|
|
||||||
|
Module header for nameserver functions
|
||||||
|
*/
|
||||||
|
|
||||||
|
|
||||||
|
#ifndef GOT_NAMESERV_H
|
||||||
|
#define GOT_NAMESERV_H
|
||||||
|
|
||||||
|
#include "addressing.h"
|
||||||
|
|
||||||
|
typedef enum {
|
||||||
|
DNS_Success,
|
||||||
|
DNS_TryAgain,
|
||||||
|
DNS_Failure
|
||||||
|
} DNS_Status;
|
||||||
|
|
||||||
|
/* Resolve names only to selected address family */
|
||||||
|
extern void DNS_SetAddressFamily(int family);
|
||||||
|
|
||||||
|
/* Maximum number of addresses returned by DNS_Name2IPAddress */
|
||||||
|
#define DNS_MAX_ADDRESSES 16
|
||||||
|
|
||||||
|
extern DNS_Status DNS_Name2IPAddress(const char *name, IPAddr *ip_addrs, int max_addrs);
|
||||||
|
|
||||||
|
extern int DNS_IPAddress2Name(IPAddr *ip_addr, char *name, int len);
|
||||||
|
|
||||||
|
extern void DNS_Reload(void);
|
||||||
|
|
||||||
|
#endif /* GOT_NAMESERV_H */
|
||||||
|
|
130
nameserv_async.c
Normal file
130
nameserv_async.c
Normal file
|
@ -0,0 +1,130 @@
|
||||||
|
/*
|
||||||
|
chronyd/chronyc - Programs for keeping computer clocks accurate.
|
||||||
|
|
||||||
|
**********************************************************************
|
||||||
|
* Copyright (C) Miroslav Lichvar 2014
|
||||||
|
*
|
||||||
|
* This program is free software; you can redistribute it and/or modify
|
||||||
|
* it under the terms of version 2 of the GNU General Public License as
|
||||||
|
* published by the Free Software Foundation.
|
||||||
|
*
|
||||||
|
* This program is distributed in the hope that it will be useful, but
|
||||||
|
* WITHOUT ANY WARRANTY; without even the implied warranty of
|
||||||
|
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
|
||||||
|
* General Public License for more details.
|
||||||
|
*
|
||||||
|
* You should have received a copy of the GNU General Public License along
|
||||||
|
* with this program; if not, write to the Free Software Foundation, Inc.,
|
||||||
|
* 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA.
|
||||||
|
*
|
||||||
|
**********************************************************************
|
||||||
|
|
||||||
|
=======================================================================
|
||||||
|
|
||||||
|
Functions to asynchronously convert name to IP address
|
||||||
|
|
||||||
|
*/
|
||||||
|
|
||||||
|
#include "config.h"
|
||||||
|
#include "sysincl.h"
|
||||||
|
|
||||||
|
#include "nameserv_async.h"
|
||||||
|
#include "logging.h"
|
||||||
|
#include "memory.h"
|
||||||
|
#include "privops.h"
|
||||||
|
#include "sched.h"
|
||||||
|
#include "util.h"
|
||||||
|
|
||||||
|
#ifdef USE_PTHREAD_ASYNCDNS
|
||||||
|
#include <pthread.h>
|
||||||
|
|
||||||
|
/* ================================================== */
|
||||||
|
|
||||||
|
struct DNS_Async_Instance {
|
||||||
|
const char *name;
|
||||||
|
DNS_Status status;
|
||||||
|
IPAddr addresses[DNS_MAX_ADDRESSES];
|
||||||
|
DNS_NameResolveHandler handler;
|
||||||
|
void *arg;
|
||||||
|
|
||||||
|
pthread_t thread;
|
||||||
|
int pipe[2];
|
||||||
|
};
|
||||||
|
|
||||||
|
static pthread_mutex_t privops_lock = PTHREAD_MUTEX_INITIALIZER;
|
||||||
|
|
||||||
|
/* ================================================== */
|
||||||
|
|
||||||
|
static void *
|
||||||
|
start_resolving(void *anything)
|
||||||
|
{
|
||||||
|
struct DNS_Async_Instance *inst = (struct DNS_Async_Instance *)anything;
|
||||||
|
|
||||||
|
pthread_mutex_lock(&privops_lock);
|
||||||
|
inst->status = PRV_Name2IPAddress(inst->name, inst->addresses, DNS_MAX_ADDRESSES);
|
||||||
|
pthread_mutex_unlock(&privops_lock);
|
||||||
|
|
||||||
|
/* Notify the main thread that the result is ready */
|
||||||
|
if (write(inst->pipe[1], "", 1) < 0)
|
||||||
|
;
|
||||||
|
|
||||||
|
return NULL;
|
||||||
|
}
|
||||||
|
|
||||||
|
/* ================================================== */
|
||||||
|
|
||||||
|
static void
|
||||||
|
end_resolving(int fd, int event, void *anything)
|
||||||
|
{
|
||||||
|
struct DNS_Async_Instance *inst = (struct DNS_Async_Instance *)anything;
|
||||||
|
int i;
|
||||||
|
|
||||||
|
if (pthread_join(inst->thread, NULL)) {
|
||||||
|
LOG_FATAL("pthread_join() failed");
|
||||||
|
}
|
||||||
|
|
||||||
|
SCH_RemoveFileHandler(inst->pipe[0]);
|
||||||
|
close(inst->pipe[0]);
|
||||||
|
close(inst->pipe[1]);
|
||||||
|
|
||||||
|
for (i = 0; inst->status == DNS_Success && i < DNS_MAX_ADDRESSES &&
|
||||||
|
inst->addresses[i].family != IPADDR_UNSPEC; i++)
|
||||||
|
;
|
||||||
|
|
||||||
|
(inst->handler)(inst->status, i, inst->addresses, inst->arg);
|
||||||
|
|
||||||
|
Free(inst);
|
||||||
|
}
|
||||||
|
|
||||||
|
/* ================================================== */
|
||||||
|
|
||||||
|
void
|
||||||
|
DNS_Name2IPAddressAsync(const char *name, DNS_NameResolveHandler handler, void *anything)
|
||||||
|
{
|
||||||
|
struct DNS_Async_Instance *inst;
|
||||||
|
|
||||||
|
inst = MallocNew(struct DNS_Async_Instance);
|
||||||
|
inst->name = name;
|
||||||
|
inst->handler = handler;
|
||||||
|
inst->arg = anything;
|
||||||
|
inst->status = DNS_Failure;
|
||||||
|
|
||||||
|
if (pipe(inst->pipe)) {
|
||||||
|
LOG_FATAL("pipe() failed");
|
||||||
|
}
|
||||||
|
|
||||||
|
UTI_FdSetCloexec(inst->pipe[0]);
|
||||||
|
UTI_FdSetCloexec(inst->pipe[1]);
|
||||||
|
|
||||||
|
if (pthread_create(&inst->thread, NULL, start_resolving, inst)) {
|
||||||
|
LOG_FATAL("pthread_create() failed");
|
||||||
|
}
|
||||||
|
|
||||||
|
SCH_AddFileHandler(inst->pipe[0], SCH_FILE_INPUT, end_resolving, inst);
|
||||||
|
}
|
||||||
|
|
||||||
|
/* ================================================== */
|
||||||
|
|
||||||
|
#else
|
||||||
|
#error
|
||||||
|
#endif
|
40
nameserv_async.h
Normal file
40
nameserv_async.h
Normal file
|
@ -0,0 +1,40 @@
|
||||||
|
/*
|
||||||
|
chronyd/chronyc - Programs for keeping computer clocks accurate.
|
||||||
|
|
||||||
|
**********************************************************************
|
||||||
|
* Copyright (C) Miroslav Lichvar 2014
|
||||||
|
*
|
||||||
|
* This program is free software; you can redistribute it and/or modify
|
||||||
|
* it under the terms of version 2 of the GNU General Public License as
|
||||||
|
* published by the Free Software Foundation.
|
||||||
|
*
|
||||||
|
* This program is distributed in the hope that it will be useful, but
|
||||||
|
* WITHOUT ANY WARRANTY; without even the implied warranty of
|
||||||
|
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
|
||||||
|
* General Public License for more details.
|
||||||
|
*
|
||||||
|
* You should have received a copy of the GNU General Public License along
|
||||||
|
* with this program; if not, write to the Free Software Foundation, Inc.,
|
||||||
|
* 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA.
|
||||||
|
*
|
||||||
|
**********************************************************************
|
||||||
|
|
||||||
|
=======================================================================
|
||||||
|
|
||||||
|
Header for asynchronous nameserver functions
|
||||||
|
*/
|
||||||
|
|
||||||
|
|
||||||
|
#ifndef GOT_NAMESERV_ASYNC_H
|
||||||
|
#define GOT_NAMESERV_ASYNC_H
|
||||||
|
|
||||||
|
#include "nameserv.h"
|
||||||
|
|
||||||
|
/* Function type for callback to process the result */
|
||||||
|
typedef void (*DNS_NameResolveHandler)(DNS_Status status, int n_addrs, IPAddr *ip_addrs, void *anything);
|
||||||
|
|
||||||
|
/* Request resolving of a name to IP address. The handler will be
|
||||||
|
called when the result is available. */
|
||||||
|
extern void DNS_Name2IPAddressAsync(const char *name, DNS_NameResolveHandler handler, void *anything);
|
||||||
|
|
||||||
|
#endif
|
200
ntp.h
Normal file
200
ntp.h
Normal file
|
@ -0,0 +1,200 @@
|
||||||
|
/*
|
||||||
|
chronyd/chronyc - Programs for keeping computer clocks accurate.
|
||||||
|
|
||||||
|
**********************************************************************
|
||||||
|
* Copyright (C) Richard P. Curnow 1997-2003
|
||||||
|
*
|
||||||
|
* This program is free software; you can redistribute it and/or modify
|
||||||
|
* it under the terms of version 2 of the GNU General Public License as
|
||||||
|
* published by the Free Software Foundation.
|
||||||
|
*
|
||||||
|
* This program is distributed in the hope that it will be useful, but
|
||||||
|
* WITHOUT ANY WARRANTY; without even the implied warranty of
|
||||||
|
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
|
||||||
|
* General Public License for more details.
|
||||||
|
*
|
||||||
|
* You should have received a copy of the GNU General Public License along
|
||||||
|
* with this program; if not, write to the Free Software Foundation, Inc.,
|
||||||
|
* 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA.
|
||||||
|
*
|
||||||
|
**********************************************************************
|
||||||
|
|
||||||
|
=======================================================================
|
||||||
|
|
||||||
|
Header file containing common NTP bits and pieces
|
||||||
|
*/
|
||||||
|
|
||||||
|
#ifndef GOT_NTP_H
|
||||||
|
#define GOT_NTP_H
|
||||||
|
|
||||||
|
#include "sysincl.h"
|
||||||
|
|
||||||
|
#include "hash.h"
|
||||||
|
|
||||||
|
typedef struct {
|
||||||
|
uint32_t hi;
|
||||||
|
uint32_t lo;
|
||||||
|
} NTP_int64;
|
||||||
|
|
||||||
|
typedef uint32_t NTP_int32;
|
||||||
|
|
||||||
|
/* The UDP port number used by NTP */
|
||||||
|
#define NTP_PORT 123
|
||||||
|
|
||||||
|
/* The NTP protocol version that we support */
|
||||||
|
#define NTP_VERSION 4
|
||||||
|
|
||||||
|
/* Maximum stratum number (infinity) */
|
||||||
|
#define NTP_MAX_STRATUM 16
|
||||||
|
|
||||||
|
/* Invalid stratum number */
|
||||||
|
#define NTP_INVALID_STRATUM 0
|
||||||
|
|
||||||
|
/* The minimum and maximum supported length of MAC */
|
||||||
|
#define NTP_MIN_MAC_LENGTH (4 + 16)
|
||||||
|
#define NTP_MAX_MAC_LENGTH (4 + MAX_HASH_LENGTH)
|
||||||
|
|
||||||
|
/* The minimum valid length of an extension field */
|
||||||
|
#define NTP_MIN_EF_LENGTH 16
|
||||||
|
|
||||||
|
/* The maximum assumed length of all extension fields in an NTP packet,
|
||||||
|
including a MAC (RFC 5905 doesn't specify a limit on length or number of
|
||||||
|
extension fields in one packet) */
|
||||||
|
#define NTP_MAX_EXTENSIONS_LENGTH (1024 + NTP_MAX_MAC_LENGTH)
|
||||||
|
|
||||||
|
/* The maximum length of MAC in NTPv4 packets which allows deterministic
|
||||||
|
parsing of extension fields (RFC 7822) */
|
||||||
|
#define NTP_MAX_V4_MAC_LENGTH (4 + 20)
|
||||||
|
|
||||||
|
/* Type definition for leap bits */
|
||||||
|
typedef enum {
|
||||||
|
LEAP_Normal = 0,
|
||||||
|
LEAP_InsertSecond = 1,
|
||||||
|
LEAP_DeleteSecond = 2,
|
||||||
|
LEAP_Unsynchronised = 3
|
||||||
|
} NTP_Leap;
|
||||||
|
|
||||||
|
typedef enum {
|
||||||
|
MODE_UNDEFINED = 0,
|
||||||
|
MODE_ACTIVE = 1,
|
||||||
|
MODE_PASSIVE = 2,
|
||||||
|
MODE_CLIENT = 3,
|
||||||
|
MODE_SERVER = 4,
|
||||||
|
MODE_BROADCAST = 5
|
||||||
|
} NTP_Mode;
|
||||||
|
|
||||||
|
typedef struct {
|
||||||
|
uint8_t lvm;
|
||||||
|
uint8_t stratum;
|
||||||
|
int8_t poll;
|
||||||
|
int8_t precision;
|
||||||
|
NTP_int32 root_delay;
|
||||||
|
NTP_int32 root_dispersion;
|
||||||
|
NTP_int32 reference_id;
|
||||||
|
NTP_int64 reference_ts;
|
||||||
|
NTP_int64 originate_ts;
|
||||||
|
NTP_int64 receive_ts;
|
||||||
|
NTP_int64 transmit_ts;
|
||||||
|
|
||||||
|
uint8_t extensions[NTP_MAX_EXTENSIONS_LENGTH];
|
||||||
|
} NTP_Packet;
|
||||||
|
|
||||||
|
#define NTP_HEADER_LENGTH (int)offsetof(NTP_Packet, extensions)
|
||||||
|
|
||||||
|
/* Macros to work with the lvm field */
|
||||||
|
#define NTP_LVM_TO_LEAP(lvm) (((lvm) >> 6) & 0x3)
|
||||||
|
#define NTP_LVM_TO_VERSION(lvm) (((lvm) >> 3) & 0x7)
|
||||||
|
#define NTP_LVM_TO_MODE(lvm) ((lvm) & 0x7)
|
||||||
|
#define NTP_LVM(leap, version, mode) \
|
||||||
|
((((leap) << 6) & 0xc0) | (((version) << 3) & 0x38) | ((mode) & 0x07))
|
||||||
|
|
||||||
|
/* Special NTP reference IDs */
|
||||||
|
#define NTP_REFID_UNSYNC 0x0UL
|
||||||
|
#define NTP_REFID_LOCAL 0x7F7F0101UL /* 127.127.1.1 */
|
||||||
|
#define NTP_REFID_SMOOTH 0x7F7F01FFUL /* 127.127.1.255 */
|
||||||
|
|
||||||
|
/* Non-authentication extension fields and corresponding internal flags */
|
||||||
|
|
||||||
|
#define NTP_EF_EXP_MONO_ROOT 0xF323
|
||||||
|
#define NTP_EF_EXP_NET_CORRECTION 0xF324
|
||||||
|
|
||||||
|
#define NTP_EF_FLAG_EXP_MONO_ROOT 0x1
|
||||||
|
#define NTP_EF_FLAG_EXP_NET_CORRECTION 0x2
|
||||||
|
|
||||||
|
/* Pre-NTPv5 experimental extension field */
|
||||||
|
typedef struct {
|
||||||
|
uint32_t magic;
|
||||||
|
NTP_int32 root_delay;
|
||||||
|
NTP_int32 root_dispersion;
|
||||||
|
NTP_int64 mono_receive_ts;
|
||||||
|
uint32_t mono_epoch;
|
||||||
|
} NTP_EFExpMonoRoot;
|
||||||
|
|
||||||
|
#define NTP_EF_EXP_MONO_ROOT_MAGIC 0xF5BEDD9AU
|
||||||
|
|
||||||
|
/* Experimental extension field to provide PTP corrections */
|
||||||
|
typedef struct {
|
||||||
|
uint32_t magic;
|
||||||
|
NTP_int64 correction;
|
||||||
|
uint32_t reserved[3];
|
||||||
|
} NTP_EFExpNetCorrection;
|
||||||
|
|
||||||
|
#define NTP_EF_EXP_NET_CORRECTION_MAGIC 0x07AC2CEBU
|
||||||
|
|
||||||
|
/* Authentication extension fields */
|
||||||
|
|
||||||
|
#define NTP_EF_NTS_UNIQUE_IDENTIFIER 0x0104
|
||||||
|
#define NTP_EF_NTS_COOKIE 0x0204
|
||||||
|
#define NTP_EF_NTS_COOKIE_PLACEHOLDER 0x0304
|
||||||
|
#define NTP_EF_NTS_AUTH_AND_EEF 0x0404
|
||||||
|
|
||||||
|
/* Enumeration for authentication modes of NTP packets */
|
||||||
|
typedef enum {
|
||||||
|
NTP_AUTH_NONE = 0, /* No authentication */
|
||||||
|
NTP_AUTH_SYMMETRIC, /* NTP MAC or CMAC using a symmetric key
|
||||||
|
(RFC 1305, RFC 5905, RFC 8573) */
|
||||||
|
NTP_AUTH_MSSNTP, /* MS-SNTP authenticator field */
|
||||||
|
NTP_AUTH_MSSNTP_EXT, /* MS-SNTP extended authenticator field */
|
||||||
|
NTP_AUTH_NTS, /* Network Time Security (RFC 8915) */
|
||||||
|
} NTP_AuthMode;
|
||||||
|
|
||||||
|
/* Structure describing an NTP packet */
|
||||||
|
typedef struct {
|
||||||
|
int length;
|
||||||
|
int version;
|
||||||
|
NTP_Mode mode;
|
||||||
|
|
||||||
|
int ext_fields;
|
||||||
|
int ext_field_flags;
|
||||||
|
|
||||||
|
struct {
|
||||||
|
NTP_AuthMode mode;
|
||||||
|
struct {
|
||||||
|
int start;
|
||||||
|
int length;
|
||||||
|
uint32_t key_id;
|
||||||
|
} mac;
|
||||||
|
} auth;
|
||||||
|
} NTP_PacketInfo;
|
||||||
|
|
||||||
|
/* Structure used to save NTP measurements. time is the local time at which
|
||||||
|
the sample is to be considered to have been made and offset is the offset at
|
||||||
|
the time (positive indicates that the local clock is slow relative to the
|
||||||
|
source). root_delay/root_dispersion include peer_delay/peer_dispersion. */
|
||||||
|
typedef struct {
|
||||||
|
struct timespec time;
|
||||||
|
double offset;
|
||||||
|
double peer_delay;
|
||||||
|
double peer_dispersion;
|
||||||
|
double root_delay;
|
||||||
|
double root_dispersion;
|
||||||
|
} NTP_Sample;
|
||||||
|
|
||||||
|
/* Possible sources of timestamps */
|
||||||
|
typedef enum {
|
||||||
|
NTP_TS_DAEMON = 0,
|
||||||
|
NTP_TS_KERNEL,
|
||||||
|
NTP_TS_HARDWARE
|
||||||
|
} NTP_Timestamp_Source;
|
||||||
|
|
||||||
|
#endif /* GOT_NTP_H */
|
386
ntp_auth.c
Normal file
386
ntp_auth.c
Normal file
|
@ -0,0 +1,386 @@
|
||||||
|
/*
|
||||||
|
chronyd/chronyc - Programs for keeping computer clocks accurate.
|
||||||
|
|
||||||
|
**********************************************************************
|
||||||
|
* Copyright (C) Miroslav Lichvar 2019-2020
|
||||||
|
*
|
||||||
|
* This program is free software; you can redistribute it and/or modify
|
||||||
|
* it under the terms of version 2 of the GNU General Public License as
|
||||||
|
* published by the Free Software Foundation.
|
||||||
|
*
|
||||||
|
* This program is distributed in the hope that it will be useful, but
|
||||||
|
* WITHOUT ANY WARRANTY; without even the implied warranty of
|
||||||
|
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
|
||||||
|
* General Public License for more details.
|
||||||
|
*
|
||||||
|
* You should have received a copy of the GNU General Public License along
|
||||||
|
* with this program; if not, write to the Free Software Foundation, Inc.,
|
||||||
|
* 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA.
|
||||||
|
*
|
||||||
|
**********************************************************************
|
||||||
|
|
||||||
|
=======================================================================
|
||||||
|
|
||||||
|
NTP authentication
|
||||||
|
*/
|
||||||
|
|
||||||
|
#include "config.h"
|
||||||
|
|
||||||
|
#include "sysincl.h"
|
||||||
|
|
||||||
|
#include "keys.h"
|
||||||
|
#include "logging.h"
|
||||||
|
#include "memory.h"
|
||||||
|
#include "ntp_auth.h"
|
||||||
|
#include "ntp_signd.h"
|
||||||
|
#include "nts_ntp.h"
|
||||||
|
#include "nts_ntp_client.h"
|
||||||
|
#include "nts_ntp_server.h"
|
||||||
|
#include "srcparams.h"
|
||||||
|
#include "util.h"
|
||||||
|
|
||||||
|
/* Structure to hold authentication configuration and state */
|
||||||
|
|
||||||
|
struct NAU_Instance_Record {
|
||||||
|
NTP_AuthMode mode; /* Authentication mode of NTP packets */
|
||||||
|
uint32_t key_id; /* Identifier of a symmetric key */
|
||||||
|
NNC_Instance nts; /* Client NTS state */
|
||||||
|
};
|
||||||
|
|
||||||
|
/* ================================================== */
|
||||||
|
|
||||||
|
static int
|
||||||
|
generate_symmetric_auth(uint32_t key_id, NTP_Packet *packet, NTP_PacketInfo *info)
|
||||||
|
{
|
||||||
|
int auth_len, max_auth_len;
|
||||||
|
|
||||||
|
if (info->length + NTP_MIN_MAC_LENGTH > sizeof (*packet)) {
|
||||||
|
DEBUG_LOG("Packet too long");
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
/* Truncate long MACs in NTPv4 packets to allow deterministic parsing
|
||||||
|
of extension fields (RFC 7822) */
|
||||||
|
max_auth_len = (info->version == 4 ? NTP_MAX_V4_MAC_LENGTH : NTP_MAX_MAC_LENGTH) - 4;
|
||||||
|
max_auth_len = MIN(max_auth_len, sizeof (*packet) - info->length - 4);
|
||||||
|
|
||||||
|
auth_len = KEY_GenerateAuth(key_id, packet, info->length,
|
||||||
|
(unsigned char *)packet + info->length + 4, max_auth_len);
|
||||||
|
if (auth_len < NTP_MIN_MAC_LENGTH - 4) {
|
||||||
|
DEBUG_LOG("Could not generate auth data with key %"PRIu32, key_id);
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
*(uint32_t *)((unsigned char *)packet + info->length) = htonl(key_id);
|
||||||
|
|
||||||
|
info->auth.mac.start = info->length;
|
||||||
|
info->auth.mac.length = 4 + auth_len;
|
||||||
|
info->auth.mac.key_id = key_id;
|
||||||
|
info->length += info->auth.mac.length;
|
||||||
|
|
||||||
|
return 1;
|
||||||
|
}
|
||||||
|
|
||||||
|
/* ================================================== */
|
||||||
|
|
||||||
|
static int
|
||||||
|
check_symmetric_auth(NTP_Packet *packet, NTP_PacketInfo *info)
|
||||||
|
{
|
||||||
|
int trunc_len;
|
||||||
|
|
||||||
|
if (info->auth.mac.length < NTP_MIN_MAC_LENGTH)
|
||||||
|
return 0;
|
||||||
|
|
||||||
|
trunc_len = info->version == 4 && info->auth.mac.length <= NTP_MAX_V4_MAC_LENGTH ?
|
||||||
|
NTP_MAX_V4_MAC_LENGTH : NTP_MAX_MAC_LENGTH;
|
||||||
|
|
||||||
|
if (!KEY_CheckAuth(info->auth.mac.key_id, packet, info->auth.mac.start,
|
||||||
|
(unsigned char *)packet + info->auth.mac.start + 4,
|
||||||
|
info->auth.mac.length - 4, trunc_len - 4))
|
||||||
|
return 0;
|
||||||
|
|
||||||
|
return 1;
|
||||||
|
}
|
||||||
|
|
||||||
|
/* ================================================== */
|
||||||
|
|
||||||
|
static NAU_Instance
|
||||||
|
create_instance(NTP_AuthMode mode)
|
||||||
|
{
|
||||||
|
NAU_Instance instance;
|
||||||
|
|
||||||
|
instance = MallocNew(struct NAU_Instance_Record);
|
||||||
|
instance->mode = mode;
|
||||||
|
instance->key_id = INACTIVE_AUTHKEY;
|
||||||
|
instance->nts = NULL;
|
||||||
|
|
||||||
|
assert(sizeof (instance->key_id) == 4);
|
||||||
|
|
||||||
|
return instance;
|
||||||
|
}
|
||||||
|
|
||||||
|
/* ================================================== */
|
||||||
|
|
||||||
|
NAU_Instance
|
||||||
|
NAU_CreateNoneInstance(void)
|
||||||
|
{
|
||||||
|
return create_instance(NTP_AUTH_NONE);
|
||||||
|
}
|
||||||
|
|
||||||
|
/* ================================================== */
|
||||||
|
|
||||||
|
NAU_Instance
|
||||||
|
NAU_CreateSymmetricInstance(uint32_t key_id)
|
||||||
|
{
|
||||||
|
NAU_Instance instance = create_instance(NTP_AUTH_SYMMETRIC);
|
||||||
|
|
||||||
|
instance->key_id = key_id;
|
||||||
|
|
||||||
|
if (!KEY_KeyKnown(key_id))
|
||||||
|
LOG(LOGS_WARN, "Key %"PRIu32" is %s", key_id, "missing");
|
||||||
|
else if (!KEY_CheckKeyLength(key_id))
|
||||||
|
LOG(LOGS_WARN, "Key %"PRIu32" is %s", key_id, "too short");
|
||||||
|
|
||||||
|
return instance;
|
||||||
|
}
|
||||||
|
|
||||||
|
/* ================================================== */
|
||||||
|
|
||||||
|
NAU_Instance
|
||||||
|
NAU_CreateNtsInstance(IPSockAddr *nts_address, const char *name, uint32_t cert_set,
|
||||||
|
uint16_t ntp_port)
|
||||||
|
{
|
||||||
|
NAU_Instance instance = create_instance(NTP_AUTH_NTS);
|
||||||
|
|
||||||
|
instance->nts = NNC_CreateInstance(nts_address, name, cert_set, ntp_port);
|
||||||
|
|
||||||
|
return instance;
|
||||||
|
}
|
||||||
|
|
||||||
|
/* ================================================== */
|
||||||
|
|
||||||
|
void
|
||||||
|
NAU_DestroyInstance(NAU_Instance instance)
|
||||||
|
{
|
||||||
|
if (instance->mode == NTP_AUTH_NTS)
|
||||||
|
NNC_DestroyInstance(instance->nts);
|
||||||
|
Free(instance);
|
||||||
|
}
|
||||||
|
|
||||||
|
/* ================================================== */
|
||||||
|
|
||||||
|
int
|
||||||
|
NAU_IsAuthEnabled(NAU_Instance instance)
|
||||||
|
{
|
||||||
|
return instance->mode != NTP_AUTH_NONE;
|
||||||
|
}
|
||||||
|
|
||||||
|
/* ================================================== */
|
||||||
|
|
||||||
|
int
|
||||||
|
NAU_GetSuggestedNtpVersion(NAU_Instance instance)
|
||||||
|
{
|
||||||
|
/* If the MAC in NTPv4 packets would be truncated, prefer NTPv3 for
|
||||||
|
compatibility with older chronyd servers */
|
||||||
|
if (instance->mode == NTP_AUTH_SYMMETRIC &&
|
||||||
|
KEY_GetAuthLength(instance->key_id) + sizeof (instance->key_id) > NTP_MAX_V4_MAC_LENGTH)
|
||||||
|
return 3;
|
||||||
|
|
||||||
|
return NTP_VERSION;
|
||||||
|
}
|
||||||
|
|
||||||
|
/* ================================================== */
|
||||||
|
|
||||||
|
int
|
||||||
|
NAU_PrepareRequestAuth(NAU_Instance instance)
|
||||||
|
{
|
||||||
|
switch (instance->mode) {
|
||||||
|
case NTP_AUTH_NTS:
|
||||||
|
if (!NNC_PrepareForAuth(instance->nts))
|
||||||
|
return 0;
|
||||||
|
break;
|
||||||
|
default:
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
|
||||||
|
return 1;
|
||||||
|
}
|
||||||
|
|
||||||
|
/* ================================================== */
|
||||||
|
|
||||||
|
int
|
||||||
|
NAU_GenerateRequestAuth(NAU_Instance instance, NTP_Packet *request, NTP_PacketInfo *info)
|
||||||
|
{
|
||||||
|
switch (instance->mode) {
|
||||||
|
case NTP_AUTH_NONE:
|
||||||
|
break;
|
||||||
|
case NTP_AUTH_SYMMETRIC:
|
||||||
|
if (!generate_symmetric_auth(instance->key_id, request, info))
|
||||||
|
return 0;
|
||||||
|
break;
|
||||||
|
case NTP_AUTH_NTS:
|
||||||
|
if (!NNC_GenerateRequestAuth(instance->nts, request, info))
|
||||||
|
return 0;
|
||||||
|
break;
|
||||||
|
default:
|
||||||
|
assert(0);
|
||||||
|
}
|
||||||
|
|
||||||
|
info->auth.mode = instance->mode;
|
||||||
|
|
||||||
|
return 1;
|
||||||
|
}
|
||||||
|
|
||||||
|
/* ================================================== */
|
||||||
|
|
||||||
|
int
|
||||||
|
NAU_CheckRequestAuth(NTP_Packet *request, NTP_PacketInfo *info, uint32_t *kod)
|
||||||
|
{
|
||||||
|
*kod = 0;
|
||||||
|
|
||||||
|
switch (info->auth.mode) {
|
||||||
|
case NTP_AUTH_NONE:
|
||||||
|
break;
|
||||||
|
case NTP_AUTH_SYMMETRIC:
|
||||||
|
if (!check_symmetric_auth(request, info))
|
||||||
|
return 0;
|
||||||
|
break;
|
||||||
|
case NTP_AUTH_MSSNTP:
|
||||||
|
/* MS-SNTP requests are not authenticated */
|
||||||
|
break;
|
||||||
|
case NTP_AUTH_MSSNTP_EXT:
|
||||||
|
/* Not supported yet */
|
||||||
|
return 0;
|
||||||
|
case NTP_AUTH_NTS:
|
||||||
|
if (!NNS_CheckRequestAuth(request, info, kod))
|
||||||
|
return 0;
|
||||||
|
break;
|
||||||
|
default:
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
return 1;
|
||||||
|
}
|
||||||
|
|
||||||
|
/* ================================================== */
|
||||||
|
|
||||||
|
int
|
||||||
|
NAU_GenerateResponseAuth(NTP_Packet *request, NTP_PacketInfo *request_info,
|
||||||
|
NTP_Packet *response, NTP_PacketInfo *response_info,
|
||||||
|
NTP_Remote_Address *remote_addr, NTP_Local_Address *local_addr,
|
||||||
|
uint32_t kod)
|
||||||
|
{
|
||||||
|
switch (request_info->auth.mode) {
|
||||||
|
case NTP_AUTH_NONE:
|
||||||
|
break;
|
||||||
|
case NTP_AUTH_SYMMETRIC:
|
||||||
|
if (!generate_symmetric_auth(request_info->auth.mac.key_id, response, response_info))
|
||||||
|
return 0;
|
||||||
|
break;
|
||||||
|
case NTP_AUTH_MSSNTP:
|
||||||
|
/* Sign the packet asynchronously by ntp_signd */
|
||||||
|
if (!NSD_SignAndSendPacket(request_info->auth.mac.key_id, response, response_info,
|
||||||
|
remote_addr, local_addr))
|
||||||
|
return 0;
|
||||||
|
/* Don't send the original packet */
|
||||||
|
return 0;
|
||||||
|
case NTP_AUTH_NTS:
|
||||||
|
if (!NNS_GenerateResponseAuth(request, request_info, response, response_info, kod))
|
||||||
|
return 0;
|
||||||
|
break;
|
||||||
|
default:
|
||||||
|
DEBUG_LOG("Could not authenticate response auth_mode=%d", (int)request_info->auth.mode);
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
response_info->auth.mode = request_info->auth.mode;
|
||||||
|
|
||||||
|
return 1;
|
||||||
|
}
|
||||||
|
|
||||||
|
/* ================================================== */
|
||||||
|
|
||||||
|
int
|
||||||
|
NAU_CheckResponseAuth(NAU_Instance instance, NTP_Packet *response, NTP_PacketInfo *info)
|
||||||
|
{
|
||||||
|
/* The authentication must match the expected mode */
|
||||||
|
if (info->auth.mode != instance->mode)
|
||||||
|
return 0;
|
||||||
|
|
||||||
|
switch (info->auth.mode) {
|
||||||
|
case NTP_AUTH_NONE:
|
||||||
|
break;
|
||||||
|
case NTP_AUTH_SYMMETRIC:
|
||||||
|
/* Check if it is authenticated with the specified key */
|
||||||
|
if (info->auth.mac.key_id != instance->key_id)
|
||||||
|
return 0;
|
||||||
|
/* and that the MAC is valid */
|
||||||
|
if (!check_symmetric_auth(response, info))
|
||||||
|
return 0;
|
||||||
|
break;
|
||||||
|
case NTP_AUTH_NTS:
|
||||||
|
if (!NNC_CheckResponseAuth(instance->nts, response, info))
|
||||||
|
return 0;
|
||||||
|
break;
|
||||||
|
default:
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
return 1;
|
||||||
|
}
|
||||||
|
|
||||||
|
/* ================================================== */
|
||||||
|
|
||||||
|
void
|
||||||
|
NAU_ChangeAddress(NAU_Instance instance, IPAddr *address)
|
||||||
|
{
|
||||||
|
switch (instance->mode) {
|
||||||
|
case NTP_AUTH_NONE:
|
||||||
|
case NTP_AUTH_SYMMETRIC:
|
||||||
|
break;
|
||||||
|
case NTP_AUTH_NTS:
|
||||||
|
NNC_ChangeAddress(instance->nts, address);
|
||||||
|
break;
|
||||||
|
default:
|
||||||
|
assert(0);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/* ================================================== */
|
||||||
|
|
||||||
|
void
|
||||||
|
NAU_DumpData(NAU_Instance instance)
|
||||||
|
{
|
||||||
|
switch (instance->mode) {
|
||||||
|
case NTP_AUTH_NTS:
|
||||||
|
NNC_DumpData(instance->nts);
|
||||||
|
break;
|
||||||
|
default:
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/* ================================================== */
|
||||||
|
|
||||||
|
void
|
||||||
|
NAU_GetReport(NAU_Instance instance, RPT_AuthReport *report)
|
||||||
|
{
|
||||||
|
memset(report, 0, sizeof (*report));
|
||||||
|
|
||||||
|
report->mode = instance->mode;
|
||||||
|
report->last_ke_ago = -1;
|
||||||
|
|
||||||
|
switch (instance->mode) {
|
||||||
|
case NTP_AUTH_NONE:
|
||||||
|
break;
|
||||||
|
case NTP_AUTH_SYMMETRIC:
|
||||||
|
report->key_id = instance->key_id;
|
||||||
|
KEY_GetKeyInfo(instance->key_id, &report->key_type, &report->key_length);
|
||||||
|
break;
|
||||||
|
case NTP_AUTH_NTS:
|
||||||
|
NNC_GetReport(instance->nts, report);
|
||||||
|
break;
|
||||||
|
default:
|
||||||
|
assert(0);
|
||||||
|
}
|
||||||
|
}
|
84
ntp_auth.h
Normal file
84
ntp_auth.h
Normal file
|
@ -0,0 +1,84 @@
|
||||||
|
/*
|
||||||
|
chronyd/chronyc - Programs for keeping computer clocks accurate.
|
||||||
|
|
||||||
|
**********************************************************************
|
||||||
|
* Copyright (C) Miroslav Lichvar 2019
|
||||||
|
*
|
||||||
|
* This program is free software; you can redistribute it and/or modify
|
||||||
|
* it under the terms of version 2 of the GNU General Public License as
|
||||||
|
* published by the Free Software Foundation.
|
||||||
|
*
|
||||||
|
* This program is distributed in the hope that it will be useful, but
|
||||||
|
* WITHOUT ANY WARRANTY; without even the implied warranty of
|
||||||
|
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
|
||||||
|
* General Public License for more details.
|
||||||
|
*
|
||||||
|
* You should have received a copy of the GNU General Public License along
|
||||||
|
* with this program; if not, write to the Free Software Foundation, Inc.,
|
||||||
|
* 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA.
|
||||||
|
*
|
||||||
|
**********************************************************************
|
||||||
|
|
||||||
|
=======================================================================
|
||||||
|
|
||||||
|
Header file for NTP authentication
|
||||||
|
*/
|
||||||
|
|
||||||
|
#ifndef GOT_NTP_AUTH_H
|
||||||
|
#define GOT_NTP_AUTH_H
|
||||||
|
|
||||||
|
#include "addressing.h"
|
||||||
|
#include "ntp.h"
|
||||||
|
#include "reports.h"
|
||||||
|
|
||||||
|
typedef struct NAU_Instance_Record *NAU_Instance;
|
||||||
|
|
||||||
|
/* Create an authenticator instance in a specific mode */
|
||||||
|
extern NAU_Instance NAU_CreateNoneInstance(void);
|
||||||
|
extern NAU_Instance NAU_CreateSymmetricInstance(uint32_t key_id);
|
||||||
|
extern NAU_Instance NAU_CreateNtsInstance(IPSockAddr *nts_address, const char *name,
|
||||||
|
uint32_t cert_set, uint16_t ntp_port);
|
||||||
|
|
||||||
|
/* Destroy an instance */
|
||||||
|
extern void NAU_DestroyInstance(NAU_Instance instance);
|
||||||
|
|
||||||
|
/* Check if an instance is not in the None mode */
|
||||||
|
extern int NAU_IsAuthEnabled(NAU_Instance instance);
|
||||||
|
|
||||||
|
/* Get NTP version recommended for better compatibility */
|
||||||
|
extern int NAU_GetSuggestedNtpVersion(NAU_Instance instance);
|
||||||
|
|
||||||
|
/* Perform operations necessary for NAU_GenerateRequestAuth() */
|
||||||
|
extern int NAU_PrepareRequestAuth(NAU_Instance instance);
|
||||||
|
|
||||||
|
/* Extend a request with data required by the authentication mode */
|
||||||
|
extern int NAU_GenerateRequestAuth(NAU_Instance instance, NTP_Packet *request,
|
||||||
|
NTP_PacketInfo *info);
|
||||||
|
|
||||||
|
/* Verify that a request is authentic. If it is not authentic and a non-zero
|
||||||
|
kod code is returned, a KoD response should be sent back. */
|
||||||
|
extern int NAU_CheckRequestAuth(NTP_Packet *request, NTP_PacketInfo *info, uint32_t *kod);
|
||||||
|
|
||||||
|
/* Extend a response with data required by the authentication mode. This
|
||||||
|
function can be called only if the previous call of NAU_CheckRequestAuth()
|
||||||
|
was on the same request. */
|
||||||
|
extern int NAU_GenerateResponseAuth(NTP_Packet *request, NTP_PacketInfo *request_info,
|
||||||
|
NTP_Packet *response, NTP_PacketInfo *response_info,
|
||||||
|
NTP_Remote_Address *remote_addr,
|
||||||
|
NTP_Local_Address *local_addr,
|
||||||
|
uint32_t kod);
|
||||||
|
|
||||||
|
/* Verify that a response is authentic */
|
||||||
|
extern int NAU_CheckResponseAuth(NAU_Instance instance, NTP_Packet *response,
|
||||||
|
NTP_PacketInfo *info);
|
||||||
|
|
||||||
|
/* Change an authentication-specific address (e.g. after replacing a source) */
|
||||||
|
extern void NAU_ChangeAddress(NAU_Instance instance, IPAddr *address);
|
||||||
|
|
||||||
|
/* Save authentication-specific data to speed up the next start */
|
||||||
|
extern void NAU_DumpData(NAU_Instance instance);
|
||||||
|
|
||||||
|
/* Provide a report about the current authentication state */
|
||||||
|
extern void NAU_GetReport(NAU_Instance instance, RPT_AuthReport *report);
|
||||||
|
|
||||||
|
#endif
|
3299
ntp_core.c
Normal file
3299
ntp_core.c
Normal file
File diff suppressed because it is too large
Load diff
142
ntp_core.h
Normal file
142
ntp_core.h
Normal file
|
@ -0,0 +1,142 @@
|
||||||
|
/*
|
||||||
|
chronyd/chronyc - Programs for keeping computer clocks accurate.
|
||||||
|
|
||||||
|
**********************************************************************
|
||||||
|
* Copyright (C) Richard P. Curnow 1997-2002
|
||||||
|
*
|
||||||
|
* This program is free software; you can redistribute it and/or modify
|
||||||
|
* it under the terms of version 2 of the GNU General Public License as
|
||||||
|
* published by the Free Software Foundation.
|
||||||
|
*
|
||||||
|
* This program is distributed in the hope that it will be useful, but
|
||||||
|
* WITHOUT ANY WARRANTY; without even the implied warranty of
|
||||||
|
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
|
||||||
|
* General Public License for more details.
|
||||||
|
*
|
||||||
|
* You should have received a copy of the GNU General Public License along
|
||||||
|
* with this program; if not, write to the Free Software Foundation, Inc.,
|
||||||
|
* 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA.
|
||||||
|
*
|
||||||
|
**********************************************************************
|
||||||
|
|
||||||
|
=======================================================================
|
||||||
|
|
||||||
|
Header file for the main NTP protocol engine
|
||||||
|
*/
|
||||||
|
|
||||||
|
#ifndef GOT_NTP_CORE_H
|
||||||
|
#define GOT_NTP_CORE_H
|
||||||
|
|
||||||
|
#include "sysincl.h"
|
||||||
|
|
||||||
|
#include "addressing.h"
|
||||||
|
#include "srcparams.h"
|
||||||
|
#include "ntp.h"
|
||||||
|
#include "reports.h"
|
||||||
|
|
||||||
|
typedef enum {
|
||||||
|
NTP_SERVER, NTP_PEER
|
||||||
|
} NTP_Source_Type;
|
||||||
|
|
||||||
|
typedef struct {
|
||||||
|
struct timespec ts;
|
||||||
|
double err;
|
||||||
|
NTP_Timestamp_Source source;
|
||||||
|
double rx_duration;
|
||||||
|
double net_correction;
|
||||||
|
} NTP_Local_Timestamp;
|
||||||
|
|
||||||
|
/* This is a private data type used for storing the instance record for
|
||||||
|
each source that we are chiming with */
|
||||||
|
typedef struct NCR_Instance_Record *NCR_Instance;
|
||||||
|
|
||||||
|
/* Init and fini functions */
|
||||||
|
extern void NCR_Initialise(void);
|
||||||
|
extern void NCR_Finalise(void);
|
||||||
|
|
||||||
|
/* Get a new instance for a server or peer */
|
||||||
|
extern NCR_Instance NCR_CreateInstance(NTP_Remote_Address *remote_addr, NTP_Source_Type type,
|
||||||
|
SourceParameters *params, const char *name);
|
||||||
|
|
||||||
|
/* Destroy an instance */
|
||||||
|
extern void NCR_DestroyInstance(NCR_Instance instance);
|
||||||
|
|
||||||
|
/* Start an instance */
|
||||||
|
extern void NCR_StartInstance(NCR_Instance instance);
|
||||||
|
|
||||||
|
/* Reset an instance */
|
||||||
|
extern void NCR_ResetInstance(NCR_Instance inst);
|
||||||
|
|
||||||
|
/* Reset polling interval of an instance */
|
||||||
|
extern void NCR_ResetPoll(NCR_Instance instance);
|
||||||
|
|
||||||
|
/* Change the remote address of an instance */
|
||||||
|
extern void NCR_ChangeRemoteAddress(NCR_Instance inst, NTP_Remote_Address *remote_addr,
|
||||||
|
int ntp_only);
|
||||||
|
|
||||||
|
/* This routine is called when a new packet arrives off the network,
|
||||||
|
and it relates to a source we have an ongoing protocol exchange with */
|
||||||
|
extern int NCR_ProcessRxKnown(NCR_Instance inst, NTP_Local_Address *local_addr,
|
||||||
|
NTP_Local_Timestamp *rx_ts, NTP_Packet *message, int length);
|
||||||
|
|
||||||
|
/* This routine is called when a new packet arrives off the network,
|
||||||
|
and we do not recognize its source */
|
||||||
|
extern void NCR_ProcessRxUnknown(NTP_Remote_Address *remote_addr, NTP_Local_Address *local_addr,
|
||||||
|
NTP_Local_Timestamp *rx_ts, NTP_Packet *message, int length);
|
||||||
|
|
||||||
|
/* This routine is called when a packet is sent to a source we have
|
||||||
|
an ongoing protocol exchange with */
|
||||||
|
extern void NCR_ProcessTxKnown(NCR_Instance inst, NTP_Local_Address *local_addr,
|
||||||
|
NTP_Local_Timestamp *tx_ts, NTP_Packet *message, int length);
|
||||||
|
|
||||||
|
/* This routine is called when a packet is sent to a destination we
|
||||||
|
do not recognize */
|
||||||
|
extern void NCR_ProcessTxUnknown(NTP_Remote_Address *remote_addr, NTP_Local_Address *local_addr,
|
||||||
|
NTP_Local_Timestamp *tx_ts, NTP_Packet *message, int length);
|
||||||
|
|
||||||
|
/* Slew receive and transmit times in instance records */
|
||||||
|
extern void NCR_SlewTimes(NCR_Instance inst, struct timespec *when, double dfreq, double doffset);
|
||||||
|
|
||||||
|
/* Take a particular source online (i.e. start sampling it) or offline
|
||||||
|
(i.e. stop sampling it) */
|
||||||
|
extern void NCR_SetConnectivity(NCR_Instance inst, SRC_Connectivity connectivity);
|
||||||
|
|
||||||
|
extern void NCR_ModifyMinpoll(NCR_Instance inst, int new_minpoll);
|
||||||
|
|
||||||
|
extern void NCR_ModifyMaxpoll(NCR_Instance inst, int new_maxpoll);
|
||||||
|
|
||||||
|
extern void NCR_ModifyMaxdelay(NCR_Instance inst, double new_max_delay);
|
||||||
|
|
||||||
|
extern void NCR_ModifyMaxdelayratio(NCR_Instance inst, double new_max_delay_ratio);
|
||||||
|
|
||||||
|
extern void NCR_ModifyMaxdelaydevratio(NCR_Instance inst, double new_max_delay_dev_ratio);
|
||||||
|
|
||||||
|
extern void NCR_ModifyMinstratum(NCR_Instance inst, int new_min_stratum);
|
||||||
|
|
||||||
|
extern void NCR_ModifyOffset(NCR_Instance inst, double new_offset);
|
||||||
|
|
||||||
|
extern void NCR_ModifyPolltarget(NCR_Instance inst, int new_poll_target);
|
||||||
|
|
||||||
|
extern void NCR_InitiateSampleBurst(NCR_Instance inst, int n_good_samples, int n_total_samples);
|
||||||
|
|
||||||
|
extern void NCR_ReportSource(NCR_Instance inst, RPT_SourceReport *report, struct timespec *now);
|
||||||
|
extern void NCR_GetAuthReport(NCR_Instance inst, RPT_AuthReport *report);
|
||||||
|
extern void NCR_GetNTPReport(NCR_Instance inst, RPT_NTPReport *report);
|
||||||
|
|
||||||
|
extern int NCR_AddAccessRestriction(IPAddr *ip_addr, int subnet_bits, int allow, int all);
|
||||||
|
extern int NCR_CheckAccessRestriction(IPAddr *ip_addr);
|
||||||
|
|
||||||
|
extern void NCR_IncrementActivityCounters(NCR_Instance inst, int *online, int *offline,
|
||||||
|
int *burst_online, int *burst_offline);
|
||||||
|
|
||||||
|
extern NTP_Remote_Address *NCR_GetRemoteAddress(NCR_Instance instance);
|
||||||
|
|
||||||
|
extern uint32_t NCR_GetLocalRefid(NCR_Instance inst);
|
||||||
|
|
||||||
|
extern int NCR_IsSyncPeer(NCR_Instance instance);
|
||||||
|
|
||||||
|
extern void NCR_DumpAuthData(NCR_Instance inst);
|
||||||
|
|
||||||
|
extern void NCR_AddBroadcastDestination(NTP_Remote_Address *addr, int interval);
|
||||||
|
|
||||||
|
#endif /* GOT_NTP_CORE_H */
|
192
ntp_ext.c
Normal file
192
ntp_ext.c
Normal file
|
@ -0,0 +1,192 @@
|
||||||
|
/*
|
||||||
|
chronyd/chronyc - Programs for keeping computer clocks accurate.
|
||||||
|
|
||||||
|
**********************************************************************
|
||||||
|
* Copyright (C) Miroslav Lichvar 2019-2020
|
||||||
|
*
|
||||||
|
* This program is free software; you can redistribute it and/or modify
|
||||||
|
* it under the terms of version 2 of the GNU General Public License as
|
||||||
|
* published by the Free Software Foundation.
|
||||||
|
*
|
||||||
|
* This program is distributed in the hope that it will be useful, but
|
||||||
|
* WITHOUT ANY WARRANTY; without even the implied warranty of
|
||||||
|
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
|
||||||
|
* General Public License for more details.
|
||||||
|
*
|
||||||
|
* You should have received a copy of the GNU General Public License along
|
||||||
|
* with this program; if not, write to the Free Software Foundation, Inc.,
|
||||||
|
* 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA.
|
||||||
|
*
|
||||||
|
**********************************************************************
|
||||||
|
|
||||||
|
=======================================================================
|
||||||
|
|
||||||
|
Functions for adding and parsing NTPv4 extension fields
|
||||||
|
*/
|
||||||
|
|
||||||
|
#include "config.h"
|
||||||
|
|
||||||
|
#include "sysincl.h"
|
||||||
|
|
||||||
|
#include "ntp_ext.h"
|
||||||
|
|
||||||
|
struct ExtFieldHeader {
|
||||||
|
uint16_t type;
|
||||||
|
uint16_t length;
|
||||||
|
};
|
||||||
|
|
||||||
|
/* ================================================== */
|
||||||
|
|
||||||
|
static int
|
||||||
|
format_field(unsigned char *buffer, int buffer_length, int start,
|
||||||
|
int type, int body_length, int *length, void **body)
|
||||||
|
{
|
||||||
|
struct ExtFieldHeader *header;
|
||||||
|
|
||||||
|
if (buffer_length < 0 || start < 0 || buffer_length <= start ||
|
||||||
|
buffer_length - start < sizeof (*header) || start % 4 != 0)
|
||||||
|
return 0;
|
||||||
|
|
||||||
|
header = (struct ExtFieldHeader *)(buffer + start);
|
||||||
|
|
||||||
|
if (body_length < 0 || sizeof (*header) + body_length > 0xffff ||
|
||||||
|
start + sizeof (*header) + body_length > buffer_length || body_length % 4 != 0)
|
||||||
|
return 0;
|
||||||
|
|
||||||
|
header->type = htons(type);
|
||||||
|
header->length = htons(sizeof (*header) + body_length);
|
||||||
|
*length = sizeof (*header) + body_length;
|
||||||
|
*body = header + 1;
|
||||||
|
|
||||||
|
return 1;
|
||||||
|
}
|
||||||
|
|
||||||
|
/* ================================================== */
|
||||||
|
|
||||||
|
int
|
||||||
|
NEF_SetField(unsigned char *buffer, int buffer_length, int start,
|
||||||
|
int type, void *body, int body_length, int *length)
|
||||||
|
{
|
||||||
|
void *ef_body;
|
||||||
|
|
||||||
|
if (!format_field(buffer, buffer_length, start, type, body_length, length, &ef_body))
|
||||||
|
return 0;
|
||||||
|
|
||||||
|
memcpy(ef_body, body, body_length);
|
||||||
|
|
||||||
|
return 1;
|
||||||
|
}
|
||||||
|
|
||||||
|
/* ================================================== */
|
||||||
|
|
||||||
|
int
|
||||||
|
NEF_AddBlankField(NTP_Packet *packet, NTP_PacketInfo *info, int type, int body_length, void **body)
|
||||||
|
{
|
||||||
|
int ef_length, length = info->length;
|
||||||
|
|
||||||
|
if (length < NTP_HEADER_LENGTH || length >= sizeof (*packet) || length % 4 != 0)
|
||||||
|
return 0;
|
||||||
|
|
||||||
|
/* Only NTPv4 packets can have extension fields */
|
||||||
|
if (info->version != 4)
|
||||||
|
return 0;
|
||||||
|
|
||||||
|
if (!format_field((unsigned char *)packet, sizeof (*packet), length,
|
||||||
|
type, body_length, &ef_length, body))
|
||||||
|
return 0;
|
||||||
|
|
||||||
|
if (ef_length < NTP_MIN_EF_LENGTH)
|
||||||
|
return 0;
|
||||||
|
|
||||||
|
info->length += ef_length;
|
||||||
|
info->ext_fields++;
|
||||||
|
|
||||||
|
return 1;
|
||||||
|
}
|
||||||
|
|
||||||
|
/* ================================================== */
|
||||||
|
|
||||||
|
int
|
||||||
|
NEF_AddField(NTP_Packet *packet, NTP_PacketInfo *info,
|
||||||
|
int type, void *body, int body_length)
|
||||||
|
{
|
||||||
|
void *ef_body;
|
||||||
|
|
||||||
|
if (!NEF_AddBlankField(packet, info, type, body_length, &ef_body))
|
||||||
|
return 0;
|
||||||
|
|
||||||
|
memcpy(ef_body, body, body_length);
|
||||||
|
|
||||||
|
return 1;
|
||||||
|
}
|
||||||
|
|
||||||
|
/* ================================================== */
|
||||||
|
|
||||||
|
int
|
||||||
|
NEF_ParseSingleField(unsigned char *buffer, int buffer_length, int start,
|
||||||
|
int *length, int *type, void **body, int *body_length)
|
||||||
|
{
|
||||||
|
struct ExtFieldHeader *header;
|
||||||
|
int ef_length;
|
||||||
|
|
||||||
|
if (buffer_length < 0 || start < 0 || buffer_length <= start ||
|
||||||
|
buffer_length - start < sizeof (*header))
|
||||||
|
return 0;
|
||||||
|
|
||||||
|
header = (struct ExtFieldHeader *)(buffer + start);
|
||||||
|
|
||||||
|
assert(sizeof (*header) == 4);
|
||||||
|
|
||||||
|
ef_length = ntohs(header->length);
|
||||||
|
|
||||||
|
if (ef_length < (int)(sizeof (*header)) || start + ef_length > buffer_length ||
|
||||||
|
ef_length % 4 != 0)
|
||||||
|
return 0;
|
||||||
|
|
||||||
|
if (length)
|
||||||
|
*length = ef_length;
|
||||||
|
if (type)
|
||||||
|
*type = ntohs(header->type);
|
||||||
|
if (body)
|
||||||
|
*body = header + 1;
|
||||||
|
if (body_length)
|
||||||
|
*body_length = ef_length - sizeof (*header);
|
||||||
|
|
||||||
|
return 1;
|
||||||
|
}
|
||||||
|
|
||||||
|
/* ================================================== */
|
||||||
|
|
||||||
|
int
|
||||||
|
NEF_ParseField(NTP_Packet *packet, int packet_length, int start,
|
||||||
|
int *length, int *type, void **body, int *body_length)
|
||||||
|
{
|
||||||
|
int ef_length;
|
||||||
|
|
||||||
|
if (packet_length <= NTP_HEADER_LENGTH || packet_length > sizeof (*packet) ||
|
||||||
|
packet_length <= start || packet_length % 4 != 0 ||
|
||||||
|
start < NTP_HEADER_LENGTH || start % 4 != 0)
|
||||||
|
return 0;
|
||||||
|
|
||||||
|
/* Only NTPv4 packets have extension fields */
|
||||||
|
if (NTP_LVM_TO_VERSION(packet->lvm) != 4)
|
||||||
|
return 0;
|
||||||
|
|
||||||
|
/* Check if the remaining data is a MAC. RFC 7822 specifies the maximum
|
||||||
|
length of a MAC in NTPv4 packets in order to enable deterministic
|
||||||
|
parsing. */
|
||||||
|
if (packet_length - start <= NTP_MAX_V4_MAC_LENGTH)
|
||||||
|
return 0;
|
||||||
|
|
||||||
|
if (!NEF_ParseSingleField((unsigned char *)packet, packet_length, start,
|
||||||
|
&ef_length, type, body, body_length))
|
||||||
|
return 0;
|
||||||
|
|
||||||
|
if (ef_length < NTP_MIN_EF_LENGTH)
|
||||||
|
return 0;
|
||||||
|
|
||||||
|
if (length)
|
||||||
|
*length = ef_length;
|
||||||
|
|
||||||
|
return 1;
|
||||||
|
}
|
43
ntp_ext.h
Normal file
43
ntp_ext.h
Normal file
|
@ -0,0 +1,43 @@
|
||||||
|
/*
|
||||||
|
chronyd/chronyc - Programs for keeping computer clocks accurate.
|
||||||
|
|
||||||
|
**********************************************************************
|
||||||
|
* Copyright (C) Miroslav Lichvar 2019
|
||||||
|
*
|
||||||
|
* This program is free software; you can redistribute it and/or modify
|
||||||
|
* it under the terms of version 2 of the GNU General Public License as
|
||||||
|
* published by the Free Software Foundation.
|
||||||
|
*
|
||||||
|
* This program is distributed in the hope that it will be useful, but
|
||||||
|
* WITHOUT ANY WARRANTY; without even the implied warranty of
|
||||||
|
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
|
||||||
|
* General Public License for more details.
|
||||||
|
*
|
||||||
|
* You should have received a copy of the GNU General Public License along
|
||||||
|
* with this program; if not, write to the Free Software Foundation, Inc.,
|
||||||
|
* 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA.
|
||||||
|
*
|
||||||
|
**********************************************************************
|
||||||
|
|
||||||
|
=======================================================================
|
||||||
|
|
||||||
|
Header file for NTP extension fields
|
||||||
|
*/
|
||||||
|
|
||||||
|
#ifndef GOT_NTP_EXT_H
|
||||||
|
#define GOT_NTP_EXT_H
|
||||||
|
|
||||||
|
#include "ntp.h"
|
||||||
|
|
||||||
|
extern int NEF_SetField(unsigned char *buffer, int buffer_length, int start,
|
||||||
|
int type, void *body, int body_length, int *length);
|
||||||
|
extern int NEF_AddBlankField(NTP_Packet *packet, NTP_PacketInfo *info, int type,
|
||||||
|
int body_length, void **body);
|
||||||
|
extern int NEF_AddField(NTP_Packet *packet, NTP_PacketInfo *info,
|
||||||
|
int type, void *body, int body_length);
|
||||||
|
extern int NEF_ParseSingleField(unsigned char *buffer, int buffer_length, int start,
|
||||||
|
int *length, int *type, void **body, int *body_length);
|
||||||
|
extern int NEF_ParseField(NTP_Packet *packet, int packet_length, int start,
|
||||||
|
int *length, int *type, void **body, int *body_length);
|
||||||
|
|
||||||
|
#endif
|
636
ntp_io.c
Normal file
636
ntp_io.c
Normal file
|
@ -0,0 +1,636 @@
|
||||||
|
/*
|
||||||
|
chronyd/chronyc - Programs for keeping computer clocks accurate.
|
||||||
|
|
||||||
|
**********************************************************************
|
||||||
|
* Copyright (C) Richard P. Curnow 1997-2003
|
||||||
|
* Copyright (C) Timo Teras 2009
|
||||||
|
* Copyright (C) Miroslav Lichvar 2009, 2013-2016, 2018-2021
|
||||||
|
*
|
||||||
|
* This program is free software; you can redistribute it and/or modify
|
||||||
|
* it under the terms of version 2 of the GNU General Public License as
|
||||||
|
* published by the Free Software Foundation.
|
||||||
|
*
|
||||||
|
* This program is distributed in the hope that it will be useful, but
|
||||||
|
* WITHOUT ANY WARRANTY; without even the implied warranty of
|
||||||
|
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
|
||||||
|
* General Public License for more details.
|
||||||
|
*
|
||||||
|
* You should have received a copy of the GNU General Public License along
|
||||||
|
* with this program; if not, write to the Free Software Foundation, Inc.,
|
||||||
|
* 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA.
|
||||||
|
*
|
||||||
|
**********************************************************************
|
||||||
|
|
||||||
|
=======================================================================
|
||||||
|
|
||||||
|
This file deals with the IO aspects of reading and writing NTP packets
|
||||||
|
*/
|
||||||
|
|
||||||
|
#include "config.h"
|
||||||
|
|
||||||
|
#include "sysincl.h"
|
||||||
|
|
||||||
|
#include "memory.h"
|
||||||
|
#include "ntp_io.h"
|
||||||
|
#include "ntp_core.h"
|
||||||
|
#include "ntp_sources.h"
|
||||||
|
#include "ptp.h"
|
||||||
|
#include "sched.h"
|
||||||
|
#include "socket.h"
|
||||||
|
#include "local.h"
|
||||||
|
#include "logging.h"
|
||||||
|
#include "conf.h"
|
||||||
|
#include "privops.h"
|
||||||
|
#include "util.h"
|
||||||
|
|
||||||
|
#ifdef HAVE_LINUX_TIMESTAMPING
|
||||||
|
#include "ntp_io_linux.h"
|
||||||
|
#endif
|
||||||
|
|
||||||
|
#define INVALID_SOCK_FD -1
|
||||||
|
|
||||||
|
/* The server/peer and client sockets for IPv4 and IPv6 */
|
||||||
|
static int server_sock_fd4;
|
||||||
|
static int server_sock_fd6;
|
||||||
|
static int client_sock_fd4;
|
||||||
|
static int client_sock_fd6;
|
||||||
|
|
||||||
|
/* Reference counters for server sockets to keep them open only when needed */
|
||||||
|
static int server_sock_ref4;
|
||||||
|
static int server_sock_ref6;
|
||||||
|
|
||||||
|
/* Flag indicating we create a new connected client socket for each
|
||||||
|
server instead of sharing client_sock_fd4 and client_sock_fd6 */
|
||||||
|
static int separate_client_sockets;
|
||||||
|
|
||||||
|
/* Flag indicating the server sockets are not created dynamically when needed,
|
||||||
|
either to have a socket for client requests when separate client sockets
|
||||||
|
are disabled and client port is equal to server port, or the server port is
|
||||||
|
disabled */
|
||||||
|
static int permanent_server_sockets;
|
||||||
|
|
||||||
|
/* Flag indicating the server IPv4 socket is bound to an address */
|
||||||
|
static int bound_server_sock_fd4;
|
||||||
|
|
||||||
|
/* PTP event port, or 0 if disabled */
|
||||||
|
static int ptp_port;
|
||||||
|
|
||||||
|
/* Shared server/client sockets for NTP-over-PTP */
|
||||||
|
static int ptp_sock_fd4;
|
||||||
|
static int ptp_sock_fd6;
|
||||||
|
|
||||||
|
/* Buffer for transmitted NTP-over-PTP messages */
|
||||||
|
static PTP_NtpMessage *ptp_message;
|
||||||
|
|
||||||
|
/* Flag indicating that we have been initialised */
|
||||||
|
static int initialised=0;
|
||||||
|
|
||||||
|
/* ================================================== */
|
||||||
|
|
||||||
|
/* Forward prototypes */
|
||||||
|
static void read_from_socket(int sock_fd, int event, void *anything);
|
||||||
|
|
||||||
|
/* ================================================== */
|
||||||
|
|
||||||
|
static int
|
||||||
|
open_socket(int family, int local_port, int client_only, IPSockAddr *remote_addr)
|
||||||
|
{
|
||||||
|
int sock_fd, sock_flags, dscp, events = SCH_FILE_INPUT;
|
||||||
|
IPSockAddr local_addr;
|
||||||
|
char *iface;
|
||||||
|
|
||||||
|
if (!SCK_IsIpFamilyEnabled(family))
|
||||||
|
return INVALID_SOCK_FD;
|
||||||
|
|
||||||
|
if (!client_only) {
|
||||||
|
CNF_GetBindAddress(family, &local_addr.ip_addr);
|
||||||
|
iface = CNF_GetBindNtpInterface();
|
||||||
|
} else {
|
||||||
|
CNF_GetBindAcquisitionAddress(family, &local_addr.ip_addr);
|
||||||
|
iface = CNF_GetBindAcquisitionInterface();
|
||||||
|
}
|
||||||
|
|
||||||
|
local_addr.port = local_port;
|
||||||
|
|
||||||
|
sock_flags = SCK_FLAG_RX_DEST_ADDR | SCK_FLAG_PRIV_BIND;
|
||||||
|
if (!client_only)
|
||||||
|
sock_flags |= SCK_FLAG_BROADCAST;
|
||||||
|
|
||||||
|
sock_fd = SCK_OpenUdpSocket(remote_addr, &local_addr, iface, sock_flags);
|
||||||
|
if (sock_fd < 0) {
|
||||||
|
if (!client_only)
|
||||||
|
LOG(LOGS_ERR, "Could not open NTP socket on %s", UTI_IPSockAddrToString(&local_addr));
|
||||||
|
return INVALID_SOCK_FD;
|
||||||
|
}
|
||||||
|
|
||||||
|
dscp = CNF_GetNtpDscp();
|
||||||
|
if (dscp > 0 && dscp < 64) {
|
||||||
|
#ifdef IP_TOS
|
||||||
|
if (family == IPADDR_INET4)
|
||||||
|
if (!SCK_SetIntOption(sock_fd, IPPROTO_IP, IP_TOS, dscp << 2))
|
||||||
|
;
|
||||||
|
#endif
|
||||||
|
#if defined(FEAT_IPV6) && defined(IPV6_TCLASS)
|
||||||
|
if (family == IPADDR_INET6)
|
||||||
|
if (!SCK_SetIntOption(sock_fd, IPPROTO_IPV6, IPV6_TCLASS, dscp << 2))
|
||||||
|
;
|
||||||
|
#endif
|
||||||
|
}
|
||||||
|
|
||||||
|
if (!client_only && family == IPADDR_INET4 && local_addr.port > 0)
|
||||||
|
bound_server_sock_fd4 = local_addr.ip_addr.addr.in4 != INADDR_ANY;
|
||||||
|
|
||||||
|
/* Enable kernel/HW timestamping of packets */
|
||||||
|
#ifdef HAVE_LINUX_TIMESTAMPING
|
||||||
|
if (!NIO_Linux_SetTimestampSocketOptions(sock_fd, client_only, &events))
|
||||||
|
#endif
|
||||||
|
if (!SCK_EnableKernelRxTimestamping(sock_fd))
|
||||||
|
;
|
||||||
|
|
||||||
|
/* Register handler for read and possibly exception events on the socket */
|
||||||
|
SCH_AddFileHandler(sock_fd, events, read_from_socket, NULL);
|
||||||
|
|
||||||
|
return sock_fd;
|
||||||
|
}
|
||||||
|
|
||||||
|
/* ================================================== */
|
||||||
|
|
||||||
|
static int
|
||||||
|
open_separate_client_socket(IPSockAddr *remote_addr)
|
||||||
|
{
|
||||||
|
return open_socket(remote_addr->ip_addr.family, 0, 1, remote_addr);
|
||||||
|
}
|
||||||
|
|
||||||
|
/* ================================================== */
|
||||||
|
|
||||||
|
static void
|
||||||
|
close_socket(int sock_fd)
|
||||||
|
{
|
||||||
|
if (sock_fd == INVALID_SOCK_FD)
|
||||||
|
return;
|
||||||
|
|
||||||
|
SCH_RemoveFileHandler(sock_fd);
|
||||||
|
SCK_CloseSocket(sock_fd);
|
||||||
|
}
|
||||||
|
|
||||||
|
/* ================================================== */
|
||||||
|
|
||||||
|
void
|
||||||
|
NIO_Initialise(void)
|
||||||
|
{
|
||||||
|
int server_port, client_port;
|
||||||
|
|
||||||
|
assert(!initialised);
|
||||||
|
initialised = 1;
|
||||||
|
|
||||||
|
#ifdef PRIVOPS_BINDSOCKET
|
||||||
|
SCK_SetPrivBind(PRV_BindSocket);
|
||||||
|
#endif
|
||||||
|
|
||||||
|
#ifdef HAVE_LINUX_TIMESTAMPING
|
||||||
|
NIO_Linux_Initialise();
|
||||||
|
#else
|
||||||
|
if (1) {
|
||||||
|
CNF_HwTsInterface *conf_iface;
|
||||||
|
if (CNF_GetHwTsInterface(0, &conf_iface))
|
||||||
|
LOG_FATAL("HW timestamping not supported");
|
||||||
|
}
|
||||||
|
#endif
|
||||||
|
|
||||||
|
server_port = CNF_GetNTPPort();
|
||||||
|
client_port = CNF_GetAcquisitionPort();
|
||||||
|
|
||||||
|
/* Use separate connected sockets if client port is negative */
|
||||||
|
separate_client_sockets = client_port < 0;
|
||||||
|
if (client_port < 0)
|
||||||
|
client_port = 0;
|
||||||
|
|
||||||
|
permanent_server_sockets = !server_port || (!separate_client_sockets &&
|
||||||
|
client_port == server_port);
|
||||||
|
|
||||||
|
server_sock_fd4 = INVALID_SOCK_FD;
|
||||||
|
server_sock_fd6 = INVALID_SOCK_FD;
|
||||||
|
client_sock_fd4 = INVALID_SOCK_FD;
|
||||||
|
client_sock_fd6 = INVALID_SOCK_FD;
|
||||||
|
server_sock_ref4 = 0;
|
||||||
|
server_sock_ref6 = 0;
|
||||||
|
|
||||||
|
if (permanent_server_sockets && server_port) {
|
||||||
|
server_sock_fd4 = open_socket(IPADDR_INET4, server_port, 0, NULL);
|
||||||
|
server_sock_fd6 = open_socket(IPADDR_INET6, server_port, 0, NULL);
|
||||||
|
}
|
||||||
|
|
||||||
|
if (!separate_client_sockets) {
|
||||||
|
if (client_port != server_port || !server_port) {
|
||||||
|
client_sock_fd4 = open_socket(IPADDR_INET4, client_port, 1, NULL);
|
||||||
|
client_sock_fd6 = open_socket(IPADDR_INET6, client_port, 1, NULL);
|
||||||
|
} else {
|
||||||
|
client_sock_fd4 = server_sock_fd4;
|
||||||
|
client_sock_fd6 = server_sock_fd6;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
if ((server_port && permanent_server_sockets &&
|
||||||
|
server_sock_fd4 == INVALID_SOCK_FD && server_sock_fd6 == INVALID_SOCK_FD) ||
|
||||||
|
(!separate_client_sockets &&
|
||||||
|
client_sock_fd4 == INVALID_SOCK_FD && client_sock_fd6 == INVALID_SOCK_FD)) {
|
||||||
|
LOG_FATAL("Could not open NTP sockets");
|
||||||
|
}
|
||||||
|
|
||||||
|
ptp_port = CNF_GetPtpPort();
|
||||||
|
ptp_sock_fd4 = INVALID_SOCK_FD;
|
||||||
|
ptp_sock_fd6 = INVALID_SOCK_FD;
|
||||||
|
ptp_message = NULL;
|
||||||
|
|
||||||
|
if (ptp_port > 0) {
|
||||||
|
ptp_sock_fd4 = open_socket(IPADDR_INET4, ptp_port, 0, NULL);
|
||||||
|
ptp_sock_fd6 = open_socket(IPADDR_INET6, ptp_port, 0, NULL);
|
||||||
|
ptp_message = MallocNew(PTP_NtpMessage);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/* ================================================== */
|
||||||
|
|
||||||
|
void
|
||||||
|
NIO_Finalise(void)
|
||||||
|
{
|
||||||
|
if (server_sock_fd4 != client_sock_fd4)
|
||||||
|
close_socket(client_sock_fd4);
|
||||||
|
close_socket(server_sock_fd4);
|
||||||
|
server_sock_fd4 = client_sock_fd4 = INVALID_SOCK_FD;
|
||||||
|
|
||||||
|
if (server_sock_fd6 != client_sock_fd6)
|
||||||
|
close_socket(client_sock_fd6);
|
||||||
|
close_socket(server_sock_fd6);
|
||||||
|
server_sock_fd6 = client_sock_fd6 = INVALID_SOCK_FD;
|
||||||
|
|
||||||
|
close_socket(ptp_sock_fd4);
|
||||||
|
close_socket(ptp_sock_fd6);
|
||||||
|
ptp_sock_fd4 = ptp_sock_fd6 = INVALID_SOCK_FD;
|
||||||
|
Free(ptp_message);
|
||||||
|
|
||||||
|
#ifdef HAVE_LINUX_TIMESTAMPING
|
||||||
|
NIO_Linux_Finalise();
|
||||||
|
#endif
|
||||||
|
|
||||||
|
initialised = 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
/* ================================================== */
|
||||||
|
|
||||||
|
int
|
||||||
|
NIO_IsHwTsEnabled(void)
|
||||||
|
{
|
||||||
|
#ifdef HAVE_LINUX_TIMESTAMPING
|
||||||
|
return NIO_Linux_IsHwTsEnabled();
|
||||||
|
#else
|
||||||
|
return 0;
|
||||||
|
#endif
|
||||||
|
}
|
||||||
|
|
||||||
|
/* ================================================== */
|
||||||
|
|
||||||
|
int
|
||||||
|
NIO_OpenClientSocket(NTP_Remote_Address *remote_addr)
|
||||||
|
{
|
||||||
|
switch (remote_addr->ip_addr.family) {
|
||||||
|
case IPADDR_INET4:
|
||||||
|
if (ptp_port > 0 && remote_addr->port == ptp_port)
|
||||||
|
return ptp_sock_fd4;
|
||||||
|
if (separate_client_sockets)
|
||||||
|
return open_separate_client_socket(remote_addr);
|
||||||
|
return client_sock_fd4;
|
||||||
|
case IPADDR_INET6:
|
||||||
|
if (ptp_port > 0 && remote_addr->port == ptp_port)
|
||||||
|
return ptp_sock_fd6;
|
||||||
|
if (separate_client_sockets)
|
||||||
|
return open_separate_client_socket(remote_addr);
|
||||||
|
return client_sock_fd6;
|
||||||
|
default:
|
||||||
|
return INVALID_SOCK_FD;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/* ================================================== */
|
||||||
|
|
||||||
|
int
|
||||||
|
NIO_OpenServerSocket(NTP_Remote_Address *remote_addr)
|
||||||
|
{
|
||||||
|
switch (remote_addr->ip_addr.family) {
|
||||||
|
case IPADDR_INET4:
|
||||||
|
if (ptp_port > 0 && remote_addr->port == ptp_port)
|
||||||
|
return ptp_sock_fd4;
|
||||||
|
if (permanent_server_sockets)
|
||||||
|
return server_sock_fd4;
|
||||||
|
if (server_sock_fd4 == INVALID_SOCK_FD)
|
||||||
|
server_sock_fd4 = open_socket(IPADDR_INET4, CNF_GetNTPPort(), 0, NULL);
|
||||||
|
if (server_sock_fd4 != INVALID_SOCK_FD)
|
||||||
|
server_sock_ref4++;
|
||||||
|
return server_sock_fd4;
|
||||||
|
case IPADDR_INET6:
|
||||||
|
if (ptp_port > 0 && remote_addr->port == ptp_port)
|
||||||
|
return ptp_sock_fd6;
|
||||||
|
if (permanent_server_sockets)
|
||||||
|
return server_sock_fd6;
|
||||||
|
if (server_sock_fd6 == INVALID_SOCK_FD)
|
||||||
|
server_sock_fd6 = open_socket(IPADDR_INET6, CNF_GetNTPPort(), 0, NULL);
|
||||||
|
if (server_sock_fd6 != INVALID_SOCK_FD)
|
||||||
|
server_sock_ref6++;
|
||||||
|
return server_sock_fd6;
|
||||||
|
default:
|
||||||
|
return INVALID_SOCK_FD;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/* ================================================== */
|
||||||
|
|
||||||
|
static int
|
||||||
|
is_ptp_socket(int sock_fd)
|
||||||
|
{
|
||||||
|
return ptp_port > 0 && sock_fd != INVALID_SOCK_FD &&
|
||||||
|
(sock_fd == ptp_sock_fd4 || sock_fd == ptp_sock_fd6);
|
||||||
|
}
|
||||||
|
|
||||||
|
/* ================================================== */
|
||||||
|
|
||||||
|
void
|
||||||
|
NIO_CloseClientSocket(int sock_fd)
|
||||||
|
{
|
||||||
|
if (is_ptp_socket(sock_fd))
|
||||||
|
return;
|
||||||
|
|
||||||
|
if (separate_client_sockets)
|
||||||
|
close_socket(sock_fd);
|
||||||
|
}
|
||||||
|
|
||||||
|
/* ================================================== */
|
||||||
|
|
||||||
|
void
|
||||||
|
NIO_CloseServerSocket(int sock_fd)
|
||||||
|
{
|
||||||
|
if (permanent_server_sockets || sock_fd == INVALID_SOCK_FD || is_ptp_socket(sock_fd))
|
||||||
|
return;
|
||||||
|
|
||||||
|
if (sock_fd == server_sock_fd4) {
|
||||||
|
if (--server_sock_ref4 <= 0) {
|
||||||
|
close_socket(server_sock_fd4);
|
||||||
|
server_sock_fd4 = INVALID_SOCK_FD;
|
||||||
|
}
|
||||||
|
} else if (sock_fd == server_sock_fd6) {
|
||||||
|
if (--server_sock_ref6 <= 0) {
|
||||||
|
close_socket(server_sock_fd6);
|
||||||
|
server_sock_fd6 = INVALID_SOCK_FD;
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
assert(0);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/* ================================================== */
|
||||||
|
|
||||||
|
int
|
||||||
|
NIO_IsServerSocket(int sock_fd)
|
||||||
|
{
|
||||||
|
return sock_fd != INVALID_SOCK_FD &&
|
||||||
|
(sock_fd == server_sock_fd4 || sock_fd == server_sock_fd6 || is_ptp_socket(sock_fd));
|
||||||
|
}
|
||||||
|
|
||||||
|
/* ================================================== */
|
||||||
|
|
||||||
|
int
|
||||||
|
NIO_IsServerSocketOpen(void)
|
||||||
|
{
|
||||||
|
return server_sock_fd4 != INVALID_SOCK_FD || server_sock_fd6 != INVALID_SOCK_FD ||
|
||||||
|
ptp_sock_fd4 != INVALID_SOCK_FD || ptp_sock_fd6 != INVALID_SOCK_FD;
|
||||||
|
}
|
||||||
|
|
||||||
|
/* ================================================== */
|
||||||
|
|
||||||
|
int
|
||||||
|
NIO_IsServerConnectable(NTP_Remote_Address *remote_addr)
|
||||||
|
{
|
||||||
|
int sock_fd;
|
||||||
|
|
||||||
|
sock_fd = open_separate_client_socket(remote_addr);
|
||||||
|
if (sock_fd == INVALID_SOCK_FD)
|
||||||
|
return 0;
|
||||||
|
|
||||||
|
close_socket(sock_fd);
|
||||||
|
|
||||||
|
return 1;
|
||||||
|
}
|
||||||
|
|
||||||
|
/* ================================================== */
|
||||||
|
|
||||||
|
static void
|
||||||
|
process_message(SCK_Message *message, int sock_fd, int event)
|
||||||
|
{
|
||||||
|
NTP_Local_Address local_addr;
|
||||||
|
NTP_Local_Timestamp local_ts;
|
||||||
|
struct timespec sched_ts;
|
||||||
|
|
||||||
|
SCH_GetLastEventTime(&local_ts.ts, &local_ts.err, NULL);
|
||||||
|
local_ts.source = NTP_TS_DAEMON;
|
||||||
|
local_ts.rx_duration = 0.0;
|
||||||
|
local_ts.net_correction = 0.0;
|
||||||
|
|
||||||
|
sched_ts = local_ts.ts;
|
||||||
|
|
||||||
|
if (message->addr_type != SCK_ADDR_IP) {
|
||||||
|
DEBUG_LOG("Unexpected address type");
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
local_addr.ip_addr = message->local_addr.ip;
|
||||||
|
local_addr.if_index = message->if_index;;
|
||||||
|
local_addr.sock_fd = sock_fd;
|
||||||
|
|
||||||
|
#ifdef HAVE_LINUX_TIMESTAMPING
|
||||||
|
if (NIO_Linux_ProcessMessage(message, &local_addr, &local_ts, event))
|
||||||
|
return;
|
||||||
|
#else
|
||||||
|
if (!UTI_IsZeroTimespec(&message->timestamp.kernel)) {
|
||||||
|
LCL_CookTime(&message->timestamp.kernel, &local_ts.ts, &local_ts.err);
|
||||||
|
local_ts.source = NTP_TS_KERNEL;
|
||||||
|
}
|
||||||
|
#endif
|
||||||
|
|
||||||
|
if (local_ts.source != NTP_TS_DAEMON)
|
||||||
|
DEBUG_LOG("Updated RX timestamp delay=%.9f tss=%u",
|
||||||
|
UTI_DiffTimespecsToDouble(&sched_ts, &local_ts.ts), local_ts.source);
|
||||||
|
|
||||||
|
if (!NIO_UnwrapMessage(message, sock_fd, &local_ts.net_correction))
|
||||||
|
return;
|
||||||
|
|
||||||
|
/* Just ignore the packet if it's not of a recognized length */
|
||||||
|
if (message->length < NTP_HEADER_LENGTH || message->length > sizeof (NTP_Packet)) {
|
||||||
|
DEBUG_LOG("Unexpected length");
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
NSR_ProcessRx(&message->remote_addr.ip, &local_addr, &local_ts, message->data, message->length);
|
||||||
|
}
|
||||||
|
|
||||||
|
/* ================================================== */
|
||||||
|
|
||||||
|
static void
|
||||||
|
read_from_socket(int sock_fd, int event, void *anything)
|
||||||
|
{
|
||||||
|
SCK_Message *messages;
|
||||||
|
int i, received, flags = 0;
|
||||||
|
|
||||||
|
if (event == SCH_FILE_EXCEPTION) {
|
||||||
|
#ifdef HAVE_LINUX_TIMESTAMPING
|
||||||
|
flags |= SCK_FLAG_MSG_ERRQUEUE;
|
||||||
|
#else
|
||||||
|
assert(0);
|
||||||
|
#endif
|
||||||
|
}
|
||||||
|
|
||||||
|
messages = SCK_ReceiveMessages(sock_fd, flags, &received);
|
||||||
|
if (!messages)
|
||||||
|
return;
|
||||||
|
|
||||||
|
for (i = 0; i < received; i++)
|
||||||
|
process_message(&messages[i], sock_fd, event);
|
||||||
|
}
|
||||||
|
|
||||||
|
/* ================================================== */
|
||||||
|
|
||||||
|
int
|
||||||
|
NIO_UnwrapMessage(SCK_Message *message, int sock_fd, double *net_correction)
|
||||||
|
{
|
||||||
|
double ptp_correction;
|
||||||
|
PTP_NtpMessage *msg;
|
||||||
|
|
||||||
|
if (!is_ptp_socket(sock_fd))
|
||||||
|
return 1;
|
||||||
|
|
||||||
|
if (message->length <= PTP_NTP_PREFIX_LENGTH) {
|
||||||
|
DEBUG_LOG("Unexpected length");
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
msg = message->data;
|
||||||
|
|
||||||
|
if ((msg->header.type != PTP_TYPE_DELAY_REQ && msg->header.type != PTP_TYPE_SYNC) ||
|
||||||
|
(msg->header.version != PTP_VERSION_2 &&
|
||||||
|
(msg->header.version != PTP_VERSION_2_1 || msg->header.min_sdoid != 0)) ||
|
||||||
|
ntohs(msg->header.length) != message->length ||
|
||||||
|
msg->header.domain != CNF_GetPtpDomain() ||
|
||||||
|
ntohs(msg->header.flags) != PTP_FLAG_UNICAST ||
|
||||||
|
ntohs(msg->tlv_header.type) != PTP_TLV_NTP ||
|
||||||
|
ntohs(msg->tlv_header.length) != message->length - PTP_NTP_PREFIX_LENGTH) {
|
||||||
|
DEBUG_LOG("Unexpected PTP message");
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
message->data = (char *)message->data + PTP_NTP_PREFIX_LENGTH;
|
||||||
|
message->length -= PTP_NTP_PREFIX_LENGTH;
|
||||||
|
|
||||||
|
ptp_correction = UTI_Integer64NetworkToHost(*(Integer64 *)msg->header.correction) /
|
||||||
|
((1 << 16) * 1.0e9);
|
||||||
|
|
||||||
|
/* Use the correction only if the RX duration is known (i.e. HW timestamp) */
|
||||||
|
if (*net_correction > 0.0)
|
||||||
|
*net_correction += ptp_correction;
|
||||||
|
|
||||||
|
DEBUG_LOG("Unwrapped PTP->NTP len=%d corr=%.9f", message->length, ptp_correction);
|
||||||
|
|
||||||
|
return 1;
|
||||||
|
}
|
||||||
|
|
||||||
|
/* ================================================== */
|
||||||
|
|
||||||
|
static int
|
||||||
|
wrap_message(SCK_Message *message, int sock_fd)
|
||||||
|
{
|
||||||
|
static uint16_t sequence_id = 0;
|
||||||
|
|
||||||
|
assert(PTP_NTP_PREFIX_LENGTH == 48);
|
||||||
|
|
||||||
|
if (!is_ptp_socket(sock_fd))
|
||||||
|
return 1;
|
||||||
|
|
||||||
|
if (!ptp_message)
|
||||||
|
return 0;
|
||||||
|
|
||||||
|
if (message->length < NTP_HEADER_LENGTH ||
|
||||||
|
message->length + PTP_NTP_PREFIX_LENGTH > sizeof (*ptp_message)) {
|
||||||
|
DEBUG_LOG("Unexpected length");
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
memset(ptp_message, 0, PTP_NTP_PREFIX_LENGTH);
|
||||||
|
ptp_message->header.type = PTP_TYPE_DELAY_REQ;
|
||||||
|
ptp_message->header.version = PTP_VERSION_2;
|
||||||
|
ptp_message->header.length = htons(PTP_NTP_PREFIX_LENGTH + message->length);
|
||||||
|
ptp_message->header.domain = CNF_GetPtpDomain();
|
||||||
|
ptp_message->header.flags = htons(PTP_FLAG_UNICAST);
|
||||||
|
ptp_message->header.sequence_id = htons(sequence_id++);
|
||||||
|
ptp_message->tlv_header.type = htons(PTP_TLV_NTP);
|
||||||
|
ptp_message->tlv_header.length = htons(message->length);
|
||||||
|
memcpy((char *)ptp_message + PTP_NTP_PREFIX_LENGTH, message->data, message->length);
|
||||||
|
|
||||||
|
message->data = ptp_message;
|
||||||
|
message->length += PTP_NTP_PREFIX_LENGTH;
|
||||||
|
|
||||||
|
DEBUG_LOG("Wrapped NTP->PTP len=%d", message->length - PTP_NTP_PREFIX_LENGTH);
|
||||||
|
|
||||||
|
return 1;
|
||||||
|
}
|
||||||
|
|
||||||
|
/* ================================================== */
|
||||||
|
/* Send a packet to remote address from local address */
|
||||||
|
|
||||||
|
int
|
||||||
|
NIO_SendPacket(NTP_Packet *packet, NTP_Remote_Address *remote_addr,
|
||||||
|
NTP_Local_Address *local_addr, int length, int process_tx)
|
||||||
|
{
|
||||||
|
SCK_Message message;
|
||||||
|
|
||||||
|
assert(initialised);
|
||||||
|
|
||||||
|
if (local_addr->sock_fd == INVALID_SOCK_FD) {
|
||||||
|
DEBUG_LOG("No socket to send to %s", UTI_IPSockAddrToString(remote_addr));
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
SCK_InitMessage(&message, SCK_ADDR_IP);
|
||||||
|
|
||||||
|
message.data = packet;
|
||||||
|
message.length = length;
|
||||||
|
|
||||||
|
if (!wrap_message(&message, local_addr->sock_fd))
|
||||||
|
return 0;
|
||||||
|
|
||||||
|
/* Specify remote address if the socket is not connected */
|
||||||
|
if (NIO_IsServerSocket(local_addr->sock_fd) || !separate_client_sockets) {
|
||||||
|
message.remote_addr.ip.ip_addr = remote_addr->ip_addr;
|
||||||
|
message.remote_addr.ip.port = remote_addr->port;
|
||||||
|
}
|
||||||
|
|
||||||
|
message.local_addr.ip = local_addr->ip_addr;
|
||||||
|
|
||||||
|
/* Don't require responses to non-link-local addresses to use the same
|
||||||
|
interface */
|
||||||
|
message.if_index = SCK_IsLinkLocalIPAddress(&message.remote_addr.ip.ip_addr) ?
|
||||||
|
local_addr->if_index : INVALID_IF_INDEX;
|
||||||
|
|
||||||
|
#if !defined(HAVE_IN_PKTINFO) && defined(IP_SENDSRCADDR)
|
||||||
|
/* On FreeBSD a local IPv4 address cannot be specified on bound socket */
|
||||||
|
if (message.local_addr.ip.family == IPADDR_INET4 &&
|
||||||
|
(bound_server_sock_fd4 || !NIO_IsServerSocket(local_addr->sock_fd)))
|
||||||
|
message.local_addr.ip.family = IPADDR_UNSPEC;
|
||||||
|
#endif
|
||||||
|
|
||||||
|
#ifdef HAVE_LINUX_TIMESTAMPING
|
||||||
|
if (process_tx)
|
||||||
|
NIO_Linux_RequestTxTimestamp(&message, local_addr->sock_fd);
|
||||||
|
#endif
|
||||||
|
|
||||||
|
if (!SCK_SendMessage(local_addr->sock_fd, &message, 0))
|
||||||
|
return 0;
|
||||||
|
|
||||||
|
return 1;
|
||||||
|
}
|
73
ntp_io.h
Normal file
73
ntp_io.h
Normal file
|
@ -0,0 +1,73 @@
|
||||||
|
/*
|
||||||
|
chronyd/chronyc - Programs for keeping computer clocks accurate.
|
||||||
|
|
||||||
|
**********************************************************************
|
||||||
|
* Copyright (C) Richard P. Curnow 1997-2002
|
||||||
|
* Copyright (C) Miroslav Lichvar 2014
|
||||||
|
*
|
||||||
|
* This program is free software; you can redistribute it and/or modify
|
||||||
|
* it under the terms of version 2 of the GNU General Public License as
|
||||||
|
* published by the Free Software Foundation.
|
||||||
|
*
|
||||||
|
* This program is distributed in the hope that it will be useful, but
|
||||||
|
* WITHOUT ANY WARRANTY; without even the implied warranty of
|
||||||
|
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
|
||||||
|
* General Public License for more details.
|
||||||
|
*
|
||||||
|
* You should have received a copy of the GNU General Public License along
|
||||||
|
* with this program; if not, write to the Free Software Foundation, Inc.,
|
||||||
|
* 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA.
|
||||||
|
*
|
||||||
|
**********************************************************************
|
||||||
|
|
||||||
|
=======================================================================
|
||||||
|
|
||||||
|
This is the header file for the NTP socket I/O bits.
|
||||||
|
|
||||||
|
*/
|
||||||
|
|
||||||
|
#ifndef GOT_NTP_IO_H
|
||||||
|
#define GOT_NTP_IO_H
|
||||||
|
|
||||||
|
#include "ntp.h"
|
||||||
|
#include "addressing.h"
|
||||||
|
#include "socket.h"
|
||||||
|
|
||||||
|
/* Function to initialise the module. */
|
||||||
|
extern void NIO_Initialise(void);
|
||||||
|
|
||||||
|
/* Function to finalise the module */
|
||||||
|
extern void NIO_Finalise(void);
|
||||||
|
|
||||||
|
/* Function to check if HW timestamping is enabled on any interface */
|
||||||
|
extern int NIO_IsHwTsEnabled(void);
|
||||||
|
|
||||||
|
/* Function to obtain a socket for sending client packets */
|
||||||
|
extern int NIO_OpenClientSocket(NTP_Remote_Address *remote_addr);
|
||||||
|
|
||||||
|
/* Function to obtain a socket for sending server/peer packets */
|
||||||
|
extern int NIO_OpenServerSocket(NTP_Remote_Address *remote_addr);
|
||||||
|
|
||||||
|
/* Function to close a socket returned by NIO_OpenClientSocket() */
|
||||||
|
extern void NIO_CloseClientSocket(int sock_fd);
|
||||||
|
|
||||||
|
/* Function to close a socket returned by NIO_OpenServerSocket() */
|
||||||
|
extern void NIO_CloseServerSocket(int sock_fd);
|
||||||
|
|
||||||
|
/* Function to check if socket is a server socket */
|
||||||
|
extern int NIO_IsServerSocket(int sock_fd);
|
||||||
|
|
||||||
|
/* Function to check if a server socket is currently open */
|
||||||
|
extern int NIO_IsServerSocketOpen(void);
|
||||||
|
|
||||||
|
/* Function to check if client packets can be sent to a server */
|
||||||
|
extern int NIO_IsServerConnectable(NTP_Remote_Address *remote_addr);
|
||||||
|
|
||||||
|
/* Function to unwrap an NTP message from non-native transport (e.g. PTP) */
|
||||||
|
extern int NIO_UnwrapMessage(SCK_Message *message, int sock_fd, double *net_correction);
|
||||||
|
|
||||||
|
/* Function to transmit a packet */
|
||||||
|
extern int NIO_SendPacket(NTP_Packet *packet, NTP_Remote_Address *remote_addr,
|
||||||
|
NTP_Local_Address *local_addr, int length, int process_tx);
|
||||||
|
|
||||||
|
#endif /* GOT_NTP_IO_H */
|
815
ntp_io_linux.c
Normal file
815
ntp_io_linux.c
Normal file
|
@ -0,0 +1,815 @@
|
||||||
|
/*
|
||||||
|
chronyd/chronyc - Programs for keeping computer clocks accurate.
|
||||||
|
|
||||||
|
**********************************************************************
|
||||||
|
* Copyright (C) Miroslav Lichvar 2016-2019, 2021-2023
|
||||||
|
*
|
||||||
|
* This program is free software; you can redistribute it and/or modify
|
||||||
|
* it under the terms of version 2 of the GNU General Public License as
|
||||||
|
* published by the Free Software Foundation.
|
||||||
|
*
|
||||||
|
* This program is distributed in the hope that it will be useful, but
|
||||||
|
* WITHOUT ANY WARRANTY; without even the implied warranty of
|
||||||
|
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
|
||||||
|
* General Public License for more details.
|
||||||
|
*
|
||||||
|
* You should have received a copy of the GNU General Public License along
|
||||||
|
* with this program; if not, write to the Free Software Foundation, Inc.,
|
||||||
|
* 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA.
|
||||||
|
*
|
||||||
|
**********************************************************************
|
||||||
|
|
||||||
|
=======================================================================
|
||||||
|
|
||||||
|
Functions for NTP I/O specific to Linux
|
||||||
|
*/
|
||||||
|
|
||||||
|
#include "config.h"
|
||||||
|
|
||||||
|
#include "sysincl.h"
|
||||||
|
|
||||||
|
#include <ifaddrs.h>
|
||||||
|
#include <linux/ethtool.h>
|
||||||
|
#include <linux/net_tstamp.h>
|
||||||
|
#include <linux/sockios.h>
|
||||||
|
#include <net/if.h>
|
||||||
|
|
||||||
|
#include "array.h"
|
||||||
|
#include "conf.h"
|
||||||
|
#include "hwclock.h"
|
||||||
|
#include "local.h"
|
||||||
|
#include "logging.h"
|
||||||
|
#include "memory.h"
|
||||||
|
#include "ntp_core.h"
|
||||||
|
#include "ntp_io.h"
|
||||||
|
#include "ntp_io_linux.h"
|
||||||
|
#include "ntp_sources.h"
|
||||||
|
#include "sched.h"
|
||||||
|
#include "socket.h"
|
||||||
|
#include "sys_linux.h"
|
||||||
|
#include "util.h"
|
||||||
|
|
||||||
|
struct Interface {
|
||||||
|
char name[IF_NAMESIZE];
|
||||||
|
int if_index;
|
||||||
|
int phc_fd;
|
||||||
|
int phc_mode;
|
||||||
|
int phc_nocrossts;
|
||||||
|
/* Link speed in mbit/s */
|
||||||
|
int link_speed;
|
||||||
|
/* Start of UDP data at layer 2 for IPv4 and IPv6 */
|
||||||
|
int l2_udp4_ntp_start;
|
||||||
|
int l2_udp6_ntp_start;
|
||||||
|
/* Compensation of errors in TX and RX timestamping */
|
||||||
|
double tx_comp;
|
||||||
|
double rx_comp;
|
||||||
|
HCL_Instance clock;
|
||||||
|
int maxpoll;
|
||||||
|
SCH_TimeoutID poll_timeout_id;
|
||||||
|
};
|
||||||
|
|
||||||
|
/* Number of PHC readings per HW clock sample */
|
||||||
|
#define PHC_READINGS 25
|
||||||
|
|
||||||
|
/* Minimum and maximum interval between PHC readings */
|
||||||
|
#define MIN_PHC_POLL -6
|
||||||
|
#define MAX_PHC_POLL 20
|
||||||
|
|
||||||
|
/* Maximum acceptable offset between SW/HW and daemon timestamp */
|
||||||
|
#define MAX_TS_DELAY 1.0
|
||||||
|
|
||||||
|
/* Array of Interfaces */
|
||||||
|
static ARR_Instance interfaces;
|
||||||
|
|
||||||
|
/* RX/TX and TX-specific timestamping socket options */
|
||||||
|
static int ts_flags;
|
||||||
|
static int ts_tx_flags;
|
||||||
|
|
||||||
|
/* Flag indicating the socket options can't be changed in control messages */
|
||||||
|
static int permanent_ts_options;
|
||||||
|
|
||||||
|
/* Unbound socket keeping the kernel RX timestamping permanently enabled
|
||||||
|
in order to avoid a race condition between receiving a server response
|
||||||
|
and the kernel actually starting to timestamp received packets after
|
||||||
|
enabling the timestamping and sending a request */
|
||||||
|
static int dummy_rxts_socket;
|
||||||
|
|
||||||
|
#define INVALID_SOCK_FD -3
|
||||||
|
|
||||||
|
/* ================================================== */
|
||||||
|
|
||||||
|
static void poll_phc(struct Interface *iface, struct timespec *now);
|
||||||
|
|
||||||
|
/* ================================================== */
|
||||||
|
|
||||||
|
static int
|
||||||
|
add_interface(CNF_HwTsInterface *conf_iface)
|
||||||
|
{
|
||||||
|
int sock_fd, if_index, minpoll, phc_fd, req_hwts_flags, rx_filter;
|
||||||
|
struct ethtool_ts_info ts_info;
|
||||||
|
struct hwtstamp_config ts_config;
|
||||||
|
struct ifreq req;
|
||||||
|
unsigned int i;
|
||||||
|
struct Interface *iface;
|
||||||
|
|
||||||
|
/* Check if the interface was not already added */
|
||||||
|
for (i = 0; i < ARR_GetSize(interfaces); i++) {
|
||||||
|
if (!strcmp(conf_iface->name, ((struct Interface *)ARR_GetElement(interfaces, i))->name))
|
||||||
|
return 1;
|
||||||
|
}
|
||||||
|
|
||||||
|
sock_fd = SCK_OpenUdpSocket(NULL, NULL, NULL, 0);
|
||||||
|
if (sock_fd < 0)
|
||||||
|
return 0;
|
||||||
|
|
||||||
|
memset(&req, 0, sizeof (req));
|
||||||
|
memset(&ts_info, 0, sizeof (ts_info));
|
||||||
|
|
||||||
|
if (snprintf(req.ifr_name, sizeof (req.ifr_name), "%s", conf_iface->name) >=
|
||||||
|
sizeof (req.ifr_name)) {
|
||||||
|
SCK_CloseSocket(sock_fd);
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (ioctl(sock_fd, SIOCGIFINDEX, &req)) {
|
||||||
|
DEBUG_LOG("ioctl(%s) failed : %s", "SIOCGIFINDEX", strerror(errno));
|
||||||
|
SCK_CloseSocket(sock_fd);
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
if_index = req.ifr_ifindex;
|
||||||
|
|
||||||
|
ts_info.cmd = ETHTOOL_GET_TS_INFO;
|
||||||
|
req.ifr_data = (char *)&ts_info;
|
||||||
|
|
||||||
|
if (ioctl(sock_fd, SIOCETHTOOL, &req)) {
|
||||||
|
DEBUG_LOG("ioctl(%s) failed : %s", "SIOCETHTOOL", strerror(errno));
|
||||||
|
SCK_CloseSocket(sock_fd);
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
req_hwts_flags = SOF_TIMESTAMPING_RX_HARDWARE | SOF_TIMESTAMPING_TX_HARDWARE |
|
||||||
|
SOF_TIMESTAMPING_RAW_HARDWARE;
|
||||||
|
if ((ts_info.so_timestamping & req_hwts_flags) != req_hwts_flags) {
|
||||||
|
DEBUG_LOG("HW timestamping not supported on %s", req.ifr_name);
|
||||||
|
SCK_CloseSocket(sock_fd);
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (ts_info.phc_index < 0) {
|
||||||
|
DEBUG_LOG("PHC missing on %s", req.ifr_name);
|
||||||
|
SCK_CloseSocket(sock_fd);
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
switch (conf_iface->rxfilter) {
|
||||||
|
case CNF_HWTS_RXFILTER_ANY:
|
||||||
|
#ifdef HAVE_LINUX_TIMESTAMPING_RXFILTER_NTP
|
||||||
|
if (ts_info.rx_filters & (1 << HWTSTAMP_FILTER_NTP_ALL))
|
||||||
|
rx_filter = HWTSTAMP_FILTER_NTP_ALL;
|
||||||
|
else
|
||||||
|
#endif
|
||||||
|
if (ts_info.rx_filters & (1 << HWTSTAMP_FILTER_ALL))
|
||||||
|
rx_filter = HWTSTAMP_FILTER_ALL;
|
||||||
|
else
|
||||||
|
rx_filter = HWTSTAMP_FILTER_NONE;
|
||||||
|
break;
|
||||||
|
case CNF_HWTS_RXFILTER_NONE:
|
||||||
|
rx_filter = HWTSTAMP_FILTER_NONE;
|
||||||
|
break;
|
||||||
|
#ifdef HAVE_LINUX_TIMESTAMPING_RXFILTER_NTP
|
||||||
|
case CNF_HWTS_RXFILTER_NTP:
|
||||||
|
rx_filter = HWTSTAMP_FILTER_NTP_ALL;
|
||||||
|
break;
|
||||||
|
#endif
|
||||||
|
case CNF_HWTS_RXFILTER_PTP:
|
||||||
|
if (ts_info.rx_filters & (1 << HWTSTAMP_FILTER_PTP_V2_L4_EVENT))
|
||||||
|
rx_filter = HWTSTAMP_FILTER_PTP_V2_L4_EVENT;
|
||||||
|
else if (ts_info.rx_filters & (1 << HWTSTAMP_FILTER_PTP_V2_EVENT))
|
||||||
|
rx_filter = HWTSTAMP_FILTER_PTP_V2_EVENT;
|
||||||
|
else
|
||||||
|
rx_filter = HWTSTAMP_FILTER_NONE;
|
||||||
|
break;
|
||||||
|
default:
|
||||||
|
rx_filter = HWTSTAMP_FILTER_ALL;
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
|
||||||
|
ts_config.flags = 0;
|
||||||
|
ts_config.tx_type = HWTSTAMP_TX_ON;
|
||||||
|
ts_config.rx_filter = rx_filter;
|
||||||
|
req.ifr_data = (char *)&ts_config;
|
||||||
|
|
||||||
|
if (ioctl(sock_fd, SIOCSHWTSTAMP, &req)) {
|
||||||
|
LOG(errno == EPERM ? LOGS_ERR : LOGS_DEBUG,
|
||||||
|
"ioctl(%s) failed : %s", "SIOCSHWTSTAMP", strerror(errno));
|
||||||
|
|
||||||
|
/* Check the current timestamping configuration in case this interface
|
||||||
|
allows only reading of the configuration and it was already configured
|
||||||
|
as requested */
|
||||||
|
req.ifr_data = (char *)&ts_config;
|
||||||
|
#ifdef SIOCGHWTSTAMP
|
||||||
|
if (ioctl(sock_fd, SIOCGHWTSTAMP, &req) ||
|
||||||
|
ts_config.tx_type != HWTSTAMP_TX_ON || ts_config.rx_filter != rx_filter)
|
||||||
|
#endif
|
||||||
|
{
|
||||||
|
SCK_CloseSocket(sock_fd);
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
SCK_CloseSocket(sock_fd);
|
||||||
|
|
||||||
|
phc_fd = SYS_Linux_OpenPHC(NULL, ts_info.phc_index);
|
||||||
|
if (phc_fd < 0)
|
||||||
|
return 0;
|
||||||
|
|
||||||
|
iface = ARR_GetNewElement(interfaces);
|
||||||
|
|
||||||
|
snprintf(iface->name, sizeof (iface->name), "%s", conf_iface->name);
|
||||||
|
iface->if_index = if_index;
|
||||||
|
iface->phc_fd = phc_fd;
|
||||||
|
iface->phc_mode = 0;
|
||||||
|
iface->phc_nocrossts = conf_iface->nocrossts;
|
||||||
|
|
||||||
|
/* Start with 1 gbit and no VLANs or IPv4/IPv6 options */
|
||||||
|
iface->link_speed = 1000;
|
||||||
|
iface->l2_udp4_ntp_start = 42;
|
||||||
|
iface->l2_udp6_ntp_start = 62;
|
||||||
|
|
||||||
|
iface->tx_comp = conf_iface->tx_comp;
|
||||||
|
iface->rx_comp = conf_iface->rx_comp;
|
||||||
|
|
||||||
|
minpoll = CLAMP(MIN_PHC_POLL, conf_iface->minpoll, MAX_PHC_POLL);
|
||||||
|
iface->clock = HCL_CreateInstance(conf_iface->min_samples, conf_iface->max_samples,
|
||||||
|
UTI_Log2ToDouble(minpoll), conf_iface->precision);
|
||||||
|
|
||||||
|
iface->maxpoll = CLAMP(minpoll, conf_iface->maxpoll, MAX_PHC_POLL);
|
||||||
|
|
||||||
|
/* Do not schedule the first poll timeout here! The argument (interface) can
|
||||||
|
move until all interfaces are added. Wait for the first HW timestamp. */
|
||||||
|
iface->poll_timeout_id = 0;
|
||||||
|
|
||||||
|
LOG(LOGS_INFO, "Enabled HW timestamping %son %s",
|
||||||
|
ts_config.rx_filter == HWTSTAMP_FILTER_NONE ? "(TX only) " : "", iface->name);
|
||||||
|
|
||||||
|
return 1;
|
||||||
|
}
|
||||||
|
|
||||||
|
/* ================================================== */
|
||||||
|
|
||||||
|
static int
|
||||||
|
add_all_interfaces(CNF_HwTsInterface *conf_iface_all)
|
||||||
|
{
|
||||||
|
CNF_HwTsInterface conf_iface;
|
||||||
|
struct ifaddrs *ifaddr, *ifa;
|
||||||
|
int r;
|
||||||
|
|
||||||
|
conf_iface = *conf_iface_all;
|
||||||
|
|
||||||
|
if (getifaddrs(&ifaddr)) {
|
||||||
|
DEBUG_LOG("getifaddrs() failed : %s", strerror(errno));
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
for (r = 0, ifa = ifaddr; ifa; ifa = ifa->ifa_next) {
|
||||||
|
conf_iface.name = ifa->ifa_name;
|
||||||
|
if (add_interface(&conf_iface))
|
||||||
|
r = 1;
|
||||||
|
}
|
||||||
|
|
||||||
|
freeifaddrs(ifaddr);
|
||||||
|
|
||||||
|
/* Return success if at least one interface was added */
|
||||||
|
return r;
|
||||||
|
}
|
||||||
|
|
||||||
|
/* ================================================== */
|
||||||
|
|
||||||
|
static void
|
||||||
|
update_interface_speed(struct Interface *iface)
|
||||||
|
{
|
||||||
|
struct ethtool_cmd cmd;
|
||||||
|
struct ifreq req;
|
||||||
|
int sock_fd, link_speed;
|
||||||
|
|
||||||
|
sock_fd = SCK_OpenUdpSocket(NULL, NULL, NULL, 0);
|
||||||
|
if (sock_fd < 0)
|
||||||
|
return;
|
||||||
|
|
||||||
|
memset(&req, 0, sizeof (req));
|
||||||
|
memset(&cmd, 0, sizeof (cmd));
|
||||||
|
|
||||||
|
snprintf(req.ifr_name, sizeof (req.ifr_name), "%s", iface->name);
|
||||||
|
cmd.cmd = ETHTOOL_GSET;
|
||||||
|
req.ifr_data = (char *)&cmd;
|
||||||
|
|
||||||
|
if (ioctl(sock_fd, SIOCETHTOOL, &req)) {
|
||||||
|
DEBUG_LOG("ioctl(%s) failed : %s", "SIOCETHTOOL", strerror(errno));
|
||||||
|
SCK_CloseSocket(sock_fd);
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
SCK_CloseSocket(sock_fd);
|
||||||
|
|
||||||
|
link_speed = ethtool_cmd_speed(&cmd);
|
||||||
|
|
||||||
|
if (iface->link_speed != link_speed) {
|
||||||
|
iface->link_speed = link_speed;
|
||||||
|
DEBUG_LOG("Updated speed of %s to %d Mb/s", iface->name, link_speed);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/* ================================================== */
|
||||||
|
|
||||||
|
#if defined(HAVE_LINUX_TIMESTAMPING_OPT_PKTINFO) || defined(HAVE_LINUX_TIMESTAMPING_OPT_TX_SWHW)
|
||||||
|
static int
|
||||||
|
check_timestamping_option(int option)
|
||||||
|
{
|
||||||
|
int sock_fd;
|
||||||
|
|
||||||
|
sock_fd = SCK_OpenUdpSocket(NULL, NULL, NULL, 0);
|
||||||
|
if (sock_fd < 0)
|
||||||
|
return 0;
|
||||||
|
|
||||||
|
if (!SCK_SetIntOption(sock_fd, SOL_SOCKET, SO_TIMESTAMPING, option)) {
|
||||||
|
SCK_CloseSocket(sock_fd);
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
SCK_CloseSocket(sock_fd);
|
||||||
|
return 1;
|
||||||
|
}
|
||||||
|
#endif
|
||||||
|
|
||||||
|
/* ================================================== */
|
||||||
|
|
||||||
|
static int
|
||||||
|
open_dummy_socket(void)
|
||||||
|
{
|
||||||
|
int sock_fd, events = 0;
|
||||||
|
|
||||||
|
sock_fd = SCK_OpenUdpSocket(NULL, NULL, NULL, 0);
|
||||||
|
if (sock_fd < 0)
|
||||||
|
return INVALID_SOCK_FD;
|
||||||
|
|
||||||
|
if (!NIO_Linux_SetTimestampSocketOptions(sock_fd, 1, &events)) {
|
||||||
|
SCK_CloseSocket(sock_fd);
|
||||||
|
return INVALID_SOCK_FD;
|
||||||
|
}
|
||||||
|
|
||||||
|
return sock_fd;
|
||||||
|
}
|
||||||
|
|
||||||
|
/* ================================================== */
|
||||||
|
|
||||||
|
void
|
||||||
|
NIO_Linux_Initialise(void)
|
||||||
|
{
|
||||||
|
CNF_HwTsInterface *conf_iface;
|
||||||
|
unsigned int i;
|
||||||
|
int hwts;
|
||||||
|
|
||||||
|
interfaces = ARR_CreateInstance(sizeof (struct Interface));
|
||||||
|
|
||||||
|
/* Enable HW timestamping on specified interfaces. If "*" was specified, try
|
||||||
|
all interfaces. If no interface was specified, enable SW timestamping. */
|
||||||
|
|
||||||
|
for (i = hwts = 0; CNF_GetHwTsInterface(i, &conf_iface); i++) {
|
||||||
|
if (!strcmp("*", conf_iface->name))
|
||||||
|
continue;
|
||||||
|
if (!add_interface(conf_iface))
|
||||||
|
LOG_FATAL("Could not enable HW timestamping on %s", conf_iface->name);
|
||||||
|
hwts = 1;
|
||||||
|
}
|
||||||
|
|
||||||
|
for (i = 0; CNF_GetHwTsInterface(i, &conf_iface); i++) {
|
||||||
|
if (strcmp("*", conf_iface->name))
|
||||||
|
continue;
|
||||||
|
if (add_all_interfaces(conf_iface))
|
||||||
|
hwts = 1;
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
|
||||||
|
ts_flags = SOF_TIMESTAMPING_SOFTWARE | SOF_TIMESTAMPING_RX_SOFTWARE;
|
||||||
|
ts_tx_flags = SOF_TIMESTAMPING_TX_SOFTWARE;
|
||||||
|
|
||||||
|
if (hwts) {
|
||||||
|
ts_flags |= SOF_TIMESTAMPING_RAW_HARDWARE | SOF_TIMESTAMPING_RX_HARDWARE;
|
||||||
|
ts_tx_flags |= SOF_TIMESTAMPING_TX_HARDWARE;
|
||||||
|
#ifdef HAVE_LINUX_TIMESTAMPING_OPT_PKTINFO
|
||||||
|
if (check_timestamping_option(SOF_TIMESTAMPING_OPT_PKTINFO))
|
||||||
|
ts_flags |= SOF_TIMESTAMPING_OPT_PKTINFO;
|
||||||
|
#endif
|
||||||
|
#ifdef HAVE_LINUX_TIMESTAMPING_OPT_TX_SWHW
|
||||||
|
if (check_timestamping_option(SOF_TIMESTAMPING_OPT_TX_SWHW))
|
||||||
|
ts_flags |= SOF_TIMESTAMPING_OPT_TX_SWHW;
|
||||||
|
#endif
|
||||||
|
}
|
||||||
|
|
||||||
|
/* Enable IP_PKTINFO in messages looped back to the error queue */
|
||||||
|
ts_flags |= SOF_TIMESTAMPING_OPT_CMSG;
|
||||||
|
|
||||||
|
/* Kernels before 4.7 ignore timestamping flags set in control messages */
|
||||||
|
permanent_ts_options = !SYS_Linux_CheckKernelVersion(4, 7);
|
||||||
|
|
||||||
|
dummy_rxts_socket = INVALID_SOCK_FD;
|
||||||
|
}
|
||||||
|
|
||||||
|
/* ================================================== */
|
||||||
|
|
||||||
|
void
|
||||||
|
NIO_Linux_Finalise(void)
|
||||||
|
{
|
||||||
|
struct Interface *iface;
|
||||||
|
unsigned int i;
|
||||||
|
|
||||||
|
if (dummy_rxts_socket != INVALID_SOCK_FD)
|
||||||
|
SCK_CloseSocket(dummy_rxts_socket);
|
||||||
|
|
||||||
|
for (i = 0; i < ARR_GetSize(interfaces); i++) {
|
||||||
|
iface = ARR_GetElement(interfaces, i);
|
||||||
|
SCH_RemoveTimeout(iface->poll_timeout_id);
|
||||||
|
HCL_DestroyInstance(iface->clock);
|
||||||
|
close(iface->phc_fd);
|
||||||
|
}
|
||||||
|
|
||||||
|
ARR_DestroyInstance(interfaces);
|
||||||
|
}
|
||||||
|
|
||||||
|
/* ================================================== */
|
||||||
|
|
||||||
|
int
|
||||||
|
NIO_Linux_IsHwTsEnabled(void)
|
||||||
|
{
|
||||||
|
return ARR_GetSize(interfaces) > 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
/* ================================================== */
|
||||||
|
|
||||||
|
int
|
||||||
|
NIO_Linux_SetTimestampSocketOptions(int sock_fd, int client_only, int *events)
|
||||||
|
{
|
||||||
|
int val, flags;
|
||||||
|
|
||||||
|
if (!ts_flags)
|
||||||
|
return 0;
|
||||||
|
|
||||||
|
/* Enable SCM_TIMESTAMPING control messages and the socket's error queue in
|
||||||
|
order to receive our transmitted packets with more accurate timestamps */
|
||||||
|
|
||||||
|
val = 1;
|
||||||
|
flags = ts_flags;
|
||||||
|
|
||||||
|
if (client_only || permanent_ts_options)
|
||||||
|
flags |= ts_tx_flags;
|
||||||
|
|
||||||
|
if (!SCK_SetIntOption(sock_fd, SOL_SOCKET, SO_SELECT_ERR_QUEUE, val)) {
|
||||||
|
ts_flags = 0;
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (!SCK_SetIntOption(sock_fd, SOL_SOCKET, SO_TIMESTAMPING, flags)) {
|
||||||
|
ts_flags = 0;
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
*events |= SCH_FILE_EXCEPTION;
|
||||||
|
return 1;
|
||||||
|
}
|
||||||
|
|
||||||
|
/* ================================================== */
|
||||||
|
|
||||||
|
static struct Interface *
|
||||||
|
get_interface(int if_index)
|
||||||
|
{
|
||||||
|
struct Interface *iface;
|
||||||
|
unsigned int i;
|
||||||
|
|
||||||
|
for (i = 0; i < ARR_GetSize(interfaces); i++) {
|
||||||
|
iface = ARR_GetElement(interfaces, i);
|
||||||
|
if (iface->if_index != if_index)
|
||||||
|
continue;
|
||||||
|
|
||||||
|
return iface;
|
||||||
|
}
|
||||||
|
|
||||||
|
return NULL;
|
||||||
|
}
|
||||||
|
|
||||||
|
/* ================================================== */
|
||||||
|
|
||||||
|
static void
|
||||||
|
poll_timeout(void *arg)
|
||||||
|
{
|
||||||
|
struct Interface *iface = arg;
|
||||||
|
struct timespec now;
|
||||||
|
|
||||||
|
iface->poll_timeout_id = 0;
|
||||||
|
|
||||||
|
SCH_GetLastEventTime(&now, NULL, NULL);
|
||||||
|
poll_phc(iface, &now);
|
||||||
|
}
|
||||||
|
|
||||||
|
/* ================================================== */
|
||||||
|
|
||||||
|
static void
|
||||||
|
poll_phc(struct Interface *iface, struct timespec *now)
|
||||||
|
{
|
||||||
|
struct timespec sample_phc_ts, sample_sys_ts, sample_local_ts;
|
||||||
|
struct timespec phc_readings[PHC_READINGS][3];
|
||||||
|
double phc_err, local_err, interval;
|
||||||
|
int n_readings;
|
||||||
|
|
||||||
|
if (!HCL_NeedsNewSample(iface->clock, now))
|
||||||
|
return;
|
||||||
|
|
||||||
|
DEBUG_LOG("Polling PHC on %s%s",
|
||||||
|
iface->name, iface->poll_timeout_id != 0 ? " before timeout" : "");
|
||||||
|
|
||||||
|
n_readings = SYS_Linux_GetPHCReadings(iface->phc_fd, iface->phc_nocrossts,
|
||||||
|
&iface->phc_mode, PHC_READINGS, phc_readings);
|
||||||
|
|
||||||
|
/* Add timeout for the next poll in case no HW timestamp will be captured
|
||||||
|
between the minpoll and maxpoll. Separate reading of different PHCs to
|
||||||
|
avoid long intervals between handling I/O events. */
|
||||||
|
SCH_RemoveTimeout(iface->poll_timeout_id);
|
||||||
|
interval = UTI_Log2ToDouble(iface->maxpoll);
|
||||||
|
iface->poll_timeout_id = SCH_AddTimeoutInClass(interval, interval /
|
||||||
|
ARR_GetSize(interfaces) / 4, 0.1,
|
||||||
|
SCH_PhcPollClass, poll_timeout, iface);
|
||||||
|
|
||||||
|
if (n_readings <= 0)
|
||||||
|
return;
|
||||||
|
|
||||||
|
if (!HCL_ProcessReadings(iface->clock, n_readings, phc_readings,
|
||||||
|
&sample_phc_ts, &sample_sys_ts, &phc_err))
|
||||||
|
return;
|
||||||
|
|
||||||
|
LCL_CookTime(&sample_sys_ts, &sample_local_ts, &local_err);
|
||||||
|
HCL_AccumulateSample(iface->clock, &sample_phc_ts, &sample_local_ts, phc_err + local_err);
|
||||||
|
|
||||||
|
update_interface_speed(iface);
|
||||||
|
}
|
||||||
|
|
||||||
|
/* ================================================== */
|
||||||
|
|
||||||
|
static void
|
||||||
|
process_hw_timestamp(struct Interface *iface, struct timespec *hw_ts,
|
||||||
|
NTP_Local_Timestamp *local_ts, int rx_ntp_length, int family,
|
||||||
|
int l2_length)
|
||||||
|
{
|
||||||
|
double rx_correction = 0.0, ts_delay, local_err;
|
||||||
|
struct timespec ts;
|
||||||
|
|
||||||
|
poll_phc(iface, &local_ts->ts);
|
||||||
|
|
||||||
|
/* We need to transpose RX timestamps as hardware timestamps are normally
|
||||||
|
preamble timestamps and RX timestamps in NTP are supposed to be trailer
|
||||||
|
timestamps. If we don't know the length of the packet at layer 2, we
|
||||||
|
make an assumption that UDP data start at the same position as in the
|
||||||
|
last transmitted packet which had a HW TX timestamp. */
|
||||||
|
if (rx_ntp_length && iface->link_speed) {
|
||||||
|
if (!l2_length)
|
||||||
|
l2_length = (family == IPADDR_INET4 ? iface->l2_udp4_ntp_start :
|
||||||
|
iface->l2_udp6_ntp_start) + rx_ntp_length;
|
||||||
|
|
||||||
|
/* Include the frame check sequence (FCS) */
|
||||||
|
l2_length += 4;
|
||||||
|
|
||||||
|
rx_correction = l2_length / (1.0e6 / 8 * iface->link_speed);
|
||||||
|
|
||||||
|
UTI_AddDoubleToTimespec(hw_ts, rx_correction, hw_ts);
|
||||||
|
}
|
||||||
|
|
||||||
|
if (!HCL_CookTime(iface->clock, hw_ts, &ts, &local_err))
|
||||||
|
return;
|
||||||
|
|
||||||
|
if (!rx_ntp_length && iface->tx_comp)
|
||||||
|
UTI_AddDoubleToTimespec(&ts, iface->tx_comp, &ts);
|
||||||
|
else if (rx_ntp_length && iface->rx_comp)
|
||||||
|
UTI_AddDoubleToTimespec(&ts, -iface->rx_comp, &ts);
|
||||||
|
|
||||||
|
ts_delay = UTI_DiffTimespecsToDouble(&local_ts->ts, &ts);
|
||||||
|
|
||||||
|
if (fabs(ts_delay) > MAX_TS_DELAY) {
|
||||||
|
DEBUG_LOG("Unacceptable timestamp delay %.9f", ts_delay);
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
local_ts->ts = ts;
|
||||||
|
local_ts->err = local_err;
|
||||||
|
local_ts->source = NTP_TS_HARDWARE;
|
||||||
|
local_ts->rx_duration = rx_correction;
|
||||||
|
/* Network correction needs to include the RX duration to avoid
|
||||||
|
asymmetric correction with asymmetric link speeds */
|
||||||
|
local_ts->net_correction = rx_correction;
|
||||||
|
}
|
||||||
|
|
||||||
|
/* ================================================== */
|
||||||
|
|
||||||
|
static void
|
||||||
|
process_sw_timestamp(struct timespec *sw_ts, NTP_Local_Timestamp *local_ts)
|
||||||
|
{
|
||||||
|
double ts_delay, local_err;
|
||||||
|
struct timespec ts;
|
||||||
|
|
||||||
|
LCL_CookTime(sw_ts, &ts, &local_err);
|
||||||
|
|
||||||
|
ts_delay = UTI_DiffTimespecsToDouble(&local_ts->ts, &ts);
|
||||||
|
|
||||||
|
if (fabs(ts_delay) > MAX_TS_DELAY) {
|
||||||
|
DEBUG_LOG("Unacceptable timestamp delay %.9f", ts_delay);
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
local_ts->ts = ts;
|
||||||
|
local_ts->err = local_err;
|
||||||
|
local_ts->source = NTP_TS_KERNEL;
|
||||||
|
}
|
||||||
|
|
||||||
|
/* ================================================== */
|
||||||
|
/* Extract UDP data from a layer 2 message. Supported is Ethernet
|
||||||
|
with optional VLAN tags. */
|
||||||
|
|
||||||
|
static int
|
||||||
|
extract_udp_data(unsigned char *msg, NTP_Remote_Address *remote_addr, int len)
|
||||||
|
{
|
||||||
|
unsigned char *msg_start = msg;
|
||||||
|
|
||||||
|
remote_addr->ip_addr.family = IPADDR_UNSPEC;
|
||||||
|
remote_addr->port = 0;
|
||||||
|
|
||||||
|
/* Skip MACs */
|
||||||
|
if (len < 12)
|
||||||
|
return 0;
|
||||||
|
len -= 12, msg += 12;
|
||||||
|
|
||||||
|
/* Skip VLAN tag(s) if present */
|
||||||
|
while (len >= 4 && msg[0] == 0x81 && msg[1] == 0x00)
|
||||||
|
len -= 4, msg += 4;
|
||||||
|
|
||||||
|
/* Skip IPv4 or IPv6 ethertype */
|
||||||
|
if (len < 2 || !((msg[0] == 0x08 && msg[1] == 0x00) ||
|
||||||
|
(msg[0] == 0x86 && msg[1] == 0xdd)))
|
||||||
|
return 0;
|
||||||
|
len -= 2, msg += 2;
|
||||||
|
|
||||||
|
/* Parse destination address and port from IPv4/IPv6 and UDP headers */
|
||||||
|
if (len >= 20 && msg[0] >> 4 == 4) {
|
||||||
|
int ihl = (msg[0] & 0xf) * 4;
|
||||||
|
uint32_t addr;
|
||||||
|
|
||||||
|
if (len < ihl + 8 || msg[9] != 17)
|
||||||
|
return 0;
|
||||||
|
|
||||||
|
memcpy(&addr, msg + 16, sizeof (addr));
|
||||||
|
remote_addr->ip_addr.addr.in4 = ntohl(addr);
|
||||||
|
remote_addr->port = ntohs(*(uint16_t *)(msg + ihl + 2));
|
||||||
|
remote_addr->ip_addr.family = IPADDR_INET4;
|
||||||
|
len -= ihl + 8, msg += ihl + 8;
|
||||||
|
#ifdef FEAT_IPV6
|
||||||
|
} else if (len >= 48 && msg[0] >> 4 == 6) {
|
||||||
|
int eh_len, next_header = msg[6];
|
||||||
|
|
||||||
|
memcpy(&remote_addr->ip_addr.addr.in6, msg + 24, sizeof (remote_addr->ip_addr.addr.in6));
|
||||||
|
len -= 40, msg += 40;
|
||||||
|
|
||||||
|
/* Skip IPv6 extension headers if present */
|
||||||
|
while (next_header != 17) {
|
||||||
|
switch (next_header) {
|
||||||
|
case 44: /* Fragment Header */
|
||||||
|
/* Process only the first fragment */
|
||||||
|
if (ntohs(*(uint16_t *)(msg + 2)) >> 3 != 0)
|
||||||
|
return 0;
|
||||||
|
eh_len = 8;
|
||||||
|
break;
|
||||||
|
case 0: /* Hop-by-Hop Options */
|
||||||
|
case 43: /* Routing Header */
|
||||||
|
case 60: /* Destination Options */
|
||||||
|
case 135: /* Mobility Header */
|
||||||
|
eh_len = 8 * (msg[1] + 1);
|
||||||
|
break;
|
||||||
|
case 51: /* Authentication Header */
|
||||||
|
eh_len = 4 * (msg[1] + 2);
|
||||||
|
break;
|
||||||
|
default:
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (eh_len < 8 || len < eh_len + 8)
|
||||||
|
return 0;
|
||||||
|
|
||||||
|
next_header = msg[0];
|
||||||
|
len -= eh_len, msg += eh_len;
|
||||||
|
}
|
||||||
|
|
||||||
|
remote_addr->port = ntohs(*(uint16_t *)(msg + 2));
|
||||||
|
remote_addr->ip_addr.family = IPADDR_INET6;
|
||||||
|
len -= 8, msg += 8;
|
||||||
|
#endif
|
||||||
|
} else {
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
/* Move the message to fix alignment of its fields */
|
||||||
|
if (len > 0)
|
||||||
|
memmove(msg_start, msg, len);
|
||||||
|
|
||||||
|
return len;
|
||||||
|
}
|
||||||
|
|
||||||
|
/* ================================================== */
|
||||||
|
|
||||||
|
int
|
||||||
|
NIO_Linux_ProcessMessage(SCK_Message *message, NTP_Local_Address *local_addr,
|
||||||
|
NTP_Local_Timestamp *local_ts, int event)
|
||||||
|
{
|
||||||
|
struct Interface *iface;
|
||||||
|
int is_tx, ts_if_index, l2_length;
|
||||||
|
double c = 0.0;
|
||||||
|
|
||||||
|
is_tx = event == SCH_FILE_EXCEPTION;
|
||||||
|
iface = NULL;
|
||||||
|
|
||||||
|
ts_if_index = message->timestamp.if_index;
|
||||||
|
if (ts_if_index == INVALID_IF_INDEX)
|
||||||
|
ts_if_index = message->if_index;
|
||||||
|
l2_length = message->timestamp.l2_length;
|
||||||
|
|
||||||
|
if (!UTI_IsZeroTimespec(&message->timestamp.hw)) {
|
||||||
|
iface = get_interface(ts_if_index);
|
||||||
|
if (iface) {
|
||||||
|
process_hw_timestamp(iface, &message->timestamp.hw, local_ts, !is_tx ? message->length : 0,
|
||||||
|
message->remote_addr.ip.ip_addr.family, l2_length);
|
||||||
|
} else {
|
||||||
|
DEBUG_LOG("HW clock not found for interface %d", ts_if_index);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
if (local_ts->source == NTP_TS_DAEMON && !UTI_IsZeroTimespec(&message->timestamp.kernel) &&
|
||||||
|
(!is_tx || UTI_IsZeroTimespec(&message->timestamp.hw))) {
|
||||||
|
process_sw_timestamp(&message->timestamp.kernel, local_ts);
|
||||||
|
}
|
||||||
|
|
||||||
|
/* If the kernel is slow with enabling RX timestamping, open a dummy
|
||||||
|
socket to keep the kernel RX timestamping permanently enabled */
|
||||||
|
if (!is_tx && local_ts->source == NTP_TS_DAEMON && ts_flags) {
|
||||||
|
DEBUG_LOG("Missing kernel RX timestamp");
|
||||||
|
if (dummy_rxts_socket == INVALID_SOCK_FD)
|
||||||
|
dummy_rxts_socket = open_dummy_socket();
|
||||||
|
}
|
||||||
|
|
||||||
|
/* Return the message if it's not received from the error queue */
|
||||||
|
if (!is_tx)
|
||||||
|
return 0;
|
||||||
|
|
||||||
|
/* The data from the error queue includes all layers up to UDP. We have to
|
||||||
|
extract the UDP data and also the destination address with port as there
|
||||||
|
currently doesn't seem to be a better way to get them both. */
|
||||||
|
l2_length = message->length;
|
||||||
|
message->length = extract_udp_data(message->data, &message->remote_addr.ip, message->length);
|
||||||
|
|
||||||
|
DEBUG_LOG("Extracted message for %s fd=%d len=%d",
|
||||||
|
UTI_IPSockAddrToString(&message->remote_addr.ip),
|
||||||
|
local_addr->sock_fd, message->length);
|
||||||
|
|
||||||
|
/* Update assumed position of UDP data at layer 2 for next received packet */
|
||||||
|
if (iface && message->length) {
|
||||||
|
if (message->remote_addr.ip.ip_addr.family == IPADDR_INET4)
|
||||||
|
iface->l2_udp4_ntp_start = l2_length - message->length;
|
||||||
|
else if (message->remote_addr.ip.ip_addr.family == IPADDR_INET6)
|
||||||
|
iface->l2_udp6_ntp_start = l2_length - message->length;
|
||||||
|
}
|
||||||
|
|
||||||
|
/* Drop the message if it has no timestamp or its processing failed */
|
||||||
|
if (local_ts->source == NTP_TS_DAEMON) {
|
||||||
|
DEBUG_LOG("Missing TX timestamp");
|
||||||
|
return 1;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (!NIO_UnwrapMessage(message, local_addr->sock_fd, &c))
|
||||||
|
return 1;
|
||||||
|
|
||||||
|
if (message->length < NTP_HEADER_LENGTH || message->length > sizeof (NTP_Packet))
|
||||||
|
return 1;
|
||||||
|
|
||||||
|
NSR_ProcessTx(&message->remote_addr.ip, local_addr, local_ts, message->data, message->length);
|
||||||
|
|
||||||
|
return 1;
|
||||||
|
}
|
||||||
|
|
||||||
|
/* ================================================== */
|
||||||
|
|
||||||
|
void
|
||||||
|
NIO_Linux_RequestTxTimestamp(SCK_Message *message, int sock_fd)
|
||||||
|
{
|
||||||
|
if (!ts_flags)
|
||||||
|
return;
|
||||||
|
|
||||||
|
/* Check if TX timestamping is disabled on this socket */
|
||||||
|
if (permanent_ts_options || !NIO_IsServerSocket(sock_fd))
|
||||||
|
return;
|
||||||
|
|
||||||
|
message->timestamp.tx_flags = ts_tx_flags;
|
||||||
|
}
|
45
ntp_io_linux.h
Normal file
45
ntp_io_linux.h
Normal file
|
@ -0,0 +1,45 @@
|
||||||
|
/*
|
||||||
|
chronyd/chronyc - Programs for keeping computer clocks accurate.
|
||||||
|
|
||||||
|
**********************************************************************
|
||||||
|
* Copyright (C) Miroslav Lichvar 2016
|
||||||
|
*
|
||||||
|
* This program is free software; you can redistribute it and/or modify
|
||||||
|
* it under the terms of version 2 of the GNU General Public License as
|
||||||
|
* published by the Free Software Foundation.
|
||||||
|
*
|
||||||
|
* This program is distributed in the hope that it will be useful, but
|
||||||
|
* WITHOUT ANY WARRANTY; without even the implied warranty of
|
||||||
|
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
|
||||||
|
* General Public License for more details.
|
||||||
|
*
|
||||||
|
* You should have received a copy of the GNU General Public License along
|
||||||
|
* with this program; if not, write to the Free Software Foundation, Inc.,
|
||||||
|
* 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA.
|
||||||
|
*
|
||||||
|
**********************************************************************
|
||||||
|
|
||||||
|
=======================================================================
|
||||||
|
|
||||||
|
This is the header file for the Linux-specific NTP socket I/O bits.
|
||||||
|
*/
|
||||||
|
|
||||||
|
#ifndef GOT_NTP_IO_LINUX_H
|
||||||
|
#define GOT_NTP_IO_LINUX_H
|
||||||
|
|
||||||
|
#include "socket.h"
|
||||||
|
|
||||||
|
extern void NIO_Linux_Initialise(void);
|
||||||
|
|
||||||
|
extern void NIO_Linux_Finalise(void);
|
||||||
|
|
||||||
|
extern int NIO_Linux_IsHwTsEnabled(void);
|
||||||
|
|
||||||
|
extern int NIO_Linux_SetTimestampSocketOptions(int sock_fd, int client_only, int *events);
|
||||||
|
|
||||||
|
extern int NIO_Linux_ProcessMessage(SCK_Message *message, NTP_Local_Address *local_addr,
|
||||||
|
NTP_Local_Timestamp *local_ts, int event);
|
||||||
|
|
||||||
|
extern void NIO_Linux_RequestTxTimestamp(SCK_Message *message, int sock_fd);
|
||||||
|
|
||||||
|
#endif
|
354
ntp_signd.c
Normal file
354
ntp_signd.c
Normal file
|
@ -0,0 +1,354 @@
|
||||||
|
/*
|
||||||
|
chronyd/chronyc - Programs for keeping computer clocks accurate.
|
||||||
|
|
||||||
|
**********************************************************************
|
||||||
|
* Copyright (C) Miroslav Lichvar 2016
|
||||||
|
*
|
||||||
|
* This program is free software; you can redistribute it and/or modify
|
||||||
|
* it under the terms of version 2 of the GNU General Public License as
|
||||||
|
* published by the Free Software Foundation.
|
||||||
|
*
|
||||||
|
* This program is distributed in the hope that it will be useful, but
|
||||||
|
* WITHOUT ANY WARRANTY; without even the implied warranty of
|
||||||
|
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
|
||||||
|
* General Public License for more details.
|
||||||
|
*
|
||||||
|
* You should have received a copy of the GNU General Public License along
|
||||||
|
* with this program; if not, write to the Free Software Foundation, Inc.,
|
||||||
|
* 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA.
|
||||||
|
*
|
||||||
|
**********************************************************************
|
||||||
|
|
||||||
|
=======================================================================
|
||||||
|
|
||||||
|
Support for MS-SNTP authentication in Samba (ntp_signd)
|
||||||
|
*/
|
||||||
|
|
||||||
|
#include "config.h"
|
||||||
|
|
||||||
|
#include "sysincl.h"
|
||||||
|
|
||||||
|
#include "array.h"
|
||||||
|
#include "conf.h"
|
||||||
|
#include "logging.h"
|
||||||
|
#include "ntp_io.h"
|
||||||
|
#include "ntp_signd.h"
|
||||||
|
#include "sched.h"
|
||||||
|
#include "socket.h"
|
||||||
|
#include "util.h"
|
||||||
|
|
||||||
|
/* Declarations per samba/source4/librpc/idl/ntp_signd.idl */
|
||||||
|
|
||||||
|
#define SIGND_VERSION 0
|
||||||
|
|
||||||
|
typedef enum {
|
||||||
|
SIGN_TO_CLIENT = 0,
|
||||||
|
ASK_SERVER_TO_SIGN = 1,
|
||||||
|
CHECK_SERVER_SIGNATURE = 2,
|
||||||
|
SIGNING_SUCCESS = 3,
|
||||||
|
SIGNING_FAILURE = 4,
|
||||||
|
} SigndOp;
|
||||||
|
|
||||||
|
typedef struct {
|
||||||
|
uint32_t length;
|
||||||
|
uint32_t version;
|
||||||
|
uint32_t op;
|
||||||
|
uint16_t packet_id;
|
||||||
|
uint16_t _pad;
|
||||||
|
uint32_t key_id;
|
||||||
|
NTP_Packet packet_to_sign;
|
||||||
|
} SigndRequest;
|
||||||
|
|
||||||
|
typedef struct {
|
||||||
|
uint32_t length;
|
||||||
|
uint32_t version;
|
||||||
|
uint32_t op;
|
||||||
|
uint32_t packet_id;
|
||||||
|
NTP_Packet signed_packet;
|
||||||
|
} SigndResponse;
|
||||||
|
|
||||||
|
typedef struct {
|
||||||
|
NTP_Remote_Address remote_addr;
|
||||||
|
NTP_Local_Address local_addr;
|
||||||
|
|
||||||
|
int sent;
|
||||||
|
int received;
|
||||||
|
int request_length;
|
||||||
|
struct timespec request_ts;
|
||||||
|
SigndRequest request;
|
||||||
|
SigndResponse response;
|
||||||
|
} SignInstance;
|
||||||
|
|
||||||
|
/* As the communication with ntp_signd is asynchronous, incoming packets are
|
||||||
|
saved in a queue in order to avoid loss when they come in bursts */
|
||||||
|
|
||||||
|
#define MAX_QUEUE_LENGTH 16U
|
||||||
|
#define NEXT_QUEUE_INDEX(index) (((index) + 1) % MAX_QUEUE_LENGTH)
|
||||||
|
#define IS_QUEUE_EMPTY() (queue_head == queue_tail)
|
||||||
|
|
||||||
|
/* Fixed-size array of SignInstance */
|
||||||
|
static ARR_Instance queue;
|
||||||
|
static unsigned int queue_head;
|
||||||
|
static unsigned int queue_tail;
|
||||||
|
|
||||||
|
#define INVALID_SOCK_FD (-6)
|
||||||
|
|
||||||
|
/* Unix domain socket connected to ntp_signd */
|
||||||
|
static int sock_fd;
|
||||||
|
|
||||||
|
/* Flag indicating if the MS-SNTP authentication is enabled */
|
||||||
|
static int enabled;
|
||||||
|
|
||||||
|
/* Flag limiting logging of connection error messages */
|
||||||
|
static int logged_connection_error;
|
||||||
|
|
||||||
|
/* ================================================== */
|
||||||
|
|
||||||
|
static void read_write_socket(int sock_fd, int event, void *anything);
|
||||||
|
|
||||||
|
/* ================================================== */
|
||||||
|
|
||||||
|
static void
|
||||||
|
close_socket(void)
|
||||||
|
{
|
||||||
|
SCH_RemoveFileHandler(sock_fd);
|
||||||
|
SCK_CloseSocket(sock_fd);
|
||||||
|
sock_fd = INVALID_SOCK_FD;
|
||||||
|
|
||||||
|
/* Empty the queue */
|
||||||
|
queue_head = queue_tail = 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
/* ================================================== */
|
||||||
|
|
||||||
|
static int
|
||||||
|
open_socket(void)
|
||||||
|
{
|
||||||
|
char path[PATH_MAX];
|
||||||
|
|
||||||
|
if (sock_fd != INVALID_SOCK_FD)
|
||||||
|
return 1;
|
||||||
|
|
||||||
|
if (snprintf(path, sizeof (path), "%s/socket", CNF_GetNtpSigndSocket()) >= sizeof (path)) {
|
||||||
|
DEBUG_LOG("signd socket path too long");
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
sock_fd = SCK_OpenUnixStreamSocket(path, NULL, 0);
|
||||||
|
if (sock_fd < 0) {
|
||||||
|
sock_fd = INVALID_SOCK_FD;
|
||||||
|
|
||||||
|
/* Log an error only once before a successful exchange to avoid
|
||||||
|
flooding the system log */
|
||||||
|
if (!logged_connection_error) {
|
||||||
|
LOG(LOGS_ERR, "Could not connect to signd socket %s : %s", path, strerror(errno));
|
||||||
|
logged_connection_error = 1;
|
||||||
|
}
|
||||||
|
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
SCH_AddFileHandler(sock_fd, SCH_FILE_INPUT, read_write_socket, NULL);
|
||||||
|
|
||||||
|
return 1;
|
||||||
|
}
|
||||||
|
|
||||||
|
/* ================================================== */
|
||||||
|
|
||||||
|
static void
|
||||||
|
process_response(SignInstance *inst)
|
||||||
|
{
|
||||||
|
struct timespec ts;
|
||||||
|
double delay;
|
||||||
|
|
||||||
|
if (ntohs(inst->request.packet_id) != ntohl(inst->response.packet_id)) {
|
||||||
|
DEBUG_LOG("Invalid response ID");
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (ntohl(inst->response.op) != SIGNING_SUCCESS) {
|
||||||
|
DEBUG_LOG("Signing failed");
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
logged_connection_error = 0;
|
||||||
|
|
||||||
|
/* Check if the file descriptor is still valid */
|
||||||
|
if (!NIO_IsServerSocket(inst->local_addr.sock_fd)) {
|
||||||
|
DEBUG_LOG("Invalid NTP socket");
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
SCH_GetLastEventTime(NULL, NULL, &ts);
|
||||||
|
delay = UTI_DiffTimespecsToDouble(&ts, &inst->request_ts);
|
||||||
|
|
||||||
|
DEBUG_LOG("Signing succeeded (delay %f)", delay);
|
||||||
|
|
||||||
|
/* Send the signed NTP packet */
|
||||||
|
NIO_SendPacket(&inst->response.signed_packet, &inst->remote_addr, &inst->local_addr,
|
||||||
|
ntohl(inst->response.length) + sizeof (inst->response.length) -
|
||||||
|
offsetof(SigndResponse, signed_packet), 0);
|
||||||
|
}
|
||||||
|
|
||||||
|
/* ================================================== */
|
||||||
|
|
||||||
|
static void
|
||||||
|
read_write_socket(int sock_fd, int event, void *anything)
|
||||||
|
{
|
||||||
|
SignInstance *inst;
|
||||||
|
uint32_t response_length;
|
||||||
|
int s;
|
||||||
|
|
||||||
|
inst = ARR_GetElement(queue, queue_head);
|
||||||
|
|
||||||
|
if (event == SCH_FILE_OUTPUT) {
|
||||||
|
assert(!IS_QUEUE_EMPTY());
|
||||||
|
assert(inst->sent < inst->request_length);
|
||||||
|
|
||||||
|
if (!inst->sent)
|
||||||
|
SCH_GetLastEventTime(NULL, NULL, &inst->request_ts);
|
||||||
|
|
||||||
|
s = SCK_Send(sock_fd, (char *)&inst->request + inst->sent,
|
||||||
|
inst->request_length - inst->sent, 0);
|
||||||
|
|
||||||
|
if (s < 0) {
|
||||||
|
close_socket();
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
inst->sent += s;
|
||||||
|
|
||||||
|
/* Try again later if the request is not complete yet */
|
||||||
|
if (inst->sent < inst->request_length)
|
||||||
|
return;
|
||||||
|
|
||||||
|
/* Disable output and wait for a response */
|
||||||
|
SCH_SetFileHandlerEvent(sock_fd, SCH_FILE_OUTPUT, 0);
|
||||||
|
}
|
||||||
|
|
||||||
|
if (event == SCH_FILE_INPUT) {
|
||||||
|
if (IS_QUEUE_EMPTY()) {
|
||||||
|
DEBUG_LOG("Unexpected signd response");
|
||||||
|
close_socket();
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
assert(inst->received < sizeof (inst->response));
|
||||||
|
s = SCK_Receive(sock_fd, (char *)&inst->response + inst->received,
|
||||||
|
sizeof (inst->response) - inst->received, 0);
|
||||||
|
|
||||||
|
if (s <= 0) {
|
||||||
|
close_socket();
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
inst->received += s;
|
||||||
|
|
||||||
|
if (inst->received < sizeof (inst->response.length))
|
||||||
|
return;
|
||||||
|
|
||||||
|
response_length = ntohl(inst->response.length) + sizeof (inst->response.length);
|
||||||
|
|
||||||
|
if (response_length < offsetof(SigndResponse, signed_packet) ||
|
||||||
|
response_length > sizeof (SigndResponse)) {
|
||||||
|
DEBUG_LOG("Invalid response length");
|
||||||
|
close_socket();
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
/* Wait for more data if not complete yet */
|
||||||
|
if (inst->received < response_length)
|
||||||
|
return;
|
||||||
|
|
||||||
|
process_response(inst);
|
||||||
|
|
||||||
|
/* Move the head and enable output for the next packet */
|
||||||
|
queue_head = NEXT_QUEUE_INDEX(queue_head);
|
||||||
|
if (!IS_QUEUE_EMPTY())
|
||||||
|
SCH_SetFileHandlerEvent(sock_fd, SCH_FILE_OUTPUT, 1);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/* ================================================== */
|
||||||
|
|
||||||
|
void
|
||||||
|
NSD_Initialise()
|
||||||
|
{
|
||||||
|
sock_fd = INVALID_SOCK_FD;
|
||||||
|
enabled = CNF_GetNtpSigndSocket() && CNF_GetNtpSigndSocket()[0];
|
||||||
|
|
||||||
|
if (!enabled)
|
||||||
|
return;
|
||||||
|
|
||||||
|
queue = ARR_CreateInstance(sizeof (SignInstance));
|
||||||
|
ARR_SetSize(queue, MAX_QUEUE_LENGTH);
|
||||||
|
queue_head = queue_tail = 0;
|
||||||
|
|
||||||
|
LOG(LOGS_INFO, "MS-SNTP authentication enabled");
|
||||||
|
}
|
||||||
|
|
||||||
|
/* ================================================== */
|
||||||
|
|
||||||
|
void
|
||||||
|
NSD_Finalise()
|
||||||
|
{
|
||||||
|
if (!enabled)
|
||||||
|
return;
|
||||||
|
if (sock_fd != INVALID_SOCK_FD)
|
||||||
|
close_socket();
|
||||||
|
ARR_DestroyInstance(queue);
|
||||||
|
}
|
||||||
|
|
||||||
|
/* ================================================== */
|
||||||
|
|
||||||
|
int
|
||||||
|
NSD_SignAndSendPacket(uint32_t key_id, NTP_Packet *packet, NTP_PacketInfo *info,
|
||||||
|
NTP_Remote_Address *remote_addr, NTP_Local_Address *local_addr)
|
||||||
|
{
|
||||||
|
SignInstance *inst;
|
||||||
|
|
||||||
|
if (!enabled) {
|
||||||
|
DEBUG_LOG("signd disabled");
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (queue_head == NEXT_QUEUE_INDEX(queue_tail)) {
|
||||||
|
DEBUG_LOG("signd queue full");
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (info->length != NTP_HEADER_LENGTH) {
|
||||||
|
DEBUG_LOG("Invalid packet length");
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (!open_socket())
|
||||||
|
return 0;
|
||||||
|
|
||||||
|
inst = ARR_GetElement(queue, queue_tail);
|
||||||
|
inst->remote_addr = *remote_addr;
|
||||||
|
inst->local_addr = *local_addr;
|
||||||
|
inst->sent = 0;
|
||||||
|
inst->received = 0;
|
||||||
|
inst->request_length = offsetof(SigndRequest, packet_to_sign) + info->length;
|
||||||
|
|
||||||
|
/* The length field doesn't include itself */
|
||||||
|
inst->request.length = htonl(inst->request_length - sizeof (inst->request.length));
|
||||||
|
inst->request.version = htonl(SIGND_VERSION);
|
||||||
|
inst->request.op = htonl(SIGN_TO_CLIENT);
|
||||||
|
inst->request.packet_id = htons(queue_tail);
|
||||||
|
inst->request._pad = 0;
|
||||||
|
inst->request.key_id = htonl(key_id);
|
||||||
|
|
||||||
|
memcpy(&inst->request.packet_to_sign, packet, info->length);
|
||||||
|
|
||||||
|
/* Enable output if there was no pending request */
|
||||||
|
if (IS_QUEUE_EMPTY())
|
||||||
|
SCH_SetFileHandlerEvent(sock_fd, SCH_FILE_OUTPUT, 1);
|
||||||
|
|
||||||
|
queue_tail = NEXT_QUEUE_INDEX(queue_tail);
|
||||||
|
|
||||||
|
DEBUG_LOG("Packet added to signd queue (%u:%u)", queue_head, queue_tail);
|
||||||
|
|
||||||
|
return 1;
|
||||||
|
}
|
42
ntp_signd.h
Normal file
42
ntp_signd.h
Normal file
|
@ -0,0 +1,42 @@
|
||||||
|
/*
|
||||||
|
chronyd/chronyc - Programs for keeping computer clocks accurate.
|
||||||
|
|
||||||
|
**********************************************************************
|
||||||
|
* Copyright (C) Miroslav Lichvar 2016
|
||||||
|
*
|
||||||
|
* This program is free software; you can redistribute it and/or modify
|
||||||
|
* it under the terms of version 2 of the GNU General Public License as
|
||||||
|
* published by the Free Software Foundation.
|
||||||
|
*
|
||||||
|
* This program is distributed in the hope that it will be useful, but
|
||||||
|
* WITHOUT ANY WARRANTY; without even the implied warranty of
|
||||||
|
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
|
||||||
|
* General Public License for more details.
|
||||||
|
*
|
||||||
|
* You should have received a copy of the GNU General Public License along
|
||||||
|
* with this program; if not, write to the Free Software Foundation, Inc.,
|
||||||
|
* 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA.
|
||||||
|
*
|
||||||
|
**********************************************************************
|
||||||
|
|
||||||
|
=======================================================================
|
||||||
|
|
||||||
|
Header for MS-SNTP authentication via Samba (ntp_signd) */
|
||||||
|
|
||||||
|
#ifndef GOT_NTP_SIGND_H
|
||||||
|
#define GOT_NTP_SIGND_H
|
||||||
|
|
||||||
|
#include "addressing.h"
|
||||||
|
#include "ntp.h"
|
||||||
|
|
||||||
|
/* Initialisation function */
|
||||||
|
extern void NSD_Initialise(void);
|
||||||
|
|
||||||
|
/* Finalisation function */
|
||||||
|
extern void NSD_Finalise(void);
|
||||||
|
|
||||||
|
/* Function to sign an NTP packet and send it */
|
||||||
|
extern int NSD_SignAndSendPacket(uint32_t key_id, NTP_Packet *packet, NTP_PacketInfo *info,
|
||||||
|
NTP_Remote_Address *remote_addr, NTP_Local_Address *local_addr);
|
||||||
|
|
||||||
|
#endif
|
Some files were not shown because too many files have changed in this diff Show more
Loading…
Add table
Add a link
Reference in a new issue