diff -Nru qtcreator-2.5.0/README qtcreator-2.5.2/README --- qtcreator-2.5.0/README 2012-05-09 12:13:19.000000000 +0000 +++ qtcreator-2.5.2/README 2012-08-08 13:47:06.000000000 +0000 @@ -248,28 +248,6 @@ QtCreator/src/libs/3rdparty -* NetSieben SSH Library is a Secure Shell client library for C++. Version 1.3.2 - - Commercial License: For organizations who do not want to release the source - code for their applications as open source/ free software; in other words - they do not want to comply with the GNU General Public License (GPL) or Q - Public License. - - Non Commercial / Open Source License: NetSieben believes in contributing back - to the open source community, thus it has released the SSH Library under Q - Public License as it is defined by Trolltech AS of Norway. The Open Source - License allows the user to use software under an open source / free software - license, and distribute it freely. The software can be used at no charge with - the condition that if the user uses the SSH Library in an application they - wish to redistribute, then the complete source code for your application must - be available and freely redistributable under reasonable conditions. For more - information on the used QPL License see: - QtCreator/src/libs/3rdparty/net7ssh/LICENSE.QPL - - The source code of NetSieben Secure Shell C++ Library can be found in - QtCreator/src/libs/3rdparty. - - * ClassView and ImageViewer plugins Copyright (C) 2012 Nokia Corporation and/or its subsidiary(-ies). diff -Nru qtcreator-2.5.0/debian/changelog qtcreator-2.5.2/debian/changelog --- qtcreator-2.5.0/debian/changelog 2012-06-28 13:13:39.000000000 +0000 +++ qtcreator-2.5.2/debian/changelog 2013-04-28 17:51:00.000000000 +0000 @@ -1,8 +1,42 @@ -qtcreator (2.5.0-0ubuntu1~precise1) precise-backports; urgency=low +qtcreator (2.5.2-0ubuntu1~ubuntu12.04.1) precise-backports; urgency=low - * No-change backport to precise + * No-change backport to precise (LP: #1039436) - -- Felix Geyer Thu, 28 Jun 2012 15:13:39 +0200 + -- Felix Geyer Sun, 28 Apr 2013 19:51:00 +0200 + +qtcreator (2.5.2-0ubuntu1) quantal; urgency=low + + * New upstream release. + * Re-apply Ubuntu delta that has been inadvertently dropped in the + last upload: + - Build qtcreator out-of-source. + - Don't call dh_makeshlibs as qtcreator doesn't provide any public + shared libraries. + - Compress binary packages with xz. + + -- Felix Geyer Fri, 17 Aug 2012 20:09:31 +0200 + +qtcreator (2.5.0-1ubuntu1) quantal; urgency=low + + * Merge with Debian unstable, remaining changes: + - Recommend, not Suggest g++. + - Keep 04_vcs_log_fix_cursor.diff + + -- Jonathan Thomas Fri, 08 Jun 2012 12:44:29 -0400 + +qtcreator (2.5.0-1) unstable; urgency=low + + * New upstream release. + + [ Pino Toscano ] + * Fix the installation of the upstream changelogs: use dh_installdocs + for them, and install them only in qtcreator. + * Build the documentation only when qtcreator-doc is being built too. + * Remove unuseful ${shlibs:Depends} from qtcreator-dbg. + * Enable RPATH also on kFreeBSD and Hurd; patch rpath_nonlinux.diff. + (Closes: #672527) + + -- Fathi Boudra Thu, 10 May 2012 21:57:24 +0300 qtcreator (2.5.0-0ubuntu1) quantal; urgency=low @@ -12,6 +46,20 @@ -- Felix Geyer Wed, 09 May 2012 18:31:04 +0200 +qtcreator (2.5.0~rc-1) unstable; urgency=low + + * New upstream release. + * Add patches, cheery-picked upstream: + - Use_bzr_branch_instead_of_bzr_clone.patch + - Fix_revno_detection_in_bzr_log.patch + * Update debian/control: + - bump debhelper build dependency to 9. + - bump build dependencies to Qt >= 4.8.1. + - bump Standards-Version to 3.9.3 (no changes needed). + * Update installed files and adjust paths for multiarch. + + -- Fathi Boudra Wed, 02 May 2012 09:47:02 +0300 + qtcreator (2.4.1-0ubuntu2) precise; urgency=low [ Aurélien Gâteau ] @@ -32,6 +80,41 @@ -- Felix Geyer Thu, 02 Feb 2012 12:08:40 +0100 +qtcreator (2.4.0-1) experimental; urgency=low + + * New upstream release (Closes: #653634): + - FTBFS: error: no matching function for call to 'qMin(double&, qreal)'. + (Closes: #638813) + * Drop patches: + - install_application_icons_according_to_freedesktop_spec.diff + merged upstream. + - 01_fix_installation_paths.diff + use INSTALL_ROOT=$(CURDIR)/debian/tmp/usr to avoid to refresh this patch + continously. + - 04_append_Debian_search_path_for_pre-built_gdbmacros.diff + we don't ship pre-built gdbmacros anymore. + * Update debian/compat: bump to 9 for multiarch support. + * Update debian/control: + - bump build dependencies to Qt >= 4.8.0, version with multiarch support. + - fix Homepage. (Closes: 627890) + - add qt4-qmlviewer to Recommends. + - add g++ to Suggests. (Closes: #649212) + - suggests git instead of git-core (transitional package). + * Update debian/rules: + - remove QMAKE export using qmake_qt4 buildsystem (requires + debhelper >= 8.9.1) + - remove pre-built gdbmacros to simplify multiarch support. + - install the documentation, not handled by make install. + - do not remove bin/qtcreator.sh in override_dh_auto_clean target. + - drop override_dh_{makeshlibs,shlibdeps}, Qt Creator provides private + libraries only. + * Add missing binaries: usr/lib/qmldesigner/*.so and usr/bin/qmlpuppet.* + (Closes: #618339, #633915) + * Enable multiarch support and update debian/*.install files accordingly. + * Fix debian/watch file: download the tarball instead of zip archive. + + -- Fathi Boudra Thu, 29 Dec 2011 22:11:49 +0200 + qtcreator (2.4.0-0ubuntu1) precise; urgency=low * New upstream release. diff -Nru qtcreator-2.5.0/debian/control qtcreator-2.5.2/debian/control --- qtcreator-2.5.0/debian/control 2012-05-09 14:16:02.000000000 +0000 +++ qtcreator-2.5.2/debian/control 2012-08-17 18:14:29.000000000 +0000 @@ -7,12 +7,13 @@ Fathi Boudra , Jeremy Lainé Build-Depends: debhelper (>= 9), - libqt4-dev (>= 4:4.7.1), - libqt4-private-dev (>= 4:4.7.3-3), + libicu-dev, + libqt4-dev (>= 4:4.8.1), + libqt4-private-dev (>= 4:4.8.1), libqtwebkit-dev (>= 2.1.0~2011week13-2), - qt4-dev-tools (>= 4:4.7.1) -Standards-Version: 3.9.2 -Homepage: https://qt.nokia.com/products/developer-tools + qt4-dev-tools (>= 4:4.8.1) +Standards-Version: 3.9.3 +Homepage: http://qt.nokia.com/products/developer-tools/ Vcs-Browser: http://git.debian.org/?p=collab-maint/qt-creator.git Vcs-Git: git://git.debian.org/collab-maint/qt-creator.git @@ -49,7 +50,7 @@ Architecture: any Section: debug Pre-Depends: dpkg (>= 1.15.6) -Depends: qtcreator (= ${binary:Version}), ${misc:Depends}, ${shlibs:Depends} +Depends: qtcreator (= ${binary:Version}), ${misc:Depends} Description: debugging symbols for Qt Creator IDE Qt Creator is a new, lightweight, cross-platform integrated development environment (IDE) designed to make development with the Qt application diff -Nru qtcreator-2.5.0/debian/not-installed qtcreator-2.5.2/debian/not-installed --- qtcreator-2.5.0/debian/not-installed 1970-01-01 00:00:00.000000000 +0000 +++ qtcreator-2.5.2/debian/not-installed 2012-06-08 16:41:26.000000000 +0000 @@ -0,0 +1,7 @@ +# Installed through qtcreator.install in usr/lib/qtcreator +./usr/share/qtcreator/gdbmacros/Makefile +./usr/share/qtcreator/gdbmacros/gdbmacros.o +./usr/share/qtcreator/gdbmacros/libgdbmacros.so +./usr/share/qtcreator/gdbmacros/libgdbmacros.so.1 +./usr/share/qtcreator/gdbmacros/libgdbmacros.so.1.0 +./usr/share/qtcreator/gdbmacros/libgdbmacros.so.1.0.0 diff -Nru qtcreator-2.5.0/debian/patches/02_use_x-terminal-emulator.diff qtcreator-2.5.2/debian/patches/02_use_x-terminal-emulator.diff --- qtcreator-2.5.0/debian/patches/02_use_x-terminal-emulator.diff 2011-10-21 08:53:24.000000000 +0000 +++ qtcreator-2.5.2/debian/patches/02_use_x-terminal-emulator.diff 2012-06-08 16:22:40.000000000 +0000 @@ -1,9 +1,13 @@ Description: use x-terminal-emulator on Debian based distributions Author: Fathi Boudra +--- + src/libs/utils/consoleprocess_unix.cpp | 2 +- + 1 file changed, 1 insertion(+), 1 deletion(-) + --- a/src/libs/utils/consoleprocess_unix.cpp +++ b/src/libs/utils/consoleprocess_unix.cpp -@@ -336,7 +336,7 @@ QString ConsoleProcess::defaultTerminalE +@@ -284,7 +284,7 @@ QString ConsoleProcess::defaultTerminalE #ifdef Q_OS_MAC return QLatin1String("/usr/X11/bin/xterm"); #else diff -Nru qtcreator-2.5.0/debian/patches/03_fix_DOCPATH.diff qtcreator-2.5.2/debian/patches/03_fix_DOCPATH.diff --- qtcreator-2.5.0/debian/patches/03_fix_DOCPATH.diff 2011-10-21 08:54:25.000000000 +0000 +++ qtcreator-2.5.2/debian/patches/03_fix_DOCPATH.diff 2012-06-08 16:22:50.000000000 +0000 @@ -1,9 +1,13 @@ Description: fix DOCPATH for Debian based distributions Author: Fathi Boudra +--- + src/plugins/help/helpplugin.cpp | 2 +- + 1 file changed, 1 insertion(+), 1 deletion(-) + --- a/src/plugins/help/helpplugin.cpp +++ b/src/plugins/help/helpplugin.cpp -@@ -110,7 +110,7 @@ const char * const SB_OPENPAGES = "OpenP +@@ -109,7 +109,7 @@ const char SB_OPENPAGES[] = "OpenPages"; #if defined(Q_OS_MAC) # define DOCPATH "/../Resources/doc/" #else diff -Nru qtcreator-2.5.0/debian/patches/05_use_bzr_branch.diff qtcreator-2.5.2/debian/patches/05_use_bzr_branch.diff --- qtcreator-2.5.0/debian/patches/05_use_bzr_branch.diff 2012-05-09 18:42:07.000000000 +0000 +++ qtcreator-2.5.2/debian/patches/05_use_bzr_branch.diff 1970-01-01 00:00:00.000000000 +0000 @@ -1,42 +0,0 @@ -From: Aurélien Gâteau -Date: Thu, 26 Jan 2012 14:12:41 +0100 -Subject: [PATCH] Use "bzr branch" instead of "bzr clone" as CloneCommand - -"bzr clone" is deprecated, using it causes a warning message to appear in the -import dialog. - -bzr-revid: aurelien.gateau@canonical.com-20120126131241-bbvsyc218glaefc2 - -diff --git a/src/plugins/bazaar/bazaarclient.h b/src/plugins/bazaar/bazaarclient.h -index 832757f..b3a2411 100644 ---- a/src/plugins/bazaar/bazaarclient.h -+++ b/src/plugins/bazaar/bazaarclient.h -@@ -64,6 +64,7 @@ - - protected: - QString vcsEditorKind(VcsCommand cmd) const; -+ QString vcsCommandString(VcsCommand cmd) const; - QStringList revisionSpec(const QString &revision) const; - VcsBase::VcsBaseEditorParameterWidget *createDiffEditor(const QString &workingDir, - const QStringList &files, -diff --git a/src/plugins/bazaar/bazaarclient.cpp b/src/plugins/bazaar/bazaarclient.cpp -index 1e4f8f9..3821dac 100644 ---- a/src/plugins/bazaar/bazaarclient.cpp -+++ b/src/plugins/bazaar/bazaarclient.cpp -@@ -138,6 +138,16 @@ - return QLatin1String(Constants::FILELOG); - default: - return QString(); -+ } -+} -+ -+QString BazaarClient::vcsCommandString(VcsCommand cmd) const -+{ -+ switch(cmd) { -+ case CloneCommand: -+ return QLatin1String("branch"); -+ default: -+ return VcsBaseClient::vcsCommandString(cmd); - } - } - diff -Nru qtcreator-2.5.0/debian/patches/06_bzr_log_no_false_positive.diff qtcreator-2.5.2/debian/patches/06_bzr_log_no_false_positive.diff --- qtcreator-2.5.0/debian/patches/06_bzr_log_no_false_positive.diff 2012-05-09 15:02:22.000000000 +0000 +++ qtcreator-2.5.2/debian/patches/06_bzr_log_no_false_positive.diff 1970-01-01 00:00:00.000000000 +0000 @@ -1,99 +0,0 @@ -Description: Fix revno detection in bzr log - Avoid false-positives when looking for changes under cursor in log view. -Forwarded: http://codereview.qt-project.org/#change,20196 -Author: Aurélien Gâteau -Applied-Upstream: acd49da4925103dc83bd2d8053d915d81cf26492 -Last-Update: 2012-03-21 - -Index: qtcreator-2.4.1/src/plugins/bazaar/annotationhighlighter.cpp -=================================================================== ---- qtcreator-2.4.1.orig/src/plugins/bazaar/annotationhighlighter.cpp 2012-01-24 15:37:13.000000000 +0100 -+++ qtcreator-2.4.1/src/plugins/bazaar/annotationhighlighter.cpp 2012-03-21 14:04:14.000000000 +0100 -@@ -39,7 +39,7 @@ - const QColor &bg, - QTextDocument *document) - : VcsBase::BaseAnnotationHighlighter(changeNumbers, bg, document), -- m_changeset(QLatin1String(Constants::CHANGESET_ID)) -+ m_changeset(QLatin1String(Constants::ANNOTATE_CHANGESET_ID)) - { - } - -Index: qtcreator-2.4.1/src/plugins/bazaar/bazaareditor.cpp -=================================================================== ---- qtcreator-2.4.1.orig/src/plugins/bazaar/bazaareditor.cpp 2012-03-21 13:57:23.000000000 +0100 -+++ qtcreator-2.4.1/src/plugins/bazaar/bazaareditor.cpp 2012-03-21 14:03:50.000000000 +0100 -@@ -55,6 +55,7 @@ - - BazaarEditor::BazaarEditor(const VcsBase::VcsBaseEditorParameters *type, QWidget *parent) - : VcsBase::VcsBaseEditorWidget(type, parent), -+ m_changesetId(QLatin1String(Constants::CHANGESET_ID)), - m_exactChangesetId(QLatin1String(Constants::CHANGESET_ID_EXACT)), - m_diffFileId(QLatin1String("^=== [a-z]+ [a-z]+ '(.*)'\\s*")) - { -@@ -86,12 +87,30 @@ - - QString BazaarEditor::changeUnderCursor(const QTextCursor &cursorIn) const - { -+ // The test is done in two steps: first we check if the line contains a -+ // changesetId. Then we check if the cursor is over the changesetId itself -+ // and not over "revno" or another part of the line. -+ // The two steps are necessary because matching only for the changesetId -+ // leads to many false-positives (a regex like "[0-9]+" matches a lot of text). -+ const int cursorCol = cursorIn.columnNumber(); - QTextCursor cursor = cursorIn; -- cursor.select(QTextCursor::WordUnderCursor); -+ cursor.select(QTextCursor::LineUnderCursor); - if (cursor.hasSelection()) { -- const QString change = cursor.selectedText(); -- if (m_exactChangesetId.exactMatch(change)) -- return change; -+ const QString line = cursor.selectedText(); -+ const int start = m_changesetId.indexIn(line); -+ if (start > -1) { -+ const QString match = m_changesetId.cap(0); -+ const int stop = start + match.length(); -+ if (start <= cursorCol && cursorCol <= stop) { -+ cursor = cursorIn; -+ cursor.select(QTextCursor::WordUnderCursor); -+ if (cursor.hasSelection()) { -+ const QString change = cursor.selectedText(); -+ if (m_exactChangesetId.exactMatch(change)) -+ return change; -+ } -+ } -+ } - } - return QString(); - } -Index: qtcreator-2.4.1/src/plugins/bazaar/bazaareditor.h -=================================================================== ---- qtcreator-2.4.1.orig/src/plugins/bazaar/bazaareditor.h 2012-03-21 13:57:23.000000000 +0100 -+++ qtcreator-2.4.1/src/plugins/bazaar/bazaareditor.h 2012-03-21 14:00:12.000000000 +0100 -@@ -54,6 +54,7 @@ - virtual VcsBase::BaseAnnotationHighlighter *createAnnotationHighlighter(const QSet &changes, const QColor &bg) const; - virtual QString fileNameFromDiffSpecification(const QTextBlock &diffFileSpec) const; - -+ const QRegExp m_changesetId; - const QRegExp m_exactChangesetId; - const QRegExp m_diffFileId; - }; -Index: qtcreator-2.4.1/src/plugins/bazaar/constants.h -=================================================================== ---- qtcreator-2.4.1.orig/src/plugins/bazaar/constants.h 2012-03-21 13:57:23.000000000 +0100 -+++ qtcreator-2.4.1/src/plugins/bazaar/constants.h 2012-03-21 15:21:27.000000000 +0100 -@@ -41,8 +41,13 @@ - const char BAZAARDEFAULT[] = "bzr"; - - // Changeset identifiers --const char CHANGESET_ID[] = "([0-9]+)"; // match and capture --const char CHANGESET_ID_EXACT[] = "[0-9]+"; // match -+const char * const CHANGESET_ID = "^(" -+ "revno: [.0-9]+" // detailed -+ "| +[.0-9]+" // short -+ "|[.0-9]+: " // line -+ ")"; -+const char * const CHANGESET_ID_EXACT = "([.0-9]+)"; -+const char * const ANNOTATE_CHANGESET_ID = "([.0-9]+)"; - - // Base editor parameters - const char COMMANDLOG_ID[] = "Bazaar Command Log Editor"; diff -Nru qtcreator-2.5.0/debian/patches/Fix_revno_detection_in_bzr_log.patch qtcreator-2.5.2/debian/patches/Fix_revno_detection_in_bzr_log.patch --- qtcreator-2.5.0/debian/patches/Fix_revno_detection_in_bzr_log.patch 1970-01-01 00:00:00.000000000 +0000 +++ qtcreator-2.5.2/debian/patches/Fix_revno_detection_in_bzr_log.patch 2012-06-08 16:24:08.000000000 +0000 @@ -0,0 +1,102 @@ +commit acd49da4925103dc83bd2d8053d915d81cf26492 +Author: Aurélien Gâteau +Date: Thu Jan 26 14:18:59 2012 +0100 + + Fix revno detection in bzr log + + Avoid false-positives when looking for changes under cursor in log view. + + Change-Id: Ic0f9c105747ee2c1792b372f9cb3757017633861 + Reviewed-by: Hugues Delorme + +--- + src/plugins/bazaar/annotationhighlighter.cpp | 2 +- + src/plugins/bazaar/bazaareditor.cpp | 27 +++++++++++++++++++++++---- + src/plugins/bazaar/bazaareditor.h | 1 + + src/plugins/bazaar/constants.h | 9 +++++++-- + 4 files changed, 32 insertions(+), 7 deletions(-) + +--- a/src/plugins/bazaar/annotationhighlighter.cpp ++++ b/src/plugins/bazaar/annotationhighlighter.cpp +@@ -40,7 +40,7 @@ BazaarAnnotationHighlighter::BazaarAnnot + const QColor &bg, + QTextDocument *document) + : VcsBase::BaseAnnotationHighlighter(changeNumbers, bg, document), +- m_changeset(QLatin1String(Constants::CHANGESET_ID)) ++ m_changeset(QLatin1String(Constants::ANNOTATE_CHANGESET_ID)) + { + } + +--- a/src/plugins/bazaar/bazaareditor.cpp ++++ b/src/plugins/bazaar/bazaareditor.cpp +@@ -55,6 +55,7 @@ using namespace Bazaar; + + BazaarEditor::BazaarEditor(const VcsBase::VcsBaseEditorParameters *type, QWidget *parent) + : VcsBase::VcsBaseEditorWidget(type, parent), ++ m_changesetId(QLatin1String(Constants::CHANGESET_ID)), + m_exactChangesetId(QLatin1String(Constants::CHANGESET_ID_EXACT)), + m_diffFileId(QLatin1String("^=== [a-z]+ [a-z]+ '(.*)'\\s*")) + { +@@ -86,12 +87,30 @@ QSet BazaarEditor::annotationCh + + QString BazaarEditor::changeUnderCursor(const QTextCursor &cursorIn) const + { ++ // The test is done in two steps: first we check if the line contains a ++ // changesetId. Then we check if the cursor is over the changesetId itself ++ // and not over "revno" or another part of the line. ++ // The two steps are necessary because matching only for the changesetId ++ // leads to many false-positives (a regex like "[0-9]+" matches a lot of text). ++ const int cursorCol = cursorIn.columnNumber(); + QTextCursor cursor = cursorIn; +- cursor.select(QTextCursor::WordUnderCursor); ++ cursor.select(QTextCursor::LineUnderCursor); + if (cursor.hasSelection()) { +- const QString change = cursor.selectedText(); +- if (m_exactChangesetId.exactMatch(change)) +- return change; ++ const QString line = cursor.selectedText(); ++ const int start = m_changesetId.indexIn(line); ++ if (start > -1) { ++ const QString match = m_changesetId.cap(0); ++ const int stop = start + match.length(); ++ if (start <= cursorCol && cursorCol <= stop) { ++ cursor = cursorIn; ++ cursor.select(QTextCursor::WordUnderCursor); ++ if (cursor.hasSelection()) { ++ const QString change = cursor.selectedText(); ++ if (m_exactChangesetId.exactMatch(change)) ++ return change; ++ } ++ } ++ } + } + return QString(); + } +--- a/src/plugins/bazaar/bazaareditor.h ++++ b/src/plugins/bazaar/bazaareditor.h +@@ -54,6 +54,7 @@ private: + virtual VcsBase::BaseAnnotationHighlighter *createAnnotationHighlighter(const QSet &changes, const QColor &bg) const; + virtual QString fileNameFromDiffSpecification(const QTextBlock &diffFileSpec) const; + ++ const QRegExp m_changesetId; + const QRegExp m_exactChangesetId; + const QRegExp m_diffFileId; + }; +--- a/src/plugins/bazaar/constants.h ++++ b/src/plugins/bazaar/constants.h +@@ -41,8 +41,13 @@ const char BAZAARREPO[] = ".bzr"; + const char BAZAARDEFAULT[] = "bzr"; + + // Changeset identifiers +-const char CHANGESET_ID[] = "([0-9]+)"; // match and capture +-const char CHANGESET_ID_EXACT[] = "[0-9]+"; // match ++const char CHANGESET_ID[] = "^(" ++ "revno: [.0-9]+" // detailed ++ "| +[.0-9]+" // short ++ "|[.0-9]+: " // line ++ ")"; ++const char CHANGESET_ID_EXACT[] = "([.0-9]+)"; ++const char ANNOTATE_CHANGESET_ID[] = "([.0-9]+)"; + + // Base editor parameters + const char COMMANDLOG_ID[] = "Bazaar Command Log Editor"; diff -Nru qtcreator-2.5.0/debian/patches/Use_bzr_branch_instead_of_bzr_clone.patch qtcreator-2.5.2/debian/patches/Use_bzr_branch_instead_of_bzr_clone.patch --- qtcreator-2.5.0/debian/patches/Use_bzr_branch_instead_of_bzr_clone.patch 1970-01-01 00:00:00.000000000 +0000 +++ qtcreator-2.5.2/debian/patches/Use_bzr_branch_instead_of_bzr_clone.patch 2012-06-08 16:33:16.000000000 +0000 @@ -0,0 +1,45 @@ +commit 4790641934fea76a8930c4157ceb54a6536cdd57 +Author: Aurélien Gâteau +Date: Wed Feb 1 11:24:54 2012 +0100 + + Use "bzr branch" instead of "bzr clone" + + "bzr clone" is deprecated. "bzr branch" does the same thing. + + Change-Id: Iafed249ff213fbd2734a900cb410fa7f23f3e84f + Reviewed-by: Hugues Delorme + +--- + src/plugins/bazaar/bazaarclient.cpp | 10 ++++++++++ + src/plugins/bazaar/bazaarclient.h | 1 + + 2 files changed, 11 insertions(+) + +--- a/src/plugins/bazaar/bazaarclient.cpp ++++ b/src/plugins/bazaar/bazaarclient.cpp +@@ -140,6 +140,16 @@ QString BazaarClient::vcsEditorKind(VcsC + } + } + ++QString BazaarClient::vcsCommandString(VcsCommand cmd) const ++{ ++ switch (cmd) { ++ case CloneCommand: ++ return QLatin1String("branch"); ++ default: ++ return VcsBaseClient::vcsCommandString(cmd); ++ } ++} ++ + QStringList BazaarClient::revisionSpec(const QString &revision) const + { + QStringList args; +--- a/src/plugins/bazaar/bazaarclient.h ++++ b/src/plugins/bazaar/bazaarclient.h +@@ -64,6 +64,7 @@ public: + + protected: + QString vcsEditorKind(VcsCommand cmd) const; ++ QString vcsCommandString(VcsCommand cmd) const; + QStringList revisionSpec(const QString &revision) const; + VcsBase::VcsBaseEditorParameterWidget *createDiffEditor(const QString &workingDir, + const QStringList &files, diff -Nru qtcreator-2.5.0/debian/patches/rpath_nonlinux.diff qtcreator-2.5.2/debian/patches/rpath_nonlinux.diff --- qtcreator-2.5.0/debian/patches/rpath_nonlinux.diff 1970-01-01 00:00:00.000000000 +0000 +++ qtcreator-2.5.2/debian/patches/rpath_nonlinux.diff 2012-06-08 16:32:07.000000000 +0000 @@ -0,0 +1,64 @@ +Author: Pino Toscano +Description: Enable RPATH also on kFreeBSD and Hurd + qtcreator's plugins use RPATH to locate the qtcreator libraries, so enable it + also on kFreeBSD and Hurd. +Last-Update: 2012-05-19 +Forwarded: no + +--- + src/qtcreatorplugin.pri | 4 ++-- + src/rpath.pri | 2 +- + src/tools/qtcreatorwidgets/qtcreatorwidgets.pro | 4 ++-- + 3 files changed, 5 insertions(+), 5 deletions(-) + +--- a/src/rpath.pri ++++ b/src/rpath.pri +@@ -5,7 +5,7 @@ macx { + QMAKE_LFLAGS_SONAME = -Wl,-install_name,@rpath/PlugIns/ + QMAKE_LFLAGS += -Wl,-rpath,@loader_path/../,-rpath,@executable_path/../ + } +-} else:linux-* { ++} else:linux-*|glibc-*|hurd-* { + #do the rpath by hand since it's not possible to use ORIGIN in QMAKE_RPATHDIR + # this expands to $ORIGIN (after qmake and make), it does NOT read a qmake var + QMAKE_RPATHDIR += \$\$ORIGIN +--- a/src/qtcreatorplugin.pri ++++ b/src/qtcreatorplugin.pri +@@ -64,7 +64,7 @@ macx { + QMAKE_LFLAGS_SONAME = -Wl,-install_name,@rpath/PlugIns/$${PROVIDER}/ + QMAKE_LFLAGS += -Wl,-rpath,@loader_path/../../,-rpath,@executable_path/../ + } +-} else:linux-* { ++} else:linux-*|glibc-*|hurd-* { + #do the rpath by hand since it's not possible to use ORIGIN in QMAKE_RPATHDIR + QMAKE_RPATHDIR += \$\$ORIGIN + QMAKE_RPATHDIR += \$\$ORIGIN/.. +@@ -78,7 +78,7 @@ macx { + contains(QT_CONFIG, reduce_exports):CONFIG += hide_symbols + + CONFIG += plugin plugin_with_soname +-linux*:QMAKE_LFLAGS += $$QMAKE_LFLAGS_NOUNDEF ++linux*|glibc-*|hurd-*:QMAKE_LFLAGS += $$QMAKE_LFLAGS_NOUNDEF + + !macx { + target.path = /$$IDE_LIBRARY_BASENAME/qtcreator/plugins/$$PROVIDER +--- a/src/tools/qtcreatorwidgets/qtcreatorwidgets.pro ++++ b/src/tools/qtcreatorwidgets/qtcreatorwidgets.pro +@@ -13,7 +13,7 @@ isEmpty(IDE_LIBRARY_BASENAME) { + IDE_LIBRARY_BASENAME = lib + } + +-linux-*||win32 { ++linux-*||win32||glibc-*||hurd-* { + # form abs path to qtcreator lib dir + QTC_LIBS=$$dirname(OUT_PWD) + QTC_LIBS=$$dirname(QTC_LIBS) +@@ -21,7 +21,7 @@ linux-*||win32 { + QTC_LIBS=$$QTC_LIBS/$$IDE_LIBRARY_BASENAME/qtcreator + } + +-linux-*{ ++linux-*|glibc-*|hurd-*{ + QMAKE_RPATHDIR *= $$QTC_LIBS + } + diff -Nru qtcreator-2.5.0/debian/patches/series qtcreator-2.5.2/debian/patches/series --- qtcreator-2.5.0/debian/patches/series 2012-05-09 15:03:18.000000000 +0000 +++ qtcreator-2.5.2/debian/patches/series 2012-06-08 16:33:46.000000000 +0000 @@ -1,5 +1,6 @@ 02_use_x-terminal-emulator.diff 03_fix_DOCPATH.diff 04_vcs_log_fix_cursor.diff -05_use_bzr_branch.diff -06_bzr_log_no_false_positive.diff +Use_bzr_branch_instead_of_bzr_clone.patch +Fix_revno_detection_in_bzr_log.patch +rpath_nonlinux.diff diff -Nru qtcreator-2.5.0/debian/qtcreator.install qtcreator-2.5.2/debian/qtcreator.install --- qtcreator-2.5.0/debian/qtcreator.install 2012-03-21 17:30:36.000000000 +0000 +++ qtcreator-2.5.2/debian/qtcreator.install 2012-06-08 16:41:05.000000000 +0000 @@ -1,40 +1,297 @@ debian/Nokia-QtCreator.xml usr/share/mime/packages/ debian/qtcreator.desktop usr/share/applications/ -#share/qtcreator/dumper/libdumper.so* usr/lib/${DEB_HOST_MULTIARCH}/qtcreator/dumper/ +usr/bin/qmlprofiler +usr/bin/qmlpuppet usr/bin/qtcreator usr/bin/qtcreator_process_stub usr/bin/qtpromaker -usr/bin/qmlprofiler -usr/bin/qmlpuppet -usr/lib/*/qtcreator/qmldesigner -usr/lib/*/qtcreator/qtcomponents -usr/lib/*/qtcreator/*.so* -usr/lib/*/qtcreator/plugins/Nokia/*.pluginspec -usr/lib/*/qtcreator/plugins/Nokia/*.so -usr/share/icons/hicolor/*/apps/qtcreator.png -usr/share/qtcreator/designer -usr/share/qtcreator/externaltools -usr/share/qtcreator/dumper/dumper.h +usr/lib/*/qtcreator/libAggregation.* +usr/lib/*/qtcreator/libBotan.* +usr/lib/*/qtcreator/libCPlusPlus.* +usr/lib/*/qtcreator/libDebuggingHelper.* +usr/lib/*/qtcreator/libExtensionSystem.* +usr/lib/*/qtcreator/libGLSL.* +usr/lib/*/qtcreator/libLanguageUtils.* +usr/lib/*/qtcreator/libQmlEditorWidgets.* +usr/lib/*/qtcreator/libQmlJS.* +usr/lib/*/qtcreator/libQmlJSDebugClient.* +usr/lib/*/qtcreator/libUtils.* +usr/lib/*/qtcreator/libsymbianutils.* +usr/lib/*/qtcreator/libzeroconf.* +usr/lib/*/qtcreator/plugins/Nokia/AnalyzerBase.pluginspec +usr/lib/*/qtcreator/plugins/Nokia/AutotoolsProjectManager.pluginspec +usr/lib/*/qtcreator/plugins/Nokia/Bazaar.pluginspec +usr/lib/*/qtcreator/plugins/Nokia/BinEditor.pluginspec +usr/lib/*/qtcreator/plugins/Nokia/Bookmarks.pluginspec +usr/lib/*/qtcreator/plugins/Nokia/CMakeProjectManager.pluginspec +usr/lib/*/qtcreator/plugins/Nokia/CVS.pluginspec +usr/lib/*/qtcreator/plugins/Nokia/ClassView.pluginspec +usr/lib/*/qtcreator/plugins/Nokia/CodePaster.pluginspec +usr/lib/*/qtcreator/plugins/Nokia/Core.pluginspec +usr/lib/*/qtcreator/plugins/Nokia/CppEditor.pluginspec +usr/lib/*/qtcreator/plugins/Nokia/CppTools.pluginspec +usr/lib/*/qtcreator/plugins/Nokia/Debugger.pluginspec +usr/lib/*/qtcreator/plugins/Nokia/Designer.pluginspec +usr/lib/*/qtcreator/plugins/Nokia/FakeVim.pluginspec +usr/lib/*/qtcreator/plugins/Nokia/Find.pluginspec +usr/lib/*/qtcreator/plugins/Nokia/GLSLEditor.pluginspec +usr/lib/*/qtcreator/plugins/Nokia/GenericProjectManager.pluginspec +usr/lib/*/qtcreator/plugins/Nokia/Git.pluginspec +usr/lib/*/qtcreator/plugins/Nokia/HelloWorld.pluginspec +usr/lib/*/qtcreator/plugins/Nokia/Help.pluginspec +usr/lib/*/qtcreator/plugins/Nokia/ImageViewer.pluginspec +usr/lib/*/qtcreator/plugins/Nokia/Locator.pluginspec +usr/lib/*/qtcreator/plugins/Nokia/Macros.pluginspec +usr/lib/*/qtcreator/plugins/Nokia/Madde.pluginspec +usr/lib/*/qtcreator/plugins/Nokia/Mercurial.pluginspec +usr/lib/*/qtcreator/plugins/Nokia/Perforce.pluginspec +usr/lib/*/qtcreator/plugins/Nokia/ProjectExplorer.pluginspec +usr/lib/*/qtcreator/plugins/Nokia/QmlDesigner.pluginspec +usr/lib/*/qtcreator/plugins/Nokia/QmlJSEditor.pluginspec +usr/lib/*/qtcreator/plugins/Nokia/QmlJSInspector.pluginspec +usr/lib/*/qtcreator/plugins/Nokia/QmlJSTools.pluginspec +usr/lib/*/qtcreator/plugins/Nokia/QmlProfiler.pluginspec +usr/lib/*/qtcreator/plugins/Nokia/QmlProjectManager.pluginspec +usr/lib/*/qtcreator/plugins/Nokia/Qt4ProjectManager.pluginspec +usr/lib/*/qtcreator/plugins/Nokia/QtSupport.pluginspec +usr/lib/*/qtcreator/plugins/Nokia/RemoteLinux.pluginspec +usr/lib/*/qtcreator/plugins/Nokia/ResourceEditor.pluginspec +usr/lib/*/qtcreator/plugins/Nokia/Subversion.pluginspec +usr/lib/*/qtcreator/plugins/Nokia/TaskList.pluginspec +usr/lib/*/qtcreator/plugins/Nokia/TextEditor.pluginspec +usr/lib/*/qtcreator/plugins/Nokia/Todo.pluginspec +usr/lib/*/qtcreator/plugins/Nokia/UpdateInfo.pluginspec +usr/lib/*/qtcreator/plugins/Nokia/Valgrind.pluginspec +usr/lib/*/qtcreator/plugins/Nokia/VcsBase.pluginspec +usr/lib/*/qtcreator/plugins/Nokia/Welcome.pluginspec +usr/lib/*/qtcreator/plugins/Nokia/libAnalyzerBase.so +usr/lib/*/qtcreator/plugins/Nokia/libAutotoolsProjectManager.so +usr/lib/*/qtcreator/plugins/Nokia/libBazaar.so +usr/lib/*/qtcreator/plugins/Nokia/libBinEditor.so +usr/lib/*/qtcreator/plugins/Nokia/libBookmarks.so +usr/lib/*/qtcreator/plugins/Nokia/libCMakeProjectManager.so +usr/lib/*/qtcreator/plugins/Nokia/libCVS.so +usr/lib/*/qtcreator/plugins/Nokia/libClassView.so +usr/lib/*/qtcreator/plugins/Nokia/libCodePaster.so +usr/lib/*/qtcreator/plugins/Nokia/libCore.so +usr/lib/*/qtcreator/plugins/Nokia/libCppEditor.so +usr/lib/*/qtcreator/plugins/Nokia/libCppTools.so +usr/lib/*/qtcreator/plugins/Nokia/libDebugger.so +usr/lib/*/qtcreator/plugins/Nokia/libDesigner.so +usr/lib/*/qtcreator/plugins/Nokia/libFakeVim.so +usr/lib/*/qtcreator/plugins/Nokia/libFind.so +usr/lib/*/qtcreator/plugins/Nokia/libGLSLEditor.so +usr/lib/*/qtcreator/plugins/Nokia/libGenericProjectManager.so +usr/lib/*/qtcreator/plugins/Nokia/libGit.so +usr/lib/*/qtcreator/plugins/Nokia/libHelloWorld.so +usr/lib/*/qtcreator/plugins/Nokia/libHelp.so +usr/lib/*/qtcreator/plugins/Nokia/libImageViewer.so +usr/lib/*/qtcreator/plugins/Nokia/libLocator.so +usr/lib/*/qtcreator/plugins/Nokia/libMacros.so +usr/lib/*/qtcreator/plugins/Nokia/libMadde.so +usr/lib/*/qtcreator/plugins/Nokia/libMercurial.so +usr/lib/*/qtcreator/plugins/Nokia/libPerforce.so +usr/lib/*/qtcreator/plugins/Nokia/libProjectExplorer.so +usr/lib/*/qtcreator/plugins/Nokia/libQmlDesigner.so +usr/lib/*/qtcreator/plugins/Nokia/libQmlJSEditor.so +usr/lib/*/qtcreator/plugins/Nokia/libQmlJSInspector.so +usr/lib/*/qtcreator/plugins/Nokia/libQmlJSTools.so +usr/lib/*/qtcreator/plugins/Nokia/libQmlProfiler.so +usr/lib/*/qtcreator/plugins/Nokia/libQmlProjectManager.so +usr/lib/*/qtcreator/plugins/Nokia/libQt4ProjectManager.so +usr/lib/*/qtcreator/plugins/Nokia/libQtSupport.so +usr/lib/*/qtcreator/plugins/Nokia/libRemoteLinux.so +usr/lib/*/qtcreator/plugins/Nokia/libResourceEditor.so +usr/lib/*/qtcreator/plugins/Nokia/libSubversion.so +usr/lib/*/qtcreator/plugins/Nokia/libTaskList.so +usr/lib/*/qtcreator/plugins/Nokia/libTextEditor.so +usr/lib/*/qtcreator/plugins/Nokia/libTodo.so +usr/lib/*/qtcreator/plugins/Nokia/libUpdateInfo.so +usr/lib/*/qtcreator/plugins/Nokia/libValgrind.so +usr/lib/*/qtcreator/plugins/Nokia/libVcsBase.so +usr/lib/*/qtcreator/plugins/Nokia/libWelcome.so +usr/lib/*/qtcreator/qmldesigner/libcustomstyleplugin.so +usr/lib/*/qtcreator/qmldesigner/libdesktopplugin.so +usr/lib/*/qtcreator/qmldesigner/libextrasplugin.so +usr/lib/*/qtcreator/qmldesigner/libmeegoplugin.so +usr/lib/*/qtcreator/qmldesigner/libqtquickplugin.so +usr/lib/*/qtcreator/qmldesigner/libsymbianplugin.so +usr/lib/*/qtcreator/qtcomponents/Button.qml +usr/lib/*/qtcreator/qtcomponents/ButtonRow.qml +usr/lib/*/qtcreator/qtcomponents/CheckBox.qml +usr/lib/*/qtcreator/qtcomponents/ChoiceList.qml +usr/lib/*/qtcreator/qtcomponents/ContextMenu.qml +usr/lib/*/qtcreator/qtcomponents/Dial.qml +usr/lib/*/qtcreator/qtcomponents/Frame.qml +usr/lib/*/qtcreator/qtcomponents/GroupBox.qml +usr/lib/*/qtcreator/qtcomponents/Menu.qml +usr/lib/*/qtcreator/qtcomponents/MenuItem.qml +usr/lib/*/qtcreator/qtcomponents/ProgressBar.qml +usr/lib/*/qtcreator/qtcomponents/RadioButton.qml +usr/lib/*/qtcreator/qtcomponents/ScrollArea.qml +usr/lib/*/qtcreator/qtcomponents/ScrollBar.qml +usr/lib/*/qtcreator/qtcomponents/Slider.qml +usr/lib/*/qtcreator/qtcomponents/SpinBox.qml +usr/lib/*/qtcreator/qtcomponents/SplitterRow.qml +usr/lib/*/qtcreator/qtcomponents/Switch.qml +usr/lib/*/qtcreator/qtcomponents/Tab.qml +usr/lib/*/qtcreator/qtcomponents/TabBar.qml +usr/lib/*/qtcreator/qtcomponents/TabFrame.qml +usr/lib/*/qtcreator/qtcomponents/TableColumn.qml +usr/lib/*/qtcreator/qtcomponents/TableView.qml +usr/lib/*/qtcreator/qtcomponents/TextArea.qml +usr/lib/*/qtcreator/qtcomponents/TextField.qml +usr/lib/*/qtcreator/qtcomponents/ToolBar.qml +usr/lib/*/qtcreator/qtcomponents/ToolButton.qml +usr/lib/*/qtcreator/qtcomponents/custom/BasicButton.qml +usr/lib/*/qtcreator/qtcomponents/custom/Button.qml +usr/lib/*/qtcreator/qtcomponents/custom/ButtonColumn.qml +usr/lib/*/qtcreator/qtcomponents/custom/ButtonGroup.js +usr/lib/*/qtcreator/qtcomponents/custom/ButtonRow.qml +usr/lib/*/qtcreator/qtcomponents/custom/CheckBox.qml +usr/lib/*/qtcreator/qtcomponents/custom/ChoiceList.qml +usr/lib/*/qtcreator/qtcomponents/custom/GroupBox.qml +usr/lib/*/qtcreator/qtcomponents/custom/ProgressBar.qml +usr/lib/*/qtcreator/qtcomponents/custom/Slider.qml +usr/lib/*/qtcreator/qtcomponents/custom/SpinBox.qml +usr/lib/*/qtcreator/qtcomponents/custom/SplitterRow.qml +usr/lib/*/qtcreator/qtcomponents/custom/TextField.qml +usr/lib/*/qtcreator/qtcomponents/custom/behaviors/ButtonBehavior.qml +usr/lib/*/qtcreator/qtcomponents/custom/behaviors/ModalPopupBehavior.qml +usr/lib/*/qtcreator/qtcomponents/custom/components.pro +usr/lib/*/qtcreator/qtcomponents/custom/private/ChoiceListPopup.qml +usr/lib/*/qtcreator/qtcomponents/custom/qmldir +usr/lib/*/qtcreator/qtcomponents/images/folder_new.png +usr/lib/*/qtcreator/qtcomponents/plugin/libstyleplugin.so +usr/lib/*/qtcreator/qtcomponents/qmldir +usr/share/icons/hicolor/128x128/apps/qtcreator.png +usr/share/icons/hicolor/16x16/apps/qtcreator.png +usr/share/icons/hicolor/24x24/apps/qtcreator.png +usr/share/icons/hicolor/256x256/apps/qtcreator.png +usr/share/icons/hicolor/32x32/apps/qtcreator.png +usr/share/icons/hicolor/48x48/apps/qtcreator.png +usr/share/icons/hicolor/512x512/apps/qtcreator.png +usr/share/icons/hicolor/64x64/apps/qtcreator.png +usr/share/qtcreator/designer/templates.xml +usr/share/qtcreator/designer/templates/Dialog_with_Buttons_Bottom.ui +usr/share/qtcreator/designer/templates/Dialog_with_Buttons_Right.ui +usr/share/qtcreator/designer/templates/Dialog_without_Buttons.ui +usr/share/qtcreator/designer/templates/Main_Window.ui +usr/share/qtcreator/designer/templates/Widget.ui +usr/share/qtcreator/dumper/LGPL_EXCEPTION.TXT +usr/share/qtcreator/dumper/LICENSE.LGPL usr/share/qtcreator/dumper/bridge.py +usr/share/qtcreator/dumper/dumper.cpp +usr/share/qtcreator/dumper/dumper.h usr/share/qtcreator/dumper/dumper.pro usr/share/qtcreator/dumper/dumper.py usr/share/qtcreator/dumper/dumper_p.h -usr/share/qtcreator/dumper/dumper.cpp usr/share/qtcreator/dumper/pdumper.py usr/share/qtcreator/dumper/qttypes.py usr/share/qtcreator/dumper/test/dumpertest.pro usr/share/qtcreator/dumper/test/main.cpp -usr/share/qtcreator/generic-highlighter -usr/share/qtcreator/glsl -usr/share/qtcreator/qml-type-descriptions +usr/share/qtcreator/externaltools/lrelease.xml +usr/share/qtcreator/externaltools/lupdate.xml +usr/share/qtcreator/externaltools/qmlviewer.xml +usr/share/qtcreator/externaltools/sort.xml +usr/share/qtcreator/externaltools/vi.xml +usr/share/qtcreator/generic-highlighter/alert.xml +usr/share/qtcreator/generic-highlighter/autoconf.xml +usr/share/qtcreator/generic-highlighter/bash.xml +usr/share/qtcreator/generic-highlighter/cmake.xml +usr/share/qtcreator/generic-highlighter/css.xml +usr/share/qtcreator/generic-highlighter/doxygen.xml +usr/share/qtcreator/generic-highlighter/dtd.xml +usr/share/qtcreator/generic-highlighter/html.xml +usr/share/qtcreator/generic-highlighter/ini.xml +usr/share/qtcreator/generic-highlighter/java.xml +usr/share/qtcreator/generic-highlighter/javadoc.xml +usr/share/qtcreator/generic-highlighter/perl.xml +usr/share/qtcreator/generic-highlighter/ruby.xml +usr/share/qtcreator/generic-highlighter/valgrind-suppression.xml +usr/share/qtcreator/generic-highlighter/xml.xml +usr/share/qtcreator/generic-highlighter/yacc.xml +usr/share/qtcreator/glsl/glsl_120.frag +usr/share/qtcreator/glsl/glsl_120.vert +usr/share/qtcreator/glsl/glsl_120_common.glsl +usr/share/qtcreator/glsl/glsl_es_100.frag +usr/share/qtcreator/glsl/glsl_es_100.vert +usr/share/qtcreator/glsl/glsl_es_100_common.glsl +usr/share/qtcreator/qml-type-descriptions/builtins.qmltypes +usr/share/qtcreator/qml-type-descriptions/qmlproject.qmltypes +usr/share/qtcreator/qml-type-descriptions/qmlruntime.qmltypes +usr/share/qtcreator/qml-type-descriptions/qt-labs-folderlistmodel.qmltypes +usr/share/qtcreator/qml-type-descriptions/qt-labs-gestures.qmltypes +usr/share/qtcreator/qml-type-descriptions/qt-labs-particles.qmltypes +usr/share/qtcreator/qml-type-descriptions/qtmobility-connectivity.qmltypes +usr/share/qtcreator/qml-type-descriptions/qtmobility-contacts.qmltypes +usr/share/qtcreator/qml-type-descriptions/qtmobility-feedback.qmltypes +usr/share/qtcreator/qml-type-descriptions/qtmobility-gallery.qmltypes +usr/share/qtcreator/qml-type-descriptions/qtmobility-location.qmltypes +usr/share/qtcreator/qml-type-descriptions/qtmobility-messaging.qmltypes +usr/share/qtcreator/qml-type-descriptions/qtmobility-organizer.qmltypes +usr/share/qtcreator/qml-type-descriptions/qtmobility-publishsubscribe.qmltypes +usr/share/qtcreator/qml-type-descriptions/qtmobility-sensors.qmltypes +usr/share/qtcreator/qml-type-descriptions/qtmobility-serviceframework.qmltypes +usr/share/qtcreator/qml-type-descriptions/qtmobility-systeminfo.qmltypes +usr/share/qtcreator/qml-type-descriptions/qtmultimediakit.qmltypes +usr/share/qtcreator/qml-type-descriptions/qtwebkit.qmltypes usr/share/qtcreator/qml/qmldump/Info.plist.in +usr/share/qtcreator/qml/qmldump/LGPL_EXCEPTION.TXT +usr/share/qtcreator/qml/qmldump/LICENSE.LGPL +usr/share/qtcreator/qml/qmldump/main.cpp usr/share/qtcreator/qml/qmldump/qmldump.pro usr/share/qtcreator/qml/qmldump/qmlstreamwriter.cpp usr/share/qtcreator/qml/qmldump/qmlstreamwriter.h -usr/share/qtcreator/qml/qmldump/main.cpp -usr/share/qtcreator/qml/qmljsdebugger +usr/share/qtcreator/qml/qmljsdebugger/editor/abstractliveedittool.cpp +usr/share/qtcreator/qml/qmljsdebugger/editor/abstractliveedittool.h +usr/share/qtcreator/qml/qmljsdebugger/editor/boundingrecthighlighter.cpp +usr/share/qtcreator/qml/qmljsdebugger/editor/boundingrecthighlighter.h +usr/share/qtcreator/qml/qmljsdebugger/editor/colorpickertool.cpp +usr/share/qtcreator/qml/qmljsdebugger/editor/colorpickertool.h +usr/share/qtcreator/qml/qmljsdebugger/editor/livelayeritem.cpp +usr/share/qtcreator/qml/qmljsdebugger/editor/livelayeritem.h +usr/share/qtcreator/qml/qmljsdebugger/editor/liverubberbandselectionmanipulator.cpp +usr/share/qtcreator/qml/qmljsdebugger/editor/liverubberbandselectionmanipulator.h +usr/share/qtcreator/qml/qmljsdebugger/editor/liveselectionindicator.cpp +usr/share/qtcreator/qml/qmljsdebugger/editor/liveselectionindicator.h +usr/share/qtcreator/qml/qmljsdebugger/editor/liveselectionrectangle.cpp +usr/share/qtcreator/qml/qmljsdebugger/editor/liveselectionrectangle.h +usr/share/qtcreator/qml/qmljsdebugger/editor/liveselectiontool.cpp +usr/share/qtcreator/qml/qmljsdebugger/editor/liveselectiontool.h +usr/share/qtcreator/qml/qmljsdebugger/editor/livesingleselectionmanipulator.cpp +usr/share/qtcreator/qml/qmljsdebugger/editor/livesingleselectionmanipulator.h +usr/share/qtcreator/qml/qmljsdebugger/editor/subcomponentmasklayeritem.cpp +usr/share/qtcreator/qml/qmljsdebugger/editor/subcomponentmasklayeritem.h +usr/share/qtcreator/qml/qmljsdebugger/editor/zoomtool.cpp +usr/share/qtcreator/qml/qmljsdebugger/editor/zoomtool.h +usr/share/qtcreator/qml/qmljsdebugger/include/jsdebuggeragent.h +usr/share/qtcreator/qml/qmljsdebugger/include/qdeclarativeinspectorservice.h +usr/share/qtcreator/qml/qmljsdebugger/include/qdeclarativeviewinspector.h +usr/share/qtcreator/qml/qmljsdebugger/include/qdeclarativeviewobserver.h +usr/share/qtcreator/qml/qmljsdebugger/include/qmlinspectorconstants.h +usr/share/qtcreator/qml/qmljsdebugger/include/qmljsdebugger_global.h +usr/share/qtcreator/qml/qmljsdebugger/include/qt_private/qdeclarativedebughelper_p.h +usr/share/qtcreator/qml/qmljsdebugger/include/qt_private/qdeclarativedebugservice_p.h +usr/share/qtcreator/qml/qmljsdebugger/include/qt_private/qdeclarativestate_p.h +usr/share/qtcreator/qml/qmljsdebugger/jsdebuggeragent.cpp +usr/share/qtcreator/qml/qmljsdebugger/protocol/inspectorprotocol.h +usr/share/qtcreator/qml/qmljsdebugger/protocol/protocol.pri +usr/share/qtcreator/qml/qmljsdebugger/qdeclarativeinspectorservice.cpp +usr/share/qtcreator/qml/qmljsdebugger/qdeclarativeviewinspector.cpp +usr/share/qtcreator/qml/qmljsdebugger/qdeclarativeviewinspector_p.h +usr/share/qtcreator/qml/qmljsdebugger/qmljsdebugger-lib.pri +usr/share/qtcreator/qml/qmljsdebugger/qmljsdebugger-src.pri +usr/share/qtcreator/qml/qmljsdebugger/qmljsdebugger.pro usr/share/qtcreator/qml/qmlobserver/Info.plist.in -usr/share/qtcreator/qml/qmlobserver/browser +usr/share/qtcreator/qml/qmlobserver/LGPL_EXCEPTION.TXT +usr/share/qtcreator/qml/qmlobserver/LICENSE.LGPL +usr/share/qtcreator/qml/qmlobserver/browser/Browser.qml +usr/share/qtcreator/qml/qmlobserver/browser/browser.qrc +usr/share/qtcreator/qml/qmlobserver/browser/images/folder.png +usr/share/qtcreator/qml/qmlobserver/browser/images/titlebar.png +usr/share/qtcreator/qml/qmlobserver/browser/images/titlebar.sci +usr/share/qtcreator/qml/qmlobserver/browser/images/up.png usr/share/qtcreator/qml/qmlobserver/deviceorientation.cpp usr/share/qtcreator/qml/qmlobserver/deviceorientation.h usr/share/qtcreator/qml/qmlobserver/deviceorientation_harmattan.cpp @@ -56,14 +313,544 @@ usr/share/qtcreator/qml/qmlobserver/qmlruntime.h usr/share/qtcreator/qml/qmlobserver/recopts.ui usr/share/qtcreator/qml/qmlobserver/recopts_maemo5.ui -usr/share/qtcreator/qml/qmlobserver/startup +usr/share/qtcreator/qml/qmlobserver/startup/Logo.qml +usr/share/qtcreator/qml/qmlobserver/startup/qt-back.png +usr/share/qtcreator/qml/qmlobserver/startup/qt-blue.jpg +usr/share/qtcreator/qml/qmlobserver/startup/qt-front.png +usr/share/qtcreator/qml/qmlobserver/startup/qt-sketch.jpg +usr/share/qtcreator/qml/qmlobserver/startup/qt-text.png +usr/share/qtcreator/qml/qmlobserver/startup/quick-blur.png +usr/share/qtcreator/qml/qmlobserver/startup/quick-regular.png +usr/share/qtcreator/qml/qmlobserver/startup/shadow.png +usr/share/qtcreator/qml/qmlobserver/startup/startup.qml +usr/share/qtcreator/qml/qmlobserver/startup/startup.qrc +usr/share/qtcreator/qml/qmlobserver/startup/white-star.png usr/share/qtcreator/qml/qmlobserver/texteditautoresizer_maemo5.h -usr/share/qtcreator/qml/qmlpuppet -usr/share/qtcreator/qmldesigner -usr/share/qtcreator/qmlicons -usr/share/qtcreator/schemes -usr/share/qtcreator/snippets -usr/share/qtcreator/styles -usr/share/qtcreator/templates -usr/share/qtcreator/welcomescreen +usr/share/qtcreator/qml/qmlpuppet/commands/changeauxiliarycommand.cpp +usr/share/qtcreator/qml/qmlpuppet/commands/changeauxiliarycommand.h +usr/share/qtcreator/qml/qmlpuppet/commands/changebindingscommand.cpp +usr/share/qtcreator/qml/qmlpuppet/commands/changebindingscommand.h +usr/share/qtcreator/qml/qmlpuppet/commands/changefileurlcommand.cpp +usr/share/qtcreator/qml/qmlpuppet/commands/changefileurlcommand.h +usr/share/qtcreator/qml/qmlpuppet/commands/changeidscommand.cpp +usr/share/qtcreator/qml/qmlpuppet/commands/changeidscommand.h +usr/share/qtcreator/qml/qmlpuppet/commands/changenodesourcecommand.cpp +usr/share/qtcreator/qml/qmlpuppet/commands/changenodesourcecommand.h +usr/share/qtcreator/qml/qmlpuppet/commands/changestatecommand.cpp +usr/share/qtcreator/qml/qmlpuppet/commands/changestatecommand.h +usr/share/qtcreator/qml/qmlpuppet/commands/changevaluescommand.cpp +usr/share/qtcreator/qml/qmlpuppet/commands/changevaluescommand.h +usr/share/qtcreator/qml/qmlpuppet/commands/childrenchangedcommand.cpp +usr/share/qtcreator/qml/qmlpuppet/commands/childrenchangedcommand.h +usr/share/qtcreator/qml/qmlpuppet/commands/clearscenecommand.cpp +usr/share/qtcreator/qml/qmlpuppet/commands/clearscenecommand.h +usr/share/qtcreator/qml/qmlpuppet/commands/commands.pri +usr/share/qtcreator/qml/qmlpuppet/commands/completecomponentcommand.cpp +usr/share/qtcreator/qml/qmlpuppet/commands/completecomponentcommand.h +usr/share/qtcreator/qml/qmlpuppet/commands/componentcompletedcommand.cpp +usr/share/qtcreator/qml/qmlpuppet/commands/componentcompletedcommand.h +usr/share/qtcreator/qml/qmlpuppet/commands/createinstancescommand.cpp +usr/share/qtcreator/qml/qmlpuppet/commands/createinstancescommand.h +usr/share/qtcreator/qml/qmlpuppet/commands/createscenecommand.cpp +usr/share/qtcreator/qml/qmlpuppet/commands/createscenecommand.h +usr/share/qtcreator/qml/qmlpuppet/commands/informationchangedcommand.cpp +usr/share/qtcreator/qml/qmlpuppet/commands/informationchangedcommand.h +usr/share/qtcreator/qml/qmlpuppet/commands/pixmapchangedcommand.cpp +usr/share/qtcreator/qml/qmlpuppet/commands/pixmapchangedcommand.h +usr/share/qtcreator/qml/qmlpuppet/commands/removeinstancescommand.cpp +usr/share/qtcreator/qml/qmlpuppet/commands/removeinstancescommand.h +usr/share/qtcreator/qml/qmlpuppet/commands/removepropertiescommand.cpp +usr/share/qtcreator/qml/qmlpuppet/commands/removepropertiescommand.h +usr/share/qtcreator/qml/qmlpuppet/commands/reparentinstancescommand.cpp +usr/share/qtcreator/qml/qmlpuppet/commands/reparentinstancescommand.h +usr/share/qtcreator/qml/qmlpuppet/commands/statepreviewimagechangedcommand.cpp +usr/share/qtcreator/qml/qmlpuppet/commands/statepreviewimagechangedcommand.h +usr/share/qtcreator/qml/qmlpuppet/commands/synchronizecommand.cpp +usr/share/qtcreator/qml/qmlpuppet/commands/synchronizecommand.h +usr/share/qtcreator/qml/qmlpuppet/commands/tokencommand.cpp +usr/share/qtcreator/qml/qmlpuppet/commands/tokencommand.h +usr/share/qtcreator/qml/qmlpuppet/commands/valueschangedcommand.cpp +usr/share/qtcreator/qml/qmlpuppet/commands/valueschangedcommand.h +usr/share/qtcreator/qml/qmlpuppet/container/addimportcontainer.cpp +usr/share/qtcreator/qml/qmlpuppet/container/addimportcontainer.h +usr/share/qtcreator/qml/qmlpuppet/container/container.pri +usr/share/qtcreator/qml/qmlpuppet/container/idcontainer.cpp +usr/share/qtcreator/qml/qmlpuppet/container/idcontainer.h +usr/share/qtcreator/qml/qmlpuppet/container/imagecontainer.cpp +usr/share/qtcreator/qml/qmlpuppet/container/imagecontainer.h +usr/share/qtcreator/qml/qmlpuppet/container/informationcontainer.cpp +usr/share/qtcreator/qml/qmlpuppet/container/informationcontainer.h +usr/share/qtcreator/qml/qmlpuppet/container/instancecontainer.cpp +usr/share/qtcreator/qml/qmlpuppet/container/instancecontainer.h +usr/share/qtcreator/qml/qmlpuppet/container/propertyabstractcontainer.cpp +usr/share/qtcreator/qml/qmlpuppet/container/propertyabstractcontainer.h +usr/share/qtcreator/qml/qmlpuppet/container/propertybindingcontainer.cpp +usr/share/qtcreator/qml/qmlpuppet/container/propertybindingcontainer.h +usr/share/qtcreator/qml/qmlpuppet/container/propertyvaluecontainer.cpp +usr/share/qtcreator/qml/qmlpuppet/container/propertyvaluecontainer.h +usr/share/qtcreator/qml/qmlpuppet/container/reparentcontainer.cpp +usr/share/qtcreator/qml/qmlpuppet/container/reparentcontainer.h +usr/share/qtcreator/qml/qmlpuppet/html/welcome.html +usr/share/qtcreator/qml/qmlpuppet/images/template_image.png +usr/share/qtcreator/qml/qmlpuppet/images/webkit.png +usr/share/qtcreator/qml/qmlpuppet/instances/anchorchangesnodeinstance.cpp +usr/share/qtcreator/qml/qmlpuppet/instances/anchorchangesnodeinstance.h +usr/share/qtcreator/qml/qmlpuppet/instances/behaviornodeinstance.cpp +usr/share/qtcreator/qml/qmlpuppet/instances/behaviornodeinstance.h +usr/share/qtcreator/qml/qmlpuppet/instances/childrenchangeeventfilter.cpp +usr/share/qtcreator/qml/qmlpuppet/instances/childrenchangeeventfilter.h +usr/share/qtcreator/qml/qmlpuppet/instances/componentnodeinstance.cpp +usr/share/qtcreator/qml/qmlpuppet/instances/componentnodeinstance.h +usr/share/qtcreator/qml/qmlpuppet/instances/dummycontextobject.cpp +usr/share/qtcreator/qml/qmlpuppet/instances/dummycontextobject.h +usr/share/qtcreator/qml/qmlpuppet/instances/dummynodeinstance.cpp +usr/share/qtcreator/qml/qmlpuppet/instances/dummynodeinstance.h +usr/share/qtcreator/qml/qmlpuppet/instances/instances.pri +usr/share/qtcreator/qml/qmlpuppet/instances/nodeinstanceclientproxy.cpp +usr/share/qtcreator/qml/qmlpuppet/instances/nodeinstanceclientproxy.h +usr/share/qtcreator/qml/qmlpuppet/instances/nodeinstancemetaobject.cpp +usr/share/qtcreator/qml/qmlpuppet/instances/nodeinstancemetaobject.h +usr/share/qtcreator/qml/qmlpuppet/instances/nodeinstanceserver.cpp +usr/share/qtcreator/qml/qmlpuppet/instances/nodeinstanceserver.h +usr/share/qtcreator/qml/qmlpuppet/instances/nodeinstancesignalspy.cpp +usr/share/qtcreator/qml/qmlpuppet/instances/nodeinstancesignalspy.h +usr/share/qtcreator/qml/qmlpuppet/instances/objectnodeinstance.cpp +usr/share/qtcreator/qml/qmlpuppet/instances/objectnodeinstance.h +usr/share/qtcreator/qml/qmlpuppet/instances/qmlpropertychangesnodeinstance.cpp +usr/share/qtcreator/qml/qmlpuppet/instances/qmlpropertychangesnodeinstance.h +usr/share/qtcreator/qml/qmlpuppet/instances/qmlstatenodeinstance.cpp +usr/share/qtcreator/qml/qmlpuppet/instances/qmlstatenodeinstance.h +usr/share/qtcreator/qml/qmlpuppet/instances/qmltransitionnodeinstance.cpp +usr/share/qtcreator/qml/qmlpuppet/instances/qmltransitionnodeinstance.h +usr/share/qtcreator/qml/qmlpuppet/instances/servernodeinstance.cpp +usr/share/qtcreator/qml/qmlpuppet/instances/servernodeinstance.h +usr/share/qtcreator/qml/qmlpuppet/interfaces/commondefines.h +usr/share/qtcreator/qml/qmlpuppet/interfaces/interfaces.pri +usr/share/qtcreator/qml/qmlpuppet/interfaces/nodeinstanceclientinterface.h +usr/share/qtcreator/qml/qmlpuppet/interfaces/nodeinstanceserverinterface.cpp +usr/share/qtcreator/qml/qmlpuppet/interfaces/nodeinstanceserverinterface.h +usr/share/qtcreator/qml/qmlpuppet/qml2puppet/Info.plist.in +usr/share/qtcreator/qml/qmlpuppet/qml2puppet/instances/instances.pri +usr/share/qtcreator/qml/qmlpuppet/qml2puppet/instances/qt5informationnodeinstanceserver.cpp +usr/share/qtcreator/qml/qmlpuppet/qml2puppet/instances/qt5informationnodeinstanceserver.h +usr/share/qtcreator/qml/qmlpuppet/qml2puppet/instances/qt5nodeinstanceclientproxy.cpp +usr/share/qtcreator/qml/qmlpuppet/qml2puppet/instances/qt5nodeinstanceclientproxy.h +usr/share/qtcreator/qml/qmlpuppet/qml2puppet/instances/qt5nodeinstanceserver.cpp +usr/share/qtcreator/qml/qmlpuppet/qml2puppet/instances/qt5nodeinstanceserver.h +usr/share/qtcreator/qml/qmlpuppet/qml2puppet/instances/qt5previewnodeinstanceserver.cpp +usr/share/qtcreator/qml/qmlpuppet/qml2puppet/instances/qt5previewnodeinstanceserver.h +usr/share/qtcreator/qml/qmlpuppet/qml2puppet/instances/qt5rendernodeinstanceserver.cpp +usr/share/qtcreator/qml/qmlpuppet/qml2puppet/instances/qt5rendernodeinstanceserver.h +usr/share/qtcreator/qml/qmlpuppet/qml2puppet/instances/sgitemnodeinstance.cpp +usr/share/qtcreator/qml/qmlpuppet/qml2puppet/instances/sgitemnodeinstance.h +usr/share/qtcreator/qml/qmlpuppet/qml2puppet/main.cpp +usr/share/qtcreator/qml/qmlpuppet/qml2puppet/qml2puppet.pro +usr/share/qtcreator/qml/qmlpuppet/qmlpuppet.pro +usr/share/qtcreator/qml/qmlpuppet/qmlpuppet.qrc +usr/share/qtcreator/qml/qmlpuppet/qmlpuppet/Info.plist.in +usr/share/qtcreator/qml/qmlpuppet/qmlpuppet/instances/graphicsobjectnodeinstance.cpp +usr/share/qtcreator/qml/qmlpuppet/qmlpuppet/instances/graphicsobjectnodeinstance.h +usr/share/qtcreator/qml/qmlpuppet/qmlpuppet/instances/instances.pri +usr/share/qtcreator/qml/qmlpuppet/qmlpuppet/instances/positionernodeinstance.cpp +usr/share/qtcreator/qml/qmlpuppet/qmlpuppet/instances/positionernodeinstance.h +usr/share/qtcreator/qml/qmlpuppet/qmlpuppet/instances/qmlgraphicsitemnodeinstance.cpp +usr/share/qtcreator/qml/qmlpuppet/qmlpuppet/instances/qmlgraphicsitemnodeinstance.h +usr/share/qtcreator/qml/qmlpuppet/qmlpuppet/instances/qt4informationnodeinstanceserver.cpp +usr/share/qtcreator/qml/qmlpuppet/qmlpuppet/instances/qt4informationnodeinstanceserver.h +usr/share/qtcreator/qml/qmlpuppet/qmlpuppet/instances/qt4nodeinstanceclientproxy.cpp +usr/share/qtcreator/qml/qmlpuppet/qmlpuppet/instances/qt4nodeinstanceclientproxy.h +usr/share/qtcreator/qml/qmlpuppet/qmlpuppet/instances/qt4nodeinstanceserver.cpp +usr/share/qtcreator/qml/qmlpuppet/qmlpuppet/instances/qt4nodeinstanceserver.h +usr/share/qtcreator/qml/qmlpuppet/qmlpuppet/instances/qt4previewnodeinstanceserver.cpp +usr/share/qtcreator/qml/qmlpuppet/qmlpuppet/instances/qt4previewnodeinstanceserver.h +usr/share/qtcreator/qml/qmlpuppet/qmlpuppet/instances/qt4rendernodeinstanceserver.cpp +usr/share/qtcreator/qml/qmlpuppet/qmlpuppet/instances/qt4rendernodeinstanceserver.h +usr/share/qtcreator/qml/qmlpuppet/qmlpuppet/main.cpp +usr/share/qtcreator/qml/qmlpuppet/qmlpuppet/qmlpuppet.pri +usr/share/qtcreator/qml/qmlpuppet/qmlpuppet/qmlpuppet.pro +usr/share/qtcreator/qml/qmlpuppet/qmlpuppet_utilities.pri +usr/share/qtcreator/qmldesigner/propertyeditor/Qt/AlignmentHorizontalButtons.qml +usr/share/qtcreator/qmldesigner/propertyeditor/Qt/AlignmentVerticalButtons.qml +usr/share/qtcreator/qmldesigner/propertyeditor/Qt/AnchorBox.qml +usr/share/qtcreator/qmldesigner/propertyeditor/Qt/AnchorButtons.qml +usr/share/qtcreator/qmldesigner/propertyeditor/Qt/BorderImageSpecifics.qml +usr/share/qtcreator/qmldesigner/propertyeditor/Qt/CheckBox.qml +usr/share/qtcreator/qmldesigner/propertyeditor/Qt/ColorGroupBox.qml +usr/share/qtcreator/qmldesigner/propertyeditor/Qt/ColorLabel.qml +usr/share/qtcreator/qmldesigner/propertyeditor/Qt/ColorScheme.qml +usr/share/qtcreator/qmldesigner/propertyeditor/Qt/ColorTypeButtons.qml +usr/share/qtcreator/qmldesigner/propertyeditor/Qt/ComboBox.qml +usr/share/qtcreator/qmldesigner/propertyeditor/Qt/DoubleSpinBox.qml +usr/share/qtcreator/qmldesigner/propertyeditor/Qt/DoubleSpinBoxAlternate.qml +usr/share/qtcreator/qmldesigner/propertyeditor/Qt/ExpressionEditor.qml +usr/share/qtcreator/qmldesigner/propertyeditor/Qt/Extended.qml +usr/share/qtcreator/qmldesigner/propertyeditor/Qt/ExtendedFunctionButton.qml +usr/share/qtcreator/qmldesigner/propertyeditor/Qt/ExtendedPane.qml +usr/share/qtcreator/qmldesigner/propertyeditor/Qt/ExtendedSwitches.qml +usr/share/qtcreator/qmldesigner/propertyeditor/Qt/FlagedButton.qml +usr/share/qtcreator/qmldesigner/propertyeditor/Qt/FlickableGroupBox.qml +usr/share/qtcreator/qmldesigner/propertyeditor/Qt/FlickableSpecifics.qml +usr/share/qtcreator/qmldesigner/propertyeditor/Qt/FlipableSpecifics.qml +usr/share/qtcreator/qmldesigner/propertyeditor/Qt/FlowSpecifics.qml +usr/share/qtcreator/qmldesigner/propertyeditor/Qt/FontComboBox.qml +usr/share/qtcreator/qmldesigner/propertyeditor/Qt/FontGroupBox.qml +usr/share/qtcreator/qmldesigner/propertyeditor/Qt/FontStyleButtons.qml +usr/share/qtcreator/qmldesigner/propertyeditor/Qt/Geometry.qml +usr/share/qtcreator/qmldesigner/propertyeditor/Qt/GridSpecifics.qml +usr/share/qtcreator/qmldesigner/propertyeditor/Qt/GridViewSpecifics.qml +usr/share/qtcreator/qmldesigner/propertyeditor/Qt/GroupBox.qml +usr/share/qtcreator/qmldesigner/propertyeditor/Qt/GroupBoxOption.qml +usr/share/qtcreator/qmldesigner/propertyeditor/Qt/HorizontalLayout.qml +usr/share/qtcreator/qmldesigner/propertyeditor/Qt/HorizontalWhiteLine.qml +usr/share/qtcreator/qmldesigner/propertyeditor/Qt/ImageSpecifics.qml +usr/share/qtcreator/qmldesigner/propertyeditor/Qt/IntEditor.qml +usr/share/qtcreator/qmldesigner/propertyeditor/Qt/ItemPane.qml +usr/share/qtcreator/qmldesigner/propertyeditor/Qt/Label.qml +usr/share/qtcreator/qmldesigner/propertyeditor/Qt/Layout.qml +usr/share/qtcreator/qmldesigner/propertyeditor/Qt/LayoutPane.qml +usr/share/qtcreator/qmldesigner/propertyeditor/Qt/LineEdit.qml +usr/share/qtcreator/qmldesigner/propertyeditor/Qt/ListViewSpecifics.qml +usr/share/qtcreator/qmldesigner/propertyeditor/Qt/Modifiers.qml +usr/share/qtcreator/qmldesigner/propertyeditor/Qt/MouseAreaSpecifics.qml +usr/share/qtcreator/qmldesigner/propertyeditor/Qt/PathViewSpecifics.qml +usr/share/qtcreator/qmldesigner/propertyeditor/Qt/PlaceHolder.qml +usr/share/qtcreator/qmldesigner/propertyeditor/Qt/PropertyFrame.qml +usr/share/qtcreator/qmldesigner/propertyeditor/Qt/RectangleColorGroupBox.qml +usr/share/qtcreator/qmldesigner/propertyeditor/Qt/RectangleSpecifics.qml +usr/share/qtcreator/qmldesigner/propertyeditor/Qt/RowSpecifics.qml +usr/share/qtcreator/qmldesigner/propertyeditor/Qt/ScrollArea.qml +usr/share/qtcreator/qmldesigner/propertyeditor/Qt/SliderWidget.qml +usr/share/qtcreator/qmldesigner/propertyeditor/Qt/SpinBox.qml +usr/share/qtcreator/qmldesigner/propertyeditor/Qt/StandardTextColorGroupBox.qml +usr/share/qtcreator/qmldesigner/propertyeditor/Qt/StandardTextGroupBox.qml +usr/share/qtcreator/qmldesigner/propertyeditor/Qt/Switches.qml +usr/share/qtcreator/qmldesigner/propertyeditor/Qt/TextEditSpecifics.qml +usr/share/qtcreator/qmldesigner/propertyeditor/Qt/TextEditor.qml +usr/share/qtcreator/qmldesigner/propertyeditor/Qt/TextInputGroupBox.qml +usr/share/qtcreator/qmldesigner/propertyeditor/Qt/TextInputSpecifics.qml +usr/share/qtcreator/qmldesigner/propertyeditor/Qt/TextSpecifics.qml +usr/share/qtcreator/qmldesigner/propertyeditor/Qt/Transformation.qml +usr/share/qtcreator/qmldesigner/propertyeditor/Qt/Type.qml +usr/share/qtcreator/qmldesigner/propertyeditor/Qt/VerticalLayout.qml +usr/share/qtcreator/qmldesigner/propertyeditor/Qt/Visibility.qml +usr/share/qtcreator/qmldesigner/propertyeditor/Qt/anchorbottom.css +usr/share/qtcreator/qmldesigner/propertyeditor/Qt/anchorbox.css +usr/share/qtcreator/qmldesigner/propertyeditor/Qt/anchorfill.css +usr/share/qtcreator/qmldesigner/propertyeditor/Qt/anchorhorizontal.css +usr/share/qtcreator/qmldesigner/propertyeditor/Qt/anchorleft.css +usr/share/qtcreator/qmldesigner/propertyeditor/Qt/anchorright.css +usr/share/qtcreator/qmldesigner/propertyeditor/Qt/anchorspacer.css +usr/share/qtcreator/qmldesigner/propertyeditor/Qt/anchortop.css +usr/share/qtcreator/qmldesigner/propertyeditor/Qt/anchorvertical.css +usr/share/qtcreator/qmldesigner/propertyeditor/Qt/applybutton.css +usr/share/qtcreator/qmldesigner/propertyeditor/Qt/aspectlock.css +usr/share/qtcreator/qmldesigner/propertyeditor/Qt/cancelbutton.css +usr/share/qtcreator/qmldesigner/propertyeditor/Qt/checkbox_tr.css +usr/share/qtcreator/qmldesigner/propertyeditor/Qt/emptyPane.qml +usr/share/qtcreator/qmldesigner/propertyeditor/Qt/images/alignmentbottom-h-icon.png +usr/share/qtcreator/qmldesigner/propertyeditor/Qt/images/alignmentbottom-icon.png +usr/share/qtcreator/qmldesigner/propertyeditor/Qt/images/alignmentcenterh-h-icon.png +usr/share/qtcreator/qmldesigner/propertyeditor/Qt/images/alignmentcenterh-icon.png +usr/share/qtcreator/qmldesigner/propertyeditor/Qt/images/alignmentleft-h-icon.png +usr/share/qtcreator/qmldesigner/propertyeditor/Qt/images/alignmentleft-icon.png +usr/share/qtcreator/qmldesigner/propertyeditor/Qt/images/alignmentmiddle-h-icon.png +usr/share/qtcreator/qmldesigner/propertyeditor/Qt/images/alignmentmiddle-icon.png +usr/share/qtcreator/qmldesigner/propertyeditor/Qt/images/alignmentright-h-icon.png +usr/share/qtcreator/qmldesigner/propertyeditor/Qt/images/alignmentright-icon.png +usr/share/qtcreator/qmldesigner/propertyeditor/Qt/images/alignmenttop-h-icon.png +usr/share/qtcreator/qmldesigner/propertyeditor/Qt/images/alignmenttop-icon.png +usr/share/qtcreator/qmldesigner/propertyeditor/Qt/images/apply.png +usr/share/qtcreator/qmldesigner/propertyeditor/Qt/images/behaivour.png +usr/share/qtcreator/qmldesigner/propertyeditor/Qt/images/blended-image-icon.png +usr/share/qtcreator/qmldesigner/propertyeditor/Qt/images/bold-h-icon.png +usr/share/qtcreator/qmldesigner/propertyeditor/Qt/images/bold-icon.png +usr/share/qtcreator/qmldesigner/propertyeditor/Qt/images/button.png +usr/share/qtcreator/qmldesigner/propertyeditor/Qt/images/cancel.png +usr/share/qtcreator/qmldesigner/propertyeditor/Qt/images/default-icon.png +usr/share/qtcreator/qmldesigner/propertyeditor/Qt/images/downArrow.png +usr/share/qtcreator/qmldesigner/propertyeditor/Qt/images/expression.png +usr/share/qtcreator/qmldesigner/propertyeditor/Qt/images/extended.png +usr/share/qtcreator/qmldesigner/propertyeditor/Qt/images/grid-icon.png +usr/share/qtcreator/qmldesigner/propertyeditor/Qt/images/icon_color_gradient.png +usr/share/qtcreator/qmldesigner/propertyeditor/Qt/images/icon_color_none.png +usr/share/qtcreator/qmldesigner/propertyeditor/Qt/images/icon_color_solid.png +usr/share/qtcreator/qmldesigner/propertyeditor/Qt/images/image-icon.png +usr/share/qtcreator/qmldesigner/propertyeditor/Qt/images/italic-h-icon.png +usr/share/qtcreator/qmldesigner/propertyeditor/Qt/images/italic-icon.png +usr/share/qtcreator/qmldesigner/propertyeditor/Qt/images/item-icon.png +usr/share/qtcreator/qmldesigner/propertyeditor/Qt/images/layout.png +usr/share/qtcreator/qmldesigner/propertyeditor/Qt/images/leftArrow.png +usr/share/qtcreator/qmldesigner/propertyeditor/Qt/images/list-icon.png +usr/share/qtcreator/qmldesigner/propertyeditor/Qt/images/mouse-area-icon.png +usr/share/qtcreator/qmldesigner/propertyeditor/Qt/images/placeholder.png +usr/share/qtcreator/qmldesigner/propertyeditor/Qt/images/rect-icon.png +usr/share/qtcreator/qmldesigner/propertyeditor/Qt/images/reset-button.png +usr/share/qtcreator/qmldesigner/propertyeditor/Qt/images/rightArrow.png +usr/share/qtcreator/qmldesigner/propertyeditor/Qt/images/standard.png +usr/share/qtcreator/qmldesigner/propertyeditor/Qt/images/strikeout-h-icon.png +usr/share/qtcreator/qmldesigner/propertyeditor/Qt/images/strikeout-icon.png +usr/share/qtcreator/qmldesigner/propertyeditor/Qt/images/submenu.png +usr/share/qtcreator/qmldesigner/propertyeditor/Qt/images/text-edit-icon.png +usr/share/qtcreator/qmldesigner/propertyeditor/Qt/images/text-icon.png +usr/share/qtcreator/qmldesigner/propertyeditor/Qt/images/underline-h-icon.png +usr/share/qtcreator/qmldesigner/propertyeditor/Qt/images/underline-icon.png +usr/share/qtcreator/qmldesigner/propertyeditor/Qt/images/upArrow.png +usr/share/qtcreator/qmldesigner/propertyeditor/Qt/layoutWidget.css +usr/share/qtcreator/qmldesigner/propertyeditor/Qt/propertyEditor.css +usr/share/qtcreator/qmldesigner/propertyeditor/Qt/specialCheckBox.css +usr/share/qtcreator/qmldesigner/propertyeditor/Qt/styledbuttonleft.css +usr/share/qtcreator/qmldesigner/propertyeditor/Qt/styledbuttonmiddle.css +usr/share/qtcreator/qmldesigner/propertyeditor/Qt/styledbuttonright.css +usr/share/qtcreator/qmldesigner/propertyeditor/Qt/switch.css +usr/share/qtcreator/qmldesigner/propertyeditor/Qt/typeLabel.css +usr/share/qtcreator/qmldesigner/propertyeditor/QtWebKit/WebViewSpecifics.qml +usr/share/qtcreator/qmlicons/Qt/16x16/BorderImage.png +usr/share/qtcreator/qmlicons/Qt/16x16/BusyIndicator.png +usr/share/qtcreator/qmlicons/Qt/16x16/Button.png +usr/share/qtcreator/qmlicons/Qt/16x16/ButtonColumn.png +usr/share/qtcreator/qmlicons/Qt/16x16/ButtonRow.png +usr/share/qtcreator/qmlicons/Qt/16x16/CheckBox.png +usr/share/qtcreator/qmlicons/Qt/16x16/ChoiceList.png +usr/share/qtcreator/qmlicons/Qt/16x16/ColorAnimation.png +usr/share/qtcreator/qmlicons/Qt/16x16/Component.png +usr/share/qtcreator/qmlicons/Qt/16x16/CountBubble.png +usr/share/qtcreator/qmlicons/Qt/16x16/DatePickerDialog.png +usr/share/qtcreator/qmlicons/Qt/16x16/Flickable.png +usr/share/qtcreator/qmlicons/Qt/16x16/Flipable.png +usr/share/qtcreator/qmlicons/Qt/16x16/FocusScope.png +usr/share/qtcreator/qmlicons/Qt/16x16/GridView.png +usr/share/qtcreator/qmlicons/Qt/16x16/Image.png +usr/share/qtcreator/qmlicons/Qt/16x16/InfoBanner.png +usr/share/qtcreator/qmlicons/Qt/16x16/Item.png +usr/share/qtcreator/qmlicons/Qt/16x16/ListButton.png +usr/share/qtcreator/qmlicons/Qt/16x16/ListDelegate.png +usr/share/qtcreator/qmlicons/Qt/16x16/ListView.png +usr/share/qtcreator/qmlicons/Qt/16x16/MoreIndicator.png +usr/share/qtcreator/qmlicons/Qt/16x16/MouseArea.png +usr/share/qtcreator/qmlicons/Qt/16x16/PageIndicator.png +usr/share/qtcreator/qmlicons/Qt/16x16/ParallelAnimation.png +usr/share/qtcreator/qmlicons/Qt/16x16/PathView.png +usr/share/qtcreator/qmlicons/Qt/16x16/PauseAnimation.png +usr/share/qtcreator/qmlicons/Qt/16x16/ProgressBar.png +usr/share/qtcreator/qmlicons/Qt/16x16/PropertyChanges.png +usr/share/qtcreator/qmlicons/Qt/16x16/RadioButton.png +usr/share/qtcreator/qmlicons/Qt/16x16/RatingIndicator.png +usr/share/qtcreator/qmlicons/Qt/16x16/Rectangle.png +usr/share/qtcreator/qmlicons/Qt/16x16/SequentialAnimation.png +usr/share/qtcreator/qmlicons/Qt/16x16/Slider.png +usr/share/qtcreator/qmlicons/Qt/16x16/State.png +usr/share/qtcreator/qmlicons/Qt/16x16/Switch.png +usr/share/qtcreator/qmlicons/Qt/16x16/TabBar.png +usr/share/qtcreator/qmlicons/Qt/16x16/TabButton.png +usr/share/qtcreator/qmlicons/Qt/16x16/Text.png +usr/share/qtcreator/qmlicons/Qt/16x16/TextArea.png +usr/share/qtcreator/qmlicons/Qt/16x16/TextEdit.png +usr/share/qtcreator/qmlicons/Qt/16x16/TextField.png +usr/share/qtcreator/qmlicons/Qt/16x16/TextInput.png +usr/share/qtcreator/qmlicons/Qt/16x16/TimePickerDialog.png +usr/share/qtcreator/qmlicons/Qt/16x16/ToolBar.png +usr/share/qtcreator/qmlicons/Qt/16x16/Transition.png +usr/share/qtcreator/qmlicons/Qt/16x16/Tumbler.png +usr/share/qtcreator/qmlicons/Qt/16x16/TumblerButton.png +usr/share/qtcreator/qmlicons/Qt/16x16/TumblerColumn.png +usr/share/qtcreator/qmlicons/Qt/16x16/TumblerDialog.png +usr/share/qtcreator/qmlicons/Qt/16x16/Window.png +usr/share/qtcreator/qmlicons/Qt/16x16/item-icon16.png +usr/share/qtcreator/qmlicons/QtWebkit/16x16/WebView.png +usr/share/qtcreator/schemes/MS_Visual_C++.kms +usr/share/qtcreator/schemes/Xcode.kms +usr/share/qtcreator/snippets/cpp.xml +usr/share/qtcreator/snippets/qml.xml +usr/share/qtcreator/snippets/text.xml +usr/share/qtcreator/styles/darkvim.xml +usr/share/qtcreator/styles/default.xml +usr/share/qtcreator/styles/grayscale.xml +usr/share/qtcreator/styles/inkpot.xml +usr/share/qtcreator/styles/intellij.xml +usr/share/qtcreator/templates/html5app/app.pro +usr/share/qtcreator/templates/html5app/html/index.html +usr/share/qtcreator/templates/html5app/html5applicationviewer/html5applicationviewer.cpp +usr/share/qtcreator/templates/html5app/html5applicationviewer/html5applicationviewer.h +usr/share/qtcreator/templates/html5app/html5applicationviewer/html5applicationviewer.pri +usr/share/qtcreator/templates/html5app/html5applicationviewer/touchnavigation/navigationcontroller.cpp +usr/share/qtcreator/templates/html5app/html5applicationviewer/touchnavigation/navigationcontroller.h +usr/share/qtcreator/templates/html5app/html5applicationviewer/touchnavigation/touchnavigation.pri +usr/share/qtcreator/templates/html5app/html5applicationviewer/touchnavigation/webnavigation.cpp +usr/share/qtcreator/templates/html5app/html5applicationviewer/touchnavigation/webnavigation.h +usr/share/qtcreator/templates/html5app/html5applicationviewer/touchnavigation/webtouchevent.cpp +usr/share/qtcreator/templates/html5app/html5applicationviewer/touchnavigation/webtouchevent.h +usr/share/qtcreator/templates/html5app/html5applicationviewer/touchnavigation/webtouchnavigation.cpp +usr/share/qtcreator/templates/html5app/html5applicationviewer/touchnavigation/webtouchnavigation.h +usr/share/qtcreator/templates/html5app/html5applicationviewer/touchnavigation/webtouchphysics.cpp +usr/share/qtcreator/templates/html5app/html5applicationviewer/touchnavigation/webtouchphysics.h +usr/share/qtcreator/templates/html5app/html5applicationviewer/touchnavigation/webtouchphysicsinterface.cpp +usr/share/qtcreator/templates/html5app/html5applicationviewer/touchnavigation/webtouchphysicsinterface.h +usr/share/qtcreator/templates/html5app/html5applicationviewer/touchnavigation/webtouchscroller.cpp +usr/share/qtcreator/templates/html5app/html5applicationviewer/touchnavigation/webtouchscroller.h +usr/share/qtcreator/templates/html5app/main.cpp +usr/share/qtcreator/templates/mobileapp/app.pro +usr/share/qtcreator/templates/mobileapp/main.cpp +usr/share/qtcreator/templates/mobileapp/mainwindow.cpp +usr/share/qtcreator/templates/mobileapp/mainwindow.h +usr/share/qtcreator/templates/mobileapp/mainwindow.ui +usr/share/qtcreator/templates/qt4project/customwidgetwizard/tpl_collection.cpp +usr/share/qtcreator/templates/qt4project/customwidgetwizard/tpl_collection.h +usr/share/qtcreator/templates/qt4project/customwidgetwizard/tpl_plugin.pro +usr/share/qtcreator/templates/qt4project/customwidgetwizard/tpl_resources.qrc +usr/share/qtcreator/templates/qt4project/customwidgetwizard/tpl_single.cpp +usr/share/qtcreator/templates/qt4project/customwidgetwizard/tpl_single.h +usr/share/qtcreator/templates/qt4project/customwidgetwizard/tpl_widget.cpp +usr/share/qtcreator/templates/qt4project/customwidgetwizard/tpl_widget.h +usr/share/qtcreator/templates/qt4project/customwidgetwizard/tpl_widget_include.pri +usr/share/qtcreator/templates/qt4project/customwidgetwizard/tpl_widget_lib.pro +usr/share/qtcreator/templates/qt4project/main.cpp +usr/share/qtcreator/templates/qt4project/mywidget.cpp +usr/share/qtcreator/templates/qt4project/mywidget.h +usr/share/qtcreator/templates/qt4project/mywidget_form.cpp +usr/share/qtcreator/templates/qt4project/mywidget_form.h +usr/share/qtcreator/templates/qt4project/widget.ui +usr/share/qtcreator/templates/qtquickapp/app.pro +usr/share/qtcreator/templates/qtquickapp/main.cpp +usr/share/qtcreator/templates/qtquickapp/qml/app/meego10/MainPage.qml +usr/share/qtcreator/templates/qtquickapp/qml/app/meego10/main.qml +usr/share/qtcreator/templates/qtquickapp/qml/app/qtquick10/main.qml +usr/share/qtcreator/templates/qtquickapp/qml/app/symbian11/MainPage.qml +usr/share/qtcreator/templates/qtquickapp/qml/app/symbian11/main.qml +usr/share/qtcreator/templates/qtquickapp/qmlapplicationviewer/qmlapplicationviewer.cpp +usr/share/qtcreator/templates/qtquickapp/qmlapplicationviewer/qmlapplicationviewer.h +usr/share/qtcreator/templates/qtquickapp/qmlapplicationviewer/qmlapplicationviewer.pri +usr/share/qtcreator/templates/shared/app.desktop +usr/share/qtcreator/templates/shared/deployment.pri +usr/share/qtcreator/templates/shared/icon64.png +usr/share/qtcreator/templates/shared/icon80.png +usr/share/qtcreator/templates/shared/manifest.aegis +usr/share/qtcreator/templates/shared/symbianicon.svg +usr/share/qtcreator/templates/wizards/README.txt +usr/share/qtcreator/templates/wizards/helloworld/console.png +usr/share/qtcreator/templates/wizards/helloworld/main.cpp +usr/share/qtcreator/templates/wizards/helloworld/project.pro +usr/share/qtcreator/templates/wizards/helloworld/wizard_sample.xml +usr/share/qtcreator/templates/wizards/listmodel/listmodel.cpp +usr/share/qtcreator/templates/wizards/listmodel/listmodel.h +usr/share/qtcreator/templates/wizards/listmodel/wizard_sample.xml +usr/share/qtcreator/templates/wizards/plaincapp-cmake/CMakeLists.txt +usr/share/qtcreator/templates/wizards/plaincapp-cmake/console.png +usr/share/qtcreator/templates/wizards/plaincapp-cmake/main.c +usr/share/qtcreator/templates/wizards/plaincapp-cmake/wizard.xml +usr/share/qtcreator/templates/wizards/plaincapp/console.png +usr/share/qtcreator/templates/wizards/plaincapp/main.c +usr/share/qtcreator/templates/wizards/plaincapp/project.pro +usr/share/qtcreator/templates/wizards/plaincapp/wizard.xml +usr/share/qtcreator/templates/wizards/plaincppapp-cmake/CMakeLists.txt +usr/share/qtcreator/templates/wizards/plaincppapp-cmake/console.png +usr/share/qtcreator/templates/wizards/plaincppapp-cmake/main.cpp +usr/share/qtcreator/templates/wizards/plaincppapp-cmake/wizard.xml +usr/share/qtcreator/templates/wizards/plaincppapp/console.png +usr/share/qtcreator/templates/wizards/plaincppapp/main.cpp +usr/share/qtcreator/templates/wizards/plaincppapp/project.pro +usr/share/qtcreator/templates/wizards/plaincppapp/wizard.xml +usr/share/qtcreator/templates/wizards/qml-extension/lib.png +usr/share/qtcreator/templates/wizards/qml-extension/object.cpp +usr/share/qtcreator/templates/wizards/qml-extension/object.h +usr/share/qtcreator/templates/wizards/qml-extension/plugin.cpp +usr/share/qtcreator/templates/wizards/qml-extension/plugin.h +usr/share/qtcreator/templates/wizards/qml-extension/project.pro +usr/share/qtcreator/templates/wizards/qml-extension/qmldir +usr/share/qtcreator/templates/wizards/qml-extension/wizard.xml +usr/share/qtcreator/templates/wizards/qtcreatorplugin/MyPlugin.pluginspec.in +usr/share/qtcreator/templates/wizards/qtcreatorplugin/myplugin.cpp +usr/share/qtcreator/templates/wizards/qtcreatorplugin/myplugin.h +usr/share/qtcreator/templates/wizards/qtcreatorplugin/myplugin.pro +usr/share/qtcreator/templates/wizards/qtcreatorplugin/myplugin_global.h +usr/share/qtcreator/templates/wizards/qtcreatorplugin/mypluginconstants.h +usr/share/qtcreator/templates/wizards/qtcreatorplugin/qtcreator_logo_24.png +usr/share/qtcreator/templates/wizards/qtcreatorplugin/wizard.xml +usr/share/qtcreator/templates/wizards/scriptgeneratedproject/generate.pl +usr/share/qtcreator/templates/wizards/scriptgeneratedproject/wizard_sample.xml usr/share/qtcreator/translations/qtcreator_*.qm +usr/share/qtcreator/welcomescreen/develop.qml +usr/share/qtcreator/welcomescreen/dummydata/examplesModel.qml +usr/share/qtcreator/welcomescreen/dummydata/pagesModel.qml +usr/share/qtcreator/welcomescreen/dummydata/projectList.qml +usr/share/qtcreator/welcomescreen/dummydata/sessionList.qml +usr/share/qtcreator/welcomescreen/dummydata/tutorialsModel.qml +usr/share/qtcreator/welcomescreen/examples.qml +usr/share/qtcreator/welcomescreen/examples_fallback.xml +usr/share/qtcreator/welcomescreen/gettingstarted.qml +usr/share/qtcreator/welcomescreen/images_areaofinterest.xml +usr/share/qtcreator/welcomescreen/qtcreator_tutorials.xml +usr/share/qtcreator/welcomescreen/tutorials.qml +usr/share/qtcreator/welcomescreen/welcomescreen.qml +usr/share/qtcreator/welcomescreen/welcomescreen.qmlproject +usr/share/qtcreator/welcomescreen/widgets/CustomColors.qml +usr/share/qtcreator/welcomescreen/widgets/CustomFonts.qml +usr/share/qtcreator/welcomescreen/widgets/CustomTab.qml +usr/share/qtcreator/welcomescreen/widgets/CustomizedGridView.qml +usr/share/qtcreator/welcomescreen/widgets/Delegate.qml +usr/share/qtcreator/welcomescreen/widgets/Feedback.qml +usr/share/qtcreator/welcomescreen/widgets/GettingStartedItem.qml +usr/share/qtcreator/welcomescreen/widgets/LinkedText.qml +usr/share/qtcreator/welcomescreen/widgets/LinksBar.qml +usr/share/qtcreator/welcomescreen/widgets/Logo.qml +usr/share/qtcreator/welcomescreen/widgets/PageCaption.qml +usr/share/qtcreator/welcomescreen/widgets/PageLoader.qml +usr/share/qtcreator/welcomescreen/widgets/ProjectItem.qml +usr/share/qtcreator/welcomescreen/widgets/RecentProjects.qml +usr/share/qtcreator/welcomescreen/widgets/SearchBar.qml +usr/share/qtcreator/welcomescreen/widgets/SessionItem.qml +usr/share/qtcreator/welcomescreen/widgets/Sessions.qml +usr/share/qtcreator/welcomescreen/widgets/ToolTip.qml +usr/share/qtcreator/welcomescreen/widgets/dummydata/context/ExampleDelegate.qml +usr/share/qtcreator/welcomescreen/widgets/dummydata/context/ExampleGridView.qml +usr/share/qtcreator/welcomescreen/widgets/dummydata/examplesModel.qml +usr/share/qtcreator/welcomescreen/widgets/dummydata/mockupTags.qml +usr/share/qtcreator/welcomescreen/widgets/dummydata/pagesModel.qml +usr/share/qtcreator/welcomescreen/widgets/dummydata/tabsModel.qml +usr/share/qtcreator/welcomescreen/widgets/images/arrowBig.png +usr/share/qtcreator/welcomescreen/widgets/images/arrow_down.png +usr/share/qtcreator/welcomescreen/widgets/images/arrow_up.png +usr/share/qtcreator/welcomescreen/widgets/images/bullet.png +usr/share/qtcreator/welcomescreen/widgets/images/dropshadow.png +usr/share/qtcreator/welcomescreen/widgets/images/gettingStarted01.png +usr/share/qtcreator/welcomescreen/widgets/images/gettingStarted02.png +usr/share/qtcreator/welcomescreen/widgets/images/gettingStarted03.png +usr/share/qtcreator/welcomescreen/widgets/images/gettingStarted04.png +usr/share/qtcreator/welcomescreen/widgets/images/icons/adressbook.png +usr/share/qtcreator/welcomescreen/widgets/images/icons/buildrun.png +usr/share/qtcreator/welcomescreen/widgets/images/icons/clone.png +usr/share/qtcreator/welcomescreen/widgets/images/icons/communityIcon.png +usr/share/qtcreator/welcomescreen/widgets/images/icons/components.png +usr/share/qtcreator/welcomescreen/widgets/images/icons/createIcon.png +usr/share/qtcreator/welcomescreen/widgets/images/icons/ddays09.png +usr/share/qtcreator/welcomescreen/widgets/images/icons/ddays10.png +usr/share/qtcreator/welcomescreen/widgets/images/icons/ddays11.png +usr/share/qtcreator/welcomescreen/widgets/images/icons/delete.png +usr/share/qtcreator/welcomescreen/widgets/images/icons/developing_with_qt_creator.png +usr/share/qtcreator/welcomescreen/widgets/images/icons/feedbackIcon.png +usr/share/qtcreator/welcomescreen/widgets/images/icons/ico_community.png +usr/share/qtcreator/welcomescreen/widgets/images/icons/labsIcon.png +usr/share/qtcreator/welcomescreen/widgets/images/icons/openIcon.png +usr/share/qtcreator/welcomescreen/widgets/images/icons/qt_quick_1.png +usr/share/qtcreator/welcomescreen/widgets/images/icons/qt_quick_2.png +usr/share/qtcreator/welcomescreen/widgets/images/icons/qt_quick_3.png +usr/share/qtcreator/welcomescreen/widgets/images/icons/qt_sdk.png +usr/share/qtcreator/welcomescreen/widgets/images/icons/qtquick.png +usr/share/qtcreator/welcomescreen/widgets/images/icons/qwidget.png +usr/share/qtcreator/welcomescreen/widgets/images/icons/rename.png +usr/share/qtcreator/welcomescreen/widgets/images/icons/userguideIcon.png +usr/share/qtcreator/welcomescreen/widgets/images/icons/videoIcon.png +usr/share/qtcreator/welcomescreen/widgets/images/info.png +usr/share/qtcreator/welcomescreen/widgets/images/mockup/designer-examples.png +usr/share/qtcreator/welcomescreen/widgets/images/mockup/desktop-examples.png +usr/share/qtcreator/welcomescreen/widgets/images/mockup/draganddrop-examples.png +usr/share/qtcreator/welcomescreen/widgets/images/mockup/itemview-examples.png +usr/share/qtcreator/welcomescreen/widgets/images/mockup/layout-examples.png +usr/share/qtcreator/welcomescreen/widgets/images/mockup/mainwindow-examples.png +usr/share/qtcreator/welcomescreen/widgets/images/mockup/network-examples.png +usr/share/qtcreator/welcomescreen/widgets/images/mockup/opengl-examples.png +usr/share/qtcreator/welcomescreen/widgets/images/mockup/penguin.png +usr/share/qtcreator/welcomescreen/widgets/images/mockup/qtscript-examples.png +usr/share/qtcreator/welcomescreen/widgets/images/mockup/thread-examples.png +usr/share/qtcreator/welcomescreen/widgets/images/more.png +usr/share/qtcreator/welcomescreen/widgets/images/qtcreator.png +usr/share/qtcreator/welcomescreen/widgets/images/tab.png diff -Nru qtcreator-2.5.0/debian/qtcreator.install.linux qtcreator-2.5.2/debian/qtcreator.install.linux --- qtcreator-2.5.0/debian/qtcreator.install.linux 1970-01-01 00:00:00.000000000 +0000 +++ qtcreator-2.5.2/debian/qtcreator.install.linux 2012-06-08 16:41:47.000000000 +0000 @@ -0,0 +1,857 @@ +debian/Nokia-QtCreator.xml usr/share/mime/packages/ +debian/qtcreator.desktop usr/share/applications/ +usr/bin/qmlprofiler +usr/bin/qmlpuppet +usr/bin/qtcreator +usr/bin/qtcreator_process_stub +usr/bin/qtpromaker +usr/lib/*/qtcreator/libAggregation.* +usr/lib/*/qtcreator/libBotan.* +usr/lib/*/qtcreator/libCPlusPlus.* +usr/lib/*/qtcreator/libDebuggingHelper.* +usr/lib/*/qtcreator/libExtensionSystem.* +usr/lib/*/qtcreator/libGLSL.* +usr/lib/*/qtcreator/libLanguageUtils.* +usr/lib/*/qtcreator/libQmlEditorWidgets.* +usr/lib/*/qtcreator/libQmlJS.* +usr/lib/*/qtcreator/libQmlJSDebugClient.* +usr/lib/*/qtcreator/libUtils.* +usr/lib/*/qtcreator/libptracepreload.* +usr/lib/*/qtcreator/libsymbianutils.* +usr/lib/*/qtcreator/libzeroconf.* +usr/lib/*/qtcreator/plugins/Nokia/AnalyzerBase.pluginspec +usr/lib/*/qtcreator/plugins/Nokia/AutotoolsProjectManager.pluginspec +usr/lib/*/qtcreator/plugins/Nokia/Bazaar.pluginspec +usr/lib/*/qtcreator/plugins/Nokia/BinEditor.pluginspec +usr/lib/*/qtcreator/plugins/Nokia/Bookmarks.pluginspec +usr/lib/*/qtcreator/plugins/Nokia/CMakeProjectManager.pluginspec +usr/lib/*/qtcreator/plugins/Nokia/CVS.pluginspec +usr/lib/*/qtcreator/plugins/Nokia/ClassView.pluginspec +usr/lib/*/qtcreator/plugins/Nokia/CodePaster.pluginspec +usr/lib/*/qtcreator/plugins/Nokia/Core.pluginspec +usr/lib/*/qtcreator/plugins/Nokia/CppEditor.pluginspec +usr/lib/*/qtcreator/plugins/Nokia/CppTools.pluginspec +usr/lib/*/qtcreator/plugins/Nokia/Debugger.pluginspec +usr/lib/*/qtcreator/plugins/Nokia/Designer.pluginspec +usr/lib/*/qtcreator/plugins/Nokia/FakeVim.pluginspec +usr/lib/*/qtcreator/plugins/Nokia/Find.pluginspec +usr/lib/*/qtcreator/plugins/Nokia/GLSLEditor.pluginspec +usr/lib/*/qtcreator/plugins/Nokia/GenericProjectManager.pluginspec +usr/lib/*/qtcreator/plugins/Nokia/Git.pluginspec +usr/lib/*/qtcreator/plugins/Nokia/HelloWorld.pluginspec +usr/lib/*/qtcreator/plugins/Nokia/Help.pluginspec +usr/lib/*/qtcreator/plugins/Nokia/ImageViewer.pluginspec +usr/lib/*/qtcreator/plugins/Nokia/Locator.pluginspec +usr/lib/*/qtcreator/plugins/Nokia/Macros.pluginspec +usr/lib/*/qtcreator/plugins/Nokia/Madde.pluginspec +usr/lib/*/qtcreator/plugins/Nokia/Mercurial.pluginspec +usr/lib/*/qtcreator/plugins/Nokia/Perforce.pluginspec +usr/lib/*/qtcreator/plugins/Nokia/ProjectExplorer.pluginspec +usr/lib/*/qtcreator/plugins/Nokia/QmlDesigner.pluginspec +usr/lib/*/qtcreator/plugins/Nokia/QmlJSEditor.pluginspec +usr/lib/*/qtcreator/plugins/Nokia/QmlJSInspector.pluginspec +usr/lib/*/qtcreator/plugins/Nokia/QmlJSTools.pluginspec +usr/lib/*/qtcreator/plugins/Nokia/QmlProfiler.pluginspec +usr/lib/*/qtcreator/plugins/Nokia/QmlProjectManager.pluginspec +usr/lib/*/qtcreator/plugins/Nokia/Qt4ProjectManager.pluginspec +usr/lib/*/qtcreator/plugins/Nokia/QtSupport.pluginspec +usr/lib/*/qtcreator/plugins/Nokia/RemoteLinux.pluginspec +usr/lib/*/qtcreator/plugins/Nokia/ResourceEditor.pluginspec +usr/lib/*/qtcreator/plugins/Nokia/Subversion.pluginspec +usr/lib/*/qtcreator/plugins/Nokia/TaskList.pluginspec +usr/lib/*/qtcreator/plugins/Nokia/TextEditor.pluginspec +usr/lib/*/qtcreator/plugins/Nokia/Todo.pluginspec +usr/lib/*/qtcreator/plugins/Nokia/UpdateInfo.pluginspec +usr/lib/*/qtcreator/plugins/Nokia/Valgrind.pluginspec +usr/lib/*/qtcreator/plugins/Nokia/VcsBase.pluginspec +usr/lib/*/qtcreator/plugins/Nokia/Welcome.pluginspec +usr/lib/*/qtcreator/plugins/Nokia/libAnalyzerBase.so +usr/lib/*/qtcreator/plugins/Nokia/libAutotoolsProjectManager.so +usr/lib/*/qtcreator/plugins/Nokia/libBazaar.so +usr/lib/*/qtcreator/plugins/Nokia/libBinEditor.so +usr/lib/*/qtcreator/plugins/Nokia/libBookmarks.so +usr/lib/*/qtcreator/plugins/Nokia/libCMakeProjectManager.so +usr/lib/*/qtcreator/plugins/Nokia/libCVS.so +usr/lib/*/qtcreator/plugins/Nokia/libClassView.so +usr/lib/*/qtcreator/plugins/Nokia/libCodePaster.so +usr/lib/*/qtcreator/plugins/Nokia/libCore.so +usr/lib/*/qtcreator/plugins/Nokia/libCppEditor.so +usr/lib/*/qtcreator/plugins/Nokia/libCppTools.so +usr/lib/*/qtcreator/plugins/Nokia/libDebugger.so +usr/lib/*/qtcreator/plugins/Nokia/libDesigner.so +usr/lib/*/qtcreator/plugins/Nokia/libFakeVim.so +usr/lib/*/qtcreator/plugins/Nokia/libFind.so +usr/lib/*/qtcreator/plugins/Nokia/libGLSLEditor.so +usr/lib/*/qtcreator/plugins/Nokia/libGenericProjectManager.so +usr/lib/*/qtcreator/plugins/Nokia/libGit.so +usr/lib/*/qtcreator/plugins/Nokia/libHelloWorld.so +usr/lib/*/qtcreator/plugins/Nokia/libHelp.so +usr/lib/*/qtcreator/plugins/Nokia/libImageViewer.so +usr/lib/*/qtcreator/plugins/Nokia/libLocator.so +usr/lib/*/qtcreator/plugins/Nokia/libMacros.so +usr/lib/*/qtcreator/plugins/Nokia/libMadde.so +usr/lib/*/qtcreator/plugins/Nokia/libMercurial.so +usr/lib/*/qtcreator/plugins/Nokia/libPerforce.so +usr/lib/*/qtcreator/plugins/Nokia/libProjectExplorer.so +usr/lib/*/qtcreator/plugins/Nokia/libQmlDesigner.so +usr/lib/*/qtcreator/plugins/Nokia/libQmlJSEditor.so +usr/lib/*/qtcreator/plugins/Nokia/libQmlJSInspector.so +usr/lib/*/qtcreator/plugins/Nokia/libQmlJSTools.so +usr/lib/*/qtcreator/plugins/Nokia/libQmlProfiler.so +usr/lib/*/qtcreator/plugins/Nokia/libQmlProjectManager.so +usr/lib/*/qtcreator/plugins/Nokia/libQt4ProjectManager.so +usr/lib/*/qtcreator/plugins/Nokia/libQtSupport.so +usr/lib/*/qtcreator/plugins/Nokia/libRemoteLinux.so +usr/lib/*/qtcreator/plugins/Nokia/libResourceEditor.so +usr/lib/*/qtcreator/plugins/Nokia/libSubversion.so +usr/lib/*/qtcreator/plugins/Nokia/libTaskList.so +usr/lib/*/qtcreator/plugins/Nokia/libTextEditor.so +usr/lib/*/qtcreator/plugins/Nokia/libTodo.so +usr/lib/*/qtcreator/plugins/Nokia/libUpdateInfo.so +usr/lib/*/qtcreator/plugins/Nokia/libValgrind.so +usr/lib/*/qtcreator/plugins/Nokia/libVcsBase.so +usr/lib/*/qtcreator/plugins/Nokia/libWelcome.so +usr/lib/*/qtcreator/qmldesigner/libcustomstyleplugin.so +usr/lib/*/qtcreator/qmldesigner/libdesktopplugin.so +usr/lib/*/qtcreator/qmldesigner/libextrasplugin.so +usr/lib/*/qtcreator/qmldesigner/libmeegoplugin.so +usr/lib/*/qtcreator/qmldesigner/libqtquickplugin.so +usr/lib/*/qtcreator/qmldesigner/libsymbianplugin.so +usr/lib/*/qtcreator/qtcomponents/Button.qml +usr/lib/*/qtcreator/qtcomponents/ButtonRow.qml +usr/lib/*/qtcreator/qtcomponents/CheckBox.qml +usr/lib/*/qtcreator/qtcomponents/ChoiceList.qml +usr/lib/*/qtcreator/qtcomponents/ContextMenu.qml +usr/lib/*/qtcreator/qtcomponents/Dial.qml +usr/lib/*/qtcreator/qtcomponents/Frame.qml +usr/lib/*/qtcreator/qtcomponents/GroupBox.qml +usr/lib/*/qtcreator/qtcomponents/Menu.qml +usr/lib/*/qtcreator/qtcomponents/MenuItem.qml +usr/lib/*/qtcreator/qtcomponents/ProgressBar.qml +usr/lib/*/qtcreator/qtcomponents/RadioButton.qml +usr/lib/*/qtcreator/qtcomponents/ScrollArea.qml +usr/lib/*/qtcreator/qtcomponents/ScrollBar.qml +usr/lib/*/qtcreator/qtcomponents/Slider.qml +usr/lib/*/qtcreator/qtcomponents/SpinBox.qml +usr/lib/*/qtcreator/qtcomponents/SplitterRow.qml +usr/lib/*/qtcreator/qtcomponents/Switch.qml +usr/lib/*/qtcreator/qtcomponents/Tab.qml +usr/lib/*/qtcreator/qtcomponents/TabBar.qml +usr/lib/*/qtcreator/qtcomponents/TabFrame.qml +usr/lib/*/qtcreator/qtcomponents/TableColumn.qml +usr/lib/*/qtcreator/qtcomponents/TableView.qml +usr/lib/*/qtcreator/qtcomponents/TextArea.qml +usr/lib/*/qtcreator/qtcomponents/TextField.qml +usr/lib/*/qtcreator/qtcomponents/ToolBar.qml +usr/lib/*/qtcreator/qtcomponents/ToolButton.qml +usr/lib/*/qtcreator/qtcomponents/custom/BasicButton.qml +usr/lib/*/qtcreator/qtcomponents/custom/Button.qml +usr/lib/*/qtcreator/qtcomponents/custom/ButtonColumn.qml +usr/lib/*/qtcreator/qtcomponents/custom/ButtonGroup.js +usr/lib/*/qtcreator/qtcomponents/custom/ButtonRow.qml +usr/lib/*/qtcreator/qtcomponents/custom/CheckBox.qml +usr/lib/*/qtcreator/qtcomponents/custom/ChoiceList.qml +usr/lib/*/qtcreator/qtcomponents/custom/GroupBox.qml +usr/lib/*/qtcreator/qtcomponents/custom/ProgressBar.qml +usr/lib/*/qtcreator/qtcomponents/custom/Slider.qml +usr/lib/*/qtcreator/qtcomponents/custom/SpinBox.qml +usr/lib/*/qtcreator/qtcomponents/custom/SplitterRow.qml +usr/lib/*/qtcreator/qtcomponents/custom/TextField.qml +usr/lib/*/qtcreator/qtcomponents/custom/behaviors/ButtonBehavior.qml +usr/lib/*/qtcreator/qtcomponents/custom/behaviors/ModalPopupBehavior.qml +usr/lib/*/qtcreator/qtcomponents/custom/components.pro +usr/lib/*/qtcreator/qtcomponents/custom/private/ChoiceListPopup.qml +usr/lib/*/qtcreator/qtcomponents/custom/qmldir +usr/lib/*/qtcreator/qtcomponents/images/folder_new.png +usr/lib/*/qtcreator/qtcomponents/plugin/libstyleplugin.so +usr/lib/*/qtcreator/qtcomponents/qmldir +usr/share/icons/hicolor/128x128/apps/qtcreator.png +usr/share/icons/hicolor/16x16/apps/qtcreator.png +usr/share/icons/hicolor/24x24/apps/qtcreator.png +usr/share/icons/hicolor/256x256/apps/qtcreator.png +usr/share/icons/hicolor/32x32/apps/qtcreator.png +usr/share/icons/hicolor/48x48/apps/qtcreator.png +usr/share/icons/hicolor/512x512/apps/qtcreator.png +usr/share/icons/hicolor/64x64/apps/qtcreator.png +usr/share/qtcreator/designer/templates.xml +usr/share/qtcreator/designer/templates/Dialog_with_Buttons_Bottom.ui +usr/share/qtcreator/designer/templates/Dialog_with_Buttons_Right.ui +usr/share/qtcreator/designer/templates/Dialog_without_Buttons.ui +usr/share/qtcreator/designer/templates/Main_Window.ui +usr/share/qtcreator/designer/templates/Widget.ui +usr/share/qtcreator/dumper/LGPL_EXCEPTION.TXT +usr/share/qtcreator/dumper/LICENSE.LGPL +usr/share/qtcreator/dumper/bridge.py +usr/share/qtcreator/dumper/dumper.cpp +usr/share/qtcreator/dumper/dumper.h +usr/share/qtcreator/dumper/dumper.pro +usr/share/qtcreator/dumper/dumper.py +usr/share/qtcreator/dumper/dumper_p.h +usr/share/qtcreator/dumper/pdumper.py +usr/share/qtcreator/dumper/qttypes.py +usr/share/qtcreator/dumper/test/dumpertest.pro +usr/share/qtcreator/dumper/test/main.cpp +usr/share/qtcreator/externaltools/lrelease.xml +usr/share/qtcreator/externaltools/lupdate.xml +usr/share/qtcreator/externaltools/qmlviewer.xml +usr/share/qtcreator/externaltools/sort.xml +usr/share/qtcreator/externaltools/vi.xml +usr/share/qtcreator/generic-highlighter/alert.xml +usr/share/qtcreator/generic-highlighter/autoconf.xml +usr/share/qtcreator/generic-highlighter/bash.xml +usr/share/qtcreator/generic-highlighter/cmake.xml +usr/share/qtcreator/generic-highlighter/css.xml +usr/share/qtcreator/generic-highlighter/doxygen.xml +usr/share/qtcreator/generic-highlighter/dtd.xml +usr/share/qtcreator/generic-highlighter/html.xml +usr/share/qtcreator/generic-highlighter/ini.xml +usr/share/qtcreator/generic-highlighter/java.xml +usr/share/qtcreator/generic-highlighter/javadoc.xml +usr/share/qtcreator/generic-highlighter/perl.xml +usr/share/qtcreator/generic-highlighter/ruby.xml +usr/share/qtcreator/generic-highlighter/valgrind-suppression.xml +usr/share/qtcreator/generic-highlighter/xml.xml +usr/share/qtcreator/generic-highlighter/yacc.xml +usr/share/qtcreator/glsl/glsl_120.frag +usr/share/qtcreator/glsl/glsl_120.vert +usr/share/qtcreator/glsl/glsl_120_common.glsl +usr/share/qtcreator/glsl/glsl_es_100.frag +usr/share/qtcreator/glsl/glsl_es_100.vert +usr/share/qtcreator/glsl/glsl_es_100_common.glsl +usr/share/qtcreator/qml-type-descriptions/builtins.qmltypes +usr/share/qtcreator/qml-type-descriptions/qmlproject.qmltypes +usr/share/qtcreator/qml-type-descriptions/qmlruntime.qmltypes +usr/share/qtcreator/qml-type-descriptions/qt-labs-folderlistmodel.qmltypes +usr/share/qtcreator/qml-type-descriptions/qt-labs-gestures.qmltypes +usr/share/qtcreator/qml-type-descriptions/qt-labs-particles.qmltypes +usr/share/qtcreator/qml-type-descriptions/qtmobility-connectivity.qmltypes +usr/share/qtcreator/qml-type-descriptions/qtmobility-contacts.qmltypes +usr/share/qtcreator/qml-type-descriptions/qtmobility-feedback.qmltypes +usr/share/qtcreator/qml-type-descriptions/qtmobility-gallery.qmltypes +usr/share/qtcreator/qml-type-descriptions/qtmobility-location.qmltypes +usr/share/qtcreator/qml-type-descriptions/qtmobility-messaging.qmltypes +usr/share/qtcreator/qml-type-descriptions/qtmobility-organizer.qmltypes +usr/share/qtcreator/qml-type-descriptions/qtmobility-publishsubscribe.qmltypes +usr/share/qtcreator/qml-type-descriptions/qtmobility-sensors.qmltypes +usr/share/qtcreator/qml-type-descriptions/qtmobility-serviceframework.qmltypes +usr/share/qtcreator/qml-type-descriptions/qtmobility-systeminfo.qmltypes +usr/share/qtcreator/qml-type-descriptions/qtmultimediakit.qmltypes +usr/share/qtcreator/qml-type-descriptions/qtwebkit.qmltypes +usr/share/qtcreator/qml/qmldump/Info.plist.in +usr/share/qtcreator/qml/qmldump/LGPL_EXCEPTION.TXT +usr/share/qtcreator/qml/qmldump/LICENSE.LGPL +usr/share/qtcreator/qml/qmldump/main.cpp +usr/share/qtcreator/qml/qmldump/qmldump.pro +usr/share/qtcreator/qml/qmldump/qmlstreamwriter.cpp +usr/share/qtcreator/qml/qmldump/qmlstreamwriter.h +usr/share/qtcreator/qml/qmljsdebugger/editor/abstractliveedittool.cpp +usr/share/qtcreator/qml/qmljsdebugger/editor/abstractliveedittool.h +usr/share/qtcreator/qml/qmljsdebugger/editor/boundingrecthighlighter.cpp +usr/share/qtcreator/qml/qmljsdebugger/editor/boundingrecthighlighter.h +usr/share/qtcreator/qml/qmljsdebugger/editor/colorpickertool.cpp +usr/share/qtcreator/qml/qmljsdebugger/editor/colorpickertool.h +usr/share/qtcreator/qml/qmljsdebugger/editor/livelayeritem.cpp +usr/share/qtcreator/qml/qmljsdebugger/editor/livelayeritem.h +usr/share/qtcreator/qml/qmljsdebugger/editor/liverubberbandselectionmanipulator.cpp +usr/share/qtcreator/qml/qmljsdebugger/editor/liverubberbandselectionmanipulator.h +usr/share/qtcreator/qml/qmljsdebugger/editor/liveselectionindicator.cpp +usr/share/qtcreator/qml/qmljsdebugger/editor/liveselectionindicator.h +usr/share/qtcreator/qml/qmljsdebugger/editor/liveselectionrectangle.cpp +usr/share/qtcreator/qml/qmljsdebugger/editor/liveselectionrectangle.h +usr/share/qtcreator/qml/qmljsdebugger/editor/liveselectiontool.cpp +usr/share/qtcreator/qml/qmljsdebugger/editor/liveselectiontool.h +usr/share/qtcreator/qml/qmljsdebugger/editor/livesingleselectionmanipulator.cpp +usr/share/qtcreator/qml/qmljsdebugger/editor/livesingleselectionmanipulator.h +usr/share/qtcreator/qml/qmljsdebugger/editor/subcomponentmasklayeritem.cpp +usr/share/qtcreator/qml/qmljsdebugger/editor/subcomponentmasklayeritem.h +usr/share/qtcreator/qml/qmljsdebugger/editor/zoomtool.cpp +usr/share/qtcreator/qml/qmljsdebugger/editor/zoomtool.h +usr/share/qtcreator/qml/qmljsdebugger/include/jsdebuggeragent.h +usr/share/qtcreator/qml/qmljsdebugger/include/qdeclarativeinspectorservice.h +usr/share/qtcreator/qml/qmljsdebugger/include/qdeclarativeviewinspector.h +usr/share/qtcreator/qml/qmljsdebugger/include/qdeclarativeviewobserver.h +usr/share/qtcreator/qml/qmljsdebugger/include/qmlinspectorconstants.h +usr/share/qtcreator/qml/qmljsdebugger/include/qmljsdebugger_global.h +usr/share/qtcreator/qml/qmljsdebugger/include/qt_private/qdeclarativedebughelper_p.h +usr/share/qtcreator/qml/qmljsdebugger/include/qt_private/qdeclarativedebugservice_p.h +usr/share/qtcreator/qml/qmljsdebugger/include/qt_private/qdeclarativestate_p.h +usr/share/qtcreator/qml/qmljsdebugger/jsdebuggeragent.cpp +usr/share/qtcreator/qml/qmljsdebugger/protocol/inspectorprotocol.h +usr/share/qtcreator/qml/qmljsdebugger/protocol/protocol.pri +usr/share/qtcreator/qml/qmljsdebugger/qdeclarativeinspectorservice.cpp +usr/share/qtcreator/qml/qmljsdebugger/qdeclarativeviewinspector.cpp +usr/share/qtcreator/qml/qmljsdebugger/qdeclarativeviewinspector_p.h +usr/share/qtcreator/qml/qmljsdebugger/qmljsdebugger-lib.pri +usr/share/qtcreator/qml/qmljsdebugger/qmljsdebugger-src.pri +usr/share/qtcreator/qml/qmljsdebugger/qmljsdebugger.pro +usr/share/qtcreator/qml/qmlobserver/Info.plist.in +usr/share/qtcreator/qml/qmlobserver/LGPL_EXCEPTION.TXT +usr/share/qtcreator/qml/qmlobserver/LICENSE.LGPL +usr/share/qtcreator/qml/qmlobserver/browser/Browser.qml +usr/share/qtcreator/qml/qmlobserver/browser/browser.qrc +usr/share/qtcreator/qml/qmlobserver/browser/images/folder.png +usr/share/qtcreator/qml/qmlobserver/browser/images/titlebar.png +usr/share/qtcreator/qml/qmlobserver/browser/images/titlebar.sci +usr/share/qtcreator/qml/qmlobserver/browser/images/up.png +usr/share/qtcreator/qml/qmlobserver/deviceorientation.cpp +usr/share/qtcreator/qml/qmlobserver/deviceorientation.h +usr/share/qtcreator/qml/qmlobserver/deviceorientation_harmattan.cpp +usr/share/qtcreator/qml/qmlobserver/deviceorientation_maemo5.cpp +usr/share/qtcreator/qml/qmlobserver/deviceorientation_symbian.cpp +usr/share/qtcreator/qml/qmlobserver/loggerwidget.cpp +usr/share/qtcreator/qml/qmlobserver/loggerwidget.h +usr/share/qtcreator/qml/qmlobserver/main.cpp +usr/share/qtcreator/qml/qmlobserver/proxysettings.cpp +usr/share/qtcreator/qml/qmlobserver/proxysettings.h +usr/share/qtcreator/qml/qmlobserver/proxysettings.ui +usr/share/qtcreator/qml/qmlobserver/proxysettings_maemo5.ui +usr/share/qtcreator/qml/qmlobserver/qdeclarativetester.cpp +usr/share/qtcreator/qml/qmlobserver/qdeclarativetester.h +usr/share/qtcreator/qml/qmlobserver/qml.icns +usr/share/qtcreator/qml/qmlobserver/qml.pri +usr/share/qtcreator/qml/qmlobserver/qmlobserver.pro +usr/share/qtcreator/qml/qmlobserver/qmlruntime.cpp +usr/share/qtcreator/qml/qmlobserver/qmlruntime.h +usr/share/qtcreator/qml/qmlobserver/recopts.ui +usr/share/qtcreator/qml/qmlobserver/recopts_maemo5.ui +usr/share/qtcreator/qml/qmlobserver/startup/Logo.qml +usr/share/qtcreator/qml/qmlobserver/startup/qt-back.png +usr/share/qtcreator/qml/qmlobserver/startup/qt-blue.jpg +usr/share/qtcreator/qml/qmlobserver/startup/qt-front.png +usr/share/qtcreator/qml/qmlobserver/startup/qt-sketch.jpg +usr/share/qtcreator/qml/qmlobserver/startup/qt-text.png +usr/share/qtcreator/qml/qmlobserver/startup/quick-blur.png +usr/share/qtcreator/qml/qmlobserver/startup/quick-regular.png +usr/share/qtcreator/qml/qmlobserver/startup/shadow.png +usr/share/qtcreator/qml/qmlobserver/startup/startup.qml +usr/share/qtcreator/qml/qmlobserver/startup/startup.qrc +usr/share/qtcreator/qml/qmlobserver/startup/white-star.png +usr/share/qtcreator/qml/qmlobserver/texteditautoresizer_maemo5.h +usr/share/qtcreator/qml/qmlpuppet/commands/changeauxiliarycommand.cpp +usr/share/qtcreator/qml/qmlpuppet/commands/changeauxiliarycommand.h +usr/share/qtcreator/qml/qmlpuppet/commands/changebindingscommand.cpp +usr/share/qtcreator/qml/qmlpuppet/commands/changebindingscommand.h +usr/share/qtcreator/qml/qmlpuppet/commands/changefileurlcommand.cpp +usr/share/qtcreator/qml/qmlpuppet/commands/changefileurlcommand.h +usr/share/qtcreator/qml/qmlpuppet/commands/changeidscommand.cpp +usr/share/qtcreator/qml/qmlpuppet/commands/changeidscommand.h +usr/share/qtcreator/qml/qmlpuppet/commands/changenodesourcecommand.cpp +usr/share/qtcreator/qml/qmlpuppet/commands/changenodesourcecommand.h +usr/share/qtcreator/qml/qmlpuppet/commands/changestatecommand.cpp +usr/share/qtcreator/qml/qmlpuppet/commands/changestatecommand.h +usr/share/qtcreator/qml/qmlpuppet/commands/changevaluescommand.cpp +usr/share/qtcreator/qml/qmlpuppet/commands/changevaluescommand.h +usr/share/qtcreator/qml/qmlpuppet/commands/childrenchangedcommand.cpp +usr/share/qtcreator/qml/qmlpuppet/commands/childrenchangedcommand.h +usr/share/qtcreator/qml/qmlpuppet/commands/clearscenecommand.cpp +usr/share/qtcreator/qml/qmlpuppet/commands/clearscenecommand.h +usr/share/qtcreator/qml/qmlpuppet/commands/commands.pri +usr/share/qtcreator/qml/qmlpuppet/commands/completecomponentcommand.cpp +usr/share/qtcreator/qml/qmlpuppet/commands/completecomponentcommand.h +usr/share/qtcreator/qml/qmlpuppet/commands/componentcompletedcommand.cpp +usr/share/qtcreator/qml/qmlpuppet/commands/componentcompletedcommand.h +usr/share/qtcreator/qml/qmlpuppet/commands/createinstancescommand.cpp +usr/share/qtcreator/qml/qmlpuppet/commands/createinstancescommand.h +usr/share/qtcreator/qml/qmlpuppet/commands/createscenecommand.cpp +usr/share/qtcreator/qml/qmlpuppet/commands/createscenecommand.h +usr/share/qtcreator/qml/qmlpuppet/commands/informationchangedcommand.cpp +usr/share/qtcreator/qml/qmlpuppet/commands/informationchangedcommand.h +usr/share/qtcreator/qml/qmlpuppet/commands/pixmapchangedcommand.cpp +usr/share/qtcreator/qml/qmlpuppet/commands/pixmapchangedcommand.h +usr/share/qtcreator/qml/qmlpuppet/commands/removeinstancescommand.cpp +usr/share/qtcreator/qml/qmlpuppet/commands/removeinstancescommand.h +usr/share/qtcreator/qml/qmlpuppet/commands/removepropertiescommand.cpp +usr/share/qtcreator/qml/qmlpuppet/commands/removepropertiescommand.h +usr/share/qtcreator/qml/qmlpuppet/commands/reparentinstancescommand.cpp +usr/share/qtcreator/qml/qmlpuppet/commands/reparentinstancescommand.h +usr/share/qtcreator/qml/qmlpuppet/commands/statepreviewimagechangedcommand.cpp +usr/share/qtcreator/qml/qmlpuppet/commands/statepreviewimagechangedcommand.h +usr/share/qtcreator/qml/qmlpuppet/commands/synchronizecommand.cpp +usr/share/qtcreator/qml/qmlpuppet/commands/synchronizecommand.h +usr/share/qtcreator/qml/qmlpuppet/commands/tokencommand.cpp +usr/share/qtcreator/qml/qmlpuppet/commands/tokencommand.h +usr/share/qtcreator/qml/qmlpuppet/commands/valueschangedcommand.cpp +usr/share/qtcreator/qml/qmlpuppet/commands/valueschangedcommand.h +usr/share/qtcreator/qml/qmlpuppet/container/addimportcontainer.cpp +usr/share/qtcreator/qml/qmlpuppet/container/addimportcontainer.h +usr/share/qtcreator/qml/qmlpuppet/container/container.pri +usr/share/qtcreator/qml/qmlpuppet/container/idcontainer.cpp +usr/share/qtcreator/qml/qmlpuppet/container/idcontainer.h +usr/share/qtcreator/qml/qmlpuppet/container/imagecontainer.cpp +usr/share/qtcreator/qml/qmlpuppet/container/imagecontainer.h +usr/share/qtcreator/qml/qmlpuppet/container/informationcontainer.cpp +usr/share/qtcreator/qml/qmlpuppet/container/informationcontainer.h +usr/share/qtcreator/qml/qmlpuppet/container/instancecontainer.cpp +usr/share/qtcreator/qml/qmlpuppet/container/instancecontainer.h +usr/share/qtcreator/qml/qmlpuppet/container/propertyabstractcontainer.cpp +usr/share/qtcreator/qml/qmlpuppet/container/propertyabstractcontainer.h +usr/share/qtcreator/qml/qmlpuppet/container/propertybindingcontainer.cpp +usr/share/qtcreator/qml/qmlpuppet/container/propertybindingcontainer.h +usr/share/qtcreator/qml/qmlpuppet/container/propertyvaluecontainer.cpp +usr/share/qtcreator/qml/qmlpuppet/container/propertyvaluecontainer.h +usr/share/qtcreator/qml/qmlpuppet/container/reparentcontainer.cpp +usr/share/qtcreator/qml/qmlpuppet/container/reparentcontainer.h +usr/share/qtcreator/qml/qmlpuppet/html/welcome.html +usr/share/qtcreator/qml/qmlpuppet/images/template_image.png +usr/share/qtcreator/qml/qmlpuppet/images/webkit.png +usr/share/qtcreator/qml/qmlpuppet/instances/anchorchangesnodeinstance.cpp +usr/share/qtcreator/qml/qmlpuppet/instances/anchorchangesnodeinstance.h +usr/share/qtcreator/qml/qmlpuppet/instances/behaviornodeinstance.cpp +usr/share/qtcreator/qml/qmlpuppet/instances/behaviornodeinstance.h +usr/share/qtcreator/qml/qmlpuppet/instances/childrenchangeeventfilter.cpp +usr/share/qtcreator/qml/qmlpuppet/instances/childrenchangeeventfilter.h +usr/share/qtcreator/qml/qmlpuppet/instances/componentnodeinstance.cpp +usr/share/qtcreator/qml/qmlpuppet/instances/componentnodeinstance.h +usr/share/qtcreator/qml/qmlpuppet/instances/dummycontextobject.cpp +usr/share/qtcreator/qml/qmlpuppet/instances/dummycontextobject.h +usr/share/qtcreator/qml/qmlpuppet/instances/dummynodeinstance.cpp +usr/share/qtcreator/qml/qmlpuppet/instances/dummynodeinstance.h +usr/share/qtcreator/qml/qmlpuppet/instances/instances.pri +usr/share/qtcreator/qml/qmlpuppet/instances/nodeinstanceclientproxy.cpp +usr/share/qtcreator/qml/qmlpuppet/instances/nodeinstanceclientproxy.h +usr/share/qtcreator/qml/qmlpuppet/instances/nodeinstancemetaobject.cpp +usr/share/qtcreator/qml/qmlpuppet/instances/nodeinstancemetaobject.h +usr/share/qtcreator/qml/qmlpuppet/instances/nodeinstanceserver.cpp +usr/share/qtcreator/qml/qmlpuppet/instances/nodeinstanceserver.h +usr/share/qtcreator/qml/qmlpuppet/instances/nodeinstancesignalspy.cpp +usr/share/qtcreator/qml/qmlpuppet/instances/nodeinstancesignalspy.h +usr/share/qtcreator/qml/qmlpuppet/instances/objectnodeinstance.cpp +usr/share/qtcreator/qml/qmlpuppet/instances/objectnodeinstance.h +usr/share/qtcreator/qml/qmlpuppet/instances/qmlpropertychangesnodeinstance.cpp +usr/share/qtcreator/qml/qmlpuppet/instances/qmlpropertychangesnodeinstance.h +usr/share/qtcreator/qml/qmlpuppet/instances/qmlstatenodeinstance.cpp +usr/share/qtcreator/qml/qmlpuppet/instances/qmlstatenodeinstance.h +usr/share/qtcreator/qml/qmlpuppet/instances/qmltransitionnodeinstance.cpp +usr/share/qtcreator/qml/qmlpuppet/instances/qmltransitionnodeinstance.h +usr/share/qtcreator/qml/qmlpuppet/instances/servernodeinstance.cpp +usr/share/qtcreator/qml/qmlpuppet/instances/servernodeinstance.h +usr/share/qtcreator/qml/qmlpuppet/interfaces/commondefines.h +usr/share/qtcreator/qml/qmlpuppet/interfaces/interfaces.pri +usr/share/qtcreator/qml/qmlpuppet/interfaces/nodeinstanceclientinterface.h +usr/share/qtcreator/qml/qmlpuppet/interfaces/nodeinstanceserverinterface.cpp +usr/share/qtcreator/qml/qmlpuppet/interfaces/nodeinstanceserverinterface.h +usr/share/qtcreator/qml/qmlpuppet/qml2puppet/Info.plist.in +usr/share/qtcreator/qml/qmlpuppet/qml2puppet/instances/instances.pri +usr/share/qtcreator/qml/qmlpuppet/qml2puppet/instances/qt5informationnodeinstanceserver.cpp +usr/share/qtcreator/qml/qmlpuppet/qml2puppet/instances/qt5informationnodeinstanceserver.h +usr/share/qtcreator/qml/qmlpuppet/qml2puppet/instances/qt5nodeinstanceclientproxy.cpp +usr/share/qtcreator/qml/qmlpuppet/qml2puppet/instances/qt5nodeinstanceclientproxy.h +usr/share/qtcreator/qml/qmlpuppet/qml2puppet/instances/qt5nodeinstanceserver.cpp +usr/share/qtcreator/qml/qmlpuppet/qml2puppet/instances/qt5nodeinstanceserver.h +usr/share/qtcreator/qml/qmlpuppet/qml2puppet/instances/qt5previewnodeinstanceserver.cpp +usr/share/qtcreator/qml/qmlpuppet/qml2puppet/instances/qt5previewnodeinstanceserver.h +usr/share/qtcreator/qml/qmlpuppet/qml2puppet/instances/qt5rendernodeinstanceserver.cpp +usr/share/qtcreator/qml/qmlpuppet/qml2puppet/instances/qt5rendernodeinstanceserver.h +usr/share/qtcreator/qml/qmlpuppet/qml2puppet/instances/sgitemnodeinstance.cpp +usr/share/qtcreator/qml/qmlpuppet/qml2puppet/instances/sgitemnodeinstance.h +usr/share/qtcreator/qml/qmlpuppet/qml2puppet/main.cpp +usr/share/qtcreator/qml/qmlpuppet/qml2puppet/qml2puppet.pro +usr/share/qtcreator/qml/qmlpuppet/qmlpuppet.pro +usr/share/qtcreator/qml/qmlpuppet/qmlpuppet.qrc +usr/share/qtcreator/qml/qmlpuppet/qmlpuppet/Info.plist.in +usr/share/qtcreator/qml/qmlpuppet/qmlpuppet/instances/graphicsobjectnodeinstance.cpp +usr/share/qtcreator/qml/qmlpuppet/qmlpuppet/instances/graphicsobjectnodeinstance.h +usr/share/qtcreator/qml/qmlpuppet/qmlpuppet/instances/instances.pri +usr/share/qtcreator/qml/qmlpuppet/qmlpuppet/instances/positionernodeinstance.cpp +usr/share/qtcreator/qml/qmlpuppet/qmlpuppet/instances/positionernodeinstance.h +usr/share/qtcreator/qml/qmlpuppet/qmlpuppet/instances/qmlgraphicsitemnodeinstance.cpp +usr/share/qtcreator/qml/qmlpuppet/qmlpuppet/instances/qmlgraphicsitemnodeinstance.h +usr/share/qtcreator/qml/qmlpuppet/qmlpuppet/instances/qt4informationnodeinstanceserver.cpp +usr/share/qtcreator/qml/qmlpuppet/qmlpuppet/instances/qt4informationnodeinstanceserver.h +usr/share/qtcreator/qml/qmlpuppet/qmlpuppet/instances/qt4nodeinstanceclientproxy.cpp +usr/share/qtcreator/qml/qmlpuppet/qmlpuppet/instances/qt4nodeinstanceclientproxy.h +usr/share/qtcreator/qml/qmlpuppet/qmlpuppet/instances/qt4nodeinstanceserver.cpp +usr/share/qtcreator/qml/qmlpuppet/qmlpuppet/instances/qt4nodeinstanceserver.h +usr/share/qtcreator/qml/qmlpuppet/qmlpuppet/instances/qt4previewnodeinstanceserver.cpp +usr/share/qtcreator/qml/qmlpuppet/qmlpuppet/instances/qt4previewnodeinstanceserver.h +usr/share/qtcreator/qml/qmlpuppet/qmlpuppet/instances/qt4rendernodeinstanceserver.cpp +usr/share/qtcreator/qml/qmlpuppet/qmlpuppet/instances/qt4rendernodeinstanceserver.h +usr/share/qtcreator/qml/qmlpuppet/qmlpuppet/main.cpp +usr/share/qtcreator/qml/qmlpuppet/qmlpuppet/qmlpuppet.pri +usr/share/qtcreator/qml/qmlpuppet/qmlpuppet/qmlpuppet.pro +usr/share/qtcreator/qml/qmlpuppet/qmlpuppet_utilities.pri +usr/share/qtcreator/qmldesigner/propertyeditor/Qt/AlignmentHorizontalButtons.qml +usr/share/qtcreator/qmldesigner/propertyeditor/Qt/AlignmentVerticalButtons.qml +usr/share/qtcreator/qmldesigner/propertyeditor/Qt/AnchorBox.qml +usr/share/qtcreator/qmldesigner/propertyeditor/Qt/AnchorButtons.qml +usr/share/qtcreator/qmldesigner/propertyeditor/Qt/BorderImageSpecifics.qml +usr/share/qtcreator/qmldesigner/propertyeditor/Qt/CheckBox.qml +usr/share/qtcreator/qmldesigner/propertyeditor/Qt/ColorGroupBox.qml +usr/share/qtcreator/qmldesigner/propertyeditor/Qt/ColorLabel.qml +usr/share/qtcreator/qmldesigner/propertyeditor/Qt/ColorScheme.qml +usr/share/qtcreator/qmldesigner/propertyeditor/Qt/ColorTypeButtons.qml +usr/share/qtcreator/qmldesigner/propertyeditor/Qt/ComboBox.qml +usr/share/qtcreator/qmldesigner/propertyeditor/Qt/DoubleSpinBox.qml +usr/share/qtcreator/qmldesigner/propertyeditor/Qt/DoubleSpinBoxAlternate.qml +usr/share/qtcreator/qmldesigner/propertyeditor/Qt/ExpressionEditor.qml +usr/share/qtcreator/qmldesigner/propertyeditor/Qt/Extended.qml +usr/share/qtcreator/qmldesigner/propertyeditor/Qt/ExtendedFunctionButton.qml +usr/share/qtcreator/qmldesigner/propertyeditor/Qt/ExtendedPane.qml +usr/share/qtcreator/qmldesigner/propertyeditor/Qt/ExtendedSwitches.qml +usr/share/qtcreator/qmldesigner/propertyeditor/Qt/FlagedButton.qml +usr/share/qtcreator/qmldesigner/propertyeditor/Qt/FlickableGroupBox.qml +usr/share/qtcreator/qmldesigner/propertyeditor/Qt/FlickableSpecifics.qml +usr/share/qtcreator/qmldesigner/propertyeditor/Qt/FlipableSpecifics.qml +usr/share/qtcreator/qmldesigner/propertyeditor/Qt/FlowSpecifics.qml +usr/share/qtcreator/qmldesigner/propertyeditor/Qt/FontComboBox.qml +usr/share/qtcreator/qmldesigner/propertyeditor/Qt/FontGroupBox.qml +usr/share/qtcreator/qmldesigner/propertyeditor/Qt/FontStyleButtons.qml +usr/share/qtcreator/qmldesigner/propertyeditor/Qt/Geometry.qml +usr/share/qtcreator/qmldesigner/propertyeditor/Qt/GridSpecifics.qml +usr/share/qtcreator/qmldesigner/propertyeditor/Qt/GridViewSpecifics.qml +usr/share/qtcreator/qmldesigner/propertyeditor/Qt/GroupBox.qml +usr/share/qtcreator/qmldesigner/propertyeditor/Qt/GroupBoxOption.qml +usr/share/qtcreator/qmldesigner/propertyeditor/Qt/HorizontalLayout.qml +usr/share/qtcreator/qmldesigner/propertyeditor/Qt/HorizontalWhiteLine.qml +usr/share/qtcreator/qmldesigner/propertyeditor/Qt/ImageSpecifics.qml +usr/share/qtcreator/qmldesigner/propertyeditor/Qt/IntEditor.qml +usr/share/qtcreator/qmldesigner/propertyeditor/Qt/ItemPane.qml +usr/share/qtcreator/qmldesigner/propertyeditor/Qt/Label.qml +usr/share/qtcreator/qmldesigner/propertyeditor/Qt/Layout.qml +usr/share/qtcreator/qmldesigner/propertyeditor/Qt/LayoutPane.qml +usr/share/qtcreator/qmldesigner/propertyeditor/Qt/LineEdit.qml +usr/share/qtcreator/qmldesigner/propertyeditor/Qt/ListViewSpecifics.qml +usr/share/qtcreator/qmldesigner/propertyeditor/Qt/Modifiers.qml +usr/share/qtcreator/qmldesigner/propertyeditor/Qt/MouseAreaSpecifics.qml +usr/share/qtcreator/qmldesigner/propertyeditor/Qt/PathViewSpecifics.qml +usr/share/qtcreator/qmldesigner/propertyeditor/Qt/PlaceHolder.qml +usr/share/qtcreator/qmldesigner/propertyeditor/Qt/PropertyFrame.qml +usr/share/qtcreator/qmldesigner/propertyeditor/Qt/RectangleColorGroupBox.qml +usr/share/qtcreator/qmldesigner/propertyeditor/Qt/RectangleSpecifics.qml +usr/share/qtcreator/qmldesigner/propertyeditor/Qt/RowSpecifics.qml +usr/share/qtcreator/qmldesigner/propertyeditor/Qt/ScrollArea.qml +usr/share/qtcreator/qmldesigner/propertyeditor/Qt/SliderWidget.qml +usr/share/qtcreator/qmldesigner/propertyeditor/Qt/SpinBox.qml +usr/share/qtcreator/qmldesigner/propertyeditor/Qt/StandardTextColorGroupBox.qml +usr/share/qtcreator/qmldesigner/propertyeditor/Qt/StandardTextGroupBox.qml +usr/share/qtcreator/qmldesigner/propertyeditor/Qt/Switches.qml +usr/share/qtcreator/qmldesigner/propertyeditor/Qt/TextEditSpecifics.qml +usr/share/qtcreator/qmldesigner/propertyeditor/Qt/TextEditor.qml +usr/share/qtcreator/qmldesigner/propertyeditor/Qt/TextInputGroupBox.qml +usr/share/qtcreator/qmldesigner/propertyeditor/Qt/TextInputSpecifics.qml +usr/share/qtcreator/qmldesigner/propertyeditor/Qt/TextSpecifics.qml +usr/share/qtcreator/qmldesigner/propertyeditor/Qt/Transformation.qml +usr/share/qtcreator/qmldesigner/propertyeditor/Qt/Type.qml +usr/share/qtcreator/qmldesigner/propertyeditor/Qt/VerticalLayout.qml +usr/share/qtcreator/qmldesigner/propertyeditor/Qt/Visibility.qml +usr/share/qtcreator/qmldesigner/propertyeditor/Qt/anchorbottom.css +usr/share/qtcreator/qmldesigner/propertyeditor/Qt/anchorbox.css +usr/share/qtcreator/qmldesigner/propertyeditor/Qt/anchorfill.css +usr/share/qtcreator/qmldesigner/propertyeditor/Qt/anchorhorizontal.css +usr/share/qtcreator/qmldesigner/propertyeditor/Qt/anchorleft.css +usr/share/qtcreator/qmldesigner/propertyeditor/Qt/anchorright.css +usr/share/qtcreator/qmldesigner/propertyeditor/Qt/anchorspacer.css +usr/share/qtcreator/qmldesigner/propertyeditor/Qt/anchortop.css +usr/share/qtcreator/qmldesigner/propertyeditor/Qt/anchorvertical.css +usr/share/qtcreator/qmldesigner/propertyeditor/Qt/applybutton.css +usr/share/qtcreator/qmldesigner/propertyeditor/Qt/aspectlock.css +usr/share/qtcreator/qmldesigner/propertyeditor/Qt/cancelbutton.css +usr/share/qtcreator/qmldesigner/propertyeditor/Qt/checkbox_tr.css +usr/share/qtcreator/qmldesigner/propertyeditor/Qt/emptyPane.qml +usr/share/qtcreator/qmldesigner/propertyeditor/Qt/images/alignmentbottom-h-icon.png +usr/share/qtcreator/qmldesigner/propertyeditor/Qt/images/alignmentbottom-icon.png +usr/share/qtcreator/qmldesigner/propertyeditor/Qt/images/alignmentcenterh-h-icon.png +usr/share/qtcreator/qmldesigner/propertyeditor/Qt/images/alignmentcenterh-icon.png +usr/share/qtcreator/qmldesigner/propertyeditor/Qt/images/alignmentleft-h-icon.png +usr/share/qtcreator/qmldesigner/propertyeditor/Qt/images/alignmentleft-icon.png +usr/share/qtcreator/qmldesigner/propertyeditor/Qt/images/alignmentmiddle-h-icon.png +usr/share/qtcreator/qmldesigner/propertyeditor/Qt/images/alignmentmiddle-icon.png +usr/share/qtcreator/qmldesigner/propertyeditor/Qt/images/alignmentright-h-icon.png +usr/share/qtcreator/qmldesigner/propertyeditor/Qt/images/alignmentright-icon.png +usr/share/qtcreator/qmldesigner/propertyeditor/Qt/images/alignmenttop-h-icon.png +usr/share/qtcreator/qmldesigner/propertyeditor/Qt/images/alignmenttop-icon.png +usr/share/qtcreator/qmldesigner/propertyeditor/Qt/images/apply.png +usr/share/qtcreator/qmldesigner/propertyeditor/Qt/images/behaivour.png +usr/share/qtcreator/qmldesigner/propertyeditor/Qt/images/blended-image-icon.png +usr/share/qtcreator/qmldesigner/propertyeditor/Qt/images/bold-h-icon.png +usr/share/qtcreator/qmldesigner/propertyeditor/Qt/images/bold-icon.png +usr/share/qtcreator/qmldesigner/propertyeditor/Qt/images/button.png +usr/share/qtcreator/qmldesigner/propertyeditor/Qt/images/cancel.png +usr/share/qtcreator/qmldesigner/propertyeditor/Qt/images/default-icon.png +usr/share/qtcreator/qmldesigner/propertyeditor/Qt/images/downArrow.png +usr/share/qtcreator/qmldesigner/propertyeditor/Qt/images/expression.png +usr/share/qtcreator/qmldesigner/propertyeditor/Qt/images/extended.png +usr/share/qtcreator/qmldesigner/propertyeditor/Qt/images/grid-icon.png +usr/share/qtcreator/qmldesigner/propertyeditor/Qt/images/icon_color_gradient.png +usr/share/qtcreator/qmldesigner/propertyeditor/Qt/images/icon_color_none.png +usr/share/qtcreator/qmldesigner/propertyeditor/Qt/images/icon_color_solid.png +usr/share/qtcreator/qmldesigner/propertyeditor/Qt/images/image-icon.png +usr/share/qtcreator/qmldesigner/propertyeditor/Qt/images/italic-h-icon.png +usr/share/qtcreator/qmldesigner/propertyeditor/Qt/images/italic-icon.png +usr/share/qtcreator/qmldesigner/propertyeditor/Qt/images/item-icon.png +usr/share/qtcreator/qmldesigner/propertyeditor/Qt/images/layout.png +usr/share/qtcreator/qmldesigner/propertyeditor/Qt/images/leftArrow.png +usr/share/qtcreator/qmldesigner/propertyeditor/Qt/images/list-icon.png +usr/share/qtcreator/qmldesigner/propertyeditor/Qt/images/mouse-area-icon.png +usr/share/qtcreator/qmldesigner/propertyeditor/Qt/images/placeholder.png +usr/share/qtcreator/qmldesigner/propertyeditor/Qt/images/rect-icon.png +usr/share/qtcreator/qmldesigner/propertyeditor/Qt/images/reset-button.png +usr/share/qtcreator/qmldesigner/propertyeditor/Qt/images/rightArrow.png +usr/share/qtcreator/qmldesigner/propertyeditor/Qt/images/standard.png +usr/share/qtcreator/qmldesigner/propertyeditor/Qt/images/strikeout-h-icon.png +usr/share/qtcreator/qmldesigner/propertyeditor/Qt/images/strikeout-icon.png +usr/share/qtcreator/qmldesigner/propertyeditor/Qt/images/submenu.png +usr/share/qtcreator/qmldesigner/propertyeditor/Qt/images/text-edit-icon.png +usr/share/qtcreator/qmldesigner/propertyeditor/Qt/images/text-icon.png +usr/share/qtcreator/qmldesigner/propertyeditor/Qt/images/underline-h-icon.png +usr/share/qtcreator/qmldesigner/propertyeditor/Qt/images/underline-icon.png +usr/share/qtcreator/qmldesigner/propertyeditor/Qt/images/upArrow.png +usr/share/qtcreator/qmldesigner/propertyeditor/Qt/layoutWidget.css +usr/share/qtcreator/qmldesigner/propertyeditor/Qt/propertyEditor.css +usr/share/qtcreator/qmldesigner/propertyeditor/Qt/specialCheckBox.css +usr/share/qtcreator/qmldesigner/propertyeditor/Qt/styledbuttonleft.css +usr/share/qtcreator/qmldesigner/propertyeditor/Qt/styledbuttonmiddle.css +usr/share/qtcreator/qmldesigner/propertyeditor/Qt/styledbuttonright.css +usr/share/qtcreator/qmldesigner/propertyeditor/Qt/switch.css +usr/share/qtcreator/qmldesigner/propertyeditor/Qt/typeLabel.css +usr/share/qtcreator/qmldesigner/propertyeditor/QtWebKit/WebViewSpecifics.qml +usr/share/qtcreator/qmlicons/Qt/16x16/BorderImage.png +usr/share/qtcreator/qmlicons/Qt/16x16/BusyIndicator.png +usr/share/qtcreator/qmlicons/Qt/16x16/Button.png +usr/share/qtcreator/qmlicons/Qt/16x16/ButtonColumn.png +usr/share/qtcreator/qmlicons/Qt/16x16/ButtonRow.png +usr/share/qtcreator/qmlicons/Qt/16x16/CheckBox.png +usr/share/qtcreator/qmlicons/Qt/16x16/ChoiceList.png +usr/share/qtcreator/qmlicons/Qt/16x16/ColorAnimation.png +usr/share/qtcreator/qmlicons/Qt/16x16/Component.png +usr/share/qtcreator/qmlicons/Qt/16x16/CountBubble.png +usr/share/qtcreator/qmlicons/Qt/16x16/DatePickerDialog.png +usr/share/qtcreator/qmlicons/Qt/16x16/Flickable.png +usr/share/qtcreator/qmlicons/Qt/16x16/Flipable.png +usr/share/qtcreator/qmlicons/Qt/16x16/FocusScope.png +usr/share/qtcreator/qmlicons/Qt/16x16/GridView.png +usr/share/qtcreator/qmlicons/Qt/16x16/Image.png +usr/share/qtcreator/qmlicons/Qt/16x16/InfoBanner.png +usr/share/qtcreator/qmlicons/Qt/16x16/Item.png +usr/share/qtcreator/qmlicons/Qt/16x16/ListButton.png +usr/share/qtcreator/qmlicons/Qt/16x16/ListDelegate.png +usr/share/qtcreator/qmlicons/Qt/16x16/ListView.png +usr/share/qtcreator/qmlicons/Qt/16x16/MoreIndicator.png +usr/share/qtcreator/qmlicons/Qt/16x16/MouseArea.png +usr/share/qtcreator/qmlicons/Qt/16x16/PageIndicator.png +usr/share/qtcreator/qmlicons/Qt/16x16/ParallelAnimation.png +usr/share/qtcreator/qmlicons/Qt/16x16/PathView.png +usr/share/qtcreator/qmlicons/Qt/16x16/PauseAnimation.png +usr/share/qtcreator/qmlicons/Qt/16x16/ProgressBar.png +usr/share/qtcreator/qmlicons/Qt/16x16/PropertyChanges.png +usr/share/qtcreator/qmlicons/Qt/16x16/RadioButton.png +usr/share/qtcreator/qmlicons/Qt/16x16/RatingIndicator.png +usr/share/qtcreator/qmlicons/Qt/16x16/Rectangle.png +usr/share/qtcreator/qmlicons/Qt/16x16/SequentialAnimation.png +usr/share/qtcreator/qmlicons/Qt/16x16/Slider.png +usr/share/qtcreator/qmlicons/Qt/16x16/State.png +usr/share/qtcreator/qmlicons/Qt/16x16/Switch.png +usr/share/qtcreator/qmlicons/Qt/16x16/TabBar.png +usr/share/qtcreator/qmlicons/Qt/16x16/TabButton.png +usr/share/qtcreator/qmlicons/Qt/16x16/Text.png +usr/share/qtcreator/qmlicons/Qt/16x16/TextArea.png +usr/share/qtcreator/qmlicons/Qt/16x16/TextEdit.png +usr/share/qtcreator/qmlicons/Qt/16x16/TextField.png +usr/share/qtcreator/qmlicons/Qt/16x16/TextInput.png +usr/share/qtcreator/qmlicons/Qt/16x16/TimePickerDialog.png +usr/share/qtcreator/qmlicons/Qt/16x16/ToolBar.png +usr/share/qtcreator/qmlicons/Qt/16x16/Transition.png +usr/share/qtcreator/qmlicons/Qt/16x16/Tumbler.png +usr/share/qtcreator/qmlicons/Qt/16x16/TumblerButton.png +usr/share/qtcreator/qmlicons/Qt/16x16/TumblerColumn.png +usr/share/qtcreator/qmlicons/Qt/16x16/TumblerDialog.png +usr/share/qtcreator/qmlicons/Qt/16x16/Window.png +usr/share/qtcreator/qmlicons/Qt/16x16/item-icon16.png +usr/share/qtcreator/qmlicons/QtWebkit/16x16/WebView.png +usr/share/qtcreator/schemes/MS_Visual_C++.kms +usr/share/qtcreator/schemes/Xcode.kms +usr/share/qtcreator/snippets/cpp.xml +usr/share/qtcreator/snippets/qml.xml +usr/share/qtcreator/snippets/text.xml +usr/share/qtcreator/styles/darkvim.xml +usr/share/qtcreator/styles/default.xml +usr/share/qtcreator/styles/grayscale.xml +usr/share/qtcreator/styles/inkpot.xml +usr/share/qtcreator/styles/intellij.xml +usr/share/qtcreator/templates/html5app/app.pro +usr/share/qtcreator/templates/html5app/html/index.html +usr/share/qtcreator/templates/html5app/html5applicationviewer/html5applicationviewer.cpp +usr/share/qtcreator/templates/html5app/html5applicationviewer/html5applicationviewer.h +usr/share/qtcreator/templates/html5app/html5applicationviewer/html5applicationviewer.pri +usr/share/qtcreator/templates/html5app/html5applicationviewer/touchnavigation/navigationcontroller.cpp +usr/share/qtcreator/templates/html5app/html5applicationviewer/touchnavigation/navigationcontroller.h +usr/share/qtcreator/templates/html5app/html5applicationviewer/touchnavigation/touchnavigation.pri +usr/share/qtcreator/templates/html5app/html5applicationviewer/touchnavigation/webnavigation.cpp +usr/share/qtcreator/templates/html5app/html5applicationviewer/touchnavigation/webnavigation.h +usr/share/qtcreator/templates/html5app/html5applicationviewer/touchnavigation/webtouchevent.cpp +usr/share/qtcreator/templates/html5app/html5applicationviewer/touchnavigation/webtouchevent.h +usr/share/qtcreator/templates/html5app/html5applicationviewer/touchnavigation/webtouchnavigation.cpp +usr/share/qtcreator/templates/html5app/html5applicationviewer/touchnavigation/webtouchnavigation.h +usr/share/qtcreator/templates/html5app/html5applicationviewer/touchnavigation/webtouchphysics.cpp +usr/share/qtcreator/templates/html5app/html5applicationviewer/touchnavigation/webtouchphysics.h +usr/share/qtcreator/templates/html5app/html5applicationviewer/touchnavigation/webtouchphysicsinterface.cpp +usr/share/qtcreator/templates/html5app/html5applicationviewer/touchnavigation/webtouchphysicsinterface.h +usr/share/qtcreator/templates/html5app/html5applicationviewer/touchnavigation/webtouchscroller.cpp +usr/share/qtcreator/templates/html5app/html5applicationviewer/touchnavigation/webtouchscroller.h +usr/share/qtcreator/templates/html5app/main.cpp +usr/share/qtcreator/templates/mobileapp/app.pro +usr/share/qtcreator/templates/mobileapp/main.cpp +usr/share/qtcreator/templates/mobileapp/mainwindow.cpp +usr/share/qtcreator/templates/mobileapp/mainwindow.h +usr/share/qtcreator/templates/mobileapp/mainwindow.ui +usr/share/qtcreator/templates/qt4project/customwidgetwizard/tpl_collection.cpp +usr/share/qtcreator/templates/qt4project/customwidgetwizard/tpl_collection.h +usr/share/qtcreator/templates/qt4project/customwidgetwizard/tpl_plugin.pro +usr/share/qtcreator/templates/qt4project/customwidgetwizard/tpl_resources.qrc +usr/share/qtcreator/templates/qt4project/customwidgetwizard/tpl_single.cpp +usr/share/qtcreator/templates/qt4project/customwidgetwizard/tpl_single.h +usr/share/qtcreator/templates/qt4project/customwidgetwizard/tpl_widget.cpp +usr/share/qtcreator/templates/qt4project/customwidgetwizard/tpl_widget.h +usr/share/qtcreator/templates/qt4project/customwidgetwizard/tpl_widget_include.pri +usr/share/qtcreator/templates/qt4project/customwidgetwizard/tpl_widget_lib.pro +usr/share/qtcreator/templates/qt4project/main.cpp +usr/share/qtcreator/templates/qt4project/mywidget.cpp +usr/share/qtcreator/templates/qt4project/mywidget.h +usr/share/qtcreator/templates/qt4project/mywidget_form.cpp +usr/share/qtcreator/templates/qt4project/mywidget_form.h +usr/share/qtcreator/templates/qt4project/widget.ui +usr/share/qtcreator/templates/qtquickapp/app.pro +usr/share/qtcreator/templates/qtquickapp/main.cpp +usr/share/qtcreator/templates/qtquickapp/qml/app/meego10/MainPage.qml +usr/share/qtcreator/templates/qtquickapp/qml/app/meego10/main.qml +usr/share/qtcreator/templates/qtquickapp/qml/app/qtquick10/main.qml +usr/share/qtcreator/templates/qtquickapp/qml/app/symbian11/MainPage.qml +usr/share/qtcreator/templates/qtquickapp/qml/app/symbian11/main.qml +usr/share/qtcreator/templates/qtquickapp/qmlapplicationviewer/qmlapplicationviewer.cpp +usr/share/qtcreator/templates/qtquickapp/qmlapplicationviewer/qmlapplicationviewer.h +usr/share/qtcreator/templates/qtquickapp/qmlapplicationviewer/qmlapplicationviewer.pri +usr/share/qtcreator/templates/shared/app.desktop +usr/share/qtcreator/templates/shared/deployment.pri +usr/share/qtcreator/templates/shared/icon64.png +usr/share/qtcreator/templates/shared/icon80.png +usr/share/qtcreator/templates/shared/manifest.aegis +usr/share/qtcreator/templates/shared/symbianicon.svg +usr/share/qtcreator/templates/wizards/README.txt +usr/share/qtcreator/templates/wizards/helloworld/console.png +usr/share/qtcreator/templates/wizards/helloworld/main.cpp +usr/share/qtcreator/templates/wizards/helloworld/project.pro +usr/share/qtcreator/templates/wizards/helloworld/wizard_sample.xml +usr/share/qtcreator/templates/wizards/listmodel/listmodel.cpp +usr/share/qtcreator/templates/wizards/listmodel/listmodel.h +usr/share/qtcreator/templates/wizards/listmodel/wizard_sample.xml +usr/share/qtcreator/templates/wizards/plaincapp-cmake/CMakeLists.txt +usr/share/qtcreator/templates/wizards/plaincapp-cmake/console.png +usr/share/qtcreator/templates/wizards/plaincapp-cmake/main.c +usr/share/qtcreator/templates/wizards/plaincapp-cmake/wizard.xml +usr/share/qtcreator/templates/wizards/plaincapp/console.png +usr/share/qtcreator/templates/wizards/plaincapp/main.c +usr/share/qtcreator/templates/wizards/plaincapp/project.pro +usr/share/qtcreator/templates/wizards/plaincapp/wizard.xml +usr/share/qtcreator/templates/wizards/plaincppapp-cmake/CMakeLists.txt +usr/share/qtcreator/templates/wizards/plaincppapp-cmake/console.png +usr/share/qtcreator/templates/wizards/plaincppapp-cmake/main.cpp +usr/share/qtcreator/templates/wizards/plaincppapp-cmake/wizard.xml +usr/share/qtcreator/templates/wizards/plaincppapp/console.png +usr/share/qtcreator/templates/wizards/plaincppapp/main.cpp +usr/share/qtcreator/templates/wizards/plaincppapp/project.pro +usr/share/qtcreator/templates/wizards/plaincppapp/wizard.xml +usr/share/qtcreator/templates/wizards/qml-extension/lib.png +usr/share/qtcreator/templates/wizards/qml-extension/object.cpp +usr/share/qtcreator/templates/wizards/qml-extension/object.h +usr/share/qtcreator/templates/wizards/qml-extension/plugin.cpp +usr/share/qtcreator/templates/wizards/qml-extension/plugin.h +usr/share/qtcreator/templates/wizards/qml-extension/project.pro +usr/share/qtcreator/templates/wizards/qml-extension/qmldir +usr/share/qtcreator/templates/wizards/qml-extension/wizard.xml +usr/share/qtcreator/templates/wizards/qtcreatorplugin/MyPlugin.pluginspec.in +usr/share/qtcreator/templates/wizards/qtcreatorplugin/myplugin.cpp +usr/share/qtcreator/templates/wizards/qtcreatorplugin/myplugin.h +usr/share/qtcreator/templates/wizards/qtcreatorplugin/myplugin.pro +usr/share/qtcreator/templates/wizards/qtcreatorplugin/myplugin_global.h +usr/share/qtcreator/templates/wizards/qtcreatorplugin/mypluginconstants.h +usr/share/qtcreator/templates/wizards/qtcreatorplugin/qtcreator_logo_24.png +usr/share/qtcreator/templates/wizards/qtcreatorplugin/wizard.xml +usr/share/qtcreator/templates/wizards/scriptgeneratedproject/generate.pl +usr/share/qtcreator/templates/wizards/scriptgeneratedproject/wizard_sample.xml +usr/share/qtcreator/translations/qtcreator_*.qm +usr/share/qtcreator/welcomescreen/develop.qml +usr/share/qtcreator/welcomescreen/dummydata/examplesModel.qml +usr/share/qtcreator/welcomescreen/dummydata/pagesModel.qml +usr/share/qtcreator/welcomescreen/dummydata/projectList.qml +usr/share/qtcreator/welcomescreen/dummydata/sessionList.qml +usr/share/qtcreator/welcomescreen/dummydata/tutorialsModel.qml +usr/share/qtcreator/welcomescreen/examples.qml +usr/share/qtcreator/welcomescreen/examples_fallback.xml +usr/share/qtcreator/welcomescreen/gettingstarted.qml +usr/share/qtcreator/welcomescreen/images_areaofinterest.xml +usr/share/qtcreator/welcomescreen/qtcreator_tutorials.xml +usr/share/qtcreator/welcomescreen/tutorials.qml +usr/share/qtcreator/welcomescreen/welcomescreen.qml +usr/share/qtcreator/welcomescreen/welcomescreen.qmlproject +usr/share/qtcreator/welcomescreen/widgets/CustomColors.qml +usr/share/qtcreator/welcomescreen/widgets/CustomFonts.qml +usr/share/qtcreator/welcomescreen/widgets/CustomTab.qml +usr/share/qtcreator/welcomescreen/widgets/CustomizedGridView.qml +usr/share/qtcreator/welcomescreen/widgets/Delegate.qml +usr/share/qtcreator/welcomescreen/widgets/Feedback.qml +usr/share/qtcreator/welcomescreen/widgets/GettingStartedItem.qml +usr/share/qtcreator/welcomescreen/widgets/LinkedText.qml +usr/share/qtcreator/welcomescreen/widgets/LinksBar.qml +usr/share/qtcreator/welcomescreen/widgets/Logo.qml +usr/share/qtcreator/welcomescreen/widgets/PageCaption.qml +usr/share/qtcreator/welcomescreen/widgets/PageLoader.qml +usr/share/qtcreator/welcomescreen/widgets/ProjectItem.qml +usr/share/qtcreator/welcomescreen/widgets/RecentProjects.qml +usr/share/qtcreator/welcomescreen/widgets/SearchBar.qml +usr/share/qtcreator/welcomescreen/widgets/SessionItem.qml +usr/share/qtcreator/welcomescreen/widgets/Sessions.qml +usr/share/qtcreator/welcomescreen/widgets/ToolTip.qml +usr/share/qtcreator/welcomescreen/widgets/dummydata/context/ExampleDelegate.qml +usr/share/qtcreator/welcomescreen/widgets/dummydata/context/ExampleGridView.qml +usr/share/qtcreator/welcomescreen/widgets/dummydata/examplesModel.qml +usr/share/qtcreator/welcomescreen/widgets/dummydata/mockupTags.qml +usr/share/qtcreator/welcomescreen/widgets/dummydata/pagesModel.qml +usr/share/qtcreator/welcomescreen/widgets/dummydata/tabsModel.qml +usr/share/qtcreator/welcomescreen/widgets/images/arrowBig.png +usr/share/qtcreator/welcomescreen/widgets/images/arrow_down.png +usr/share/qtcreator/welcomescreen/widgets/images/arrow_up.png +usr/share/qtcreator/welcomescreen/widgets/images/bullet.png +usr/share/qtcreator/welcomescreen/widgets/images/dropshadow.png +usr/share/qtcreator/welcomescreen/widgets/images/gettingStarted01.png +usr/share/qtcreator/welcomescreen/widgets/images/gettingStarted02.png +usr/share/qtcreator/welcomescreen/widgets/images/gettingStarted03.png +usr/share/qtcreator/welcomescreen/widgets/images/gettingStarted04.png +usr/share/qtcreator/welcomescreen/widgets/images/icons/adressbook.png +usr/share/qtcreator/welcomescreen/widgets/images/icons/buildrun.png +usr/share/qtcreator/welcomescreen/widgets/images/icons/clone.png +usr/share/qtcreator/welcomescreen/widgets/images/icons/communityIcon.png +usr/share/qtcreator/welcomescreen/widgets/images/icons/components.png +usr/share/qtcreator/welcomescreen/widgets/images/icons/createIcon.png +usr/share/qtcreator/welcomescreen/widgets/images/icons/ddays09.png +usr/share/qtcreator/welcomescreen/widgets/images/icons/ddays10.png +usr/share/qtcreator/welcomescreen/widgets/images/icons/ddays11.png +usr/share/qtcreator/welcomescreen/widgets/images/icons/delete.png +usr/share/qtcreator/welcomescreen/widgets/images/icons/developing_with_qt_creator.png +usr/share/qtcreator/welcomescreen/widgets/images/icons/feedbackIcon.png +usr/share/qtcreator/welcomescreen/widgets/images/icons/ico_community.png +usr/share/qtcreator/welcomescreen/widgets/images/icons/labsIcon.png +usr/share/qtcreator/welcomescreen/widgets/images/icons/openIcon.png +usr/share/qtcreator/welcomescreen/widgets/images/icons/qt_quick_1.png +usr/share/qtcreator/welcomescreen/widgets/images/icons/qt_quick_2.png +usr/share/qtcreator/welcomescreen/widgets/images/icons/qt_quick_3.png +usr/share/qtcreator/welcomescreen/widgets/images/icons/qt_sdk.png +usr/share/qtcreator/welcomescreen/widgets/images/icons/qtquick.png +usr/share/qtcreator/welcomescreen/widgets/images/icons/qwidget.png +usr/share/qtcreator/welcomescreen/widgets/images/icons/rename.png +usr/share/qtcreator/welcomescreen/widgets/images/icons/userguideIcon.png +usr/share/qtcreator/welcomescreen/widgets/images/icons/videoIcon.png +usr/share/qtcreator/welcomescreen/widgets/images/info.png +usr/share/qtcreator/welcomescreen/widgets/images/mockup/designer-examples.png +usr/share/qtcreator/welcomescreen/widgets/images/mockup/desktop-examples.png +usr/share/qtcreator/welcomescreen/widgets/images/mockup/draganddrop-examples.png +usr/share/qtcreator/welcomescreen/widgets/images/mockup/itemview-examples.png +usr/share/qtcreator/welcomescreen/widgets/images/mockup/layout-examples.png +usr/share/qtcreator/welcomescreen/widgets/images/mockup/mainwindow-examples.png +usr/share/qtcreator/welcomescreen/widgets/images/mockup/network-examples.png +usr/share/qtcreator/welcomescreen/widgets/images/mockup/opengl-examples.png +usr/share/qtcreator/welcomescreen/widgets/images/mockup/penguin.png +usr/share/qtcreator/welcomescreen/widgets/images/mockup/qtscript-examples.png +usr/share/qtcreator/welcomescreen/widgets/images/mockup/thread-examples.png +usr/share/qtcreator/welcomescreen/widgets/images/more.png +usr/share/qtcreator/welcomescreen/widgets/images/qtcreator.png +usr/share/qtcreator/welcomescreen/widgets/images/tab.png diff -Nru qtcreator-2.5.0/debian/qtcreator.links qtcreator-2.5.2/debian/qtcreator.links --- qtcreator-2.5.0/debian/qtcreator.links 2011-11-02 16:39:36.000000000 +0000 +++ qtcreator-2.5.2/debian/qtcreator.links 2012-06-08 16:42:17.000000000 +0000 @@ -1,6 +1,9 @@ -/usr/share/common-licenses/LGPL-2.1 /usr/share/qtcreator/dumper/LICENSE.LGPL -/usr/share/common-licenses/LGPL-2.1 /usr/share/qtcreator/qml/qmldump/LICENSE.LGPL -/usr/share/common-licenses/LGPL-2.1 /usr/share/qtcreator/qml/qmlobserver/LICENSE.LGPL -/usr/share/doc/qtcreator/LGPL_EXCEPTION.TXT /usr/share/qtcreator/dumper/LGPL_EXCEPTION.TXT -/usr/share/doc/qtcreator/LGPL_EXCEPTION.TXT /usr/share/qtcreator/qml/qmldump/LGPL_EXCEPTION.TXT -/usr/share/doc/qtcreator/LGPL_EXCEPTION.TXT /usr/share/qtcreator/qml/qmlobserver/LGPL_EXCEPTION.TXT +# Needed for source code of gdbmacros +usr/share/common-licenses/LGPL-2.1 usr/share/qtcreator/dumper/LICENSE.LGPL +usr/share/common-licenses/LGPL-2.1 usr/share/qtcreator/gdbmacros/LICENSE.LGPL +usr/share/common-licenses/LGPL-2.1 usr/share/qtcreator/qml/qmldump/LICENSE.LGPL +usr/share/common-licenses/LGPL-2.1 usr/share/qtcreator/qml/qmlobserver/LICENSE.LGPL +usr/share/doc/qtcreator/LGPL_EXCEPTION.TXT usr/share/qtcreator/dumper/LGPL_EXCEPTION.TXT +usr/share/doc/qtcreator/LGPL_EXCEPTION.TXT usr/share/qtcreator/gdbmacros/LGPL_EXCEPTION.TXT +usr/share/doc/qtcreator/LGPL_EXCEPTION.TXT usr/share/qtcreator/qml/qmldump/LGPL_EXCEPTION.TXT +usr/share/doc/qtcreator/LGPL_EXCEPTION.TXT usr/share/qtcreator/qml/qmlobserver/LGPL_EXCEPTION.TXT diff -Nru qtcreator-2.5.0/debian/rules qtcreator-2.5.2/debian/rules --- qtcreator-2.5.0/debian/rules 2012-05-09 14:16:07.000000000 +0000 +++ qtcreator-2.5.2/debian/rules 2012-08-17 18:27:17.000000000 +0000 @@ -8,29 +8,23 @@ # Upstream changelog upstream_changes := $(wildcard dist/changes-*) - %: - dh $@ --parallel --dbg-package=qtcreator-dbg --builddirectory=builddir + dh $@ --parallel --list-missing --dbg-package=qtcreator-dbg --buildsystem=qmake_qt4 --builddirectory=builddir override_dh_auto_configure: mkdir -p builddir - dh_auto_configure -- $(CURDIR) IDE_LIBRARY_BASENAME=lib/$(DEB_HOST_MULTIARCH) + dh_auto_configure -- $(CURDIR) IDE_LIBRARY_BASENAME=lib/$(DEB_HOST_MULTIARCH) $(extra_configure_opts) override_dh_auto_install: - # build docs in dh_auto_install so that they are only built when - # arch-all packages are generated ifneq (,$(filter qtcreator-doc, $(shell dh_listpackages))) - # Build documentations dh_auto_build -- qch_docs endif dh_auto_install --destdir=debian/tmp/usr - make -C builddir install_docs INSTALL_ROOT=$(CURDIR)/debian/tmp/usr - -override_dh_install: - dh_install --list-missing + dh_auto_build -- install_docs INSTALL_ROOT=$(CURDIR)/debian/tmp/usr -override_dh_installchangelogs: - dh_installchangelogs $(upstream_changes) +override_dh_installdocs: + dh_installdocs -pqtcreator $(upstream_changes) + dh_installdocs --remaining-packages override_dh_makeshlibs: # qtcreator doesn't provide any public libraries diff -Nru qtcreator-2.5.0/dist/changes-2.5.1 qtcreator-2.5.2/dist/changes-2.5.1 --- qtcreator-2.5.0/dist/changes-2.5.1 1970-01-01 00:00:00.000000000 +0000 +++ qtcreator-2.5.2/dist/changes-2.5.1 2012-08-08 13:47:06.000000000 +0000 @@ -0,0 +1,35 @@ +Qt Creator version 2.5.1 contains bug fixes. + +The most important changes are listed in this document. For a complete +list of changes, see the Git log for the Qt Creator sources that +you can check out from the public Git repository. For example: + +git clone git://gitorious.org/qt-creator/qt-creator.git +git log --cherry-pick --pretty=oneline v2.5.0...origin/2.5 + +Managing Projects + * Fixed crash in CMake makestep if used in the deploystep list + (QTCREATORBUG-7427) + * Fixed crash on unloading Qt4 projects + +C++ Support + * Fixed crash on invalid class name (QTCREATORBUG-7462) + * Fixed class scope completion for templates + +QML/JS Support + * Fixed crash with e.g. color picker on Mac (QTCREATORBUG-7605) + +Help + * Handle mailto links (QTCREATORBUG-4058) + +Version Control + * Fixed SVN project status command when no document is open + * Fixed committing to Mercurial repositories (QTCREATORBUG-7511) + +Platform Specific + +Linux + * Fixed default UI language on systems where that contains region information + +Mac + * Fixed font rendering problems (QTCREATORBUG-7127, fixed in Qt) diff -Nru qtcreator-2.5.0/dist/changes-2.5.2 qtcreator-2.5.2/dist/changes-2.5.2 --- qtcreator-2.5.0/dist/changes-2.5.2 1970-01-01 00:00:00.000000000 +0000 +++ qtcreator-2.5.2/dist/changes-2.5.2 2012-08-08 13:47:06.000000000 +0000 @@ -0,0 +1,13 @@ +Qt Creator version 2.5.2 contains bug fixes. + +The most important changes are listed in this document. For a complete +list of changes, see the Git log for the Qt Creator sources that +you can check out from the public Git repository. For example: + +git clone git://gitorious.org/qt-creator/qt-creator.git +git log --cherry-pick --pretty=oneline v2.5.1...v2.5.2 + +Platform Specific + +Windows + * Fixed resource leak leading to regular crashes (QTCREATORBUG-7385) diff -Nru qtcreator-2.5.0/doc/src/overview/creator-supported-platforms.qdoc qtcreator-2.5.2/doc/src/overview/creator-supported-platforms.qdoc --- qtcreator-2.5.0/doc/src/overview/creator-supported-platforms.qdoc 2012-05-09 12:13:19.000000000 +0000 +++ qtcreator-2.5.2/doc/src/overview/creator-supported-platforms.qdoc 2012-08-08 13:47:06.000000000 +0000 @@ -49,7 +49,7 @@ \o Windows Vista - \o (K)Ubuntu Linux 8.04 (32-bit and 64-bit) or later, with the + \o (K)Ubuntu Linux 10.04 (32-bit and 64-bit) or later, with the following: \list diff -Nru qtcreator-2.5.0/qtcreator.pri qtcreator-2.5.2/qtcreator.pri --- qtcreator-2.5.0/qtcreator.pri 2012-05-09 12:13:19.000000000 +0000 +++ qtcreator-2.5.2/qtcreator.pri 2012-08-08 13:47:06.000000000 +0000 @@ -1,7 +1,7 @@ !isEmpty(QTCREATOR_PRI_INCLUDED):error("qtcreator.pri already included") QTCREATOR_PRI_INCLUDED = 1 -QTCREATOR_VERSION = 2.5.0 +QTCREATOR_VERSION = 2.5.2 defineReplace(cleanPath) { win32:1 ~= s|\\\\|/|g diff -Nru qtcreator-2.5.0/qtcreator.qbp qtcreator-2.5.2/qtcreator.qbp --- qtcreator-2.5.0/qtcreator.qbp 2012-05-09 12:13:19.000000000 +0000 +++ qtcreator-2.5.2/qtcreator.qbp 2012-08-08 13:47:06.000000000 +0000 @@ -4,7 +4,7 @@ Project { property string ide_version_major: '2' property string ide_version_minor: '5' - property string ide_version_release: '0' + property string ide_version_release: '1' property string qtcreator_version: ide_version_major + '.' + ide_version_minor + '.' + ide_version_release property var additionalCppDefines: [ 'IDE_LIBRARY_BASENAME="lib"' ] moduleSearchPaths: "qbs" diff -Nru qtcreator-2.5.0/share/qtcreator/translations/qtcreator_cs.ts qtcreator-2.5.2/share/qtcreator/translations/qtcreator_cs.ts --- qtcreator-2.5.0/share/qtcreator/translations/qtcreator_cs.ts 2012-05-09 12:13:19.000000000 +0000 +++ qtcreator-2.5.2/share/qtcreator/translations/qtcreator_cs.ts 2012-08-08 13:47:06.000000000 +0000 @@ -5,7 +5,7 @@ Application Failed to load core: %1 - Přídavný modul 'core' se nepodařilo nahrát: %1 + Nepodařilo se nahrát 'core': %1 Unable to send command line arguments to the already running instance. It appears to be not responding. Do you want to start a new instance of Creator? @@ -13,7 +13,7 @@ Unable to send command line arguments to the already running instance. It appears to be not responding. - Agumenty příkazového řádku se nepodařilo předat již běžící úrovni. Zdá se, že neodpovídá. + Argumenty příkazového řádku se nepodařilo předat již běžící instanci. Zdá se, že neodpovídá. Could not send message @@ -29,14 +29,14 @@ Qt Creator - Plugin loader messages - Qt Creator - Zprávy správy přídavných modulů + Qt Creator - Zprávy zavaděče přídavných modulů AttachCoreDialog Start Debugger - Spustit ladicí program + Spustit ladicí program Executable: @@ -48,15 +48,15 @@ &Executable: - &Spustitelný soubor: + &Spustitelný soubor: &Core file: - Soubor &core: + Soubor &core: &Tool chain: - Ř&etězec nástrojů: + Sada &nástrojů: Sysroot @@ -68,18 +68,18 @@ Sys&root: - Sys&root: + Sys&root: Override &start script: - Přepsat &spouštěcí skript: + Přepsat &spouštěcí skript: AttachExternalDialog Start Debugger - Spustit ladicí program + Spustit ladicí program Attach to Process ID: @@ -99,11 +99,11 @@ Attach to &process ID: - Připojit k ID &procesu: + Připojit k ID &procesu: &Tool chain: - Řetě&zec nástrojů: + Sada &nástrojů: @@ -356,6 +356,10 @@ CMakeProjectManager::Internal::CMakeBuildSettingsWidget + Run cmake + Provést cmake + + Reconfigure project: Nastavit projekt znovu: @@ -363,6 +367,10 @@ &Change &Změnit + + Build directory: + Adresář pro sestavování: + CMakeProjectManager::Internal::CMakeOpenProjectWizard @@ -403,7 +411,7 @@ Debugger: - Ladič: + Ladič: Run Environment @@ -462,23 +470,23 @@ The directory %1 already contains a cbp file, which is recent enough. You can pass special arguments or change the used tool chain here and rerun CMake. Or simply finish the wizard directly. - Adresář %1 již obsahuje soubor cbp, který je dost čerstvý. Můžete tu podat zvláštní argumenty nebo změnit použitý řetěz nástrojů a spustit cmake znovu. Nebo průvodce jednoduše rovnou ukončete. + Adresář %1 již obsahuje dostatečně čerstvý soubor cbp. Můžete tu podat zvláštní argumenty nebo změnit použitou sadu nástrojů a spustit CMake znovu. Také lze průvodce rovnou ukončit. The directory %1 does not contain a cbp file. Qt Creator needs to create this file by running CMake. Some projects require command line arguments to the initial CMake call. - Adresář %1 neobsahuje žádný soubor cbp. Qt Creator musí tento soubor vytvořit vyvoláním cmake. U některých projektů jsou k tomu vyžadovány argumenty příkazového řádku. + Adresář %1 neobsahuje žádný soubor cbp. Qt Creator musí tento soubor vytvořit spuštěním cmake. U některých projektů jsou k tomu vyžadovány argumenty příkazového řádku. The directory %1 contains an outdated .cbp file. Qt Creator needs to update this file by running CMake. If you want to add additional command line arguments, add them below. Note that CMake remembers command line arguments from the previous runs. - Adresář %1 obsahuje zastaralý soubor cbp. Qt Creator musí tento soubor obnovit vyvoláním cmake. Dodatečné argumenty příkazového řádku lze zadat dole. Všimněte si, že cmake ukládá argumenty příkazového řádku z předchozího vyvolání. + Adresář %1 obsahuje zastaralý soubor cbp. Qt Creator musí tento soubor obnovit spuštěním cmake. Dodatečné argumenty příkazového řádku lze zadat dole. Všimněte si, že cmake ukládá argumenty příkazového řádku z předchozího spuštění. The directory %1 specified in a build-configuration, does not contain a cbp file. Qt Creator needs to recreate this file, by running CMake. Some projects require command line arguments to the initial CMake call. Note that CMake remembers command line arguments from the previous runs. - Adresář %1, který byl zadán v nastavení sestavování, neobsahuje soubor cbp. Qt Creator musí soubor vytvořit pomocí vyvolání cmake. U některých projektů jsou k tomu vyžadovány argumenty příkazového řádku. Všimněte si, že cmake ukládá argumenty příkazového řádku z předchozího vyvolání. + Adresář %1, který byl zadán v nastavení sestavování, neobsahuje soubor cbp. Qt Creator musí soubor vytvořit spuštěním cmake. U některých projektů jsou k tomu vyžadovány argumenty příkazového řádku. Všimněte si, že cmake ukládá argumenty příkazového řádku z předchozího spuštění. Qt Creator needs to run CMake in the new build directory. Some projects require command line arguments to the initial CMake call. - Qt Creator musí vyvolat cmake v novém adresáři pro sestavování. U některých projektů jsou k tomu vyžadovány argumenty příkazového řádku. + Qt Creator musí spustit cmake v novém adresáři pro sestavování. U některých projektů jsou k tomu vyžadovány argumenty příkazového řádku. Refreshing cbp file in %1. @@ -584,7 +592,7 @@ <b>Unknown tool chain</b> - <b>Neznámý řetěz nástrojů</b> + <b>Neznámá sada nástrojů</b> <b>Make:</b> %1 %2 @@ -592,7 +600,7 @@ <b>Unknown Toolchain</b> - <b>Neznámý řetěz nástrojů</b> + <b>Neznámá sada nástrojů</b> @@ -699,15 +707,15 @@ Select - Vybrat + Vybrat Change: - Změna: + Změna: Repository location: - Umístění skladiště: + Umístění skladiště: @@ -718,7 +726,7 @@ Paste Snippet... - Vložit kousek... + Uložit úryvek na... Alt+C,Alt+P @@ -726,11 +734,11 @@ Paste Clipboard... - Vložit schránku... + Uložit schránku na... Fetch Snippet... - Natáhnout kousek... + Stáhnout úryvek... Alt+C,Alt+F @@ -738,7 +746,7 @@ Empty snippet received for "%1". - Přijat prázdný kousek pro "%1". + Přijat prázdný úryvek pro "%1". This protocol supports no listing @@ -753,11 +761,11 @@ CodePaster::PasteSelectDialog Paste: - Vložit: + Vložit: Protocol: - Protokol: + Protokol: Refresh @@ -776,7 +784,7 @@ CodePaster::SettingsPage Username: - Uživatelské jméno: + Uživatelské jméno: Copy Paste URL to clipboard @@ -812,15 +820,15 @@ Display Output pane after sending a post - Po odeslání ukázat výstupní tabulku + Po odeslání ukázat výstupní tabulku Copy-paste URL to clipboard - Kopírovat URL do schránky + Kopírovat URL do schránky Default protocol: - Výchozí protokol: + Výchozí protokol: @@ -835,7 +843,7 @@ Populate source file view automatically - Aktualizovat pohled na zdrojový soubor automaticky + Aktualizovat pohled na zdrojový soubor automaticky When this option is checked, 'Step Into' compresses several steps into one in certain situations, leading to 'less noisy' debugging. So will, e.g., the atomic @@ -852,11 +860,11 @@ <unlimited> - <neomezená> + <neomezená> Use alternating row colors in debug views - Používat pro pohledy na ladění střídavých barev řádků + Používat pro pohledy na ladění střídavých barev řádků Enable reverse debugging @@ -868,7 +876,7 @@ Use tooltips in main editor while debugging - Používat v hlavním editoru během ladění nástrojových rad + Používat v hlavním editoru během ladění nástrojových rad Language @@ -888,31 +896,31 @@ Register Qt Creator for debugging crashed applications. - Zapsat Qt Creator pro ladění spadlých programů. + Zapsat Qt Creator pro ladění spadlých programů. Use Qt Creator for post-mortem debugging - Použít Qt Creator pro ladění - následný rozbor + Použít Qt Creator pro ladění - následný rozbor Behavior - Chování + Chování Change the font size in the debugger views when the font size in the main editor changes. - Změnit velikost písma v oknech pro ladění, když se velikost písma změní v hlavním editoru. + Změnit velikost písma v oknech pro ladění, když se velikost písma změní v hlavním editoru. Debugger font size follows main editor - Velikost písma ladiče následuje hlavní editor + Velikost písma ladiče následuje hlavní editor Populate the source file view automatically. This might slow down debugger startup considerably. - Aktualizovat pohled na zdrojový soubor automaticky. Toto může spuštění ladiče výrazně zpomalit. + Aktualizovat pohled na zdrojový soubor automaticky. Toto může spuštění ladiče výrazně zpomalit. Close temporary buffers on debugger exit - Při ukončení ladění zavřít editory + Při ukončení ladění zavřít editory Switch to previous mode on debugger exit. @@ -920,11 +928,11 @@ Switch to previous mode on debugger exit - Na začátku ladění činný režim při ukončení ladění obnovit + Na začátku ladění činný režim při ukončení ladění obnovit Maximum stack depth: - Největší hloubka zásobníku: + Největší hloubka zásobníku: @@ -947,11 +955,11 @@ Insert the common prefix of available completion items. - Vložit společnou předponu hodících se návrhů na doplnění. + Vložit společnou předponu hodících se návrhů na doplnění. Autocomplete common &prefix - Automaticky doplnit společnou &předponu + Automaticky doplnit společnou &předponu &Automatically insert brackets @@ -959,43 +967,43 @@ Behavior - Chování + Chování &Case-sensitivity: - &Rozlišování velkých a malých písmen: + &Rozlišování velkých a malých písmen: Full - Plné + Plné None - Žádné + Žádné First Letter - První písmeno + První písmeno Insert &space after function name - Vložit &mezeru po názvu funkce + Vložit &mezeru po názvu funkce Activate completion: - Zapnout doplňování kódu: + Zapnout doplňování kódu: Manually - Ruční + Ruční When Triggered - Na žádost + Na žádost Always - Vždy + Vždy Automatically insert brackets and semicolons when appropriate. @@ -1003,19 +1011,19 @@ Automatically insert semicolons and closing brackets, parentheses, curly braces, and quotes when appropriate. - Automaticky vkládat středníky a uzavírající závorky, kulaté závorky, složené závorky a uvozovky, když je to vhodné. + Automaticky vkládat středníky a uzavírající závorky, kulaté závorky, složené závorky a uvozovky, když je to vhodné. &Automatically insert matching characters - &Automaticky vložit odpovídající znaky + &Automaticky vložit odpovídající znaky When typing a matching character and there is a text selection, instead of removing the selection, surround it with the corresponding characters. - Při psaní odpovídajícího znaku, a když je vybrán text, místo odstranění výběru tento ohraničit příslušnými znaky. + Při psaní odpovídajícího znaku, a když je vybrán text, místo odstranění výběru tento ohraničit příslušnými znaky. Surround &text selections - Ohraničit výběry &textů + Ohraničit výběry &textů @@ -1249,6 +1257,10 @@ Otevřít + Make Writable + Udělat zapisovatelným + + Save %1 &As... Uložit %1 &jako... @@ -1310,15 +1322,15 @@ Make writable - Udělat zapisovatelným + Udělat zapisovatelným Next Open Document in History - Další otevřít dokument na seznamu + Další otevřený dokument na seznamu Previous Open Document in History - Předchozí otevřít dokument na seznamu + Předchozí otevřený dokument na seznamu Go Back @@ -1413,47 +1425,47 @@ File Error - Chyba souboru + Chyba souboru Overwrite? - Přepsat? + Přepsat? An item named '%1' already exists at this location. Do you want to overwrite it? - V tomto umístění již existuje soubor s názvem '%1'. Chcete jej přepsat? + V tomto umístění již existuje soubor s názvem '%1'. Chcete jej přepsat? Save File As - Uložit soubor jako + Uložit soubor jako Open File - Otevřít soubor + Otevřít soubor File is Read Only - Soubor je pouze pro čtení + Soubor je pouze pro čtení The file <i>%1</i> is read only. - Soubor <i>%1</i> je pouze pro čtení. + Soubor <i>%1</i> je pouze pro čtení. Open with VCS (%1) - Otevřít s pomocí systému na řízení verzí (VCS) (%1) + Otevřít s pomocí systému na řízení verzí (VCS) (%1) Make writable - Udělat zapisovatelným + Udělat zapisovatelným Save as... - Uložit jako... + Uložit jako... Cannot reload %1 - %1 se nahrát znovu nepodařilo + %1 se nahrát znovu nepodařilo @@ -1687,7 +1699,7 @@ Automatically create temporary copies of modified files. If Qt Creator is restarted after a crash or power failure, it asks whether to recover the auto-saved content. - Způsobí, že dočasné kopie změněných souborů se vytvoří automaticky. Při opětovném spuštění Qt Creatoru po pádu nebo po výpadku proudu bude položena otázka, zda se má automaticky uložený obsah obnovit. + Dočasné kopie změněných souborů vytváří automaticky. Po pádu nebo výpadku proudu se při spuštění bude Qt Creator ptát, má-li automaticky uložený obsah obnovit. Auto-save modified files @@ -1750,6 +1762,14 @@ Naposledy upravované soubory + Exit Full Screen + Ukončit na celou obrazovku + + + Enter Full Screen + Vstoupit na celou obrazovku + + Open File &With... Otevřít soubor &s... @@ -1762,6 +1782,10 @@ &Uložit + Save + Uložit + + Save &As... Uložit &jako... @@ -1770,6 +1794,10 @@ Ctrl+Shift+S + Save As... + Uložit jako... + + Save A&ll Uložit &vše @@ -1790,10 +1818,18 @@ &Zpět + Undo + Zpět + + &Redo &Znovu + Redo + Znovu + + Cu&t Vyj&mout @@ -1830,10 +1866,18 @@ &Volby... + Ctrl+, + Ctrl+, + + Minimize Zmenšit + Ctrl+M + Ctrl+M + + Zoom Zvětšení @@ -1842,6 +1886,22 @@ Ukázat postranní panel + Ctrl+0 + Ctrl+0 + + + Alt+0 + Alt+0 + + + Ctrl+Meta+F + Ctrl+Meta+F + + + Ctrl+Shift+F11 + Ctrl+Shift+F11 + + Full Screen Na celou obrazovku @@ -1950,6 +2010,22 @@ Files and Classes Soubory a třídy + + All Templates + Všechny předlohy + + + %1 Templates + %1 předlohy + + + Platform independent + Platformně nezávislé + + + Supported Platforms + Podporované platformy + Core::Internal::OpenEditorsWidget @@ -1999,6 +2075,14 @@ Open file '%1' with: Otevřít soubor '%1' s: + + Open File With... + Otevřít soubor s... + + + Open file extension with: + Otevřít souborovou příponu s: + Core::Internal::OutputPaneManager @@ -2027,6 +2111,22 @@ Výstupní &tabulky + Shift+F6 + Shift+F6 + + + F6 + F6 + + + Ctrl+9 + Ctrl+9 + + + Alt+9 + Alt+9 + + Minimize Output Pane Zmenšit výstupní tabulku @@ -2088,6 +2188,18 @@ Save Selected Uložit vybrané + + Save Changes + Uložit změny + + + The following files have unsaved changes: + Následující soubory byly změněny: + + + Automatically save all files before building + Automaticky uložit všechny změněné soubory před sestavováním + Core::Internal::ShortcutSettings @@ -2113,15 +2225,15 @@ Import Keyboard Mapping Scheme - Zavést nákres s přiřazením kláves + Zavést schéma přiřazení kláves Keyboard Mapping Scheme (*.kms) - Soubor s nákresem přiřazení kláves (*.kms) + Schéma přiřazení kláves (soubor *.kms) Export Keyboard Mapping Scheme - Vyvést nákres s přiřazením kláves + Vyvést schéma přiřazení kláves @@ -2152,7 +2264,7 @@ <h3>Qt Creator %1 %8</h3>Based on Qt %2 (%3 bit)<br/><br/>Built on %4 at %5<br /><br/>%9<br/>Copyright 2008-%6 %7. All rights reserved.<br/><br/>The program is provided AS IS with NO WARRANTY OF ANY KIND, INCLUDING THE WARRANTY OF DESIGN, MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE.<br/> - <h3>Qt Creator %1</h3>Založený na Qt %2 (%3 bit)<br/><br/>Vytvořený %4 v %5<br /><br/>%8<br/>Autorské právo 2008-%6 %7. Všechna práva vyhrazena.<br/><br/> Program je poskytován tak, JAK JE, BEZ ZÁRUKY JAKÉHOKOLI DRUHU, VČETNĚ ZÁRUKY PROVEDENÍ, PRODEJNOSTI A VHODNOSTI PRO URČITÝ ÚČEL.<br/> + <h3>Qt Creator %1 %8 </h3>Založený na Qt %2 (%3 bit)<br/><br/>Vytvořený %4 v %5<br /><br/>%9<br/>Autorské právo 2008-%6 %7. Všechna práva vyhrazena.<br/><br/> Program je poskytován tak, JAK JE, BEZ ZÁRUKY JAKÉHOKOLI DRUHU, VČETNĚ ZÁRUKY PROVEDENÍ, PRODEJNOSTI A VHODNOSTI PRO URČITÝ ÚČEL.<br/> <h3>Qt Creator %1</h3>Based on Qt %2 (%3 bit)<br/><br/>Built on %4 at %5<br /><br/>%8<br/>Copyright 2008-%6 %7. All rights reserved.<br/><br/>The program is provided AS IS with NO WARRANTY OF ANY KIND, INCLUDING THE WARRANTY OF DESIGN, MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE.<br/> @@ -2314,7 +2426,11 @@ Switch Between Method Declaration/Definition - Přepínání mezi prohlášením a vymezením metody + Přepínání mezi deklarací a definicí metody + + + Shift+F2 + Shift+F2 Find Usages @@ -2337,6 +2453,10 @@ Přejmenovat symbol pod ukazovátkem + CTRL+SHIFT+R + Ctrl+Shift+R + + Update Code Model Obnovit model kódu @@ -2353,15 +2473,15 @@ CppFileSettingsPage Header suffix: - Přípona hlavičkových souborů: + Přípona hlavičkových souborů: Source suffix: - Přípona zdrojových souborů: + Přípona zdrojových souborů: Lower case file names - Pro názvy souborů používat malých písmen + Pro názvy souborů používat malých písmen File Naming Conventions @@ -2373,7 +2493,7 @@ License template: - Předloha pro povolení: + Předloha pro povolení: @@ -2387,7 +2507,7 @@ CppTools File Naming Conventions - Zvyklosti při tvoření souborových názvů + Úmluva pro pojmenování souborů Code Style @@ -2395,7 +2515,7 @@ File Naming - Tvoření souborových názvů + Pojmenování souborů C++ @@ -2412,19 +2532,111 @@ Text Editor Textový editor + + Behavior + Chování + + + &Case-sensitivity: + &Rozlišovat velká a malá písmena: + + + Full + Plně + + + None + Vůbec ne + + + First Letter + První písmeno + + + Activate completion: + Zapnout doplňování kódu: + + + Manually + Ručně + + + When Triggered + Na žádost + + + Always + Vždy + + + Insert the common prefix of available completion items. + Vložit společnou předponu hodících se návrhů na doplnění. + + + Autocomplete common &prefix + Automaticky doplnit společnou &předponu + + + Automatically insert semicolons and closing brackets, parentheses, curly braces, and quotes when appropriate. + Když je to vhodné, automaticky vkládat středníky a uzavírající závorky, kulaté závorky, složené závorky a uvozovky. + + + &Automatically insert matching characters + &Automaticky vložit odpovídající znaky + + + When typing a matching character and there is a text selection, instead of removing the selection, surround it with the corresponding characters. + Je-li vybrán text, při psaní odpovídajícího znaku se místo odstranění výběru tento ohraničí příslušnými znaky. + + + Surround &text selections + Ohraničit výběry &textů + + + Insert &space after function name + Vložit &mezeru po názvu funkce + + + Documentation Comments + Dokumentační poznámky + + + Automatically create a Doxygen comment upon pressing enter after a /** or /*! + Automaticky vytvořit poznámku Doxygen ihned po stisknutí klávesy Enter po /** nebo /*! + + + Enable Doxygen blocks + Zapnout bloky Doxygen + + + Generate a <i>brief</i> command with an initial description for the corresponding declaration + Vytvořit <i>brief</i> příkaz s počátečním popisem odpovídající deklarace + + + Generate brief description + Vytvořit popis "brief" + + + Add leading asterisks when continuing comments on new lines + Přidat hvězdičky na začátku při pokračování poznámek na nových řádcích + + + Add leading asterisks + Přidat hvězdičky na začátku + CppTools::Internal::CppClassesFilter Classes - Třídy + Třídy CppTools::Internal::CppFunctionsFilter Methods - Postupy + Metody Methods and functions @@ -2443,7 +2655,7 @@ Parsing - Zpracování + Syntaktický rozbor unnamed @@ -2462,7 +2674,7 @@ Switch Header/Source - Přepínat mezi hlavičkovým/zdrojovým souborem + Přepnout mezi hlavičkou a zdrojem @@ -2514,7 +2726,7 @@ Unable to determine executable from core file. - Ze souboru 'core' se nepodařilo určit žádný spustitelný soubor. + Ze souboru 'core' se nepodařilo určit spustitelný soubor. The name of the binary file cannot be extracted from this core file. @@ -2610,7 +2822,7 @@ Debugging Helper - Pomocná knihovna pro výstup dat o ladění + Pomocný ladicí program Ctrl+Shift+F11 @@ -2639,6 +2851,30 @@ Select Startup Script Vybrat spouštěcí skript + + Start Debugger + Spustit ladicí program + + + &Executable: + &Spustitelný soubor: + + + &Core file: + Soubor &core: + + + &Tool chain: + Sada &nástrojů: + + + Sys&root: + Sys&root: + + + Override &start script: + Přepsat &spouštěcí skript: + Debugger::Internal::AttachExternalDialog @@ -2668,6 +2904,18 @@ Proces %1 je již pod řízením ladiče. Qt Creator se k němu nemůže připojit. + + Start Debugger + Spustit ladicí program + + + Attach to &process ID: + Připojit k ID &procesu: + + + &Tool chain: + Sada &nástrojů: + Debugger::Internal::BreakHandler @@ -2692,6 +2940,10 @@ Vlastnost + Breakpoint on QML Signal Emit + Přerušit při vyslání signálu QML + + Breakpoint will only be hit in the specified thread(s). Bod přerušení bude spuštěn jen v zadaném vláknu. @@ -2713,7 +2965,7 @@ Engine: - &Stroj: + Stroj: Breakpoint Type: @@ -2805,7 +3057,7 @@ Removal proceeding - Vymazuje se + Vymazává se Dead @@ -3268,7 +3520,7 @@ <html><body><p>Specify the path to the <a href="%1">Debugging Tools for Windows</a> (%2) here.</p><p><b>Note:</b> Restarting Qt Creator is required for these settings to take effect.</p></p></body></html> Label text for path configuration. %2 is "x-bit version". - <html><body><p>Zadejte cestu k <a href="%1">nástrojům pro ladění Windows</a> (%2).</p><p><b>Poznámka:</b> Změny začnou působit až při dalším spuštění Qt Creatoru.</p></p></body></html> + <html><body><p>Zadejte cestu k <a href="%1">nástrojům pro ladění Windows</a> (%2).</p><p><b>Poznámka:</b> Změny zapůsobí teprve při dalším spuštění Qt Creatoru.</p></p></body></html> 64-bit version @@ -3506,7 +3758,7 @@ This switches the debugger to instruction-wise operation mode. In this mode, stepping operates on single instructions and the source location view also shows the disassembled instructions. - Přikáže ladicímu programu, aby pracoval na úrovni příkazu. V tomto režimu pracuje funkce jednotlivého kroku na strojových příkazech a pohled na zdrojový text ukazuje rozložené příkazy. + Přikáže ladicímu programu, aby pracoval na úrovni instrukce. V tomto režimu se krokuje po jednotlivých instrukcích a pohled na zdrojový text také ukazuje instrukce z disasembleru. Dereference pointers automatically @@ -3562,15 +3814,15 @@ Verbose Log - Podrobný zápis + Podrobný záznam Operate by Instruction - Pracovat na úrovni příkazu + Pracovat na úrovni instrukcí Dereference Pointers Automatically - Zrušit automaticky odkazování u ukazovátek + U ukazatelů automaticky vyhodnotit odkazy Watch Expression "%1" @@ -3594,7 +3846,7 @@ Use Debugging Helpers - Používat pomocnou knihovnu pro výstup dat + Používat pomocné ladicí programy Debug Debugging Helpers @@ -3606,7 +3858,7 @@ Selecting this causes the C++ Code Model being asked for variable scope information. This might result in slightly faster debugger operation but may fail for optimized code. - Výběr tohoto způsobí, že model kódu C++ (Qt Creatoru) je dotazován na informace ohledně oblasti platnosti proměnných. Výsledkem může být o něco rychlejší oznamování hodnot (u operace ladiče), ale u optimalizovaného kódu to může selhat. + Výběr tohoto způsobí, že model kódu C++ (Qt Creatoru) je dotazován na informace ohledně oboru působnosti proměnných. Výsledkem může být o něco rychlejší činost ladiče, ale u optimalizovaného kódu to může selhat. Recheck Debugging Helper Availability @@ -3626,11 +3878,11 @@ This switches the Locals&&Watchers view to automatically dereference pointers. This saves a level in the tree view, but also loses data for the now-missing intermediate level. - Způsobí, že u ukazovátka v okně "Místní &proměnné a sledované výrazy" je automaticky zrušeno odkazování. Tím se zjednoduší stromové zobrazení, ale zase ovšem chybí informace o nyní nepřítomné mezilehlé úrovni. + Způsobí, že v okně "Místní proměnné a sledované výrazy" se u ukazatelů automaticky provede vyhodnocení odkazu. Tím se sice zjednoduší stromové zobrazení, ale zase chybí informace o nyní nepřítomné mezilehlé úrovni. Sort Members of Classes and Structs Alphabetically - Roztřídit členy tříd a staveb abecedně + Roztřídit členy tříd a struktur abecedně Adjust Breakpoint Locations @@ -3657,24 +3909,32 @@ Přerušit při "qFatal" + Break on "raise" + Přerušit při "raise" + + + Use Dynamic Object Type for Display + Použít pro zobrazení dynamický typ objektu + + Automatically Quit Debugger Automaticky ukončit ladicí program Use tooltips in main editor when debugging - Používat během ladění v hlavním editoru rady k nástrojům + Při ladění použít vysvětlivky v hlavním editoru Checking this will enable tooltips for variable values during debugging. Since this can slow down debugging and does not provide reliable information as it does not use scope information, it is switched off by default. - Tato volba během ladění zapíná rady k nástrojům pro proměnné. Ve výchozím nastavení je tato volba vypnuta, neboť zpomaluje ladění a neposkytuje spolehlivé údaje, protože nepoužívá informace o oblasti. + Tato volba zapíná při ladění vysvětlivky pro proměnné. Ve výchozím nastavení vypnuto, neboť zpomaluje ladění a neposkytuje spolehlivé údaje, protože není brán zřetel na obor působnosti. Use Tooltips in Locals View When Debugging - Používat během ladění pro zobrazení místních proměnných rady k nástrojům + Při ladění použít vysvětlivky pro zobrazení místních proměnných Use Tooltips in Breakpoints View When Debugging - Používat během ladění v okně s body přerušení rady k nástrojům + Při ladění použít vysvětlivky v okně s body přerušení Show Address Data in Breakpoints View When Debugging @@ -3710,7 +3970,7 @@ Create Full Backtrace - Vytvořit úplnou zpětnou stopu + Vytvořit úplný výpis volání Execute Line @@ -3730,7 +3990,7 @@ Checking this will enable tooltips in the locals view during debugging. - Zapne během ladění pro zobrazení místních proměnných rady k nástrojům. + Při ladění použít vysvětlivky pro zobrazení místních proměnných. Use tooltips in breakpoints view when debugging @@ -3738,7 +3998,7 @@ Checking this will enable tooltips in the breakpoints view during debugging. - Zapne během ladění v okně s body přerušení rady k nástrojům. + Při ladění použít vysvětlivky v okně s body přerušení. Show address data in breakpoints view when debugging @@ -3746,7 +4006,7 @@ Checking this will show a column with address information in the breakpoint view during debugging. - Přidá pro ukázání během ladění sloupec s adresami bodů přerušení. + Při ladění ukáže v okně bodů přerušení sloupec s adresami. Show address data in stack view when debugging @@ -3754,7 +4014,7 @@ Checking this will show a column with address information in the stack view during debugging. - Přidá v okně zásobníku pro ukázání během ladění sloupec s adresami. + Při ladění ukáže v okně zásobníku sloupec s adresami. Use debugging helper @@ -3815,6 +4075,33 @@ Ctrl+Shift+F11 Ctrl+Shift+F11 + + <html><head/><body> +<p>The debugging helper is only used to produce a nice display of objects of certain types like QString or std::map in the &quot;Locals and Expressions&quot; view. It is not strictly necessary for debugging with Qt Creator. </p></body></html> + <html><head/><body> +<p>Pomocný ladicí program slouží pouze pro přehledné zobrazení objektů určitého typu jako QString nebo std::map v okně "Místní proměnné a sledované výrazy".</p> +<p> K ladění s Qt Creatorem však není bezpodmínečně požadován. </p></body></html> + + + Use Debugging Helper + Používat pomocný ladicí program + + + Makes use of Qt Creator's code model to find out if a variable has already been assigned a value at the point the debugger interrupts. + Používá kódový model Qt Creatoru ke zjištění, zda proměnná již má nějakou hodnotu v místě kde dojde k přerušení ladicím programem. + + + Use code model + Použít model kódu + + + Displays names of QThread based threads. + Zobrazí názvy vláken založených na QThread. + + + Display thread names + Zobrazit názvy vláken + Debugger::Internal::GdbEngine @@ -3825,7 +4112,7 @@ The Gdb process crashed some time after starting successfully. -Proces Gdb po určité době od úspěšného spuštění spadl. +Proces Gdb po úspěšném spuštění za nějakou dobu spadl. The last waitFor...() function timed out. The state of QProcess is unchanged, and you can try calling waitFor...() again. @@ -3922,7 +4209,7 @@ The gdb process has not responded to a command within %1 seconds. This could mean it is stuck in an endless loop or taking longer than expected to perform the operation. You can choose between waiting longer or abort debugging. - Proces Gdb neodpověděl na příkaz během %1 sekund. Mohlo by to znamenat, že je zaseknutý v nekonečné smyčce nebo potřebuje na rovedení operace delší čas, než bylo očekáváno. + Proces Gdb neodpověděl na příkaz během %1 sekund. Mohlo by to znamenat, že je zaseknutý v nekonečné smyčce nebo potřebuje na provedení operace delší čas, než bylo očekáváno. Můžete si vybrat mezi delším čekáním nebo přerušením ladění. @@ -4021,11 +4308,11 @@ An error occurred when attempting to write to the gdb process. For example, the process may not be running, or it may have closed its input channel. - Při pokusu o zápis do procesu Gdb se vyskytla chyba. Pravděpodobně proces neběží, nebo zavřel svůj vstupní kanál. + Při pokusu o zápis do procesu Gdb se vyskytla chyba. Například je možné, že proces neběží, anebo zavřel svůj vstupní kanál. An error occurred when attempting to read from the gdb process. For example, the process may not be running. - Při pokusu o čtení z procesu Gdb se vyskytla chyba. Pravděpodobně proces neběží. + Při pokusu o čtení z procesu Gdb se vyskytla chyba. Například je možné, že proces neběží. An unknown error in the gdb process occurred. @@ -4040,6 +4327,12 @@ Byla spuštěna nějaká výjimka: + Missing debug information for %1 +Try: %2 + Chybí informace o ladění pro %1 +Zkuste: %2 + + Stopping temporarily Dočasně se zastavuje @@ -4051,11 +4344,11 @@ The gdb process has not responded to a command within %n second(s). This could mean it is stuck in an endless loop or taking longer than expected to perform the operation. You can choose between waiting longer or abort debugging. - Proces Gdb neodpověděl na příkaz po dobu %n sekundy. Mohlo by to znamenat, že je zaseknutý v nekonečné smyčce nebo potřebuje na rovedení operace delší čas, než bylo očekáváno. + Proces Gdb neodpověděl na příkaz po dobu %n sekundy. Možná je zaseknutý v nekonečné smyčce nebo potřebuje na provedení operace delší čas, než bylo očekáváno. Můžete si vybrat mezi delším čekáním nebo přerušením ladění. - Proces Gdb neodpověděl na příkaz po dobu %n sekund. Mohlo by to znamenat, že je zaseknutý v nekonečné smyčce nebo potřebuje na rovedení operace delší čas, než bylo očekáváno. + Proces Gdb neodpověděl na příkaz po dobu %n sekund. Možná je zaseknutý v nekonečné smyčce nebo potřebuje na provedení operace delší čas, než bylo očekáváno. Můžete si vybrat mezi delším čekáním nebo přerušením ladění. - Proces Gdb neodpověděl na příkaz po dobu %n sekund. Mohlo by to znamenat, že je zaseknutý v nekonečné smyčce nebo potřebuje na rovedení operace delší čas, než bylo očekáváno. + Proces Gdb neodpověděl na příkaz po dobu %n sekund. Možná je zaseknutý v nekonečné smyčce nebo potřebuje na provedení operace delší čas, než bylo očekáváno. Můžete si vybrat mezi delším čekáním nebo přerušením ladění. @@ -4085,7 +4378,7 @@ Raw structure - Surová stavba + Surová struktura Normal @@ -4105,7 +4398,7 @@ Step by instruction requested... - Požadován jednotlivý krok prostřednictvím příkazu... + Požadován krok po instrukcích... Finish function requested... @@ -4113,11 +4406,11 @@ Step next requested... - Požadován jednotlivý krok... + Požadován krok na další... Step next instruction requested... - Požadován jednotlivý krok prostřednictvím příkazu... + Požadován krok na další instrukci... Run to line %1 requested... @@ -4167,11 +4460,11 @@ The debugging helper library was not found at %1. - Pomocnou knihovnu pro výstup dat o ladění se nepodařilo nalézt v %1. + Pomocnou ladicí knihovnu se nepodařilo nalézt v %1. Disassembler failed: %1 - Chyba v překladači ze strojového jazyka do asembleru (disasembler): %1 + Chyba v disasembleru: %1 Unable to start gdb '%1': %2 @@ -4219,7 +4512,7 @@ <unknown> - <Neznámý> + <Neznámá> Jumped. Stopped @@ -4227,7 +4520,7 @@ Target line hit. Stopped - Zásah cílového řádku. Zastaveno + Dosažen cílový řádek. Zastaveno Application exited with exit code %1 @@ -4243,7 +4536,7 @@ Stopped at breakpoint %1 in thread %2. - Zastaveno při %1 ve vlákně %2. + Zastaveno na %1 ve vlákně %2. Stopped: %1 by signal %2 @@ -4369,7 +4662,7 @@ The debugger settings point to a script file at '%1' which is not accessible. If a script file is not needed, consider clearing that entry to avoid this warning. - K souboru se skriptem zadanému v nastaveních ladicího programu nelze přistupovat '%1'. Pokud není potřeba žádný skript, zvažte navrácení nastavení zpět, abyste se vyhnul tomuto upozornění. + Není přístupný souboru se skriptem jenž je podle nastaveních ladicího programu umístěn na '%1'. Pokud není potřeba žádný skript, zvažte navrácení nastavení zpět, aby se předešlo tomuto upozornění. Unable to run '%1': %2 @@ -4383,8 +4676,8 @@ Retrieving data for watch view (%n requests pending)... Přijímají se data pro zobrazení místních proměnných (ještě jeden zbývající dotaz)... - Přijímají se data pro zobrazení místních proměnných (% zbývající dotazy)... - Přijímají se data pro zobrazení místních proměnných (% zbývající dotazy)... + Přijímají se data pro zobrazení místních proměnných (%n zbývající dotazy)... + Přijímají se data pro zobrazení místních proměnných (%n zbývající dotazy)... @@ -4402,7 +4695,7 @@ Debugging helpers not found. - Pomocnou knihovnu pro výstup dat o ladění se nepodařilo nalézt. + Pomocný ladicí program se nepodařilo nalézt. Custom dumper setup: %1 @@ -4419,12 +4712,12 @@ Debugging helpers: Qt version mismatch - Pomocné knihovny pro výstup dat o ladění: nesoulad ve verzi Qt + Pomocný ladicí program: nesoulad ve verzi Qt The Qt version used to build the debugging helpers (%1) does not match the Qt version used to build the debugged application (%2). This might yield incorrect results. - Verze Qt použitá pro sestavení pomocných knihoven pro výstup dat o ladění (%1) neodpovídá verzi Qt použité pro sestavení laděného programu (%2). + Verze Qt použitá pro sestavení pomocných ladicích programů (%1) neodpovídá verzi Qt použité pro sestavení laděného programu (%2). To může dávat nesprávné výsledky. @@ -4452,12 +4745,140 @@ Vybrat umístění Gdb + General + Obecné + + + GDB timeout: + Časové omezení u GDB: + + + This is the number of seconds Qt Creator will wait before +it terminates a non-responsive GDB process. The default value of 20 seconds +should be sufficient for most applications, but there are situations when +loading big libraries or listing source files takes much longer than that +on slow machines. In this case, the value should be increased. + Toto je počet sekund, po které Qt Creator počká než ukončí neodpovídající proces GDB. +Výchozí hodnotou je 20 sekund, což by mělo postačovat v případě většiny programů. Ale jsou situace, +kdy na pomalých strojích nahrávání velkých knihoven nebo výpis zdrojových souborů zabere mnohem +více času. V takovém případě by se měla hodnota zvýšit. + + + sec + s + + + Skip known frames when stepping + Přeskočit při krokování známá místa + + + Allows 'Step Into' to compress several steps into one step +for less noisy debugging. For example, the atomic reference +counting code is skipped, and a single 'Step Into' for a signal +emission ends up directly in the slot connected to it. + Tato volba má za následek, že v určitých situacích sloučí +'Krok do' více kroků do jednoho, čímž se ladění +uspíší. Například se přeskočí kód atomického počítání odkazů, +a jednotlivý 'Krok do' pro vyslání signálu skončí přímo ve zdířce +(slot) s ním spojené. + + + Show a message box when receiving a signal + Při obdržení signálu ukázat okno se zprávami + + + This will show a message box as soon as your application +receives a signal like SIGSEGV during debugging. + Ukáže oznamovací okno, když program během ladění +přijme signál SIGSEGV. + + + Adjust breakpoint locations + Upravit umístění bodů přerušení + + + GDB allows setting breakpoints on source lines for which no code +was generated. In such situations the breakpoint is shifted to the +next source code line for which code was actually generated. +This option reflects such temporary change by moving the breakpoint +markers in the source code editor. + Ne všechny řádky se zdrojovým kódem vytváří spustitelný kód. Když se na takový řádek dá bod přerušení, chová se tak, jakoby byl napsán až na další řádek. Zapnutí nastavení "Upravit umístění bodů přerušení" způsobuje, že se v takovém případě značka bodu přerušení přesune na místo výsledného bodu přerušení. + + + Use dynamic object type for display + Použít pro zobrazení dynamický typ objektu + + + This specifies whether the dynamic or the static type of objects will bedisplayed. Choosing the dynamic type might be slower. + Udává, zda je zobrazen dynamický nebo statický typ objektu. Určení dynamického typu může trvat déle. + + + Load .gdbinit file on startup + Nahrát soubor .gdbinit na začátku + + + This allows or inhibits reading the user's default +.gdbinit file on debugger startup. + Dovolí nebo potlačí čtení výchozího uživatelského +souboru-.gdbinit při spuštění ladiče. + + + The options below should be used with care. + Nastavení níže by se měla používat opatrně. + + + Use asynchronous mode to control the inferior + Použít asynchronní režim k ovládání laděného procesu + + + Use common locations for debug information + Použít společné cesty pro informace o ladění + + + This adds common paths to locations of debug information +at debugger startup. + Toto při spuštění ladiče přidá společné cesty do míst s +informacemi o ladění. + + + Stop when a qWarning is issued + Zastavit při qWarning + + + Stop when a qFatal is issued + Zastavit při qFatal + + + Stop when raise() is called + Zastavit při vyvolání raise() + + + Enable reverse debugging + Zapnout obrácené ladění + + + <html><head/><body><p>Selecting this enables reverse debugging.</p><.p><b>Note:</b> This feature is very slow and unstable on the GDB side.It exhibits unpredictable behavior when going backwards over system calls and is very likely to destroy your debugging session.</p><body></html> + <html><head/><body><p>Výběr tohoto povolí obrácené ladění.</p><.p><b>Poznámka:</b> Tato vlastnost je velmi pomalá a na straně GDB nestálá. Projevuje při postupu zpět systémovými voláními nepředvídatelné chování a je dost pravděpodobné, že zničí vaše ladicí sezení.</p><body></html> + + + Attempt quick start + Pokusit se o rychlé spuštění + + + Checking this option will postpone reading debug information as long as possible. This can result in faster startup times at the price of not being able to set breakpoints by file and number. + Zaškrtnutí této volby odloží čtení informací o ladění na tak dlouho, jak jen to bude možné. To může mít za následek rychlejší časy spuštění za tu cenu, že body přerušení nejde nastavit podle souboru a čísla. + + + Additional Startup Commands + Dodatečné příkazy při spuštění + + GDB GDB Choose Location of Startup Script File - Vybrat umístění souboru se spouštěcím skriptem + Vybrat umístění souboru se spouštěcím skriptem @@ -4565,7 +4986,7 @@ Update Module List - Obnovit seznam s moduly + Obnovit seznam modulů Show Source Files for Module "%1" @@ -4686,7 +5107,7 @@ Open Disassembler... - Otevřít překladač ze strojového jazyka do asembleru... + Otevřít disasembler... Open Memory Editor at 0x%1 @@ -4694,11 +5115,11 @@ Open Memory View at Value of Register %1 0x%2 - Otevřít editor paměti při hodnotě registru %1 0x%2 + Otevřít náhled paměti při hodnotě registru %1 0x%2 Open Disassembler at 0x%1 - Otevřít překladač ze strojového jazyka do asembleru při %1 + Otevřít disasembler při %1 Open Memory Editor @@ -4706,11 +5127,11 @@ Open Memory View at Value of Register - Otevřít editor paměti při hodnotě registru + Otevřít náhled paměti při hodnotě registru Open Disassembler - Otevřít překladač ze strojového jazyka do asembleru + Otevřít disasembler Open Memory Editor at %1 @@ -4764,16 +5185,28 @@ Požadováno pokračování... + '%1' contains no identifier. + '%1' neobsahuje žádný identifikátor. + + + String literal %1. + Řetězec znaků tvořený písmeny %1. + + + Cowardly refusing to evaluate expression '%1' with potential side effects. + Nevyhodnocovat výraz '%1' s možnými postranními účinky. + + '%1' contains no identifier - '%1' neobsahuje žádný identifikátor + '%1' neobsahuje žádný identifikátor String literal %1 - Řetězec znaků tvořený písmeny %1 + Řetězec znaků tvořený písmeny %1 Cowardly refusing to evaluate expression '%1' with potential side effects - Nevyhodnocovat výraz '%1' s možnými postranními účinky + Nevyhodnocovat výraz '%1' s možnými postranními účinky Stopped at %1:%2. @@ -4888,6 +5321,14 @@ Zásobník + Function: + Funkce: + + + Disassemble Function + Rozložit funkci (disassemble) + + Copy Contents to Clipboard Obsah kopírovat do schránky @@ -4900,12 +5341,20 @@ Otevřít editor paměti při 0x%1 + Open Disassembler at Address... + Otevřít disasembler na adrese... + + + Disassemble Function... + Rozložit funkci (disassemble)... + + Open Disassembler... - Otevřít překladač ze strojového jazyka do asembleru... + Otevřít překladač ze strojového jazyka do asembleru... Open Disassembler at 0x%1 - Otevřít překladač ze strojového jazyka do asembleru při %1 + Otevřít disasembler při %1 Try to Load Unknown Symbols @@ -4925,7 +5374,7 @@ Open Disassembler - Otevřít překladač ze strojového jazyka do asembleru + Otevřít disasembler Open Disassembler at %1 @@ -4986,12 +5435,44 @@ Arguments: Argumenty: + + Start Debugger + Spustit ladicí program + + + &Executable: + &Spustitelný soubor: + + + &Arguments: + &Argumenty: + + + Run in &terminal: + Spustit v &terminálu: + + + &Working directory: + &Pracovní adresář: + + + &Tool chain: + Sada &nástrojů: + + + Break at '&main': + Bod přerušení při '&main': + + + &Recent: + &Poslední: + Debugger::Internal::StartRemoteDialog Select Debugger - Vybrat ladicí program + Vybrat ladicí program Select Location of Debugging Information @@ -5014,9 +5495,57 @@ Vybrat spouštěcí skript k serveru + Remote: "%1" + Vzdálený: "%1" + + Select Start Script Vybrat spouštěcí skript + + Start Debugger + Spustit ladicí program + + + Tool &chain: + S&ada nástrojů: + + + Local &executable: + &Místní spustitelný soubor: + + + &Host and port: + &Hostitelský počítač a číslo portu: + + + &Architecture: + &Architektura: + + + Sys&root: + Sys&root: + + + Location of debugging &information: + Umístění &informací o ladění: + + + Override host GDB s&tart script: + Přepsat &spouštěcí skript GDB na hostiteli: + + + &Use server start script: + &Použít spouštěcí skript k serveru: + + + &Server start script: + Spouštěcí skript k &serveru: + + + &Recent: + &Poslední: + Debugger::Internal::ThreadsHandler @@ -5119,6 +5648,10 @@ Always adjust column widths to contents Vždy přizpůsobit šířku sloupců obsahu + + Threads + Vlákna + Debugger::Internal::WatchData @@ -5164,6 +5697,14 @@ Odkazující adresa + Static Object Size + Velikost statického objektu + + + %1 bytes + %1 bytů + + Size Velikost @@ -5197,7 +5738,7 @@ Tooltip - Nástrojová rada + Kontextová nápověda <empty> @@ -5232,6 +5773,46 @@ + Raw pointer + Hodnota ukazatele + + + Latin1 string + Řetězec Latin1 + + + UTF8 string + Řetězec UTF8 + + + Local 8bit string + Řetězec znaků v místním 8bitovém kódování + + + UTF16 string + Řetězec UTF16 + + + UCS4 string + Řetězec UCS4 + + + Decimal + Desítkový + + + Hexadecimal + Šestnáctkový + + + Binary + Dvojkový + + + Octal + Osmičkový + + %1 Object at %2 Objekt typu %1 při %2 @@ -5350,7 +5931,7 @@ Change Display Format... - Změnit formát zobrazení... + Změnit formát zobrazení... Treat All Characters as Printable @@ -5374,7 +5955,7 @@ Use Display Format Based on Type - Použít formát zobrazení založený na typu + Použít formát zobrazení založený na typu Change Display for Type "%1": @@ -5406,7 +5987,7 @@ Setting a data breakpoint on an address will cause the program to stop when the data at the address is modified. - Nastavení bodu přerušení dat na adrese způsobí, že program se zastaví, když jsou tam nacházející se data změněna. + Nastavení bodu přerušení dat na určitou adresu způsobí, že program se zastaví, když jsou tam nacházející se data změněna. Add Data Breakpoint at Expression "%1" @@ -5414,7 +5995,7 @@ Setting a data breakpoint on an expression will cause the program to stop when the data at the address given by the expression is modified. - Nastavení bodu přerušení dat při výrazu způsobí, že program se zastaví, když jsou data nacházející se na adrese vyplývající z vyhodnocení výrazu změněna. + Nastavení bodu přerušení dat na určitý výraz způsobí, že program se zastaví, když jsou změněna data nacházející se na adrese dané oním výrazem. Insert New Evaluated Expression @@ -5433,6 +6014,26 @@ Odstranit všechny sledované výrazy + Change Local Display Format... + Změnit místní formát zobrazení... + + + Use Format for Type (Currently %1) + Použít formát zobrazení na typ (nyní %1) + + + Use Display Format Based on Type + Použít formát zobrazení založený na typu + + + Add Data Breakpoint at Expression + Přidat bod přerušení dat při výrazu + + + Change Global Display Formats... + Změnit celkové formáty zobrazení... + + Open Memory Editor... Otevřít editor paměti... @@ -5474,7 +6075,7 @@ Copy Contents to Clipboard - Kopírovat obsah do schránky + Obsah kopírovat do schránky Copy Value to Clipboard @@ -5486,7 +6087,7 @@ Close Editor Tooltips - Zavřít nástrojové rady k editoru + Zavřít vysvětlivky editoru Enter watch expression @@ -5616,11 +6217,11 @@ Makes use of Qt Creator's code model to find out if a variable has already been assigned a value at the point the debugger interrupts. - Používá kódový model Qt Creatoru, aby našel, zda proměnná již má nějakou hodnotu na místě přerušení ladicím programem. + Používá kódový model Qt Creatoru, aby našel, zda proměnná již má nějakou hodnotu na místě přerušení ladicím programem. Use code model - Použít model kódu + Použít model kódu <html><head/><body> @@ -5632,22 +6233,22 @@ Use Debugging Helper - Používat pomocnou knihovnu pro výstup dat + Používat pomocnou knihovnu pro výstup dat <html><head/><body> <p>The debugging helper is only used to produce a nice display of objects of certain types like QString or std::map in the &quot;Locals and Expressions&quot; view. It is not strictly necessary for debugging with Qt Creator. </p></body></html> - <html><head/><body> + <html><head/><body> <p>Pomocná knihovna pro výstup dat slouží pouze pro hezké zobrazení objektů určitého typu jako QString nebo std::map v okně &quot;Místní proměnné a sledované výrazy&quot;.</p> <p> K ladění s Qt Creatorem však není bezpodmínečně požadována. </p></body></html> Displays names of QThread based threads. - Zobrazí názvy vláken založených na QThread. + Zobrazí názvy vláken založených na QThread. Display thread names - Zobrazit názvy vláken + Zobrazit názvy vláken @@ -5693,7 +6294,7 @@ Class Generation - Vytvoření třídy + Vytváření třídy Form Editor @@ -5762,7 +6363,7 @@ Creates a Qt Designer form that you can add to a Qt Widget Project. This is useful if you already have an existing class for the UI business logic. - Vytvoří formůlář Qt Designeru, který můžete přidat do projektu doplňku Qt (Qt Widget).Tuto předlohu použijte, jestliže již existuje třída pro běh programu. + Vytvoří formulář Qt Designeru pro projekt Qt Widget.Tuto předlohu použijte, jestliže již existuje třída pro běh programu. Creates a Qt Designer form along with a matching class (C++ header and source file) for implementation purposes. You can add the form and class to an existing Qt Widget Project. @@ -5853,7 +6454,7 @@ Edit Signals/Slots - Upravit signály/místa (slot) + Upravit signály/zdířky (slots) Edit Buddies @@ -5861,7 +6462,7 @@ Edit Tab Order - Upravit pořadí zarážek + Upravit pořadí tabulátorů Meta+H @@ -5901,7 +6502,7 @@ Switch Source/Form - Přepínat mezi formulářem a zdrojovým souborem + Přepnout mezi formulářem a zdrojem Shift+F4 @@ -5913,7 +6514,7 @@ Signals && Slots Editor - Editor signálů a míst (slot) + Editor signálů a zdířek (slots) Locked @@ -5999,7 +6600,7 @@ Error finding/adding a slot. - Chyba při nalezení/přidání kódu místa (slot). + Chyba při nalezení/přidání zdířky (slot). Internal error: No project could be found for %1. @@ -6013,26 +6614,26 @@ Unable to add the method definition. - Nepodařilo se přidat vymezení metody. + Nepodařilo se přidat definici metody. DocSettingsPage Registered Documentation - Zapsaná dokumentace + Zapsaná dokumentace Add... - Přidat... + Přidat... Remove - Odstranit + Odstranit Add and remove compressed help files, .qch. - Přidat a odstranit stlačené soubory s nápovědou, .qch. + Přidat a odstranit stlačené soubory s nápovědou, .qch. Add @@ -6050,7 +6651,7 @@ Note: This adds the toolchain to the build environment and runs the program inside a virtual machine. It also automatically sets the correct Qt version. Použijte Virtual Box -Poznámka: Tímto se do prostředí pro sestavování přidá řetěz nástrojů a program se spustí uvnitř virtuálního stroje. +Poznámka: Tímto se do prostředí pro sestavování přidá sada nástrojů a program se spustí uvnitř virtuálního stroje. Také se automaticky nastaví správná verze Qt. @@ -6354,7 +6955,7 @@ File "%1" exists (add ! to override) - Soubor '%1' již existuje (Přidejte !, abyste jej přepsal) + Soubor '%1' již existuje (Přidejte ! chcete-li jej přepsat) Cannot open file "%1" for writing @@ -6404,6 +7005,14 @@ Already at newest change Dosažena poslední změna + + Unknown option: %1 + Neznámá volba: %1 + + + Argument must be positive: %1=%2 + Argument musí být kladný: %1=%2 + FakeVim::Internal::FakeVimOptionPage @@ -6415,6 +7024,94 @@ FakeVim FakeVim + + Use FakeVim + Používat FakeVim + + + Read .vimrc + Číst .vimrc + + + Vim Behavior + Chování Vim + + + Automatic indentation + Automatické odsazení + + + Start of line + Začátek řádku + + + Smart indentation + Chytré odsazování + + + Use search dialog + Použít dialog pro hledání + + + Expand tabulators + Tabulátor nahradit mezerami + + + Show position of text marks + Ukázat polohu textových značek + + + Smart tabulators + Režim chytrých tabulátorů + + + Pass key sequences like Ctrl-S to Qt Creator core instead of interpreting them in FakeVim. This gives easier access to Qt Creator core functionality at the price of losing some features of FakeVim. + Předat klávesové zkratky jako např. Ctrl-S Qt Creatoru místo FakeVim. Toto usnadňuje přístup k základním funkcím Qt Creatoru za cenu ztráty některých funkcí FakeVimu. + + + Pass control key + Klávesu Ctrl postoupit Qt Creatoru + + + Highlight search results + Zvýraznit výsledky hledání + + + Incremental search + Přírůstkové hledání + + + Shift width: + Odsazení: + + + Vim tabstop option + Nastavení tabulátorů Vim + + + Tabulator size: + Šířka tabulátoru: + + + Backspace: + Zpětná klávesa: + + + Keyword characters: + Znaky klíčového slova: + + + Copy Text Editor Settings + Kopírovat nastavení textového editoru + + + Set Qt Style + Nastavit styl Qt + + + Set Plain Style + Nastavit prostý styl + FakeVim::Internal::FakeVimPluginPrivate @@ -6447,7 +7144,7 @@ Jeden soubor nebyl uložen %n soubory nebyly uloženy - %n soubory nebyly uloženy + %n souborů nebylo uloženo @@ -6463,7 +7160,7 @@ FakeVimOptionPage Use FakeVim - Používat FakeVim + Používat FakeVim Vim style settings @@ -6483,7 +7180,7 @@ Shift width: - Odsazení: + Odsazení: Smart tabulators: @@ -6499,11 +7196,11 @@ Tabulator size: - Šířka zarážky: + Šířka zarážky: Backspace: - Zpětná klávesa: + Zpětná klávesa: VIM's "autoindent" option @@ -6531,75 +7228,75 @@ Read .vimrc - Číst .vimrc + Číst .vimrc Vim Behavior - Chování Vim + Chování Vim Automatic indentation - Automatické odsazení + Automatické odsazení Start of line - Začátek řádku + Začátek řádku Smart indentation - Chytré odsazování + Chytré odsazování Use search dialog - Použít dialog pro hledání + Použít dialog pro hledání Expand tabulators - Rozbalit zarážky + Rozbalit zarážky Show position of text marks - Ukázat polohu textových značek + Ukázat polohu textových značek Smart tabulators - Režim chytrých zarážek + Režim chytrých zarážek Highlight search results - Zvýraznit výsledky hledání + Zvýraznit výsledky hledání Incremental search - Přírůstkové hledání + Přírůstkové hledání Keyword characters: - Znaky klíčového slova: + Znaky klíčového slova: Copy Text Editor Settings - Kopírovat nastavení textového editoru + Kopírovat nastavení textového editoru Set Qt Style - Nastavit styl Qt + Nastavit styl Qt Set Plain Style - Nastavit prostý styl + Nastavit prostý styl Pass key sequences like Ctrl-S to Qt Creator core instead of interpreting them in FakeVim. This gives easier access to Qt Creator core functionality at the price of losing some features of FakeVim. - Způsobí, že s určitými klávesovými zkratkami jako je Ctrl-S se nezachází v FakeVim, nýbrž jsou postoupeny Qt Creatoru. Toto usnadňuje přístup k základním funkcím Qt Creatoru za cenu ztráty některých funkcí FakeVimu. + Způsobí, že s určitými klávesovými zkratkami jako je Ctrl-S se nezachází v FakeVim, nýbrž jsou postoupeny Qt Creatoru. Toto usnadňuje přístup k základním funkcím Qt Creatoru za cenu ztráty některých funkcí FakeVimu. Pass control key - Postoupit klávesu Ctrl + Postoupit klávesu Ctrl Vim tabstop option - Nastavení zarážek Vim + Nastavení zarážek Vim @@ -6617,37 +7314,37 @@ FilterSettingsPage Filters - Filtry + Filtry 1 - 1 + 1 Add - Přidat + Přidat Remove - Odstranit + Odstranit Attributes - Vlastnosti + Vlastnosti <html><body> <p> Add, modify, and remove document filters, which determine the documentation set displayed in the Help mode. The attributes are defined in the documents. Select them to display a set of relevant documentation. Note that some attributes are defined in several documents. </p></body></html> - <html><body> + <html><body> <p> -Přidat, upravit a odstranit dokumentové filtry, které určují v režimu nápovědy zobrazenou dokumentaci. Vlastnosti jsou popsány v dokumentech. Vyberte je, abyste zobrazil množinu související dokumentace. Všimněte si, že některé vlastnosti jsou popsány ve více dokumentech. +Přidat, upravit a odstranit dokumentové filtry, které v režimu nápovědy určují zobrazenou dokumentaci. Vlastnosti jsou popsány v dokumentech. Vyberte filtry, aby se zobrazila sada související dokumentace. Všimněte si, že některé vlastnosti jsou popsány ve více dokumentech. </p></body></html> No user defined filters available or no filter selected. - Nejsou dostupné žádné uživatelské filtry nebo není žádný filtr vybrán. + Nejsou dostupné žádné uživatelské filtry nebo není žádný filtr vybrán. @@ -6763,6 +7460,30 @@ Nahradit a najít další + Shift+Enter + Shift+Enter + + + Shift+Return + Shift+Return + + + Find Next (Selected) + Najít další (vybrané) + + + Ctrl+F3 + Ctrl+F3 + + + Find Previous (Selected) + Najít předchozí (vybrané) + + + Ctrl+Shift+F3 + Ctrl+Shift+F3 + + Replace Nahradit @@ -6909,7 +7630,7 @@ This is the slowest but safest option. - Toto je nejbezpečnější nastavení, ale zároveň i nejpomalejší. + Toto je nejbezpečnější nastavení, ale zároveň i nejpomalejší. Try to set breakpoints in plugins always automatically. @@ -6917,15 +7638,15 @@ Try to set breakpoints in selected plugins - Pokuste se nastavit body přerušení ve vybraných přídavných modulech + Pokuste se nastavit body přerušení ve vybraných přídavných modulech Matching regular expression: - Odpovídající regulárnímu výrazu: + Odpovídající regulárnímu výrazu: Never set breakpoints in plugins automatically - Nikdy nenastavovat body přerušení v přídavných modulech automaticky + Nikdy nenastavovat body přerušení v přídavných modulech automaticky This is either a full absolute path leading to the gdb binary you intend to use or the name of a gdb binary that will be searched in your PATH. @@ -6966,7 +7687,7 @@ Enable reverse debugging - Zapnout obrácené ladění + Zapnout obrácené ladění When this option is checked, 'Step Into' compresses several steps into one in certain situations, leading to 'less noisy' debugging. So will, e.g., the atomic @@ -6975,27 +7696,27 @@ Skip known frames when stepping - Přeskočit při provádění jednotlivého kroku známá místa + Přeskočit při provádění jednotlivého kroku známá místa Show a message box when receiving a signal - Při obdržení signálu ukázat okno se zprávami + Při obdržení signálu ukázat okno se zprávami Behavior of Breakpoint Setting in Plugins - Chování nastavení bodu přerušení v přídavných modulech + Chování nastavení bodu přerušení v přídavných modulech GDB - GDB + GDB This is either empty or points to a file containing GDB commands that will be executed immediately after GDB starts up. - Toto je buď prázdné nebo ukazuje na soubor obsahující příkazy, které jsou provedeny bezprostředně po spuštění GDB. + Toto je buď prázdné nebo ukazuje na soubor obsahující příkazy, které jsou provedeny bezprostředně po spuštění GDB. GDB startup script: - Skript pro spuštění GDB: + Skript pro spuštění GDB: This is the number of seconds Qt Creator will wait before @@ -7003,51 +7724,51 @@ should be sufficient for most applications, but there are situations when loading big libraries or listing source files takes much longer than that on slow machines. In this case, the value should be increased. - Toto je počet sekund, po které bude Qt Creator čekat, předtím než ukončí neodpovídající procesy GDB. + Toto je počet sekund, po které bude Qt Creator čekat, předtím než ukončí neodpovídající procesy GDB. Výchozí hodnota, jíž je 20 sekund, by měla postačovat v případě většiny programů. Ale jsou situace, kdy na pomalých strojích nahrávání velkých knihoven nebo výpis zdrojových souborů zabere mnohem více času, než je nastaveno. V takovém případě by se měla hodnota zvýšit. GDB timeout: - Překročení času u GDB: + Překročení času u GDB: Allows 'Step Into' to compress several steps into one step for less noisy debugging. For example, the atomic reference counting code is skipped, and a single 'Step Into' for a signal emission ends up directly in the slot connected to it. - Tato volba má za následek, že v určitých situacích sloučí 'Jednotlivý krok do' více kroků do jednoho, čímž se ladění uspíší. Například se přeskočí kód počítání atomárních odkazů; a jednotlivý 'Krok do' pro vyslání signálu skončí přímo v místě (slot) s ním spojeném. + Tato volba má za následek, že v určitých situacích sloučí 'Jednotlivý krok do' více kroků do jednoho, čímž se ladění uspíší. Například se přeskočí kód počítání atomárních odkazů; a jednotlivý 'Krok do' pro vyslání signálu skončí přímo v místě (slot) s ním spojeném. This will show a message box as soon as your application receives a signal like SIGSEGV during debugging. - Ukáže oznamovací okno, když program během ladění přijme signál SIGSEGV. + Ukáže oznamovací okno, když program během ladění přijme signál SIGSEGV. <html><head/></body><p>GDB allows setting breakpoints on source lines for which no code was generated. In such situations the breakpoint is shifted to the next source code line for which code was actually generated. This option reflects such temporary change by moving the breakpoint markers in the source code editor.</p></body></html> - <html><head/></body><p>Ne všechny řádky zdrojového kódu způsobují vytvoření spustitelného kódu. GDB však dovoluje, aby byl takových situacích pro takové řádky, pro něž nebyl vytvořen žádný kód, nastaven bod přerušení. Tento bod přerušení je pak posunut na další řádek se zdrojovým kódem, pro který byl kód skutečně vytvořen. Toto nastavení odráží takovou dočasnou změnu a způsobuje, že značky pro body přerušení jsou odpovídajícím způsobem posunuty v editoru zdrojového kódu.</p></body></html> + <html><head/></body><p>Ne všechny řádky zdrojového kódu způsobují vytvoření spustitelného kódu. GDB však dovoluje, aby byl takových situacích pro takové řádky, pro něž nebyl vytvořen žádný kód, nastaven bod přerušení. Tento bod přerušení je pak posunut na další řádek se zdrojovým kódem, pro který byl kód skutečně vytvořen. Toto nastavení odráží takovou dočasnou změnu a způsobuje, že značky pro body přerušení jsou odpovídajícím způsobem posunuty v editoru zdrojového kódu.</p></body></html> Adjust breakpoint locations - Upravit umístění bodů přerušení + Upravit umístění bodů přerušení This allows or inhibits reading the user's default .gdbinit file on debugger startup. - Dovolí nebo potlačí čtení výchozího uživatelského souboru-.gdbinit při spuštění ladiče. + Dovolí nebo potlačí čtení výchozího uživatelského souboru-.gdbinit při spuštění ladiče. Load .gdbinit file on startup - Nahrát soubor .gdbinit na začátku + Nahrát soubor .gdbinit na začátku Use asynchronous mode to control the inferior - Použít asynchronní režim k ovládání laděného procesu + Použít asynchronní režim k ovládání laděného procesu Stop when a qWarning is issued - Zastavit při qWarning + Zastavit při qWarning Stop when a qFatal is issued - Zastavit při qFatal + Zastavit při qFatal <html><head/><body><p>Selecting this enables reverse debugging.</p><.p><b>Note:</b>This feature is very slow and unstable on the GDB side. It exhibits unpredictable behaviour when going backwards over system calls and is very likely to destroy your debugging session.</p><body></html> @@ -7055,34 +7776,34 @@ Always try to set breakpoints in plugins automatically - Pokusit se nastavit body přerušení v přídavných modulech vždy automaticky + Pokusit se nastavit body přerušení v přídavných modulech vždy automaticky This adds common paths to locations of debug information at debugger startup. - Toto při spuštění ladiče přidá společné cesty do míst s informacemi o ladění. + Toto při spuštění ladiče přidá společné cesty do míst s informacemi o ladění. Use common locations for debug information automatically - Použít společné cesty pro informace o ladění automaticky + Použít společné cesty pro informace o ladění automaticky <html><head/><body><p>Selecting this enables reverse debugging.</p><.p><b>Note:</b> This feature is very slow and unstable on the GDB side. It exhibits unpredictable behavior when going backwards over system calls and is very likely to destroy your debugging session.</p><body></html> - <html><head/><body><p>Výběr tohoto povolí obrácené ladění.</p><.p><b>Poznámka:</b> Tato vlastnost je velmi pomalá a na straně GDB nestálá. Projevuje při postupu zpět systémovými voláními nepředvídatelné chování a je dost pravděpodobné, že zničí vaše ladicí sezení.</p><body></html> + <html><head/><body><p>Výběr tohoto povolí obrácené ladění.</p><.p><b>Poznámka:</b> Tato vlastnost je velmi pomalá a na straně GDB nestálá. Projevuje při postupu zpět systémovými voláními nepředvídatelné chování a je dost pravděpodobné, že zničí vaše ladicí sezení.</p><body></html> GenericMakeStep Override %1: - Přepsat %1: + Přepsat %1: Make arguments: - Argumenty příkazového řádku pro 'make': + Argumenty příkazového řádku pro 'make': Targets: - Cíle: + Cíle: @@ -7104,15 +7825,15 @@ Tool chain: - Řetězec nástrojů: + Sada nástrojů: <Invalid tool chain> - <Neplatný řetězec nástrojů> + <Neplatná sada nástrojů> Tool Chain: - Řetězec nástrojů: + Sada nástrojů: Generic Manager @@ -7158,8 +7879,12 @@ Zavést stávající projekt + Imports existing projects that do not use qmake, CMake or Autotools. This allows you to use Qt Creator as a code editor. + Zavede stávající projekty, které nepoužívají qmake, CMake nebo Autotools. Toto vám umožní používat Qt Creator jako editor kódu. + + Imports existing projects that do not use qmake or CMake. This allows you to use Qt Creator as a code editor. - Zavede stávající projekty, které nepoužívají qmake nebo CMake. Toto vám umožní používat Qt Creator jako editor kódu. + Zavede stávající projekty, které nepoužívají qmake nebo CMake. Toto vám umožní používat Qt Creator jako editor kódu. @@ -7233,7 +7958,7 @@ Would you like to delete the <b>unmerged</b> branch '%1'? - Chcete smazat větev '%1'? Ještě nebla provedena <b>žádná</b> slučovací operace? + Chcete smazat dosud <b>nesloučenou</b> větev '%1'? Delete Branch @@ -7305,15 +8030,15 @@ &Diff - &Rozdíly (diff) + &Rozdíly &Log - &Log + &Záznamy &Checkout - &Stáhnout (checkout) kopii + &Načíst (checkout) @@ -7328,11 +8053,15 @@ Select a Git Commit - Vyberte odeslání do Git + Vybrat zápis z Git Select Git Repository - Vyberte skladiště Git + Vyberte skladiště Git + + + Select Working Directory + Vybrat pracovní adresář Error @@ -7342,12 +8071,24 @@ Selected directory is not a Git repository Vybraný adresář není skladištěm Git + + Working directory: + Pracovní adresář: + + + Select + Vybrat + + + Change: + Revize: + Git::Internal::GitClient Note that the git plugin for QtCreator is not able to interact with the server so far. Thus, manual ssh-identification etc. will not work. - Všimněte si, že přídavný modul Git pro QtCreator stále ještě není schopen spolupracovat se serverem. Tudíž nebude pracovat ani ruční rozpoznání ssh a tak dále. + Upozornění: přídavný modul Git pro QtCreator zatím ještě není schopen spolupracovat se serverem. Tudíž nebude pracovat ani ruční rozpoznání ssh a tak dále. Unable to determine the repository for %1. @@ -7416,9 +8157,9 @@ Unable to remove %n file(s) from %1: %2 - Jeden soubor v %1 se nepodařilo odstraniit: %2 - %n soubory v %1 se nepodařilo odstraniit: %2 - %n souborů v %1 se nepodařilo odstraniit: %2 + Jeden soubor v %1 se nepodařilo odstranit: %2 + %n soubory v %1 se nepodařilo odstranit: %2 + %n souborů v %1 se nepodařilo odstranit: %2 @@ -7461,7 +8202,7 @@ Stash Description - Popis uložených změn (stash) + Popis odložených změn (stash) Description: @@ -7482,7 +8223,7 @@ Unable to run 'git clean' in %1: %2 - Příkaz pro vyčištění 'git clean' se nepodařilo provést: %1: %2 + Příkaz pro vyčištění 'git clean' se nepodařilo provést v %1: %2 There were warnings while applying %1 to %2: @@ -7562,9 +8303,9 @@ Committed %n file(s). - Odeslán jeden soubor. - %n soubory odeslány. - %n souborů odesláno. + Zapsán (commit) jeden soubor. + %n soubory zapsány (commit). + %n souborů zapsáno (commit). @@ -7597,7 +8338,7 @@ Git Log "%1" - Git Log (zápis) "%1" + Git Log (záznamy) "%1" Cannot describe "%1". @@ -7614,7 +8355,11 @@ Cannot checkout "%1" of "%2": %3 Meaning of the arguments: %1: Branch, %2: Repository, %3: Error message - Operace 'checkout' se nepodařila pro větev "%1" skladiště "%2": %3 Hlášení o chybě + Nelze načíst (checkout) větev "%1" ze skladiště "%2" Hlášení o chybě: %3 + + + Cannot obtain log of "%1": %2 + Nelze obdržet záznam pro "%1": %2 Cannot add %n file(s) to "%1": %2 @@ -7651,7 +8396,7 @@ Cannot checkout "%1" of %2 in "%3": %4 Meaning of the arguments: %1: revision, %2: files, %3: repository, %4: Error message - Soubory %2 revize "%1" ve skladišti "%3" se nepodařilo odbavit: %4 Hlášení o chybě + Nelze načíst revizi "%1" pro %2 ze skladiště "%3" Hlášení o chybě: %4 Cannot find parent revisions of "%1" in "%2": %3 @@ -7672,12 +8417,12 @@ Cannot stash in "%1": %2 - Operace uložení změn (stash) se v "%1" nepodařila: %2 + Operace odložení změn (stash) se v "%1" nepodařila: %2 Cannot resolve stash message "%1" in "%2". Look-up of a stash via its descriptive message failed. - Nepodařilo se najít žádné uložené změny (stash) s popisem "%1" ve skladišti "%2". + Nepodařilo se určit odložené změny s popisem "%1" ve skladišti "%2". Cannot run "git branch" in "%1": %2 @@ -7698,20 +8443,20 @@ There were warnings while applying "%1" to "%2": %3 - Při použití opravného souboru %1 na skladiště %2 se objevila varování: + Při použití záplaty "%1" na "%2" se objevila varování: %3 Cannot apply patch "%1" to "%2": %3 - Opravný soubor "%1" se na skladiště "%2" nepodařilo použít: %3 + Záplatu "%1" se na "%2" nepodařilo použít: %3 Would you like to stash your changes? - Chcete své změny uložit (stash)? + Chcete své změny odložit (stash)? Cannot obtain status: %1 - Stav se nepodařilo vyvolat: %1 + Stav se nepodařilo získat: %1 Cannot locate "%1". @@ -7723,43 +8468,43 @@ The repository "%1" is not initialized. - Skladiště "%1" ještě není spuštěno. + Skladiště "%1" ještě není založeno. Cannot retrieve last commit data of repository "%1". - Data z posledního odevzdání do skladiště "%1" se nepodařilo určit. + Data z posledního zápisu (commit) do skladiště "%1" se nepodařilo určit. Amended "%1" (%n file(s)). - Odevzdání "%1" změnilo jeden soubor. + Zápis "%1" poupravil jeden soubor. - Odevzdání "%1" změnilo jeden %n soubory. + Zápis "%1" poupravil %n soubory. - Odevzdání "%1" změnilo jeden %n souborů. + Zápis "%1" poupravil %n souborů. Amended "%1". - Odevzdání "%1" změněno. + Zápis "%1" poupraven. Cannot commit %n file(s): %1 - Odeslání jednoho souboru se nezdařilo: %1 + Zápis (commit) jednoho souboru se nezdařil: %1 - Odeslání jednoho %n souborů se nezdařilo: %1 + Zápis (commit) %n souborů se nezdařil: %1 - Odeslání jednoho %n souborů se nezdařilo: %1 + Zápis (commit) %n souborů se nezdařil: %1 Revert - Vrátit + Vrátit změny The file has been changed. Do you want to revert it? @@ -7771,7 +8516,7 @@ The command 'git pull --rebase' failed, aborting rebase. - Příkaz 'git pull --rebase' se nezdařil. Operace rebase se ruší. + Příkaz 'git pull --rebase' selhal. Přeskládání (rebase) se ruší. Git SVN Log @@ -7783,23 +8528,23 @@ Cannot restore stash "%1": %2 - Stash (uložené změny) "%1" nelze obnovit: %2 + Odložené změny "%1" nelze použít: %2 Cannot restore stash "%1" to branch "%2": %3 - Uložené změny "%1" nelze ve větvi "%2" obnovit: %3 + Odložené změny "%1" nelze ve větvi "%2" použít: %3 Cannot remove stashes of "%1": %2 - Odstranění uložených změn ze skladiště "%1" se nezdařilo: %2 + Odstranění odložených změn ze skladiště "%1" se nezdařilo: %2 Cannot remove stash "%1" of "%2": %3 - Odstranění uložené změny "%1" ze skladiště "%2" se nezdařilo: %3 + Odstranění odložené změny "%1" ze skladiště "%2" se nezdařilo: %3 Cannot retrieve stash list of "%1": %2 - Seznam uložených změn skladiště "%1" nelze určit: %2 + Seznam odložených změn skladiště "%1" nelze určit: %2 Cannot determine git version: %1 @@ -7814,11 +8559,11 @@ Diff Current File - Rozdíly (diff) nynějšího souboru + Rozdíly nynějšího souboru Diff "%1" - Rozdíly (diff) pro "%1" + Rozdíly pro "%1" Alt+G,Alt+D @@ -7842,7 +8587,7 @@ Log of "%1" - Zápis pro "%1" + Záznamy pro "%1" Alt+G,Alt+L @@ -7854,7 +8599,7 @@ Blame for "%1" - Blame (vina) pro "%1" + Vypsat anotace (blame) pro "%1" Alt+G,Alt+B @@ -7862,7 +8607,7 @@ Diff - Rozdíly (diff) + Rozdíly Status @@ -7874,7 +8619,7 @@ Clean... - Uklidit skladiště + Vyčistit skladiště... Launch gitk @@ -7882,11 +8627,11 @@ Remotes... - Git Remotes ... + Vzdálené... Patch - Opravný soubor + Záplaty Apply from Editor @@ -7902,11 +8647,11 @@ Amend Last Commit... - Změnit poslední odevzdání... + Poupravit poslední zápis (amend)... Diff &Selected Files - Rozdíly (diff) pro &vybrané soubory + Rozdíly pro &vybrané soubory Undo all pending changes to the repository @@ -7924,7 +8669,7 @@ Choose Patch - Vybrat cestu k opravnému souboru + Vybrat záplatu Undo Changes for "%1" @@ -7936,23 +8681,23 @@ Stage File for Commit - Připojit soubor pro odeslání (stage) + Připravit soubor pro zápis (stage) Blame Current File - Blame (obviňovat) nynější soubor + Vypsat anotace pro nynější soubor (blame) Diff of "%1" - Rozdíly (diff) pro "%1" + Rozdíly pro "%1" Log Current File - Zápis pro nynější soubor + Záznamy pro nynější soubor Stage "%1" for Commit - Připojit "%1" pro odeslání (stage) + Připravit "%1" pro zápis (stage) Alt+G,Alt+A @@ -7960,11 +8705,11 @@ Unstage File from Commit - Odstranit soubor z odeslání (unstage) + Odstranit soubor z přípravky (unstage) Unstage "%1" from Commit - Odstranit "%1" z odeslání (unstage) + Odstranit "%1" z přípravky (unstage) Undo Unstaged Changes @@ -7976,19 +8721,19 @@ Undo Uncommitted Changes - Vzít neodeslané změny zpět + Vrátit nezapsané změny zpět Undo Uncommitted Changes for "%1" - Neodeslané změny pro "%1" vrátit zpět + Nezapsané změny pro "%1" vrátit zpět Diff Current Project - Rozdíly (diff) pro nynější projekt + Rozdíly pro nynější projekt Diff Project "%1" - Rozdíly (diff) pro projekt "%1" + Rozdíly pro projekt "%1" Project Status @@ -8000,11 +8745,11 @@ Log Project - Zápis pro projekt + Záznamy pro projekt Log Project "%1" - Zápis pro projekt "%1" + Záznamy pro projekt "%1" Alt+G,Alt+K @@ -8016,7 +8761,7 @@ Stash - Ulít (stash) + Odložit (stash) Saves the current state of your work. @@ -8032,7 +8777,7 @@ Diff Repository - Rozdíly (diff) skladiště + Rozdíly skladiště (diff) Repository Status @@ -8040,7 +8785,7 @@ Log Repository - Zápis (log) skladiště + Zápis skladiště (log) Apply Patch @@ -8048,7 +8793,7 @@ Apply "%1" - Použít opravný program "%1" + Použít záplatu "%1" Apply Patch... @@ -8072,23 +8817,23 @@ Saves the current state of your work and resets the repository. - Uloží nynější stav vaší práce a nastaví znovu skladiště. + Uloží nynější stav vaší práce a znovunastaví skladiště. Pull - Vytáhnout (pull) + Přivést a začlenit (pull) Stash Pop - Ulít pop (stash pop) + Přiložit (stash pop) Restores changes saved to the stash list using "Stash". - Obnoví změny uložené do seznamu s uloženými změnami pomocí příkazu "Stash". + Přiloží změny které byly odloženy do seznamu odložených změn pomocí příkazu "Stash". Commit... - Odeslat... + Zapsat (commit)... Alt+G,Alt+C @@ -8096,27 +8841,39 @@ Push - Zatlačit (push) + Odvést (push) + + + Alt+G,Alt+Shift+D + Alt+G,Alt+Shift+D Undo Uncommitted Changes... - Vzít neodeslané změny zpět... + Vzít nezapsané změny zpět... + + + Launch repository browser + Spustit prohlížeč skladiště Branches... Větve... + Show... + Ukázat... + + Stashes... - Uložené změny (stash)... + Odložené změny... Amend %1 - Změnit odevzdání %1 + Pozměnit zápis %1 Git Commit - Git Commit (odeslat) + Zapsat (Git Commit) Closing Git Editor @@ -8124,7 +8881,7 @@ Git will not accept this commit. Do you want to continue to edit it? - Git toto odevzdání nepřijímá. Chcete je dále zpracovat? + Git tento zápis (commit) nepřijme. Chcete jej dále zpracovávat? Unable to retrieve file list @@ -8140,7 +8897,7 @@ Patches (*.patch *.diff) - Soubory s opravnými programy (*.patch *.diff) + Soubory se záplatami (*.patch *.diff) Choose patch @@ -8148,7 +8905,7 @@ Patch %1 successfully applied to %2 - Opravný soubor %1 byl s úspěchem použit na skladiště %2 + Záplata %1 byla s úspěchem použita na %2 List Stashes @@ -8156,7 +8913,7 @@ Show Commit... - Ukázat odeslání... + Ukázat odeslání... Subversion @@ -8164,15 +8921,15 @@ Log - Zápis + Záznamy Fetch - Natáhnout + Přivést (fetch) Commit - Odeslat + Zapsat (commit) Diff Selected Files @@ -8202,7 +8959,7 @@ Another submit is currently being executed. - V současnosti se již provádí jiné předložení. + V současnosti se již provádí jiné odeslání (submit). Cannot create temporary file: %1 @@ -8214,7 +8971,7 @@ Do you want to commit the change? - Chcete odeslat změnu? + Chcete zapsat změnu? The commit message check failed. Do you want to commit the change? @@ -8225,7 +8982,7 @@ Git::Internal::GitSettings The binary '%1' could not be located in the path '%2' - Spustitelný soubor '%1' se v umístění '%2' nepodařilo najít + Spustitelný soubor '%1' se v cestě '%2' nepodařilo najít @@ -8247,7 +9004,7 @@ repository - Skladiště + skladiště Branch: @@ -8255,11 +9012,11 @@ branch - Větev + větev Commit Information - Informace o odeslání + Informace o zápisu Author: @@ -8267,7 +9024,11 @@ Email: - E-mailová adresa: + E-mail: + + + By&pass hooks + O&bejít podmíněné akce (hooks) @@ -8297,7 +9058,7 @@ PATH: - Proměnná CESTA: + Proměnná CESTA: From system @@ -8317,7 +9078,7 @@ Note that huge amount of commits might take some time. - Všimněte si, že velký počet může vyvolat dlouhou čekací dobu. + Upozornění: velký počet zápisů může dlouhou trvat. Timeout (seconds): @@ -8337,7 +9098,7 @@ From System - Ze systému + Ze systému Miscellaneous @@ -8353,7 +9114,7 @@ Prompt on submit - Potvrdit předložení + Ptát se na potvrzení Ignore whitespace changes in annotation @@ -8365,7 +9126,7 @@ Pull with rebase - Vytáhnout (pull) s rebase + Přivést s přeskládáním (pull --rebase) Set "HOME" environment variable @@ -8381,7 +9142,7 @@ Customize Environment: - Přizpůsobit prostředí: + Přizpůsobit prostředí: Git needs to find Perl in the environment. @@ -8389,7 +9150,23 @@ Log count: - Počet zápisů omezit na: + Počet záznamů omezit na: + + + Configuration + Nastavení + + + Prepend to PATH: + Předpona na začátku CESTY: + + + Repository browser + Prohlížeč skladiště + + + Command: + Příkaz: @@ -8506,6 +9283,22 @@ Cannot unregister documentation file %1! Soubor s dokumentací %1 nelze odhlásit! + + Add and remove compressed help files, .qch. + Přidat a odstranit stlačené soubory s nápovědou, .qch. + + + Registered Documentation + Přihlášená dokumentace + + + Add... + Přidat... + + + Remove + Odstranit + Help::Internal::FilterSettingsPage @@ -8537,6 +9330,32 @@ Help Nápověda + + <html><body> +<p> +Add, modify, and remove document filters, which determine the documentation set displayed in the Help mode. The attributes are defined in the documents. Select them to display a set of relevant documentation. Note that some attributes are defined in several documents. +</p></body></html> + <html><body> +<p> +Přidat, upravit a odstranit dokumentové filtry, které v režimu nápovědy určují zobrazenou dokumentaci. Vlastnosti jsou popsány v dokumentech. Vyberte filtry, aby se zobrazila sada související dokumentace. Všimněte si, že některé vlastnosti jsou popsány ve více dokumentech. +</p></body></html> + + + Attributes + Vlastnosti + + + 1 + 1 + + + Add + Přidat + + + Remove + Odstranit + Help::Internal::HelpIndexFilter @@ -8684,7 +9503,7 @@ Unfiltered - Nezpracovaný + Žádný filtr <html><head><title>No Documentation</title></head><body><br/><center><b>%1</b><br/>No documentation available.</center></body></html> @@ -8696,7 +9515,7 @@ Filtered by: - Zpracováno: + Profiltrováno: @@ -9077,11 +9896,11 @@ MakeStep Override %1: - Přepsat %1: + Přepsat %1: Make arguments: - Argumenty příkazového řádku pro 'make': + Argumenty příkazového řádku pro 'make': @@ -9107,18 +9926,18 @@ Nicknames - Přezdívky + Přezdívky OpenWithDialog Open File With... - Otevřít soubor s... + Otevřít soubor s... Open file extension with: - Otevřít souborovou příponu s: + Otevřít souborovou příponu s: @@ -9152,22 +9971,22 @@ Perforce::Internal::ChangeNumberDialog Change Number - Změnit číslo + Číslo změny Change Number: - Změnit číslo: + Číslo změny: Perforce::Internal::PendingChangesDialog P4 Pending Changes - P4 zbývající změny + P4 nevyřízené změny Submit - Předložit + Odevzdat (submit) Cancel @@ -9246,19 +10065,19 @@ Diff Current File - Rozdíly (diff) nynějšího souboru + Rozdíly nynějšího souboru Diff "%1" - Rozdíly (diff) pro "%1" + Rozdíly pro "%1" Diff Current Project/Session - Rozdíly (diff) pro nynější projekt/sezení + Rozdíly pro nynější projekt/sezení Diff Project "%1" - Rozdíly (diff) pro projekt "%1" + Rozdíly pro projekt "%1" Alt+P,Alt+D @@ -9266,7 +10085,7 @@ Diff Opened Files - Rozdíly (diff) pro otevřené soubory + Rozdíly pro otevřené soubory Opened @@ -9278,7 +10097,7 @@ Submit Project - Předložit projekt + Odevzdat projekt (submit) Alt+P,Alt+S @@ -9286,7 +10105,7 @@ Pending Changes... - Zbývající změny... + Nevyřízené změny... Update Current Project/Session @@ -9294,7 +10113,7 @@ Update Project "%1" - Obnovit projekt "%1" + Aktualizovat projekt "%1" Describe... @@ -9302,23 +10121,23 @@ Annotate Current File - Opatřit nynější soubor vysvětlivkami + Opatřit nynější soubor anotacemi Annotate "%1" - Opatřit vysvětlivkami "%1" + Opatřit anotacemi "%1" Annotate... - Opatřit vysvětlivkami... + Opatřit anotacemi... Filelog Current File - Zápis k souboru pro nynější soubor + Záznamy k souboru pro nynější soubor Filelog "%1" - Zápis k souboru "%1" + Záznamy k souboru "%1" Alt+P,Alt+F @@ -9326,15 +10145,15 @@ Filelog... - Zápis k souboru... + Záznamy k souboru... Update All - Obnovit vše + Aktualizovat vše Submit - Předložit + Odevzdat (submit) Diff Selected Files @@ -9350,7 +10169,7 @@ p4 revert - Vrátit (p4 revert) + Vrátit zpět (revert) The file has been changed. Do you want to revert it? @@ -9364,7 +10183,7 @@ Another submit is currently executed. - V současnosti se již provádí jiné předložení. + V současnosti se již provádí jiné odevzdání (submit). Cannot create temporary file. @@ -9376,19 +10195,19 @@ p4 annotate - Opatřit vysvětlivkami (p4 annotate) + Opatřit anotacemi p4 annotate %1 - Opatřit vysvětlivkami (p4 annotate) "%1" + Opatřit anotacemi "%1" p4 filelog - Zápis pro soubor (p4 filelog) + Záznamy pro soubor p4 filelog %1 - Zápis k souboru (p4 filelog) %1 + Záznamy k souboru%1 The process terminated with exit code %1. @@ -9408,51 +10227,51 @@ Log Project - Zápis pro projekt + Záznamy pro projekt Log Project "%1" - Zápis pro projekt "%1" + Záznamy pro projekt "%1" Submit Project "%1" - Předložit projekt "%1" + Odevzdat projekt "%1" (submit) Update Current Project - Obnovit nynější projekt + Aktualizovat nynější projekt Revert Unchanged - Zvrátit požadované, nezměněné soubory + Vrátit požadované, nezměněné soubory Revert Unchanged Files of Project "%1" - Zvrátit požadované, nezměněné soubory projektu "%1" + Vrátit požadované, nezměněné soubory projektu "%1" Revert Project - Zvrátit změny v projektu + Vvrátit změny v projektu Revert Project "%1" - Zvrátit změny v projektu "%1" + Vrátit změny v projektu "%1" Repository Log - Zápis skladiště + Záznamy skladiště Diff &Selected Files - Rozdíly (diff) pro &vybrané soubory + Rozdíly pro &vybrané soubory Do you want to revert all changes to the project "%1"? - Chcete zvrátit všechny zbývající změny v projektu "%1"? + Chcete vrátit zpět všechny změny v projektu "%1"? Could not start perforce '%1'. Please check your settings in the preferences. - Příkaz 'Perforce' '%1' se nepodařilo spustit. Ověřte, prosím, svá nastavení v nastaveních. + Příkaz 'Perforce' '%1' se nepodařilo spustit. Ověřte, prosím, své volby v nastaveních. Perforce did not respond within timeout limit (%1 ms). @@ -9468,11 +10287,11 @@ p4 diff %1 - Lišit se (p4 diff) %1 + Rozdíly pro %1 p4 describe %1 - Popsat (p4 describe) %1 + Popsat (describe) %1 Closing p4 Editor @@ -9480,11 +10299,11 @@ Do you want to submit this change list? - Chcete předložit seznam se změnami? + Chcete odevzdat tento seznam změn (submit)? The commit message check failed. Do you want to submit this change list? - Ověření popisu týkajícího se odeslání se nezdařilo. Přesto chcete předložit tento seznam se změnami? + Přezkoušení popisu změn se nezdařilo. Přesto chcete odevzdat tento seznam změn? The commit message check failed. Do you want to submit this change list @@ -9496,7 +10315,7 @@ p4 submit failed: %1 - Chyba při předkládání: %1 + Chyba při odevzdání (submit): %1 Error running "where" on %1: %2 @@ -9526,11 +10345,11 @@ Pending change - Zbývající změny + Nevyřízené změny Could not submit the change, because your workspace was out of date. Created a pending submit instead. - Změnu se nepodařilo předložit, protože váš pracovní prostor byl zastaralý. Místo toho bylo vytvořeno předložení čekající na vyřízení. + Změnu se nepodařilo odevzdat (submit), protože váš pracovní prostor byl zastaralý. Místo toho bylo vytvořeno odevzdání čekající na vyřízení. Invalid configuration: %1 @@ -9549,7 +10368,7 @@ Perforce::Internal::PerforceSubmitEditor Perforce Submit - Předložení 'Perforce' + Odevzdání 'Perforce' (submit) @@ -9623,7 +10442,7 @@ P4 port: - Číslo přípojky P4: + Číslo portu P4: Miscellaneous @@ -9639,11 +10458,11 @@ Prompt on submit - Potvrdit předložení + Ptát se na potvrzení Log count: - Počet zápisů omezit na: + Počet záznamů omezit na: Automatically open files when editing @@ -9666,14 +10485,14 @@ Perforce Command - Příkaz 'Perforce' + Příkaz Perforce Perforce::Internal::SubmitPanel Submit - Předložit + Odevzdat (submit) Change: @@ -9715,11 +10534,11 @@ PluginManager The plugin '%1' does not exist. - Neexistuje žádnpřídavný modul '%1'. + Přídavný modul '%1' neexistuje. Unknown option %1 - Neplatný argument příkazového řádku %1 + Neplatný volba příkazového řádku %1 The option %1 requires an argument. @@ -9727,18 +10546,18 @@ Failed Plugins - Nenahrané přídavné moduly (neúspěch během nahrávání) + Přídavné moduly které selhaly při nahrávání PluginSpec '%1' misses attribute '%2' - Vlastnost '%1' chybí u '%2' + U '%1' chybí vlastnost '%2' '%1' has invalid format - '%1' je v nějakém neplatném formátu + '%1' je v neplatném formátu Invalid element '%1' @@ -9754,7 +10573,7 @@ Expected element '%1' as top level element - Kořenový prvek musí být '%1' + Prvek '%1' má být na vrchní úrovni Resolving dependencies failed because state != Read @@ -9770,7 +10589,7 @@ Plugin is not valid (does not derive from IPlugin) - Přídavný modul je neplatný (není odvozen od třídy IPlugin) + Přídavný modul je neplatný (není odvozen od IPlugin) Initializing the plugin failed because state != Loaded @@ -9792,6 +10611,10 @@ Internal error: have no plugin instance to perform extensionsInitialized Vnitřní chyba: Není žádná instance přídavného modulu k provedení 'extensionsInitialized' + + Internal error: have no plugin instance to perform delayedInitialize + Vnitřní chyba: Není žádná instance přídavného modulu k provedení 'delayedInitialize' + ProjectExplorer::AbstractProcessStep @@ -9852,7 +10675,7 @@ Finished %1 of %n build steps - + Jeden z %n kroků sestavování dokončen %1 z %n kroků sestavování dokončeny %1 z %n kroků sestavování dokončeno @@ -9871,36 +10694,36 @@ Compile Category for compiler isses listed under 'Issues' - Sestavení + Sestavení Build System Category for build system isses listed under 'Issues' - Sestavovací systém + Sestavovací systém Build canceled - Sestavování zrušeno + Sestavování zrušeno Canceled build. - Zrušené sestavování. + Zrušené sestavování. Build - Sestavování + Sestavování Error while building project %1 (target: %2) - Chyba při sestavování projektu %1 (cíl: %2) + Chyba při sestavování projektu %1 (cíl: %2) When executing build step '%1' - Při provádění sestavovacího kroku '%1' + Při provádění sestavovacího kroku '%1' Running build steps for project %1... - Běží sestavovací kroky pro projekt %1... + Běží sestavovací kroky pro projekt %1... Finished %n of %1 build steps @@ -9926,6 +10749,48 @@ <b>Running build steps for project %2...</b> <b>Provádí se kroky sestavování pro projekt %2...</b> + + Finished %1 of %n steps + + Dokončen %1 z %n kroků + Dokončeny %1 z %n kroků + Dokončeno %1 z %n kroků + + + + Compile + Category for compiler issues listed under 'Issues' + Překlad + + + Build System + Category for build system issues listed under 'Issues' + Sestavovací systém + + + Build/Deployment canceled + Sestavení/Nasazení bylo zrušeno + + + Canceled build/deployment. + Sestavení/Nasazení zrušeno. + + + Error while building/deploying project %1 (target: %2) + Chyba při sestavování/nasazování projektu %1 (cíl: %2) + + + When executing step '%1' + Při provádění kroku '%1' + + + Running steps for project %1... + Prováděn postup pro projekt %1... + + + Skipping disabled step %1. + Přeskakuje se zakázaný krok %1. + ProjectExplorer::CustomExecutableRunConfiguration @@ -9935,11 +10800,11 @@ Could not find the executable, please specify one. - Nepodařilo se najít spustitelný soubor. Jeden, prosím, zadejte. + Nepodařilo se najít spustitelný soubor; nějaký, prosím, zadejte. Clean Environment - Smazat prostředí + Vyčistit prostředí System Environment @@ -10010,7 +10875,11 @@ &Unset - &Vyprázdnit + &Vypnout + + + &Batch Edit... + Upravit jako &text... Unset <a href="%1"><b>%1</b></a> @@ -10018,7 +10887,17 @@ Set <a href="%1"><b>%1</b></a> to <b>%2</b> - Přidělit <b>%2</b> na <a href="%1"><b>%1</b></a> + Nastavit <a href="%1"><b>%1</b></a> na <b>%2</b> + + + Use <b>%1</b> + %1 is "System Environment" or some such. + Použít <b>%1</b> + + + Use <b>%1</b> and + Yup, word puzzle. The Set/Unset phrases above are appended to this. %1 is "System Environment" or some such. + Použít <b>%1</b> a Unset <b>%1</b> @@ -10030,11 +10909,11 @@ Using <b>%1</b> - Používá se <b>%1</b> + Používá se <b>%1</b> Using <b>%1</b> and - Používá se <b>%1</b> a + Používá se <b>%1</b> a Summary: No changes to Environment @@ -10092,7 +10971,7 @@ Build Steps - Kroky sestavování + Postup sestavování Edit Build Configuration: @@ -10100,7 +10979,7 @@ No build settings available - Nejsou dostupná žádná nastavení pro sestavování + Sestavování zatím není nastaveno Edit build configuration: @@ -10116,7 +10995,7 @@ Cancel Build && Remove Build Configuration - Zrušit sestavování a odstranit nastavení sestavování + Zrušit sestavování a odstranit jeho nastavení Do Not Remove @@ -10132,7 +11011,7 @@ Do you want to cancel the build process and remove the Build Configuration anyway? - Chcete zrušit proces sestavování a v každém případě odstranit nastavení sestavování? + Chcete zrušit sestavování a v každopádně odstranit jeho nastavení? Remove Build Configuration? @@ -10212,11 +11091,11 @@ Build Steps - Kroky při sestavování + Postup sestavování Clean Steps - Kroky k očistění + Postup pročišťování Move Up @@ -10228,7 +11107,7 @@ Remove Item - Odstranit prvek + Odstranit položku Removing Step failed @@ -10251,7 +11130,7 @@ ProjectExplorer::Internal::CompileOutputWindow Compile Output - Výstup sestavení + Výstup překladače @@ -10262,7 +11141,7 @@ A project is currently being built. - Právě je sestavován jeden projekt. + Právě je sestavován nějaký projekt. Close Qt Creator? @@ -10270,7 +11149,7 @@ Do you want to cancel the build process and close Qt Creator anyway? - Chcete zrušit proces sestavování a v každém případě zavřít Qt Creator? + Chcete zrušit sestavování a každopádně zavřít Qt Creator? @@ -10323,11 +11202,11 @@ Run in &Terminal - Spustit v &terminálu + Spustit v &terminálu Debugger: - Ladič: + Ladič: Run Environment @@ -10335,6 +11214,14 @@ Base environment for this runconfiguration: + Základní prostředí pro toto nastavení spuštění: + + + Run in &terminal + Spustit v &terminálu + + + Base environment for this run configuration: Základní prostředí pro toto nastavení spuštění: @@ -10356,7 +11243,7 @@ Running executable: <b>%1</b> %2 - Spouští se spustitelný soubor: <b>%1</b> %2 + Běží spustitelný soubor: <b>%1</b> %2 @@ -10481,7 +11368,7 @@ Force Quit - Ukončit + Ukončit vynuceně The application is still running. Close it first. @@ -10510,12 +11397,12 @@ Custom Process Step Default ProcessStep display name - Uživatelsky stanovený krok procesu + Uživatelsky stanovený krok Custom Process Step item in combobox - Uživatelsky stanovený krok procesu + Uživatelsky stanovený krok @@ -10534,7 +11421,7 @@ Command Arguments: - Argumenty příkazového řádku: + Argumenty příkazu: Enable Custom Process Step @@ -10542,7 +11429,7 @@ Enable custom process step - Povolit uživatelsky stanovený krok zpracování + Povolit uživatelsky stanovený krok zpracování Working directory: @@ -10550,7 +11437,7 @@ Command arguments: - Argumenty příkazového řádku: + Argumenty příkazu: @@ -10604,7 +11491,7 @@ Failed to add subproject '%1' to project '%2'. - Podřízený projekt '%1' + Dílčí projekt '%1' se projektu '%2' nepodařilo přidat. @@ -10614,11 +11501,11 @@ A version control system repository could not be created in '%1'. - V adresáři '%1' se skladiště systému na řízení verzí nepodařilo vytvořit. + V adresáři '%1' se verzovací skladiště nepodařilo vytvořit. Failed to add '%1' to the version control system. - '%1' se nepodařilo přidat do systému pro řízení verzí. + '%1' se nepodařilo přidat do verzovacího systému. @@ -10629,7 +11516,7 @@ Hide generated files - Neukazovat vytvořené soubory + Neukazovat automaticky vytvořené soubory Simplify Tree @@ -10637,7 +11524,7 @@ Hide Generated Files - Skrýt vytvořené soubory + Skrýt automaticky tvořené soubory Synchronize with Editor @@ -10663,18 +11550,18 @@ ProjectExplorer::Internal::ProjectWindow Active Build and Run Configurations - Činná nastavení sestavování a spouštění + Činná nastavení pro sestavování a spouštění No project loaded. - Nenahrán žádný projekt. + Není nahrán žádný projekt. ProjectExplorer::Internal::ProjectWizardPage Add to &VCS (%1) - Přidat do &systému pro ověřování verzí (%1) + Přidat do verzovacího &systému (%1) Summary @@ -10682,7 +11569,7 @@ Add as a subproject to project: - Přidat do projektu jako podřízený projekt: + Přidat do projektu jako dílčí projekt: Add to &project: @@ -10739,19 +11626,19 @@ Run configuration: - Nastavení spuštění: + Nastavení spuštění: Deployment: - Nasazení: + Nasazení: Add - Přidat + Přidat Remove - Odstranit + Odstranit Rename @@ -10759,7 +11646,7 @@ Rename ... - Přejmenovat... + Přejmenovat... @@ -10822,22 +11709,22 @@ Automatically restore the last session when Qt Creator is started. - Obnovit poslední sezení při spuštění Qt Creatoru automaticky. + Automaticky obnovit minulé sezení při spuštění Qt Creatoru. Restore last session on startup - Obnovit poslední sezení při spuštění + Při spuštění obnovit minulé sezení ProjectExplorer::Internal::SessionFile Session - Sezení + Sezení Failed to open project - Nepodařilo se otevřít projekt + Nepodařilo se otevřít projekt Untitled @@ -10934,6 +11821,10 @@ Manage ... + Spravovat... + + + Manage... Spravovat... @@ -11029,7 +11920,7 @@ Clean All - Vyčistit vše + Pročistit vše Build Project @@ -11045,7 +11936,7 @@ Clean Project - Vyčistit projekt + Pročistit projekt Run @@ -11106,7 +11997,7 @@ Recent Sessions - Naposledy otevřená sezení + Naposledy otevřená sezení Deploy All @@ -11138,7 +12029,31 @@ Clean Project "%1" - Vyčistit projekt "%1" + Pročistit projekt "%1" + + + Sessions + Sezení + + + Close All Projects and Editors + Zavřít všechny projekty a editory + + + Run Without Deployment + Spuštění bez nasazení + + + Build + Sestavení + + + Rebuild + Sestavit znovu + + + Clean + Pročistit Build Without Dependencies @@ -11154,7 +12069,11 @@ Clean Without Dependencies - Vyčistit s vyloučením závislostí + Pročistit s vyloučením závislostí + + + Deploy + Nasazení New Subproject... @@ -11178,12 +12097,20 @@ Nastavit jako činný projekt + Set "%1" as Active Project + Nastavit "%1" jako činný projekt + + Collapse All Složit vše Open Build/Run Target Selector... - Otevřít volič pro sestavování/spouštění cíle... + Otevřít volič pro cíl sestavování/spouštění... + + + Quick Switch Target Selector + Rychlý výběr cíle Ctrl+T @@ -11219,23 +12146,78 @@ Do you want to cancel the build process and unload the project anyway? - Chcete zrušit proces sestavování a v každém případě Zrušit nahrání projektu? + Chcete zrušit sestavování a každopádně zrušit nahrání projektu? + + + Ignore all errors? + Přehlížet všechny chyby? + + + Found some build errors in current task. +Do you want to ignore them? + V současném úkolu byly nalezeny chyby při sestavování. +Chcete je přehlížet? Always save files before build Vždy uložit soubory před sestavováním + The project %1 is not configured, skipping it. + + Projekt %1 není nastaven. Přeskakuje se. + + + No project loaded. + Není nahrán žádný projekt. + + + Currently building the active project. + Právě je sestavován činný projekt. + + + The project %1 is not configured. + Projekt %1 není nastaven. + + + Project has no build settings. + Projekt nemá žádná nastavení pro sestavování. + + No project loaded - Nenahrán žádný projekt + Není nahrán žádný projekt + + + No active project. + Žádný projekt není činný. + + + The project '%1' has no active target. + Projekt '%1' nemá žádný činný cíl. + + + The target '%1' for the project '%2' has no active run configuration. + Cíl '%1' projektu '%2' nemá žádné činné nastavení pro spuštění. + + + Cannot run '%1'. + Nelze spustit '%1'. + + + Project Editing Failed + Úpravy projektu se nepodařily + + + The file %1 was renamed to %2, but the project file %3 could not be automatically changed. + Soubor %1 byl přejmenován na %2, ale soubor s projektem %3 se automaticky změnit nepodařilo. Currently building the active project - Právě je sestavován aktivní projekt + Právě je sestavován aktivní projekt Project has no build settings - Projekt nemá žádná nastavení pro sestavování + Projekt ještě nebyl nastaven pro sestavování Building '%1' is disabled: %2<br> @@ -11243,7 +12225,7 @@ A build is in progress - Nyní běží jedno sestavování + Právě probíhá sestavování Building '%1' is disabled: %2 @@ -11265,31 +12247,35 @@ A project is currently being built. - Právě je sestavován jeden projekt. + Právě je sestavován nějaký projekt. Do you want to cancel the build process and close Qt Creator anyway? - Chcete zrušit proces sestavování a v každém případě zavřít Qt Creator? + Chcete zrušit sestavování a každopádně zavřít Qt Creator? No active project - Žádný činný projekt + Žádný projekt není činný The project '%1' has no active target - Projekt '%1' nemá žádný činný cíl + Projekt '%1' nemá žádný činný cíl The target '%1' for project '%2' has no active run configuration - Cíl '%1' projektu '%2' nemá žádné činné nastavení pro spuštění + Cíl '%1' projektu '%2' nemá žádné činné nastavení pro spuštění Cannot run '%1' in mode '%2'. - '%1' nelze spustit v režimu '%2'. + '%1' nelze spustit v režimu '%2'. A build is still in progress. - Nyní běží jedno sestavování. + Sestavování stále ještě probíhá. + + + Run %1 + Spustit %1 Adding Files to Project Failed @@ -11437,6 +12423,14 @@ Odstranit projekty ze sezení + Failed to open project + Nepodařilo se otevřít projekt + + + Session + Sezení + + Error while saving session Při ukládání sezení se vyskytla chyba @@ -11473,27 +12467,27 @@ Additional arguments: - Dodatečné argumenty: + Dodatečné argumenty: Effective qmake call: - Účinné vyvolání qmake: + Účinné vyvolání qmake: qmake build configuration: - Nastavení sestavování pro qmake: + Nastavení sestavování pro qmake: Debug - Ladění + Ladění Release - Vydání + Vydání Link QML debugging library: - Odkaz na knihovnu pro ladění QML: + Odkaz na knihovnu pro ladění QML: @@ -11701,16 +12695,16 @@ Creates a project containing a single main.cpp file with a stub implementation. Preselects a desktop Qt for building the application if available. - Vytvoří projekt, který sestává z jednoho souboru main.cpp s provedením trupu. + Vytvoří projekt, který sestává z jednoho souboru main.cpp s pahýlem provedení. -Vybere pro vývoj programu vhodnou verzi Qt, je-li dostupná. +Vybere pro sestavení programu verzi Qt pro stolní počítač, je-li dostupná. Qt4ProjectManager::Internal::ConsoleAppWizardDialog This wizard generates a Qt4 console application project. The application derives from QCoreApplication and does not provide a GUI. - Tento průvodce vytvoří projekt konzolové aplikace v Qt4.. Aplikace je odvozena z QCoreApplication a nemá žádné uživatelsk rozhraní. + Tento průvodce vytvoří projekt konzolové aplikace v Qt4. Aplikace je odvozena z QCoreApplication a nemá žádné uživatelské rozhraní. @@ -11754,7 +12748,7 @@ Qt4ProjectManager::Internal::EmptyProjectWizardDialog This wizard generates an empty Qt4 project. Add files to it later on by using the other wizards. - Tento průvodce vytvoří prázdný projekt Qt4. S pomocí dalších průvodců lze do něj později přidat další soubory. + Tento průvodce vytvoří prázdný projekt Qt4. S pomocí ostatních průvodců do něj lze později přidat další soubory. @@ -11776,7 +12770,7 @@ Specify basic information about the classes for which you want to generate skeleton source code files. - Zadejte základní informace ohledně tříd, pro které chcete vytvořit základní soubory se zdrojovým kódem. + Zadejte základní informace ohledně tříd, pro které chcete vytvořit soubory s kostrou zdrojového kódu. @@ -11825,7 +12819,7 @@ Creates a C++ library based on qmake. This can be used to create:<ul><li>a shared C++ library for use with <tt>QPluginLoader</tt> and runtime (Plugins)</li><li>a shared or static C++ library for use with another project at linktime</li></ul> - Vytvoří knihovnu C++ založenou na qmake. Tuto lze použít pro vytvoření:<ul><li>sdílené knihovny C++ pro užití s <tt>QPluginLoader</tt> a pro dobu běhu (přídavné moduly)</li><li>sdílenou nebo statickou knihovnu C++ pro použití s dalším projektem v čase spojení</li></ul> + Vytvoří knihovnu C++ založenou na qmake. Tuto lze použít pro vytvoření:<ul><li>sdílené knihovny C++ pro užití s <tt>QPluginLoader</tt> a běhového prostředí (přídavných modulů)</li><li>sdílenou nebo statickou knihovnu C++ pro použití s dalším projektem v čase zavedení</li></ul> Creates a C++ library based on qmake. This can be used to create:<ul><li>a shared C++ library for use with <tt>QPluginLoader</tt> and runtime (Plugins)</li><li>a shared or static C++ library for use with another project at linktime</li></ul>. @@ -11852,11 +12846,11 @@ Shared Library - Sdílená knihovna (dynamicky svázaná) + Sdílená knihovna Statically Linked Library - Staticky svázaná knihovna + Statická knihovna Qt 4 Plugin @@ -11942,7 +12936,7 @@ Add Scope - Přidat oblast + Přidat obor Add Block @@ -11953,7 +12947,7 @@ Qt4ProjectManager::Internal::ProEditorModel <Global Scope> - <Celková oblast> + <Celkový obor> Change Item @@ -12012,7 +13006,7 @@ Project Setup - Nastavení projektu + Nastavení projektu @@ -12164,11 +13158,11 @@ using Qt version: <b>%1</b><br>with tool chain <b>%2</b><br>building in <b>%3</b> - Používá se verze Qt: <b>%1</b><br>s řetězcem nástrojů <b>%2</b><br>sestavuje se v <b>%3</b> + Používá se verze Qt: <b>%1</b><br>se sadou nástrojů <b>%2</b><br>sestavuje se v <b>%3</b> <Invalid tool chain> - <Neplatný řetězec nástrojů> + <Neplatná sada nástrojů> General @@ -12208,7 +13202,7 @@ <No tool chain selected> - <b>Nevybrán žádný řetěz nástrojů</b> + <b>Nevybrána žádna sada nástrojů</b> Building in subdirectories of the source directory is not supported by qmake. @@ -12241,7 +13235,7 @@ Tool chain: - Řetězec nástrojů: + Sada nástrojů: Shadow build: @@ -12271,12 +13265,40 @@ Sestavit + Build "%1" + Sestavit "%1" + + Rebuild Sestavit znovu Clean - Uklidit + Pročistit + + + Build Subproject + Sestavit dílčí projekt + + + Build Subproject "%1" + Sestavit dílčí projekt "%1" + + + Rebuild Subproject + Sestavit dílčí projekt znovu + + + Rebuild Subproject "%1" + Sestavit dílčí projekt "%1" znovu + + + Clean Subproject + Pročistit dílčí projekt + + + Clean Subproject "%1" + Pročistit dílčí projekt "%1" Jump to File Under Cursor @@ -12326,9 +13348,13 @@ Prostředí pro sestavování - Qt4 RunConfiguration + Qt4 Run Configuration Nastavení spuštění Qt4 + + Qt4 RunConfiguration + Nastavení spuštění Qt4 + Qt4ProjectManager::Internal::Qt4RunConfigurationWidget @@ -12373,8 +13399,12 @@ Spustit v terminálu + Base environment for this run configuration: + Základní prostředí pro toto nastavení spuštění: + + Debugger: - Ladič: + Ladič: Run Environment @@ -12382,7 +13412,7 @@ Base environment for this runconfiguration: - Základní prostředí pro toto nastavení spuštění: + Základní prostředí pro toto nastavení spuštění: Clean Environment @@ -12441,7 +13471,7 @@ Select the CSL ARM Toolchain (GCCE) Directory - Vybrat adresář s řetězem nástrojů CSL ARM (GCCE) + Vybrat adresář se sadou nástrojů CSL ARM (GCCE) Auto-detected @@ -12462,7 +13492,7 @@ This Qt Version has a unknown toolchain. - Tato verze Qt nemá přiřazen žádný známý řetěz nástrojů. + Tato verze Qt nemá přiřazen žádnou známou sadu nástrojů. Desktop @@ -12606,7 +13636,7 @@ Toolchain: - Řetězec nástrojů: + Sada nástrojů: CSL/GCCE directory: @@ -12723,8 +13753,12 @@ Make + Qt Creator needs a build configuration set up to build. Configure a tool chain in Project mode. + Qt Creator potřebuje nastavení sestavování pro sestavení projektu. Nastavte, prosím, sadu nástrojů v projektovém režimu. + + Qt Creator needs a tool chain set up to build. Configure a tool chain in Project mode. - Qt Creator potřebuje řetěz nástrojů pro sestavení projektu. Nastavte, prosím, řetěz nástrojů v projektovém režimu. + Qt Creator potřebuje sadu nástrojů pro sestavení projektu. Nastavte, prosím, sadu nástrojů v projektovém režimu. Cannot find Makefile. Check your build settings. @@ -12750,6 +13784,14 @@ Přepsat %1: + Make: + Make: + + + No Qt4 build configuration. + Žádné nastavení sestavování Qt4. + + <b>Make:</b> %1 not found in the environment. <b>Krok Make:</b> %1 v prostředí nenalezen. @@ -12839,7 +13881,11 @@ Full path to the bin/ install directory of the current project's Qt version. - Úplná cesta k instalačnímu adresáři bin nynějším projektem používané verze Qt. + Úplná cesta k instalačnímu adresáři bin nynějším projektem používané verze Qt. + + + Full path to the bin directory of the current project's Qt version. + Úplná cesta k adresáři bin nynějším projektem používané verze Qt. Update of Generated Files @@ -12854,6 +13900,10 @@ Projekt %1 se nepodařil otevřít: Soubor s projektem neexistuje + QMake + QMake + + Failed opening project Projekt se nepodařil otevřít @@ -12917,7 +13967,37 @@ QtDumperHelper Found an outdated version of the debugging helper library (%1); version %2 is required. - Byla nalezena zastaralá verze (%1) pomocné knihovny pro výstup dat o ladění. Je požadována verze %2. + Byla nalezena zastaralá verze pomocného ladicího programu (%1). Je požadována verze %2. + + + ptrace: Operation not permitted. + +Could not attach to the process. Check the settings of +/proc/sys/kernel/yama/ptrace_scope +For more details, see/etc/sysctl.d/10-ptrace.conf + + ptrace: Operace nepovolena. + +Nepodařilo se připojit k procesu. Prověřte nastavení +/proc/sys/kernel/yama/ptrace_scope +Více podrobností hledejte v /etc/sysctl.d/10-ptrace.conf + + + + ptrace: Operation not permitted. + +Could not attach to the process. If your uid matches the uid +of the target process, check the settings of +/proc/sys/kernel/yama/ptrace_scope +For more details, see/etc/sysctl.d/10-ptrace.conf + + ptrace: Operace nepovolena. + +Nepodařilo se připojit k procesu. Pokud vaše UID odpovídá UID +cílového procesu, prověřte nastavení +/proc/sys/kernel/yama/ptrace_scope +Více podrobností hledejte v /etc/sysctl.d/10-ptrace.conf + <none> @@ -12948,7 +14028,7 @@ Graphical user interface components - Součástky názorného uživatelského rozhraní + Součástky grafického uživatelského rozhraní Classes for network programming @@ -13124,7 +14204,7 @@ Creates a Qt Resource file (.qrc) that you can add to a Qt Widget Project. - Vytvoří zdrojový soubor Qt (.qrc), který můžete přidat do projektu doplňku Qt (Qt Widget). + Vytvoří soubor s prostředky Qt (.qrc), který můžete přidat do projektu doplňku Qt (Qt Widget). &Undo @@ -13138,6 +14218,10 @@ ResourceEditor::Internal::ResourceEditorW + Open With + Otevřít s + + untitled bez názvu @@ -13146,15 +14230,15 @@ SaveItemsDialog Save Changes - Uložit změny + Uložit změny The following files have unsaved changes: - Následující soubory byly změněny: + Následující soubory byly změněny: Automatically save all files before building - Automaticky uložit všechny změněné soubory před sestavováním + Automaticky uložit všechny změněné soubory před sestavováním @@ -13216,7 +14300,7 @@ The file %1 is not in a subdirectory of the resource file. You now have the option to copy this file to a valid location. - Soubor %1 se nenachází v podadresáři zdrojového souboru. Nyní máte možnost zkopírovat tento soubor do platného umístění. + Soubor %1 se nenachází v podadresáři souboru s prostředky. Nyní máte možnost zkopírovat tento soubor do platného umístění. Choose copy location @@ -13243,27 +14327,27 @@ SharedTools::ResourceView Add Files... - Přidat soubory... + Přidat soubory... Change Alias... - Změnit přezdívku... + Změnit přezdívku... Add Prefix... - Přidat předponu... + Přidat předponu... Change Prefix... - Změnit předponu... + Změnit předponu... Change Language... - Změnit jazyk... + Změnit jazyk... Remove Item - Odstranit prvek + Odstranit prvek Open File @@ -13271,7 +14355,7 @@ Input prefix: - Předpona vstupu: + Předpona vstupu: Open file @@ -13283,7 +14367,7 @@ Change Prefix - Změnit předponu + Změnit předponu Input Prefix: @@ -13291,19 +14375,19 @@ Change Language - Změnit jazyk + Změnit jazyk Language: - Jazyk: + Jazyk: Change File Alias - Změnit soubor s přezdívkou + Změnit soubor s přezdívkou Alias: - Přezdívka: + Přezdívka: @@ -13361,7 +14445,7 @@ ShowBuildLog Debugging Helper Build Log - Zápis o vytvoření pomocné knihovny pro výstup dat o ladění + Zápis o vytvoření pomocné knihovny pro výstup dat o ladění @@ -13382,7 +14466,7 @@ StartExternalDialog Start Debugger - Spustit ladicí program + Spustit ladicí program Executable: @@ -13398,34 +14482,34 @@ &Executable: - &Spustitelný soubor: + &Spustitelný soubor: &Arguments: - &Argumenty: + &Argumenty: Run in &terminal: - Spustit v &terminálu: + Spustit v &terminálu: &Working directory: - &Pracovní adresář: + &Pracovní adresář: &Tool chain: - Ř&etězec nástrojů: + Sada &nástrojů: Break at '&main': - Bod přerušení při 'main': + Bod přerušení při 'main': StartRemoteDialog Start Debugger - Spustit ladicí program + Spustit ladicí program Architecture: @@ -13457,27 +14541,27 @@ &Debugger: - &Ladič: + &Ladič: Local &executable: - &Místní spustitelný soubor: + &Místní spustitelný soubor: &Host and port: - &Hostitelský počítač a číslo přípojky: + &Hostitelský počítač a číslo přípojky: &Architecture: - &Architektura: + &Architektura: &GNU target: - Cílová platforma &GNU: + Cílová platforma &GNU: Sys&root: - Sys&root: + Sys&root: Override s&tart script: @@ -13485,19 +14569,19 @@ &Use server start script: - &Použít spouštěcí skript k serveru: + &Použít spouštěcí skript k serveru: &Server start script: - Spouštěcí skript k &serveru: + Spouštěcí skript k &serveru: Location of debugging information: - Umístění informací o ladění: + Umístění informací o ladění: Override host GDB s&tart script: - Přepsat &spouštěcí skript GDB na hostiteli: + Přepsat &spouštěcí skript GDB na hostiteli: @@ -13552,7 +14636,7 @@ Prompt on submit - Potvrdit předložení + Ptát se na potvrzení Ignore whitespace changes in annotation @@ -13560,7 +14644,7 @@ Log count: - Počet zápisů omezit na: + Počet záznamů omezit na: @@ -13606,15 +14690,15 @@ Diff Project - Rozdíly (diff) pro projekt + Rozdíly pro projekt Diff Current File - Rozdíly (diff) nynějšího souboru + Rozdíly nynějšího souboru Diff "%1" - Rozdíly (diff) pro "%1" + Rozdíly pro "%1" Alt+S,Alt+D @@ -13622,15 +14706,15 @@ Commit All Files - Odeslat všechny soubory + Odevzdat (commit) všechny soubory Commit Current File - Odeslat nynější soubor + Odevzdat (commit) nynější soubor Commit "%1" - Odeslat "%1" + Odevzdat (commit) "%1" Alt+S,Alt+C @@ -13638,19 +14722,19 @@ Filelog Current File - Zápis k souboru pro nynější soubor + Záznamy k souboru pro nynější soubor Filelog "%1" - Zápis k souboru "%1" + Záznamy k souboru "%1" Annotate Current File - Opatřit nynější soubor vysvětlivkami + Opatřit nynější soubor anotacemi Annotate "%1" - Opatřit vysvětlivkami "%1" + Opatřit anotacemi "%1" Describe... @@ -13662,11 +14746,11 @@ Update Project - Obnovit projekt + Aktualizovat projekt Commit - Odeslat + Odevzdat (commit) Diff Selected Files @@ -13686,11 +14770,11 @@ Do you want to commit the change? - Chcete odeslat změnu? + Chcete odevzdat změnu? The commit message check failed. Do you want to commit the change? - Ověření popisu týkajícího se odeslání se nezdařilo. Přesto chcete odeslání změny provést? + Přezkoušení popisu změn se nezdařilo. Přesto chcete změnu odevzdat? The commit list spans several repositories (%1). Please commit them one by one. @@ -13724,7 +14808,7 @@ Diff Project "%1" - Rozdíly (diff) pro projekt "%1" + Rozdíly pro projekt "%1" Status of Project "%1" @@ -13732,27 +14816,27 @@ Log Project - Zápis pro projekt + Záznamy pro projekt Log Project "%1" - Zápis pro projekt "%1" + Záznamy pro projekt "%1" Update Project "%1" - Obnovit projekt "%1" + Aktualizovat projekt "%1" Commit Project - Odeslat projekt + Odevzdat (commit) projekt Commit Project "%1" - Odeslat projekt "%1" + Odevzdat (commit) projekt "%1" Diff Repository - Rozdíly (diff) skladiště + Rozdíly skladiště Repository Status @@ -13760,11 +14844,11 @@ Log Repository - Zápis (log) skladiště + Záznamy skladiště Update Repository - Obnovit skladiště + Aktualizovat skladiště Revert Repository... @@ -13772,7 +14856,7 @@ Diff &Selected Files - Rozdíly (diff) pro &vybrané soubory + Rozdíly pro &vybrané soubory Revert repository @@ -13780,7 +14864,7 @@ Revert all pending changes to the repository? - Chcete vrátit všechny čekající změny ve skladišti zpět? + Chcete vrátit zpět všechny nevyřízené změny ve skladišti? Would you like to revert all changes to the repository? @@ -13792,7 +14876,7 @@ Another commit is currently being executed. - V současnosti se již provádí jiné odeslání. + V současnosti se již provádí jiné odevzdání. There are no modified files. @@ -13808,7 +14892,7 @@ Revision number: - Číslo pozměnění: + Číslo revize: Executing in %1: %2 %3 @@ -13842,7 +14926,7 @@ Subversion::Internal::SubversionSubmitEditor Subversion Submit - Předložení Subversion + Odeslání Subversion (submit) @@ -13853,7 +14937,7 @@ %1 found - %1 nalezen + %1 nalezen List of comma separated wildcard filters @@ -13880,7 +14964,7 @@ Opening file - Otevření souboru + Otevírán soubor <em>Binary data</em> @@ -14049,43 +15133,43 @@ TextEditor::DisplaySettingsPage Display - Zobrazení + Zobrazení Display line &numbers - &Zobrazit čísla řádků + &Zobrazit čísla řádků Display &folding markers - Zobrazit znaky s&kládání kódu + Zobrazit znaky s&kládání kódu Show tabs and spaces. - Ukázat zarážky a prázdné znaky (mezery). + Ukázat zarážky a prázdné znaky (mezery). &Visualize whitespace - &Zviditelnit prázdné znaky + &Zviditelnit prázdné znaky Highlight current &line - Zvýraznit nynější řá&dek + Zvýraznit nynější řá&dek Text Wrapping - Zalomení textu + Zalomení textu Enable text &wrapping - Povolit &zalomení textu + Povolit &zalomení textu Display right &margin at column: - Zobrazit pravý &okraj sloupce: + Zobrazit pravý &okraj sloupce: Highlight &blocks - Zvýraznit &bloky + Zvýraznit &bloky Animate matching parentheses @@ -14105,19 +15189,19 @@ Mark &text changes - Vyznačit &textové změny + Vyznačit &textové změny &Animate matching parentheses - &Rozhýbat odpovídající závorky + &Rozhýbat odpovídající závorky Auto-fold first &comment - Automaticky složit první po&známku + Automaticky složit první po&známku Center &cursor on scroll - Při projíždění držet &ukazovátko vprostřed + Při projíždění držet &ukazovátko vprostřed @@ -14248,7 +15332,7 @@ Family: - &Písmová rodina + Rodina písma: Size: @@ -14256,7 +15340,7 @@ Color Scheme - Barevné schéma + Schéma barev Antialias @@ -14313,6 +15397,14 @@ Obecné + Creates a scratch buffer using a temporary file. + Vytvoří narychlo udělanou vyrovnávací paměť z dočasného souboru. + + + Scratch Buffer + Narychlo udělaná vyrovnávací paměť + + Triggers a completion in this scope Začne doplnění v této oblasti @@ -14381,7 +15473,7 @@ &Visualize Whitespace - &Zviditelnit prázdné znaky + Z&viditelnit prázdné znaky Clean Whitespace @@ -14393,7 +15485,7 @@ (Un)Comment &Selection - &Výběr opatřit poznámkou/Zrušit poznámku + Výběr opatřit &poznámkou/Zrušit poznámku Ctrl+/ @@ -14500,24 +15592,48 @@ Jít na konec bloku s výběrem - Ctrl+} - Ctrl+} + Delete Word from Cursor On + Smazat celé slovo po ukazovátku - Select Block Up - Vybrat jeden blok nahoru + Delete Word Camel Case from Cursor On + Smazat slovo po ukazovátku (Camel Case) - Ctrl+U - Ctrl+U + Delete Word up to Cursor + Smazat celé slovo před ukazovátkem - Select Block Down - Vybrat jeden blok dolů + Delete Word Camel Case up to Cursor + Smazat slovo před ukazovátkem (Camel Case) - Join Lines - Spojit řádky + Go to Block Start with Selection + Označit po začátek bloku + + + Go to Block End with Selection + Označit po konec bloku + + + Ctrl+} + Ctrl+} + + + Select Block Up + Vybrat jeden blok nahoru + + + Ctrl+U + Ctrl+U + + + Select Block Down + Vybrat jeden blok dolů + + + Join Lines + Spojit řádky Ctrl+J @@ -14525,7 +15641,7 @@ Insert Line Above Current Line - Vložit řádek nad nynějším řádkem + Vložit řádek nad nynějším Ctrl+Shift+Return @@ -14533,7 +15649,7 @@ Insert Line Below Current Line - Vložit řádek pod nynějším řádkem + Vložit řádek pod nynějším Ctrl+Return @@ -14556,86 +15672,190 @@ Alt+U - Goto Line Start + Paste from Clipboard History + Vložit z historie schránky + + + Ctrl+Shift+V + Ctrl+Shift+V + + + Indent + Odsazení + + + Unindent + Zrušit odsazení + + + Follow Symbol Under Cursor + Následovat symbol pod ukazovátkem + + + Jump To File Under Cursor + Jít na soubor pod ukazovátkem + + + Go to Line Start Jít na začátek řádku - Goto Line End + Go to Line End Jít na konec řádku - Goto Next Line + Go to Next Line Jít na další řádek - Goto Previous Line + Go to Previous Line Jít na předchozí řádek - Goto Previous Character + Go to Previous Character Jít na předchozí znak - Goto Next Character + Go to Next Character Jít na další znak - Goto Previous Word + Go to Previous Word Jít na předchozí slovo - Goto Next Word + Go to Next Word Jít na další slovo - Goto Previous Word Camel Case + Go to Previous Word Camel Case Jít na předchozí slovo (Camel Case) - Goto Next Word Camel Case + Go to Next Word Camel Case Jít na další slovo (Camel Case) - Goto Line Start With Selection + Go to Line Start with Selection Označit až po začátek řádku - Goto Line End With Selection + Go to Line End with Selection Označit až po konec řádku - Goto Next Line With Selection + Go to Next Line with Selection Označit až po další řádek - Goto Previous Line With Selection + Go to Previous Line with Selection Označit až po předchozí řádek - Goto Previous Character With Selection + Go to Previous Character with Selection Označit předchozí znak - Goto Next Character With Selection + Go to Next Character with Selection Označit další znak - Goto Previous Word With Selection + Go to Previous Word with Selection Označit předchozí slovo - Goto Next Word With Selection + Go to Next Word with Selection Označit další slovo - Goto Previous Word Camel Case With Selection + Go to Previous Word Camel Case with Selection Označit předchozí slovo (Camel Case) - Goto Next Word Camel Case With Selection + Go to Next Word Camel Case with Selection Označit další slovo (Camel Case) + Goto Line Start + Jít na začátek řádku + + + Goto Line End + Jít na konec řádku + + + Goto Next Line + Jít na další řádek + + + Goto Previous Line + Jít na předchozí řádek + + + Goto Previous Character + Jít na předchozí znak + + + Goto Next Character + Jít na další znak + + + Goto Previous Word + Jít na předchozí slovo + + + Goto Next Word + Jít na další slovo + + + Goto Previous Word Camel Case + Jít na předchozí slovo (Camel Case) + + + Goto Next Word Camel Case + Jít na další slovo (Camel Case) + + + Goto Line Start With Selection + Označit až po začátek řádku + + + Goto Line End With Selection + Označit až po konec řádku + + + Goto Next Line With Selection + Označit až po další řádek + + + Goto Previous Line With Selection + Označit až po předchozí řádek + + + Goto Previous Character With Selection + Označit předchozí znak + + + Goto Next Character With Selection + Označit další znak + + + Goto Previous Word With Selection + Označit předchozí slovo + + + Goto Next Word With Selection + Označit další slovo + + + Goto Previous Word Camel Case With Selection + Označit předchozí slovo (Camel Case) + + + Goto Next Word Camel Case With Selection + Označit další slovo (Camel Case) + + <line number> <číslo řádku> @@ -14657,19 +15877,19 @@ Delete Word From The Cursor On - Smazat celé slovo po ukazovátku + Smazat celé slovo po ukazovátku Delete Word Camel Case From The Cursor On - Smazat slovo po ukazovátku (Camel Case) + Smazat slovo po ukazovátku (Camel Case) Delete Word Up To The Cursor - Smazat celé slovo před ukazovátkem + Smazat celé slovo před ukazovátkem Delete Word Camel Case Up To The Cursor - Smazat slovo před ukazovátkem (Camel Case) + Smazat slovo před ukazovátkem (Camel Case) Fold @@ -14701,11 +15921,11 @@ Go to Block Start With Selection - Označit po začátek bloku + Označit po začátek bloku Go to Block End With Selection - Označit po konec bloku + Označit po konec bloku Ctrl+Shift+Up @@ -14816,7 +16036,7 @@ QML Binding - Svázání QML + Vázání QML QML Local Id @@ -14824,15 +16044,15 @@ QML Root Object Property - Vlastnost kořenového prvku QML + Vlastnost kořenového objektu QML QML Scope Object Property - Vlastnost objektu QML v oblasti + Vlastnost oborového objektu QML QML State Name - Název stavu v QML + Název stavu QML QML Type Name @@ -14848,7 +16068,7 @@ JavaScript Scope Var - Proměnná JavaScriptu v oblasti + Proměnná oboru JavaScriptu JavaScript Import @@ -14856,7 +16076,7 @@ JavaScript Global Variable - Celková proměnná JavaScriptu + Globální proměnná JavaScriptu Keyword @@ -14888,7 +16108,7 @@ Visual Whitespace - Zviditelnit prázdné znaky + Viditelné prázdné znaky Disabled Code @@ -14904,11 +16124,11 @@ Diff File - Rozdíly (diff): údaj o souboru + Porovnávaný soubor Diff Location - Rozdíly (diff): údaj o umístění + Porovnávané umístění Text Editor @@ -14957,12 +16177,16 @@ Správa verzí + General + Obecné + + Common Obecné Project from Version Control - Projekt ze systému na správu verzí + Projekt ze systému na správu verzí @@ -14987,6 +16211,10 @@ Cannot open '%1': %2 Nelze otevřít '%1': %2 + + Nicknames + Přezdívky + VcsBase::SubmitFileModel @@ -15026,7 +16254,7 @@ Prompt to submit - Potvrdit předložení + Nadále vyžadovat potvrzení Submit Message Check failed @@ -15042,7 +16270,7 @@ Submit Message Check Failed - Ověření zprávy o předložení se nezdařilo + Ověření popisu revize se nezdařilo Executing %1 @@ -15054,7 +16282,7 @@ The check script '%1' crashed. - Skript '%1' k ověření zprávy spadl. + Skript '%1' k ověření popisu spadl. Unable to open '%1': %2 @@ -15135,35 +16363,35 @@ ViewDialog Send to Codepaster - Poslat CodePaster + Poslat CodePaster &Username: - &Uživatelské jméno: + &Uživatelské jméno: <Username> - <Uživatelské jméno> + <Uživatelské jméno> &Description: - &Popis: + &Popis: <Description> - <Popis> + <Popis> Patch 1 - Opravný soubor 1 + Opravný soubor 1 Patch 2 - Opravný soubor 2 + Opravný soubor 2 Protocol: - Protokol: + Protokol: <!DOCTYPE HTML PUBLIC "-//W3C//DTD HTML 4.0//EN" "http://www.w3.org/TR/REC-html40/strict.dtd"> @@ -15187,7 +16415,7 @@ p, li { white-space: pre-wrap; } </style></head><body style=" font-family:'DejaVu Sans'; font-size:10pt; font-weight:400; font-style:normal;"> <p style=" margin-top:0px; margin-bottom:0px; margin-left:0px; margin-right:0px; -qt-block-indent:0; text-indent:0px;"><span style=" font-family:'Sans Serif'; font-size:9pt;">&lt;Comment&gt;</span></p></body></html> - <!DOCTYPE HTML PUBLIC "-//W3C//DTD HTML 4.0//EN" "http://www.w3.org/TR/REC-html40/strict.dtd"> + <!DOCTYPE HTML PUBLIC "-//W3C//DTD HTML 4.0//EN" "http://www.w3.org/TR/REC-html40/strict.dtd"> <html><head><meta name="qrichtext" content="1" /><style type="text/css"> p, li { white-space: pre-wrap; } </style></head><body style=" font-family:'DejaVu Sans'; font-size:10pt; font-weight:400; font-style:normal;"> @@ -15195,7 +16423,7 @@ Parts to Send to Server - Části pro poslání serveru + Části pro poslání serveru @@ -15225,7 +16453,7 @@ Utils::CheckableMessageBox Dialog - Dialog + Dialog TextLabel @@ -15235,6 +16463,10 @@ CheckBox Zaškrtávací okénko + + Do not ask again + Neptat se znovu + Utils::WizardPage @@ -15392,12 +16624,16 @@ Introduction and Project Location Uvedení a umístění projektu + + Project: + Projekt: + Utils::SubmitEditorWidget Subversion Submit - Předložení Subversion + Odeslání Subversion Des&cription @@ -15409,11 +16645,11 @@ Descriptio&n - &Popis: + &Popis Check &all - Označit &vše + Označit &vše %1 %2/%n File(s) @@ -15425,7 +16661,7 @@ &Commit - &Odeslat + &Zapsat či Odevzdat (commit) Check All @@ -15437,12 +16673,16 @@ Uncheck all for submit Odstranit označení u všeho + + Check a&ll + Označit &vše + PasteBinComSettingsWidget Form - Formulář + Formulář Server Prefix: @@ -15464,7 +16704,7 @@ Server prefix: - Předpona serveru: + Předpona serveru: <html><head/><body> @@ -15476,11 +16716,11 @@ <a href="http://pastebin.com">pastebin.com</a> allows for sending posts to custom subdomains (eg. creator.pastebin.com). Fill in the desired prefix. - <a href="http://pastebin.com">pastebin.com</a> dovoluje posílání uživatelsky stanovených subdomén (například creator.pastebin.com). Zadejte požadovanou předponu. + <a href="http://pastebin.com">pastebin.com</a> dovoluje posílání uživatelsky stanovených subdomén (například creator.pastebin.com). Zadejte požadovanou předponu. <i>Note: The plugin will use this for posting as well as fetching.</i> - <i>Poznámka: Přídavný modul toto použije pro posílání a stejně tak natahování.</i> + <i>Poznámka: Přídavný modul toto použije pro posílání a stejně tak natahování.</i> @@ -15519,7 +16759,7 @@ CVS root: - Zdroj CVS (CVSROOT): + Zdroj CVS (CVS ROOT): Miscellaneous @@ -15527,19 +16767,19 @@ Diff options: - Volby pro rozdíly (diff): + Volby pro rozdíly: Prompt on submit - Potvrdit předložení + Ptát se na potvrzení When checked, all files touched by a commit will be displayed when clicking on a revision number in the annotation view (retrieved via commit ID). Otherwise, only the respective file will be displayed. - Když je zapnuta tato volba, ukáží se při klepnutí na číslo revize v pohledu s vysvětlivkami všechny soubory přiložené k odeslání (získané využitím ID odeslání). Jinak se zobrazí pouze příslušný soubor. + Když je zapnuta tato volba, ukáží se při klepnutí na číslo revize v pohledu s anotacemi všechny soubory změněné při odevzdání (získané pomocí ID). Jinak se zobrazí pouze příslušný soubor. Describe all files matching commit id - Popsat všechny soubory patřící k ID odeslání + Popsat všechny soubory patřící k ID odevzdání Timeout: @@ -15593,15 +16833,15 @@ Embedding of the UI Class - Použití třídy UI + Vložení třídy UI Aggregation as a pointer member - Nakupení jako ukazovátko + Nakupení jako ukazatel Aggregation - Nakupení + Nakupení (aggregation) Multiple Inheritance @@ -15609,7 +16849,7 @@ Code Generation - Doplnění kódu + Vytváření kódu Support for changing languages at runtime @@ -15617,12 +16857,16 @@ Use Qt module name in #include-directive - Používat název modulu Qt v #include-directive + Používat název modulu Qt v direktivě #include Multiple inheritance Několikanásobná dědičnost + + Add Qt version #ifdef for module names + Přidat #ifdef direktivy závislé na verzi Qt pro názvy modulů + Gitorious::Internal::GitoriousHostWidget @@ -15733,31 +16977,31 @@ GeneralSettingsPage Form - Formulář + Formulář Font - Písmo + Písmo Family: - Písmová rodina: + Písmová rodina: Style: - Styl: + Styl: Size: - Velikost: + Velikost: Startup - Spuštění + Spuštění On context help: - Související nápověda: + Související nápověda: Show side-by-side if possible @@ -15773,7 +17017,7 @@ On help start: - Na začátek nápovědy: + Na začátek nápovědy: Show my home page @@ -15793,11 +17037,11 @@ Use &Current Page - Použít &nynější stranu + Použít &nynější stranu Use &Blank Page - Použít &prázdnou stranu + Použít &prázdnou stranu Restore to Default @@ -15805,67 +17049,67 @@ Help Bookmarks - Záložky v nápovědě + Záložky v nápovědě Import... - Zavést... + Zavést... Export... - Vyvést... + Vyvést... Show Side-by-Side if Possible - Ukázat, je-li to možné, vedle sebe + Ukázat, je-li to možné, vedle sebe Always Show Side-by-Side - Ukázat vždy vedle sebe + Ukázat vždy vedle sebe Always Start Full Help - Vždy spustit plnou nápovědu + Vždy spustit plnou nápovědu Show My Home Page - Ukázat moji domovskou stránku + Ukázat moji domovskou stránku Show a Blank Page - Ukázat prázdnou stránku + Ukázat prázdnou stránku Show My Tabs from Last Session - Ukázat mé karty z posledního sezení + Ukázat mé karty z posledního sezení Home page: - Domovská stránka: + Domovská stránka: Always Show Help in External Window - Vždy ukazovat nápovědu v odděleném okně + Vždy ukazovat nápovědu v odděleném okně Reset to default - Nastavit znovu výchozí + Nastavit znovu výchozí Reset - Nastavit znovu + Nastavit znovu Behaviour - Chování + Chování Switch to editor context after last help page is closed. - Přepnout do editoru po zavření poslední stránky s nápovědou. + Přepnout do editoru po zavření poslední stránky s nápovědou. Return to editor on closing the last page - Vrátit se do editoru po zavření poslední stránky s nápovědou + Vrátit se do editoru po zavření poslední stránky s nápovědou @@ -15890,7 +17134,7 @@ Specify a short word/abbreviation that can be used to restrict completions to files from this directory tree. To do this, you type this shortcut and a space in the Locator entry field, and then the word to search for. Zadejte krátké slovo nebo zkratku, které omezí nálezy na soubory nálezající se v tomto adresářovém stromu. -Abyste to provedl, napište tuto zkraku v zadávacím poli vyhledávače, následně znak mezery a hledaný pojem. +Toho se dosáhne vložením této zkratky v zadávacím poli vyhledávače, následované mezerou a hledaným výrazem. Limit to prefix @@ -15947,6 +17191,10 @@ Filter: Filtr: + + Add Filter Configuration + Přidat nastavení filtru + Locator::Internal::SettingsWidget @@ -15976,7 +17224,7 @@ Refresh interval: - Mezera mezi obnovami: + Doba mezi obnovami: @@ -16035,11 +17283,11 @@ Clear old application output on a new run - Smazat při novém spuštění výstup předchozího spuštění programu + Smazat při novém spuštění výstupy předchozího běhu <i>jom</i> is a drop-in replacement for <i>nmake</i> which distributes the compilation process to multiple CPU cores. The latest binary is available at <a href="ftp://ftp.qt.nokia.com/jom/">ftp://ftp.qt.nokia.com/jom/</a>. Disable it if you experience problems with your builds. - <i>jom</i> je zaskakující náhradou za <i>nmake</i>, která proces sestavování rozděluje mezi více jader CPU. Nejnovější binární soubor je dostupný na <a href="ftp://ftp.qt.nokia.com/jom/">ftp://ftp.qt.nokia.com/jom/</a>. Zakažte jej, pokud při vytváření svých programů narazíte na potíže. + <i>jom</i> je náhražka za <i>nmake</i>, která proces sestavování rozděluje mezi více jader CPU. Nejnovější binární soubor je dostupný na <a href="ftp://ftp.qt.nokia.com/jom/">ftp://ftp.qt.nokia.com/jom/</a>. Zakažte jej narazíte-li při vytváření svých programů na potíže. Always build project before deploying it @@ -16059,11 +17307,11 @@ Enabling this option ensures that the order of interleaved messages from stdout and stderr is preserved, at the cost of disabling highlighting of stderr. - Toto nastavení způsobuje, že pořadí členitých zpráv na běžném a chybovém výstupu je zachováno; ovšem chybový výstup už pak není zvýrazněn. + Toto nastavení zajistí, že vzájemné pořadí zpráv na běžném a chybovém výstupu je zachováno; ovšem chybový výstup už pak není zvýrazněn. Merge stderr and stdout - Smíchat běžný a chybový výstup + Sloučit běžný a chybový výstup Word-wrap application output @@ -16075,16 +17323,20 @@ lines - Řádky + řádků Ask before terminating the running application in response to clicking the stop button in Application Output. - Zeptat se před ukončením běžícího programu v odpovědi na klepnutí na zastavovací tlačítko ve výstupním panelu programu. + Po klepnutí na zastavovací tlačítko ve výstupním panelu běžícího programu se nejprve zeptat, a teprve pak ukončit běh. Always ask before stopping applications Vždy se zeptat před ukončením programu + + Open application output pane when debugging + Při ladění programu ukázat výstupní panel + ProjectExplorer::Internal::ProjectWelcomePageWidget @@ -16145,7 +17397,7 @@ ProjectWelcomePage Form - Formulář + Formulář @@ -16216,7 +17468,7 @@ Include pro&ject - Zahrnout projekt (soubor *.pri) + Zahrnout pro&jekt (soubor *.pri) &Description @@ -16228,7 +17480,7 @@ &Tooltip: - &Rada k nástroji: + Vysvě&tlivka: W&hat's this: @@ -16271,7 +17523,7 @@ Collection class: - Třída sbírka: + Třída sbírky: Collection header file: @@ -16287,7 +17539,7 @@ Resource file: - Zdrojový soubor: + Soubor s prostředky: icons.qrc @@ -16298,11 +17550,11 @@ Qt4ProjectManager::Internal::CustomWidgetWidgetsWizardPage Custom Qt Widget Wizard - Průvodce pro vytvoření uživatelsky stanoveného prvku Qt + Průvodce pro vytvoření vlastního prvku Qt Custom Widget List - Seznam uživatelsky stanovených prvků + Seznam vlastních prvků @@ -16311,7 +17563,7 @@ Specify the list of custom widgets and their properties. - Zadejte seznam uživatelsky stanovených prvků a jejich vlastnosti. + Zadejte seznam vlastních prvků a jejich vlastností. ... @@ -16342,7 +17594,7 @@ Did You Know? - Víte, že? + Víte že ...? <b>Qt Creator - A quick tour</b> @@ -16587,7 +17839,7 @@ Foreground: - Popředí: + Písmo: Erase background @@ -16599,14 +17851,14 @@ Erase foreground - Smazat popředí + Smazat písmo VcsBase::BaseCheckoutWizardPage WizardPage - WizardPage + WizardPage Checkout Directory: @@ -16618,47 +17870,47 @@ Repository - Skladiště + Skladiště The remote repository to check out. - Vzdálené skladiště ke stažení. + Vzdálené skladiště ke stažení. Branch: - Větev: + Větev: The development branch in the remote repository to check out. - Vývojářská větev ve vzdáleném skladišti ke stažení. + Vývojářská větev ve vzdáleném skladišti ke stažení. Retrieve list of branches in repository. - Získat seznam větví ve skladišti. + Získat seznam větví ve skladišti. ... - ... + ... Working Copy - Pracovní kopie + Pracovní kopie The path in which the directory containing the checkout will be created. - Cesta, ve které bude vytvořen adresář obsahující stažené. + Cesta, ve které bude vytvořen adresář obsahující stažené. Checkout path: - Cesta ke staženému: + Cesta ke staženému: The local directory that will contain the code after the checkout. - Místní adresář, který bude po stažení obsahovat kód. + Místní adresář, který bude po stažení obsahovat kód. Checkout directory: - Adresář se stažením: + Adresář se stažením: @@ -16784,7 +18036,7 @@ Utils::ClassNameValidatingLineEdit The class name must not contain namespace delimiters. - Název třídy nesmí obsahovat omezení jmenného prostoru. + Název třídy nesmí obsahovat znaky pro oddělení jmenného prostoru. Please enter a class name. @@ -16803,7 +18055,7 @@ Press <RETURN> to close this window... - Kvůli zavření tohoto okna stiskněte tlačítko <RETURN>... + Zavřete toto okno stisknutím tlačítko <RETURN>... Cannot create temporary file: %1 @@ -16811,7 +18063,7 @@ Cannot write temporary file. Disk full? - Dočasný soubor se nepodařilo zapsat. Možná že už na pevném disku není místo pro ukládání? + Dočasný soubor se nepodařilo zapsat. Možná je už pevný disk plný? Cannot create temporary directory '%1': %2 @@ -16827,7 +18079,7 @@ Cannot change to working directory '%1': %2 - Nepodařilo se provést změnu na pracovní adresář '%1': %2 + Nepodařilo se přepnout na pracovní adresář '%1': %2 Cannot execute '%1': %2 @@ -16835,15 +18087,15 @@ Quoting error in command. - Chyba v uvozovkách v příkazu. + V příkazu je chyba v uvozovkách. Debugging complex shell commands in a terminal is currently not supported. - Ladění složitých příkazů v shellu v jednom terminálu současně nyní není podporováno. + Ladění složitých příkazů shellu v terminálu zatím není podporováno. Quoting error in terminal command. - Chyba v uvozovkách v příkazu v terminálu. + V příkazu terminálu je chyba v uvozovkách. Terminal command may not be a shell command. @@ -16851,11 +18103,11 @@ Cannot start the terminal emulator '%1'. - Nepodařilo se spustit program pro napodobení jinak též emulaci terminálu '%1'. + Nepodařilo se spustit program pro emulaci terminálu '%1'. Cannot create socket '%1': %2 - Nepodařilo se vytvořit zásuvku '%1': %2 + Nepodařilo se vytvořit zásuvku (socket) '%1': %2 The process '%1' could not be started: %2 @@ -16863,11 +18115,11 @@ Cannot obtain a handle to the inferior: %1 - Proces k odladění se nepodařilo rozpoznat: %1 + Nepodařilo se získat úchyt k podřízenému: %1 Cannot obtain exit status from inferior: %1 - U procesu k odladění se nepodařilo obdržet zpětnou hodnotu: %1 + Nepodařilo se získat vrácenou hodnotu z podřízeného: %1 @@ -17126,7 +18378,7 @@ Binary Editor - Dvojkový editor + Editor binárních souborů C++ Editor @@ -17142,7 +18394,7 @@ QMLJS Editor - Editor Editor + Editor QMLJS .qmlproject Editor @@ -17158,7 +18410,7 @@ Resource Editor - Editor zdrojových souborů + Editor prostředků GLSL Editor @@ -17203,7 +18455,7 @@ <i>Note: Specify the host name for the CodePaster service without any protocol prepended (e.g. codepaster.mycompany.com).</i> - Poznámka: Zadejte název hostitelského počítače (serveru) pro službu CodePaster bez protokolové předpony (například: codepaster.mycompany.com). + <i>Poznámka: Zadejte název hostitelského počítače (serveru) pro službu CodePaster bez protokolové předpony (například: codepaster.mycompany.com).</i> Code Pasting @@ -17277,10 +18529,10 @@ **************************************************************************/ /************************************************************************** -** Hlavičková předloha povolení pro Qt Creator +** Licenční hlavičková předloha pro Qt Creator ** Zvláštní klíčová slova: %USER% %DATE% %YEAR% ** Proměnné prostředí: %$VARIABLE% -** Kvůli ochránění symbolu procenta používejte '%%'. +** Znak procenta ochráníte použitím '%%'. **************************************************************************/ @@ -17294,7 +18546,7 @@ Choose Location for New License Template File - Vybrat umístění pro nový soubor s předlohou povolení + Vybrat umístění pro nový soubor s licenční předlohou Choose a location for the new license template file @@ -17325,7 +18577,7 @@ C++ Macro Usages: - Použití makro C++: + Použití makra C++: @@ -17343,11 +18595,11 @@ Checks out a CVS repository and tries to load the contained project. - Stáhne skladiště CVS (checkout) a zkusí nahrát obsažený projekt. + Získá skladiště CVS (checkout) a zkusí nahrát obsažený projekt. CVS Checkout - Stažení (checkout) skladiště CVS + Získání (checkout) skladiště CVS @@ -17376,7 +18628,7 @@ Cvs::Internal::CvsPlugin Parsing of the log output failed - Nepodařilo se vyhodnotit výstup zapisu + Nepodařilo se vyhodnotit výstup záznamu &CVS @@ -17412,15 +18664,15 @@ Diff Project - Rozdíly (diff) pro projekt + Rozdíly pro projekt Diff Current File - Rozdíly (diff) nynějšího souboru + Rozdíly nynějšího souboru Diff "%1" - Rozdíly (diff) pro "%1" + Rozdíly pro "%1" Alt+C,Alt+D @@ -17428,15 +18680,15 @@ Commit All Files - Odeslat všechny soubory + Odevzdat (commit) všechny soubory Commit Current File - Odeslat nynější soubor + Odevzdat (commit) nynější soubor Commit "%1" - Odeslat "%1" + Odevzdat (commit) "%1" Alt+C,Alt+C @@ -17444,7 +18696,7 @@ Filelog Current File - Zápis k souboru pro nynější soubor + Záznamy k souboru pro nynější soubor Cannot find repository for '%1' @@ -17452,15 +18704,15 @@ Filelog "%1" - Zápis k souboru "%1" + Záznamy k souboru "%1" Annotate Current File - Opatřit nynější soubor vysvětlivkami + Opatřit nynější soubor anotacemi Annotate "%1" - Opatřit vysvětlivkami "%1" + Opatřit anotacemi "%1" Delete... @@ -17500,7 +18752,7 @@ Diff Project "%1" - Rozdíly (diff) pro projekt "%1" + Rozdíly pro projekt "%1" Project Status @@ -17512,31 +18764,31 @@ Log Project - Zápis pro projekt + Záznamy pro projekt Log Project "%1" - Zápis pro projekt "%1" + Záznamy pro projekt "%1" Update Project - Obnovit projekt + Aktualizovat projekt Update Project "%1" - Obnovit projekt "%1" + Aktualizovat projekt "%1" Commit Project - Odeslat projekt + Odevzdat (commit) projekt Commit Project "%1" - Odeslat projekt "%1" + Odevzdat (commit) projekt "%1" Diff Repository - Rozdíly (diff) skladiště + Rozdíly skladiště Repository Status @@ -17544,11 +18796,11 @@ Repository Log - Zápis (log) skladiště + Záznamy skladiště Update Repository - Obnovit skladiště + Aktualizovat skladiště Revert Repository... @@ -17556,23 +18808,23 @@ Commit - Odeslat + Odevzdat (commit) Diff &Selected Files - Rozdíly (diff) pro &vybrané soubory + Rozdíly pro &vybrané soubory Revert all pending changes to the repository? - Chcete vrátit všechny čekající změny ve skladišti zpět? + Chcete vrátit zpět všechny nevyřízené změny ve skladišti? Would you like to discard your changes to the repository '%1'? - Chcete vrátit všechny své změny ve skladišti '%1'? + Chcete zahodit všechny své změny ve skladišti '%1'? Would you like to discard your changes to the file '%1'? - Chcete vrátit všechny své změny v souboru '%1'? + Chcete zahodit všechny své změny v souboru '%1'? Diff Selected Files @@ -17592,11 +18844,11 @@ Do you want to commit the change? - Chcete odeslat změnu? + Chcete odevzdat změnu? The commit message check failed. Do you want to commit the change? - Ověření popisu týkajícího se odeslání se nezdařilo. Přesto chcete odeslání změn provést? + Přezkoušení popisu změn se nezdařilo. Přesto chcete změnu odevzdat? The files do not differ. @@ -17612,7 +18864,7 @@ Revert failed: %1 - Vzetí změn zpět se nezdařilo: %1 + Vrácení změn zpět se nezdařilo: %1 The file '%1' could not be deleted. @@ -17628,7 +18880,7 @@ Another commit is currently being executed. - V současnosti se již provádí jiné odeslání. + V současnosti se již provádí jiné odevzdání. There are no modified files. @@ -17648,11 +18900,11 @@ The initial revision %1 cannot be described. - První verzi (%1) nelze popsat. + Prvotní verzi (%1) nelze popsat. Could not find commits of id '%1' on %2. - Nepodařilo se najít odeslání s ID '%1' a s datem %2. + Nepodařilo se najít odevzdání s ID '%1' a s datem %2. Executing: %1 %2 @@ -17681,7 +18933,7 @@ Could not start cvs '%1'. Please check your settings in the preferences. - Příkaz 'cvs' '%1' se nepodařilo spustit. Ověřte, prosím, svá nastavení v nastaveních. + Příkaz 'cvs' '%1' se nepodařilo spustit. Ověřte, prosím, své volby v nastaveních. CVS did not respond within timeout limit (%1 ms). @@ -18060,7 +19312,7 @@ Attach to core "%1" failed: - Ladění souboru 'core' "%1" se nezdařilo. + Ladění souboru 'core' "%1" se nezdařilo: @@ -18323,43 +19575,47 @@ Raw pointer - Hodnota kurzoru + Hodnota kurzoru Latin1 string - Řetězec Latin1 + Řetězec Latin1 UTF8 string - Řetězec UTF8 + Řetězec UTF8 Local 8bit string - Řetězec znaků v místním 8bitovém zobrazení + Řetězec znaků v místním 8bitovém zobrazení UTF16 string - Řetězec UTF16 + Řetězec UTF16 UCS4 string - Řetězec UCS4 + Řetězec UCS4 Decimal - Desítkový + Desítkový Hexadecimal - Šestnáctkový + Šestnáctkový Binary - Dvojkový + Dvojkový Octal - Osmičkový + Osmičkový + + + returned value + Návratová hodnota Name @@ -18431,11 +19687,11 @@ Specify repository URL, checkout directory and path. - Zadejte adresu skladiště (URL), adresář pro stažení (checkout) a cestu. + Zadejte adresu skladiště (URL), cestu a adresář pro umístění pracovní kopie. Clone URL: - Adresa (URL) klonu: + Klonovat z adresy (URL): Delete master branch @@ -18443,7 +19699,7 @@ Delete the master branch after checking out the repository. - Způsobuje, že hlavní větev je po stažení skladiště smazána. + Způsobuje, že hlavní větev je po načtení skladiště smazána. @@ -18513,6 +19769,10 @@ Obecná nastavení + General + Obecné + + Import Bookmarks Zavést záložky @@ -18536,6 +19796,110 @@ Save File Uložit soubor + + Form + Formulář + + + Font + Písmo + + + Family: + Rodina písma: + + + Style: + Styl: + + + Size: + Velikost: + + + Startup + Spuštění + + + On context help: + Související nápověda: + + + Show Side-by-Side if Possible + Ukázat, je-li to možné, vedle sebe + + + Always Show Side-by-Side + Ukázat vždy vedle sebe + + + Always Start Full Help + Vždy spustit plnou nápovědu + + + Always Show Help in External Window + Vždy ukazovat nápovědu v odděleném okně + + + On help start: + Na začátek nápovědy: + + + Show My Home Page + Ukázat moji domovskou stránku + + + Show a Blank Page + Ukázat prázdnou stránku + + + Show My Tabs from Last Session + Ukázat mé karty z posledního sezení + + + Home page: + Domovská stránka: + + + Use &Current Page + Použít &nynější stranu + + + Use &Blank Page + Použít &prázdnou stranu + + + Reset to default + Nastavit znovu výchozí + + + Reset + Nastavit znovu + + + Help Bookmarks + Záložky v nápovědě + + + Import... + Zavést... + + + Export... + Vyvést... + + + Behaviour + Chování + + + Switch to editor context after last help page is closed. + Přepnout do editoru po zavření poslední stránky s nápovědou. + + + Return to editor on closing the last page + Vrátit se do editoru po zavření poslední stránky s nápovědou + Help::Internal::XbelReader @@ -18574,8 +19938,8 @@ %1 filter update: %n files %1 stav filtru: jeden soubor - - + %1 stav filtru: %n soubory + %1 stav filtru: %n souborů @@ -18619,6 +19983,10 @@ Locator::Internal::LocatorPlugin + Ctrl+K + Ctrl+K + + Type to locate Vzor hledání @@ -18806,7 +20174,7 @@ %1 (disabled) %1 is the custom process step summary - %1 (vypnuto) + %1 (vypnuto) @@ -19254,8 +20622,12 @@ <b>qmake:</b> %1 %2 + <b>Warning:</b> The tool chain suggests using another mkspec. + <b>Varování:</b> Řetězec znaků udává použití jiného mkspec. + + <b>Warning:</b> The tool chain suggested "%1" as mkspec. - <b>Varování:</b> Řetězec znaků udává "%1" jako mkspec. + <b>Varování:</b> Řetězec znaků udává "%1" jako mkspec. Enable QML debugging: @@ -19515,7 +20887,7 @@ Debugger: - Ladič: + Ladič: Installation file: @@ -19774,7 +21146,7 @@ Subversion Checkout - Stáhnout (checkout) kopii Subversion + Projekt ze skladiště Subversion (checkout) @@ -19800,7 +21172,7 @@ TextEditor::Internal::ColorScheme Not a color scheme file. - Není souborem znázornění barev: + Není souborem znázornění barev. @@ -19864,7 +21236,11 @@ VcsBase::Internal::CheckoutProgressWizardPage Checkout - Stáhnout (checkout) kopii + Příkaz checkout + + + No job running, please abort. + Žádná běžící práce, prosím, zrušit. Checkout started... @@ -20065,7 +21441,7 @@ CVS submit template - Předloha předložení CVS + Předloha odeslání (submit) CVS Qt Designer file @@ -20089,7 +21465,7 @@ Perforce submit template - Předloha předložení Perforce + Předloha odeslání (submit) Perforce QML file @@ -20116,6 +21492,10 @@ Soubor se skriptem Qt + Automake based Makefile + Automake založený Makefile + + GLSL Shader file Soubor GLSL Shader @@ -20192,6 +21572,10 @@ Soubor s obrázkem XPM + JSON file + Soubor JSON + + QML Project file Projektový soubor QML @@ -20201,11 +21585,11 @@ Qt Resource file - Zdrojový soubor Qt + Soubor s prostředky Qt Subversion submit template - Předloha předložení Subversion + Předloha odeslání (submit) Subversion Qt Creator task list file @@ -20236,19 +21620,19 @@ CommandMappings Command Mappings - Přiřazení příkazů + Přiřazení příkazů Command - Příkaz + Příkaz Label - Štítek + Štítek Target - Cíl + Cíl Defaults @@ -20256,65 +21640,65 @@ Import... - Zavést... + Zavést... Export... - Vyvést... + Vyvést... Target Identifier - Identifikátor cíle + Identifikátor cíle Target: - Cíl: + Cíl: Reset - Nastavit znovu + Nastavit znovu Reset all to default - Nastavit vše znovu na výchozí + Nastavit vše znovu na výchozí Reset All - Nastavit znovu vše + Nastavit znovu vše Reset to default - Nastavit znovu výchozí + Nastavit znovu výchozí CodePaster::FileShareProtocolSettingsWidget Form - Formulář + Formulář &Path: - &Cesta: + &Cesta: &Display: - &Zobrazit: + &Zobrazit: entries - Záznamy + Záznamy The fileshare-based paster protocol allows for sharing code snippets using simple files on a shared network drive. Files are never deleted. - Protokol vložení založený na sdílení souborů umožňuje sdílení kousků kódu pomocí jednoduchých souborů na sdílené síťové diskové jednotce. Soubory nejsou nikdy mazány. + Protokol vložení založený na sdílení souborů umožňuje sdílení kousků kódu pomocí jednoduchých souborů na sdílené síťové diskové jednotce. Soubory nejsou nikdy mazány. Git::Internal::StashDialog Stashes - Uložené změny + Odložené změny Name @@ -20326,7 +21710,7 @@ Message - Označení + Popis Delete all... @@ -20342,7 +21726,7 @@ Restore... - Obnovit... + Použít odložené... Restore to branch... @@ -20351,7 +21735,7 @@ Refresh - Obnovit + Obnovit seznam <No repository> @@ -20367,14 +21751,14 @@ Do you want to delete all stashes? - Chcete smazat všechny uložené změny (stash)? + Chcete smazat všechny odložené změny (stashes)? Do you want to delete %n stash(es)? - Chcete smazat jednu uloženou změnu? - Chcete smazat %n uložené změny? - Chcete smazat %n ulitých změn? + Chcete smazat jednu odloženou změnu? + Chcete smazat %n odložené změny? + Chcete smazat %n odložených změn? @@ -20388,25 +21772,25 @@ Restore to Branch... Restore a git stash to new branch to be created - Obnovit jako větev... + Použít jako větev... Delete Stashes - Smazat uložené změny + Smazat odložené změny Repository Modified - Skladiště upraveno + Skladiště pozměněno %1 cannot be restored since the repository is modified. You can choose between stashing the changes or discarding them. - %1 nelze obnovit, protože skladiště bylo upraveno. -Můžete si vybrat mezi uložením změn nebo jejich vyhozením. + %1 nelze použít, protože skladiště bylo pozměněno. +Můžete si vybrat mezi odložením změn nebo jejich vyhozením. Stash - Ulít + Odložit Discard @@ -20414,7 +21798,7 @@ Restore Stash to Branch - Obnovit uloženou změnu jako větev + Použít odloženou jako větev Branch: @@ -20422,15 +21806,15 @@ Stash Restore - Obnovení uložené změny + Použití odložené změny Would you like to restore %1? - Chcete obnovit %1? + Chcete použít %1? Error restoring %1 - Chyba při obnově %1 + Chyba při použití %1 @@ -20445,7 +21829,7 @@ repository - Skladiště + skladiště Branch: @@ -20453,11 +21837,11 @@ branch - Větev + větev Commit Information - Informace o odeslání + Informace o zápisu Author: @@ -20488,7 +21872,7 @@ Username to use by default on commit. - Uživatelské jméno, které se použije jako výchozí při odeslání. + Uživatelské jméno, které se použije jako výchozí při zápisu. Default username: @@ -20496,7 +21880,7 @@ Email to use by default on commit. - Adresa elektronické pošty, která se použije jako výchozí při odeslání. + Adresa elektronické pošty, která se použije jako výchozí při zápisu. Default email: @@ -20508,7 +21892,7 @@ Log count: - Počet zápisů omezit na: + Počet záznamů omezit na: The number of recent commit logs to show, choose 0 to see all enteries @@ -20524,7 +21908,7 @@ Prompt on submit - Potvrdit předložení + Ptát se na potvrzení Mercurial @@ -20532,7 +21916,7 @@ The number of recent commit logs to show, choose 0 to see all entries. - Počet ukázaných nedávných zápisů o odeslání. Vyberte 0, abyste viděl všechny záznamy. + Počet nedávných záznamů které se vypisují v historii změn. Při volbě 0 budou vidět všechny záznamy. @@ -20543,7 +21927,7 @@ Specify a revision other than the default? - Chcete uvést jinou revizi, než je ta výchozí? + Chcete jinou než výchozí revizi? Revision: @@ -20606,11 +21990,11 @@ BehaviorDialog Dialog - Dialog + Dialog Type: - Typ: + Typ: Id: @@ -20622,52 +22006,47 @@ Animation - Vykreslování + Vykreslování SpringFollow - SpringFollow + SpringFollow Settings - Nastavení + Nastavení Duration: - Doba trvání: + Doba trvání: Curve: - Křivka: + Křivka: easeNone - easeNone + easeNone Source: - Zdroj: + Zdroj: Velocity: - Rychlost: + Rychlost: Spring: - Pružina: - - - Damping: - Tlumení: - + Pružina: ID: - ID: + ID: Property name: - Název vlastnosti: + Název vlastnosti: @@ -20717,7 +22096,7 @@ This area allows you to edit gradient stops. Double click on the existing stop handle to duplicate it. Double click outside of the existing stop handles to create a new stop. Drag & drop the handle to reposition it. Use right mouse button to popup context menu with extra actions. - Tato oblast vám umožňuje upravovat body zastavení přechodu. Dvakrát klepněte na stávající bod zastavení pro jeho zdvojení. Dvakrát klepněte mimo stávající bod zastavení pro vytvoření nového bodu zastavení. Táhněte a pusťte bod zastavení, abyste jej přemístil. Použijte pravé tlačítko myši pro vyvolání vyskakovací nabídky se souvisejícími činnostmi navíc. + Tato oblast vám umožňuje upravovat body nastavení přechodu. Dvakrát klepněte na stávající bod nastavení pro jeho zdvojení. Dvakrát klepněte mimo stávající bod pro vytvoření nového bodu nastavení. Popotáhnutím lze bod nastavení přemístit. Pravým tlačítkem myši se vyvolá vyskakovací nabídka s dalšími úkony. Zoom @@ -21339,15 +22718,15 @@ Major: - Větší: + Větší: Minor: - Menší: + Menší: Patch: - Opravný soubor: + Opravný soubor: Files to deploy: @@ -21355,27 +22734,27 @@ Package name: - Název balíčku: + Název balíčku: Package version: - Verze balíčku: + Verze balíčku: Short package description: - Krátký popis balíčku: + Krátký popis balíčku: Name to be displayed in Package Manager: - Název k zobrazení ve správci balíčků: + Název k zobrazení ve správci balíčků: Icon to be displayed in Package Manager: - Ikona k zobrazení ve správci balíčků: + Ikona k zobrazení ve správci balíčků: Adapt Debian file: - Upravit soubor Debianu: + Upravit soubor Debianu: Edit @@ -21383,11 +22762,11 @@ Edit spec file - Upravit soubor spec + Upravit soubor spec Edit... - Upravit... + Upravit... @@ -21572,11 +22951,11 @@ Self-signed certificate - Osobně podepsané osvědčení + Osobně podepsaný certifikát Custom certificate: - Uživatelsky stanovené osvědčení: + Uživatelsky stanovený certifikát: Choose certificate file (.cer) @@ -21592,7 +22971,7 @@ Choose certificate file - Vybrat soubor s osvědčením + Vybrat soubor s certifikátem Create Smart Installer package @@ -21608,14 +22987,14 @@ Certificate's details - Podrobnosti osvědčení + Podrobnosti certifikátu Qt4ProjectManager::Internal::TargetSetupPage Setup targets for your project - Nastavte cíl pro svůj projekt + Nastavte cíl pro svůj projekt Qt Creator can set up the following targets: @@ -21677,7 +23056,15 @@ <html><head/><body><p><b>No valid Qt versions found.</b></p><p>Please add a Qt version in <i>Tools/Options</i> or via the maintenance tool of the SDK.</p></body></html> - <html><head/><body><p><b>Nepodařilo se najít žádné platné verze Qt.</b><br>Přidejte, prosím, platnou verzi verzi Qt v <i>Nástroje/Nastavení</i> nebo prostřednictvím instalačního nástroje SDK.</p></body></html> + <html><head/><body><p><b>Nepodařilo se najít žádné platné verze Qt.</b><br>Přidejte, prosím, platnou verzi verzi Qt v <i>Nástroje/Nastavení</i> nebo prostřednictvím instalačního nástroje SDK.</p></body></html> + + + Set up Targets for Your Project + Nastavte cíl pro svůj projekt + + + <html><head/><body><p><span style=" font-weight:600;">No valid Qt versions found.</span></p><p>Please add a Qt version in <span style=" font-style:italic;">Tools &gt; Options &gt; Build &amp; Run</span> (<span style=" font-style:italic;">Qt Creator &gt; Preferences &gt; Build &amp; Run</span> on Mac OS) or via the maintenance tool of the SDK.</p></body></html> + <html><head/><body><p><span style=" font-weight:600;">Nenalezeny žádné platné verze Qt.</span></p><p>Přidejte, prosím, verzi Qt v <span style=" font-style:italic;">Nástroje &gt; Volby &gt; Sestavení &amp; Spuštění</span> (<span style=" font-style:italic;">Qt Creator &gt; Nastavení &gt; Sestavení &amp; Spuštění</span> na Mac OS) nebo přes nástroj na správu SDK.</p></body></html> @@ -21717,7 +23104,7 @@ Test slot: - Zkušební místo: + Zkušební pozice: Requires QApplication @@ -21736,7 +23123,7 @@ VcsBase::CleanDialog Clean Repository - Uklidit skladiště + Uklidit skladiště The directory %1 could not be deleted. @@ -21787,15 +23174,15 @@ CommonSettingsPage Wrap submit message at: - Zalomit popis předložení na: + Zalomit popis předložení na: characters - znacích + znacích An executable which is called with the submit message in a temporary file as first argument. It should return with an exit != 0 and a message on standard error to indicate failure. - Spustitelný soubor, který je zavolán s popisem předložení v dočasném souboru jako první argument příkazového řádku. Při neúspěchu by měl vrátit zpět hodnotu rozdílnou od nuly (!= 0) a odpovídající zprávu o obvyklé chybě kvůli poukázání na selhání. + Spustitelný soubor, který je zavolán s popisem předložení v dočasném souboru jako první argument příkazového řádku. Při neúspěchu by měl vrátit zpět hodnotu rozdílnou od nuly (!= 0) a odpovídající zprávu o obvyklé chybě kvůli poukázání na selhání. Submit message check script: @@ -21804,7 +23191,7 @@ A file listing user names and email addresses in a 4-column mailmap format: name <email> alias <email> - Soubor, který obsahuje jména uživatelů a e-mailové adresy ve čtyřsloupcovém formátu (mailmap): + Soubor, který obsahuje jména uživatelů a e-mailové adresy ve čtyřsloupcovém formátu (mailmap): Jméno <E-mail> Přezdívka <E-mail> @@ -21813,7 +23200,7 @@ A simple file containing lines with field names like "Reviewed-By:" which will be added below the submit editor. - Soubor, který obsahuje řádky s názvy polí (například "Reviewed-By:"), který bude bude přidán pod okno editoru předložení. + Soubor, který obsahuje řádky s názvy polí (například "Reviewed-By:"), který bude bude přidán pod okno editoru předložení. User fields configuration file: @@ -21821,29 +23208,29 @@ Submit message &check script: - Skript k ověření popisu předložení: + Skript k ověření popisu předložení: User/&alias configuration file: - Soubor s nastavením uživatele/&přezdívky: + Soubor s nastavením uživatele/&přezdívky: User &fields configuration file: - Soubor s nastavením p&olí uživatele: + Soubor s nastavením p&olí uživatele: &Patch command: - &Příkaz pro opravný soubor: + &Příkaz pro opravný soubor: Specifies a command that is executed to graphically prompt for a password, should a repository require SSH-authentication (see documentation on SSH and the environment variable SSH_ASKPASS). - Určuje příkaz, který je spuštěn, aby graficky vyzval k zadání hesla, které je požadováno při ověření pravosti SSH skladiště + Určuje příkaz, který je spuštěn, aby graficky vyzval k zadání hesla, které je požadováno při ověření pravosti SSH skladiště (podívejte se na dokumentaci k SSH k proměnné prostředí SSH-ASKPASS). &SSH prompt command: - &Příkaz pro výzvu o heslo k SSH: + &Příkaz pro výzvu o heslo k SSH: @@ -21889,7 +23276,7 @@ None or multiple items selected. - Nevybrána žádná nebo více položek + Nevybrána žádná nebo více položek. @@ -22467,7 +23854,7 @@ QmlJS::Check unknown value for enum - Neznámá hodnota pro 'enum' + Neznámá hodnota pro 'enum' value might be 'undefined' @@ -22475,167 +23862,167 @@ enum value is not a string or number - Hodnota 'enum' není řetězcem ani číslem + Hodnota 'enum' není řetězcem ani číslem numerical value expected - Očekávána číselná hodnota + Očekávána číselná hodnota boolean value expected - Očekávána booleánská hodnota + Očekávána booleánská hodnota string value expected - Očekáván řetězec + Očekáván řetězec not a valid url - Není platná adresa (URL) + Není platná adresa (URL) file or directory does not exist - Soubor nebo adresář neexistuje + Soubor nebo adresář neexistuje not a valid color - Není platnou barvou + Není platnou barvou expected anchor line - Očekáván kotevní řádek + Očekáván kotevní řádek unreachable - nedosažitelný + nedosažitelný declarations should be at the start of a function - Prohlášení mají stát na začátku funkce + Prohlášení mají stát na začátku funkce already a formal parameter - Je již formální parametr + Je již formální parametr already declared as function - Již prohlášeno jako funkce + Již prohlášeno jako funkce duplicate declaration - Již prohlášeno + Již prohlášeno variable is used before being declared - Proměnná se používá před prohlášením + Proměnná se používá před prohlášením already declared as var - Již prohlášeno jako proměnná + Již prohlášeno jako proměnná function is used before being declared - Funkce se používá před prohlášením + Funkce se používá před prohlášením properties can only be assigned once - Vlastnosti mohou být přiřazeny jen jednou + Vlastnosti mohou být přiřazeny jen jednou unknown type - Neznámý typ + Neznámý typ could not resolve the prototype %1 of %2 - Prototyp %1 z %2 se nepodařilo vyřešit + Prototyp %1 z %2 se nepodařilo vyřešit could not resolve the prototype of %1 - Prototyp %1 se nepodařilo vyřešit + Prototyp %1 se nepodařilo vyřešit prototype cycle, the last non-repeated object is %1 - Prototypová smyčka, poslední neopakovaný vyskytující se objekt je %1 + Prototypová smyčka, poslední neopakovaný vyskytující se objekt je %1 expected id - Očekáváno ID + Očekáváno ID using string literals for ids is discouraged - Od používání řetězců znaků tvořených písmeny jako ID se odrazuje + Od používání řetězců znaků tvořených písmeny jako ID se odrazuje ids must be lower case or start with underscore - ID musí začínat malým písmenem nebo podtržítkem + ID musí začínat malým písmenem nebo podtržítkem ids must be unique - ID musí být jednoznačná + ID musí být jednoznačná '%1' is not a valid property type - '%1' není platný typ vlastnosti + '%1' není platný typ vlastnosti unknown identifier - Neznámý identifikátor + Neznámý identifikátor could not resolve - Nepodařilo se vyřešit + Nepodařilo se vyřešit does not have members - Nemá žádné členy + Nemá žádné členy unknown member - Neznámý člen + Neznámý člen == and != perform type coercion, use === or !== instead to avoid - Operátory == a != způsobují změnu typu; místo toho by se měly používat operátory === nebo !== + Operátory == a != způsobují změnu typu; místo toho by se měly používat operátory === nebo !== blocks do not introduce a new scope, avoid - Bloky neuvádějí novou oblast, a mělo by se jich z toho důvodu vyvarovat + Bloky neuvádějí novou oblast, a mělo by se jich z toho důvodu vyvarovat unintentional empty block, use ({}) for empty object literal - nezamýšlený prázdný blok, použijte ({}) jako prázdný objekt tvořený písmeny + nezamýšlený prázdný blok, použijte ({}) jako prázdný objekt tvořený písmeny use of the with statement is not recommended, use a var instead - od použití s-příkazem se zrazuje, místo toho by se mělo použít var + od použití s-příkazem se zrazuje, místo toho by se mělo použít var use of void is usually confusing and not recommended - od použití void se zrazuje, protože obvykle vede ke zmatku + od použití void se zrazuje, protože obvykle vede ke zmatku avoid comma expressions - od použití výrazů s čárkou se zrazuje + od použití výrazů s čárkou se zrazuje expression statements should be assignments, calls or delete expressions only - Příkazy s výrazy by měly být jen přiřazení, volání funkcí nebo výrazy delete + Příkazy s výrazy by měly být jen přiřazení, volání funkcí nebo výrazy delete 'new' should only be used with functions that start with an uppercase letter - 'new' by se měl používat jen s funkcemi, které začínají velkým písmenem + 'new' by se měl používat jen s funkcemi, které začínají velkým písmenem calls of functions that start with an uppercase letter should use 'new' - Funkce, které začínají velkým písmenem, by se měly volat s 'new' + Funkce, které začínají velkým písmenem, by se měly volat s 'new' avoid assignments in conditions - od přiřazení v podmínkách se zrazuje + od přiřazení v podmínkách se zrazuje case is not terminated and not empty - case není uzavřen a není prázdný + case není uzavřen a není prázdný case does not end with return, break, continue or throw @@ -22647,15 +24034,19 @@ '%1' is not a valid property name - ' %1' není jednoznačný název pro vlastnost + ' %1' není jednoznačný název pro vlastnost '%1' does not have members - ' %1' nemá členy + ' %1' nemá členy '%1' is not a member of '%2' - ' %1' nepatří k '%2' + ' %1' nepatří k '%2' + + + 'int' or 'real' + 'int' nebo 'real' @@ -22771,16 +24162,28 @@ Soubor %1 byl odstraněn. Chcete jej uložit pod jiným názvem, nebo zavřít editor? + &Close + &Zavřít + + + Save &as... + Uložit &jako... + + + &Save + &Uložit + + Close - Zavřít + Zavřít Save as... - Uložit jako... + Uložit jako... Save - Uložit + Uložit @@ -22932,10 +24335,14 @@ Prostředí pro sestavování - The executable is not built by the current buildconfiguration + The executable is not built by the current build configuration Spustitelný soubor není sestaven současným nastavením pro sestavování + The executable is not built by the current buildconfiguration + Spustitelný soubor není sestaven současným nastavením pro sestavování + + (disabled) (zakázáno) @@ -23020,14 +24427,26 @@ Kopírovat celou cestu do schránky + Split + Rozdělit + + Copy Full Path to Clipboard Kopírovat celou cestu do schránky - Make writable + Remove Split + Odstranit rozdělení + + + Make Writable Udělat zapisovatelným + Make writable + Udělat zapisovatelným + + File is writable Soubor je zapisovatelný @@ -23121,23 +24540,31 @@ It is recommended that you secure your private key with a password, which you can enter below. - Doporučuje se, abyste zabezpečil svůj soukromý klíč + Doporučuje se zabezpečit svůj soukromý klíč heslem, jež můžete zadat níže. - Encrypt key file + Encrypt Key File Zašifrovat soubor s klíčem - Do not encrypt key file + Do Not Encrypt Key File Nešifrovat soubor s klíčem + + Encrypt key file + Zašifrovat soubor s klíčem + + + Do not encrypt key file + Nešifrovat soubor s klíčem + CodePaster Code Pasting - Vkládání kódu + Úryvky kódu @@ -23178,7 +24605,7 @@ CodePaster::PasteBinDotComSettings Pastebin.com - Pastebin.com + Pastebin.com @@ -23226,7 +24653,7 @@ Split Declaration - Rozdělit prohlášení + Rozdělit deklaraci Add Curly Braces @@ -23234,7 +24661,7 @@ Move Declaration out of Condition - Odstranit prohlášení z podmínky + Odstranit deklaraci z podmínky Split if Statement @@ -23257,6 +24684,18 @@ Uzavřít do %1(...) + Convert to String Literal + Převést na řetězec znaků tvořený písmeny (string literal) + + + Convert to Character Literal and Enclose in QLatin1Char(...) + Převést na řetězec znaků tvořený písmeny (character literal) a uzavřít v QLatin1Char(...) + + + Convert to Character Literal + Převést na řetězec znaků tvořený písmeny (character literal) + + Mark as Translatable Označit jako překladatelné @@ -23282,13 +24721,25 @@ Add Local Declaration - Přidat místní prohlášení + Přidat místní deklaraci Convert to Camel Case Převést na Camel Case + Add #include %1 + Přidat #include %1 + + + Switch with Previous Parameter + Vyměnit s předchozím parametrem + + + Switch with Next Parameter + Vyměnit s dalším parametrem + + Use Fast String Concatenation with % Použít účinné zřetězení řetězce za použití operátoru % @@ -23301,15 +24752,15 @@ VCS CVS Commit Editor - Editor odeslání pro CVS + Editor odevzdání (commit) pro CVS CVS Command Log Editor - Editor zápisů příkazů pro CVS + Editor záznamů o příkazech pro CVS CVS File Log Editor - Editor zápisů souborů pro CVS + Editor záznamů souborů pro CVS CVS Annotation Editor @@ -23321,11 +24772,11 @@ Git Command Log Editor - Editor zápisů příkazů pro Git + Editor záznamů o příkazech pro Git Git File Log Editor - Editor zápisů souborů pro Git + Editor záznamů souborů pro Git Git Annotation Editor @@ -23337,15 +24788,15 @@ Git Submit Editor - Editor předložení pro Git + Editor odeslání (submit) pro Git Mercurial Command Log Editor - Editor zápisů příkazů pro Mercurial + Editor záznamů o příkazech pro Mercurial Mercurial File Log Editor - Editor zápisů souborů pro Mercurial + Editor záznamů o souborech pro Mercurial Mercurial Annotation Editor @@ -23357,19 +24808,19 @@ Mercurial Commit Log Editor - Editor zápisu odeslání pro Mercurial + Editor historie změn pro Mercurial Perforce.SubmitEditor - Editor předložení pro Perforce + Editor odevzdání (submit) pro Perforce Perforce CommandLog Editor - Editor zápisů příkazů pro Perforce + Editor záznamů o příkazech pro Perforce Perforce Log Editor - Editor zápisů pro Perforce + Editor záznamů pro Perforce Perforce Diff Editor @@ -23385,15 +24836,15 @@ Subversion Commit Editor - Editor odeslání pro Subversion + Editor odevzdání (commit) pro Subversion Subversion Command Log Editor - Editor zápisů příkazů pro Subversion + Editor záznamů o příkazech pro Subversion Subversion File Log Editor - Editor zápisů souborů pro Subversion + Editor záznamů o souborech pro Subversion Subversion Annotation Editor @@ -23405,11 +24856,11 @@ Bazaar Command Log Editor - Editor zápisů příkazů pro Bazaar + Editor záznamů o příkazech pro Bazaar Bazaar File Log Editor - Editor zápisů souborů pro Bazaar + Editor záznamů o souborech pro Bazaar Bazaar Annotation Editor @@ -23421,14 +24872,14 @@ Bazaar Commit Log Editor - Editor zápisů o odeslání pro Bazaar + Editor historie změn pro Bazaar Cvs::Internal::CvsEditor Annotate revision "%1" - Opatřit vysvětlivkami revizi "%1" + Opatřit anotacemi revizi "%1" @@ -23517,7 +24968,7 @@ Toolchains - Řetězce nástrojů + Sady nástrojů Duplicate binary @@ -23717,7 +25168,7 @@ FakeVim::Internal::FakeVimExCommandsPage Ex Command Mapping - Přiřazení příkazů + Přiřazení příkazů pro Ex FakeVim @@ -23733,7 +25184,7 @@ Ex Command - Příkaz + Příkaz pro Ex @@ -23765,6 +25216,18 @@ Make Make + + Override %1: + Přepsat %1: + + + Make arguments: + Argumenty příkazového řádku pro 'make': + + + Targets: + Cíle: + GenericProjectManager::Internal::Manager @@ -23798,11 +25261,11 @@ Git::Internal::GitEditor Blame %1 - Blame (obviňovat) %1 + Vinit (blame) %1 Blame parent revision %1 - Blame (vina) nadřazené revize %1 + Vinit (blame) nadřazenou revizi %1 @@ -23857,7 +25320,7 @@ Mercurial::Internal::CloneWizard Clones a Mercurial repository and tries to load the contained project. - Vytvoří klon skladiště Mercurial a pokusí se nahrát obsažený projekt. + Vytvoří klon skladiště pro Mercurial a pokusí se nahrát obsažený projekt. Mercurial Clone @@ -23872,18 +25335,18 @@ Specify repository URL, checkout directory and path. - Zadejte adresu skladiště (URL), adresář pro stažení (checkout) kopie a cestu. + Zadejte adresu skladiště (URL), cestu a adresář pro umístění pracovní kopie. Clone URL: - Adresa (URL) klonu: + Klonovat z adresy (URL): Mercurial::Internal::CommitEditor Commit Editor - Editor odeslání + Editor zápisu (commit) @@ -23898,7 +25361,7 @@ Hg Annotate %1 - Opatřit vysvětlivkami (Hg annotate) "%1" + Zobrazit anotace (annotate) "%1" Hg diff %1 @@ -23910,7 +25373,7 @@ Hg incoming %1 - Hg přícházející %1 + Hg přicházející %1 Hg outgoing %1 @@ -23932,11 +25395,11 @@ Mercurial::Internal::MercurialEditor Annotate %1 - Opatřit vysvětlivkami %1 + Vypsat anotace pro %1 Annotate parent revision %1 - Opatření vysvětlivkou nadřazené revize "%1" + Vypsat anotace nadřazené revize "%1" @@ -23964,19 +25427,19 @@ Annotate Current File - Opatřit nynější soubor vysvětlivkami + Vypsat anotace pro nynější soubor Annotate "%1" - Opatřit vysvětlivkami "%1" + Vypsat anotace pro "%1" Diff Current File - Rozdíly (diff) nynějšího souboru + Rozdíly nynějšího souboru Diff "%1" - Rozdíly (diff) pro "%1" + Rozdíly pro "%1" Alt+H,Alt+D @@ -23984,11 +25447,11 @@ Log Current File - Zápis pro nynější soubor + Záznamy pro nynější soubor Log "%1" - Zápis pro "%1" + Záznamy pro "%1" Alt+H,Alt+L @@ -24032,11 +25495,11 @@ Diff - Rozdíly (diff) + Rozdíly Log - Zápis + Záznamy Revert... @@ -24048,31 +25511,31 @@ Pull... - Vytáhnout (pull)... + Přivést (pull)... Push... - Zatlačit (push)... + Odvést (push)... Update... - Obnovit... + Načíst (update)... Import... - Zavést... + Dovést (import)... Incoming... - Přícházející... + Příchozí... Outgoing... - Odcházející... + Odchozí... Commit... - Odeslat... + Zapsat (commit)... Alt+H,Alt+C @@ -24084,27 +25547,27 @@ Pull Source - Zdroj pro operaci pull + Zdroj pro přivedení (pull) Push Destination - Cíl pro operaci push + Cíl pro odvedení (push) Update - Obnovit + Načíst (update) Incoming Source - Přícházející zdroj + Zdroj pro příchozí Commit - Odeslat + Zapsat (commit) Diff &Selected Files - Rozdíly (diff) pro &vybrané soubory + Rozdíly pro &vybrané soubory Diff Selected Files @@ -24120,15 +25583,15 @@ There are no changes to commit. - Nejsou zde žádné změny pro odeslání. + Nejsou zde žádné změny k zápisu. Unable to generate a temporary file for the commit editor. - Nepodařilo se vytvořit žádný dočasný soubor pro editor pro odeslání. + Nepodařilo se vytvořit dočasný soubor pro editor pro zápis. Unable to create an editor for the commit. - Nepodařilo se vytvořit žádný editor pro odeslání. + Nepodařilo se vytvořit žádný editor pro zápis. Unable to create a commit editor. @@ -24136,7 +25599,7 @@ Commit changes for "%1". - Odeslat změny pro "%1". + Zapsat změny pro "%1". Close commit editor @@ -24144,15 +25607,15 @@ Do you want to commit the changes? - Chcete odeslat změny? + Chcete zapsat změny? Close Commit Editor - Zavřít editor pro odeslání + Zavřít editor pro zápis Message check failed. Do you want to proceed? - Ověření popisu se nezdařilo. Přesto chcete soubory odeslat? + Ověření popisu se nezdařilo. Přesto chcete pokračovat? @@ -24202,7 +25665,7 @@ Perforce::Internal::PerforceEditor Annotate change list "%1" - Opatřit vysvětlivkami seznam se změnami "%1" + Opatřit anotacemi seznam se změnami "%1" @@ -24213,7 +25676,7 @@ untitled - File path suggestion for a new project. If you choose to translate it, make sure it is a valid path name without blanks. + File path suggestion for a new project. If you choose to translate it, make sure it is a valid path name without blanks and using only ascii chars. Bez názvu @@ -24227,7 +25690,7 @@ Clean Display name of the clean build step list. Used as part of the labels in the project window. - Uklidit + Pročistit System Environment @@ -24272,15 +25735,15 @@ Creates a C++ plugin to extend the funtionality of the QML runtime. - Vytvoří přídavný modul C++ pro rozšíření funkčnosti doby běhu QML. + Vytvoří přídavný modul C++ pro rozšíření funkčnosti běhového prostředí QML. QML Runtime Plug-in - Přídavný modul doby běhu QML + Přídavný modul běhového prostředí QML QML Runtime Plug-in Parameters - Parametry přídavného modulu doby běhu QML + Parametry přídavného modulu běhového prostředí QML Example Object Class-name: @@ -24288,7 +25751,7 @@ Creates a C++ plugin that makes it possible to offer extensions that can be loaded dynamically into applications using the QDeclarativeEngine class. - Vytvoří přídavný modul C++ pro rozšíření, která mohou být do programů nahrávána dynamicky, která používají třídu QDeclarativeEngine. + Vytvoří přídavný modul C++ pro rozšíření, která mohou být do programů nahrávána dynamicky s použitím třídy QDeclarativeEngine. Custom QML Extension Plugin @@ -24296,7 +25759,7 @@ QML Extension Plugin - Přídavný modul pro rozšíření QML + Přídavný modul pro rozšíření QML Custom QML Extension Plugin Parameters @@ -24320,27 +25783,55 @@ Qt Creator plugin - Přídavný modul pro Qt Creator + Přídavný modul pro Qt Creator Other Project - Jiný projekt + Jiný projekt Creates a plain C project using qmake, not using the Qt library. - Vytvoří na qmake založený čistý projekt C bez použití knihovny Qt. + Vytvoří na qmake založený projekt v čistém C bez použití knihovny Qt. Plain C Project - Čistý projekt C + Projekt v čistém C + + + Non-Qt Project + Projekty C/C++ (bez Qt) + + + Creates a plain C project using CMake, not using the Qt library. + Vytvoří na qmake založený projekt v čistém C bez použití knihovny Qt. + + + Plain C Project (CMake Build) + Projekt v čistém C (sestavovaný CMake) Creates a plain C++ project using qmake, not using the Qt library. - Vytvoří na qmake založený čistý projekt C++ bez použití knihovny Qt. + Vytvoří na qmake založený projekt v čistém C++ bez použití knihovny Qt. Plain C++ Project - Čistý projekt C++ + Projekt v čistém C++ + + + Creates a plain C++ project using CMake, not using the Qt library. + Vytvoří na qmake založený projekt v čistém C++ bez použití knihovny Qt. + + + Plain C++ Project (CMake Build) + Projekt v C++ (sestavovaný CMake) + + + Libraries + Knihovny + + + Qt Creator Plugin + Přídavný modul pro Qt Creator Plugin Information @@ -24352,7 +25843,7 @@ Vendor name: - Název prodejce: + Dodavatel: Copyright: @@ -24372,7 +25863,7 @@ Deploy into: - Nasazení do: + K nasazení do: Qt Creator build @@ -24480,11 +25971,11 @@ Show containing folder... - Ukázat obsaženou složku... + Ukázat obsahující složku... Open Command Prompt here... - Otevřít výzvu k příkazu zde... + Otevřít příkazový řádek zde... Open Terminal here... @@ -24512,7 +26003,7 @@ Open Command Prompt Here... - Otevřít výzvu k příkazu zde... + Otevřít příkazový řádek zde... Open Terminal Here... @@ -24563,19 +26054,19 @@ ProjectExplorer::Internal::MiniTargetWidget Select active build configuration - Vybrat nastavení sestavení + Vybrat nastavení sestavení Select active run configuration - Vybrat nastavení spuštění + Vybrat nastavení spuštění Build: - Sestavení: + Sestavení: Run: - Spuštění: + Spuštění: @@ -24585,32 +26076,104 @@ Projekt + Target + Cíl + + + Build + Sestavování + + + Deploy + Nasazení + + + Run + Spustit + + + Unconfigured + Nenastaveno + + + <b>Project:</b> %1 + <b>Projekt:</b> %1 + + + <b>Target:</b> %1 + <b>Cíl:</b> %1 + + + <b>Build:</b> %1 + <b>Sestavení:</b> %1 + + + <b>Deploy:</b> %1 + <b>Nasazení:</b> %1 + + + <b>Run:</b> %1 + <b>Spuštění:</b> %1 + + + %1 + %1 + + + <html><nobr>%1</html> + <html><nobr>%1</html> + + + Project: <b>%1</b><br/> + Projekt: <b>%1</b><br/> + + + Target: <b>%1</b><br/> + Cíl: <b>%1</b><br/> + + + Build: <b>%1</b><br/> + Sestavení: <b>%1</b><br/> + + + Deploy: <b>%1</b><br/> + Nasazení: <b>%1</b><br/> + + + Run: <b>%1</b><br/> + Spuštění: <b>%1</b><br/> + + + <style type=text/css>a:link {color: rgb(128, 128, 255, 240);}</style>The project <b>%1</b> is not yet configured<br/><br/>You can configure it in the <a href="projectmode">Projects mode</a><br/> + <style type=text/css>a:link {color: rgb(128, 128, 255, 240);}</style>Das Projekt <b>%1</b> ještě není nastaven<br/><br/>Můžete jej nastavit v <a href="projectmode">Projektovém režimu</a><br/> + + Select active project - Vybrat projekt + Vybrat projekt Build: - Sestavení: + Sestavení: Run: - Spuštění: + Spuštění: <html><nobr><b>Project:</b> %1<br/>%2%3<b>Run:</b> %4%5</html> - <html><nobr><b>Projekt:</b> %1<br/>%2%3<b>Spuštění:</b> %4%5</html> + <html><nobr><b>Projekt:</b> %1<br/>%2%3<b>Spuštění:</b> %4%5</html> <b>Target:</b> %1<br/> - <b>Cíl:</b> %1<br/> + <b>Cíl:</b> %1<br/> <b>Build:</b> %2<br/> - <b>Sestavení:</b> %2<br/> + <b>Sestavení:</b> %2<br/> <br/>%1 - <br/>%1 + <br/>%1 @@ -24634,6 +26197,30 @@ Other Project Jiný projekt + + Applications + Programy + + + Libraries + Knihovny + + + Non-Qt Project + Projekty C/C++ (bez Qt) + + + Import Project + Zavést projekt + + + Devices + Zařízení + + + Qt Application + Program Qt + TargetSettingsPanelFactory @@ -25346,7 +26933,7 @@ Open Documents - Otevřít dokumenty + Otevřené dokumenty Qt Quick emulation layer crashed @@ -25806,6 +27393,10 @@ Ctrl+Shift+C + Reformat File + Zformátovat soubor znovu + + Show Qt Quick Toolbar Ukázat nástrojový pruh Qt Quick @@ -25852,6 +27443,14 @@ QML project: %1 Projekt QML: %1 + + Warning while loading project file %1. + Varování při nahrávání projektového souboru %1. + + + File '%1' does not exist or is not readable. + Soubor '%1' neexistuje nebo není čitelný. + QmlProjectManager::Internal::QmlProjectApplicationWizardDialog @@ -25897,8 +27496,20 @@ Creates a Qt Quick UI project with a single QML file that contains the main view. +You can review Qt Quick UI projects in the QML Viewer and you need not build them. You do not need to have the development environment installed on your computer to create and run this type of projects. + +Requires <b>Qt 4.7.4</b> or newer. + Vytvoří projekt Qt Quick UI s jediným souborem QML, který obsahuje hlavní pohled. + +Projekty Qt Quick UI není potřeba je sestavovat a lze je spouštět přímo v prohlížeči QML. K vytvoření a ke spuštění tohoto typu projektů není potřeba, aby bylo ve vašem počítači nainstalováno vývojářské prostředí. + +Vyžaduje <b>Qt 4.7.4</b> nebo novější. + + + Creates a Qt Quick UI project with a single QML file that contains the main view. + You can review Qt Quick UI projects in the QML Viewer and you need not build them. You do not need to have the development environment installed on your computer to create and run this type of projects. - Vytvoří projekt Qt Quick UI z jednoho jednotlivého souboru QML, který obsahuje hlavní pohled. + Vytvoří projekt Qt Quick UI s jediným souborem QML, který obsahuje hlavní pohled. Projekty Qt Quick UI není potřeba je sestavovat a lze je spouštět přímo v prohlížeči QML. K vytvoření a ke spuštění tohoto typu projektů není potřeba, aby bylo ve vašem počítači nainstalováno vývojářské prostředí. @@ -25915,7 +27526,7 @@ QmlProjectManager Qt Quick Project - Projekt Qt Quick + Projekt Qt Quick @@ -26468,7 +28079,7 @@ Ignore patching for this packaging step. - Při tomto kroku balíčkování neprovádět žádný opravný soubor. + Při tomto kroku balíčkování neprovádět žádnou záplatu. Could not start process "%1" in %2 @@ -26482,11 +28093,11 @@ No certificate file specified. Please specify one in the project settings. - Nebyl zadán žádný soubor s osvědčením. Zadejte, prosím, jeden v nastavení projektu. + Nebyl zadán žádný soubor s certifikátem. Zadejte, prosím, jeden v nastavení projektu. Certificate file "%1" does not exist. Please specify an existing certificate file in the project settings. - Zadaný soubor s osvědčením "%1" neexistuje. Zadejte, prosím, existující soubor s osvědčením v nastavení projektu. + Zadaný soubor s certifikátem "%1" neexistuje. Zadejte, prosím, existující soubor s certifikátem v nastavení projektu. No key file specified. Please specify one in the project settings. @@ -26498,7 +28109,7 @@ The package created will not install on a device as some of the defined capabilities are not supported by the certificate: %1 - Vytvořený balíček nelze na mobilním zařízení nainstalovat, neboť některá ze zadaných oprávnění nejsou osvědčením podporována: %1 + Vytvořený balíček nelze na mobilním zařízení nainstalovat, neboť některá ze zadaných oprávnění nejsou certifikátem podporována: %1 The process "%1" exited normally. @@ -26532,11 +28143,11 @@ signed with the certificate "%1" using the key "%2" - Podepsáno s osvědčením "%1" za použití klíče "%2" + Podepsáno s certifikátem "%1" za použití klíče "%2" signed with a certificate and a key that need to be specified - Podepsáno s osvědčením a klíčem, které je ještě potřeba zadat + Podepsáno s certifikátem a klíčem, které je ještě potřeba zadat not signed @@ -26624,11 +28235,11 @@ The Symbian tool chain does not handle spaces in the project path '%1'. - Mezery v zadání projektové cesty '%1' mohou vést u nástrojového řetězce pro Symbian k potížím. + Mezery v zadání projektové cesty '%1' mohou vést u sady nástrojů pro Symbian k potížím. The Symbian tool chain does not handle special characters in the project name '%1' well. - Zvláštní znaky v zadání projektové cesty '%1' mohou vést u nástrojového řetězce pro Symbian k potížím. + Zvláštní znaky v zadání projektové cesty '%1' mohou vést u sady nástrojů pro Symbian k potížím. @@ -26665,7 +28276,7 @@ Qt4ProjectManager Qt4 - Qt4 + Qt4 Qt Versions @@ -26677,12 +28288,16 @@ Qt Widget Project - Projekt Qt Widget + Projekt Qt Widget Linux Devices Linuxová zařízení + + Unconfigured Project + Nenastavený projekt + Qt4ProjectManager::Internal::Qt4TargetFactory @@ -26767,8 +28382,13 @@ ABI(s) verze Qt se nepodařilo určit. + Android + Qt Version is meant for Android + Android + + ABI detection failed: Make sure to use a matching tool chain when building. - ABI se nepodařilo určit. Ujistěte se, že pro sestavování používáte odpovídající řetězec nástrojů. + ABI se nepodařilo určit. Ujistěte se, že pro sestavování používáte odpovídající sadu nástrojů. No qmlviewer installed. @@ -26809,12 +28429,18 @@ Building helper(s) with toolchain '%1' ... - Vytváří se pomocná knihovna pro výstup dat s řetězcem nástrojů '%1'... + Vytváří se pomocná knihovna pro výstup dat s řetězcem nástrojů '%1'... + + + + Building helper(s) with toolchain '%1'... + + Sestavuje se pomocná knihovna pro výstup dat pomocí sady nástrojů '%1'... Build failed. - Vytvoření se nezdařilo. + Sestavení se nezdařilo. Build succeeded. @@ -26840,6 +28466,11 @@ Qt Version is meant for Meego Meego + + Embedded Linux + Qt Version is used for embedded Linux development + Vložený Linux + Qt4ProjectManager::Internal::MobileGuiAppWizard @@ -26851,7 +28482,7 @@ Creates a Qt application optimized for mobile devices with a Qt Designer-based main window. Preselects Qt for Simulator and mobile targets if available - Vytvoří program napsaný v Qt určený pro přenosná zařízení s jedním hlavním oknem založeným na Qt Designeru. + Vytvoří program napsaný v Qt určený pro přenosná zařízení s hlavním oknem založeným na Qt Designeru. Vybere pro napodobovatele a přenosné cíle vhodné verze Qt, jsou-li dostupné. @@ -26879,7 +28510,7 @@ Creates a QTestLib-based unit test for a feature or a class. Unit tests allow you to verify that the code is fit for use and that there are no regressions. - Vytvoří na QTestLib založenou jednotkovou zkoušku pro funkci nebo třídu. Jednotkové zkoušky slouží k přezkoušení použitelnosti kódu a ke zjištění zpětného vývoje (regresí). + Vytvoří na QTestLib založenou jednotkovou zkoušku pro funkci nebo třídu. Jednotkové zkoušky slouží k přezkoušení použitelnosti kódu a ke zjištění regresí. @@ -26897,7 +28528,7 @@ Subversion::Internal::SubversionEditor Annotate revision "%1" - Opatřit vysvětlivkami revizi "%1" + Opatřit anotacemi revizi "%1" @@ -27207,51 +28838,51 @@ Remote - Vzdálený + Vzdálený Host: - Hostitelský počítač: + Hostitelský počítač: User: - Uživatel: + Uživatel: You need to pass either a password or an SSH key. - Potřebujete buď heslo nebo klíč SSH. + Potřebujete buď heslo nebo klíč SSH. Password: - Heslo: + Heslo: Port: - Přípojka: + Přípojka: Private key: - Soukromý klíč: + Soukromý klíč: Target - Cíl + Cíl Executable: - Spustitelný soubor: + Spustitelný soubor: Arguments: - Argumenty: + Argumenty: Working directory: - Pracovní adresář: + Pracovní adresář: Start Remote Analysis - Spustit vzdálený rozbor + Spustit vzdálený rozbor @@ -27267,16 +28898,16 @@ Perform a local commit in a bound branch. Local commits are not pushed to the master branch until a normal commit is performed. - Provést místní odeslání do spojené větve. -Místní odeslání nejsou použita na hlavní větev, dokud není provedeno normální odeslání. + Provést místní zápis (commit) do vázané větve. +Místní zápisy nejsou odevzdány do hlavní větve, dokud není proveden normální zápis-odevzdání. Local commit - Místní odeslání + Místní zápis Commit Information - Informace o odeslání + Informace o zápisu Author: @@ -27300,7 +28931,7 @@ By default, branch will fail if the target directory exists, but does not already have a control directory. This flag will allow branch to proceed. - Ve výchozím nastavení se operace s větví nepodaří, pokud existuje cílový adresář, ale již nemá adresář na správu verzí. + Ve výchozím nastavení se větvení (branch) nepodaří, pokud existuje cílový adresář, ale ještě nemá adresář na správu verzí. Toto nastavení dovolí za těchto okolností v operaci pokračovat. @@ -27310,12 +28941,12 @@ Create a stacked branch referring to the source branch. The new branch will depend on the availability of the source branch for all operations. - Vytvořit větev typu 'stacked', která se vztahuje ke zdrojové větvi. -Nová větev je závislá na dostupnosti zdrojové větve pro všechny operace. + Vytvořit nastavující (stacked) větev, která nastavuje zdrojovou větev. +Nová větev bude ve všech operacích závislá na dostupnosti zdrojové větve. Stacked - Stacked + Nastavující (stacked) Do not use a shared repository, even if available. @@ -27323,19 +28954,19 @@ Standalone - Samostatný + Samostatná Bind new branch to source location - Spojit novou větev s umístěním zdroje + Svázat novou větev s umístěním zdroje (bind) Switch the checkout in the current directory to the new branch. - Přepnout stažení (checkout) v nynějším adresáři na novou větev. + Přepnout získání (switch checkout) v nynějším adresáři na novou větev. Switch checkout - Přepnout stažení (checkout) + Přepnout získání (switch checkout) Hard-link working tree files where possible. @@ -27351,7 +28982,7 @@ No working-tree - Žádný pracovní souborový strom + Bez pracovního stromu Revision: @@ -27378,7 +29009,7 @@ Username to use by default on commit. - Uživatelské jméno, které se použije jako výchozí při odeslání. + Uživatelské jméno, které se použije jako výchozí při zápisu (commit). Default username: @@ -27386,7 +29017,7 @@ Email to use by default on commit. - Adresa elektronické pošty, která se použije jako výchozí při odeslání. + Adresa elektronické pošty, která se použije jako výchozí při zápisu (commit). Default email: @@ -27398,11 +29029,11 @@ Log count: - Počet zápisů omezit na: + Počet záznamů omezit na: The number of recent commit logs to show. Choose 0 to see all entries. - Počet ukázaných nedávných zápisů o odeslání. Vyberte 0, abyste viděl všechny záznamy. + Počet nedávných záznamů které se vypisují v historii změn. Při volbě 0 budou vidět všechny záznamy. Timeout: @@ -27414,7 +29045,7 @@ Prompt on submit - Potvrdit předložení + Ptát se na potvrzení Bazaar @@ -27459,7 +29090,7 @@ Ignore differences between branches and overwrite unconditionally. Nepřihlížet k rozdílům mezi větvemi -a přepisovat stále. +a bezpodmínečně přepisovat. Overwrite @@ -27468,8 +29099,8 @@ By default, push will fail if the target directory exists, but does not already have a control directory. This flag will allow push to proceed. - Ve výchozím nastavení se operace s větví nepodaří, pokud existuje cílový adresář, ale již nemá adresář na správu verzí. -Toto nastavení dovolí za těchto okolností v operaci pokračovat. + Ve výchozím nastavení se odvedení (push) nepodaří, pokud existuje cílový adresář, ale ještě nemá adresář na správu verzí. +Toto nastavení dovolí za těchto okolností v operaci odvedení pokračovat. Use existing directory @@ -27490,8 +29121,8 @@ Perform a local pull in a bound branch. Local pulls are not applied to the master branch. - Provést místní pull operaci do spojené větve. -Místní pull operace nejsou použity na hlavní větev. + Provést místní přivedení (pull) ve vázané větvi. +Místní přivedení nejsou použity na hlavní větev. Local @@ -27499,11 +29130,11 @@ Pull Source - Zdroj pro operaci pull + Zdroj pro přivedení (pull) Push Destination - Cíl pro operaci push + Cíl pro odvedení (push) @@ -27514,7 +29145,7 @@ Specify a revision other than the default? - Chcete uvést jinou revizi, než je ta výchozí? + Chcete jinou než výchozí revizi? Revision: @@ -27529,7 +29160,7 @@ Show Subprojects - Ukázat podřízené projekty + Ukázat dílčí projekty @@ -27649,86 +29280,86 @@ MimeTypeMagicDialog Dialog - Dialog + Dialog Value: - Hodnota: + Hodnota: Type - Typ + Typ String - Řetězec + Řetězec Byte - Byte + Byte Use Recommended - Použít doporučené + Použít doporučené Start range: - Začátek oblasti: + Začátek oblasti: End range: - Konec oblasti: + Konec oblasti: Priority: - Přednost: + Přednost: <i>Note: Wide range values might impact Qt Creator's performance when opening files.</i> - <i>Poznámka: Velké oblasti se mohou na rychlosti otevírání souborů odrazit záporně.</i> + <i>Poznámka: Velké oblasti se mohou na rychlosti otevírání souborů odrazit záporně.</i> MimeTypeSettingsPage Form - Formulář + Formulář Registered MIME Types - Zapsané MIME typy + Zapsané MIME typy Reset all to default. - Nastavit vše znovu na výchozí. + Nastavit vše znovu na výchozí. Reset All - Nastavit znovu vše + Nastavit znovu vše Details - Podrobnosti + Podrobnosti Patterns: - Vzory: + Vzory: Magic Header - Kouzelné záhlaví + Kouzelné záhlaví Type - Typ + Typ Range - Rozsah + Rozsah Priority - Přednost + Přednost Add @@ -27740,26 +29371,26 @@ Remove - Odstranit + Odstranit Add... - Přidat... + Přidat... Edit... - Upravit... + Upravit... Core::VariableChooser Variables - Proměnné + Proměnné Select a variable to insert. - Vyberte proměnnou, která se má vložit. + Vyberte proměnnou, která se má vložit. Insert variable @@ -27793,7 +29424,7 @@ Declarations relative to "public", "protected" and "private" - Prohlášení relativní k "public", + Deklarace relativní k "public", "protected" a "private" @@ -27807,7 +29438,7 @@ Declarations within "namespace" definition - Prohlášení ve vymezení + Deklarace v definici jmenného prostoru @@ -27820,19 +29451,19 @@ Class declarations - Prohlášení tříd + Deklarace tříd Namespace declarations - Prohlášení jmenných prostorů + Deklarace jmenných prostorů Enum declarations - Prohlášení výčtů + Deklarace výčtů Method declarations - Prohlášení metod + Deklarace metod Blocks @@ -27925,7 +29556,7 @@ c; </pre> </body></html> - <html><head/><body> + <html><head/><body> Dodatečné zarovnání obvykle ovlivní pouze podmínky příkazů if. Bez dodatečného zarovnání: <pre> if (a && @@ -27946,6 +29577,40 @@ Podmínky dále odsadit pokud by jinak byly zarovnány na dalším řádku + + <html><head/><body> +Adds an extra level of indentation to multiline conditions in the switch, if, while and foreach statements if they would otherwise have the same or less indentation than a nested statement. + +For four-spaces indentation only if statement conditions are affected. Without extra padding: +<pre> +if (a && + b) + c; +</pre> +With extra padding: +<pre> +if (a && + b) + c; +</pre> +</body></html> + <html><head/><body> +Přidá další úroveň odsazení do podmínek na víc řádků v přepínači, pokud, zatímco a pro každé prohlášení, pokud by měly stejné nebo menší odsazení než vnořené prohlášení. + +Pro čtyřmezerové odsazení, pouze pokud jsou ovlivněny podmínky prohlášení. Bez vycpání mezerami navíc: +<pre> +if (a && + b) + c; +</pre> +S vycpáním mezerami navíc: +<pre> +if (a && + b) + c; +</pre> +</body></html> + Debugger::Internal::BreakCondition @@ -28093,6 +29758,10 @@ Přerušit při přístupu k datům na adrese dané výrazem + Break on QML signal emit + Přerušit při vyslání signálu QML + + Break on QML signal handler Přerušit při manipulátoru signálu QML @@ -28123,38 +29792,38 @@ LldbOptionsPageWidget Enable LLDB - Povolit LLDB + Povolit LLDB Use GDB Python dumpers - Použít pomocné knihovny pro výstup dat Python GDB + Použít pomocné knihovny pro výstup dat Python GDB StartRemoteEngineDialog Start Remote Engine - Spustit vzdálený stroj + Spustit vzdálený stroj &Host: - &Hostitel: + &Hostitel: &Username: - &Uživatelské jméno: + &Uživatelské jméno: &Password: - He&slo: + He&slo: &Engine path: - Cesta ke &stroji: + Cesta ke &stroji: &Inferior path: - &Podřízená cesta: + &Podřízená cesta: @@ -28184,7 +29853,7 @@ Git::Internal::RemoteAdditionDialog Add Remote - Přidat Git Remote + Přidat vzdálené Name: @@ -28199,23 +29868,23 @@ Git::Internal::RemoteDialog Remotes - Git Remotes + Vzdálené (Remotes) Re&fresh - &Obnovit + O&bnovit &Add... - &Přidat... + Přid&at... F&etch - &Natáhnout + &Přivést (fetch) &Remove - &Odstranit + O&dstranit Delete Remote @@ -28225,6 +29894,10 @@ Would you like to delete the remote "%1"? Chcete smazat vzdálené '%1'? + + &Push + &Odvést (push) + Help::Internal::RemoteFilterOptions @@ -28252,24 +29925,28 @@ Double-click to edit item. Dvojité klepnutí pro úpravu položky. + + Edit Filter Configuration + Upravit nastavení filtru + ImageViewer::Internal::ImageViewerToolbar Show background - Ukázat pozadí + Ukázat pozadí Show outline - Ukázat obrys + Ukázat obrys Fit image in the screen - Přizpůsobit obrázek velikosti obrazovky + Přizpůsobit obrázek velikosti obrazovky Original size - Původní velikost + Původní velikost Zoom In @@ -28279,6 +29956,22 @@ Zoom Out Oddálit + + Show Background + Ukázat pozadí + + + Show Outline + Ukázat obrys + + + Fit to Screen + Přizpůsobit obrazovce + + + Original Size + Původní velikost + Macros::Internal::MacroOptionsWidget @@ -28334,11 +30027,11 @@ ProjectExplorer::CodeStyleSettingsPropertiesPage Form - Formulář + Formulář Language: - Jazyk: + Jazyk: @@ -28364,15 +30057,15 @@ ToolChainOptionsPage Add - Přidat + Přidat Clone - Klonovat + Klonovat Remove - Odstranit + Odstranit @@ -28400,19 +30093,19 @@ ComponentNameDialog Dialog - Dialog + Dialog Component name: - Název součástky: + Název součástky: Path: - Cesta: + Cesta: Choose... - Vybrat... + Vybrat... @@ -28427,7 +30120,7 @@ If enabled, the toolbar will remain pinned to an absolute position. - Ukotví nástrojový pruh v pevné absolutní poloze. + Ukotví panel nástrojů v pevné absolutní poloze. Pin Qt Quick Toolbar @@ -28435,11 +30128,11 @@ Always show Qt Quick Toolbar - Ukázat nástrojový pruh Qt Quick vždy + Vždy ukazovat nástrojový pruh Qt Quick Qt Quick ToolBar - Nástrojový pruh Qt Quick + Panel nástrojů Qt Quick @@ -28456,12 +30149,16 @@ Loading data Nahrávají se data + + Application stopped before loading all data + Program zastaven před nahráním všech dat + TimeDisplay length: %1 - Délka: %1 + Délka: %1 @@ -28496,7 +30193,11 @@ &Port: - &Přípojka: + &Port: + + + Sys&root: + Sys&root: @@ -28609,27 +30310,27 @@ S60CertificateDetailsDialog Details of Certificate - Podrobnosti k osvědčení + Podrobnosti k osvědčení S60PublishingBuildSettingsPageOvi Form - Formulář + Formulář Choose a build configuration: - Vybrat nastavení sestavování: + Vybrat nastavení sestavování: Choose a tool chain: - Vybrat řetězec nástrojů: + Vybrat řetězec nástrojů: Only Qt versions above 4.6.3 are made available in this wizard. Previous Qt versions have limitations in building suitable SIS files. - Tento průvodce ukáže jen verze Qt po verzi 4.6.3. + Tento průvodce ukáže jen verze Qt po verzi 4.6.3. Předchozí verze Qt mají omezení v sestavování vhodných souborů SIS. @@ -28637,58 +30338,58 @@ S60PublishingResultsPageOvi Form - Formulář + Formulář S60PublishingSisSettingsPageOvi Form - Formulář + Formulář Localised Vendor Names - Lokalizovaný název prodejce + Lokalizovaný název prodejce Current Global Vendor Name - Nynější jednoznačný název prodejce + Nynější jednoznačný název prodejce Display name: - Název k zobrazení: + Název k zobrazení: Localised vendor names: - Lokalizovaný název prodejce: + Lokalizovaný název prodejce: Capabilities: - Oprávnění: + Oprávnění: Current UID3 - Nynější UID3 + Nynější UID3 Application UID: - UID programu: + UID programu: Current Qt Version - Nynější verze Qt + Nynější verze Qt Qt version used in builds: - Při sestavování používaná verze Qt: + Při sestavování používaná verze Qt: Current set of capabilities - Nynější oprávnění + Nynější oprávnění Global vendor name: - Jednoznačný název prodejce: + Jednoznačný název prodejce: @@ -28714,133 +30415,133 @@ Html5AppWizardSourcesPage WizardPage - Stránka průvodce + Stránka průvodce Main HTML File - Hlavní soubor HTML + Hlavní soubor HTML Generate an index.html file - Vytvořit soubor index.html + Vytvořit soubor index.html Import an existing .html file - Zavést existující soubor .html + Zavést existující soubor .html Load a URL - Nahrát adresu (URL) + Nahrát adresu (URL) http:// - http:// + http:// Note: Unless you chose to load a URL, all files and directories that reside in the same directory as the main HTML file are deployed. You can modify the contents of the directory any time before deploying. - Poznámka: Když nezadáte žádnou adresu (URL), budou nasazeny všechny soubory, které se nachází v tomtéž adresáři jako hlavní soubor HTML. Obsah adresáře lze před nasazením kdykoli upravit. + Poznámka: Když nezadáte žádnou adresu (URL), budou nasazeny všechny soubory, které se nachází v tomtéž adresáři jako hlavní soubor HTML. Obsah adresáře lze před nasazením kdykoli upravit. Touch optimized navigation - Pro dotykové ovládání optimalizovaná navigace + Pro dotykové ovládání optimalizovaná navigace Enable touch optimized navigation - Zapnout pro dotykové ovládání optimalizovanou navigaci + Zapnout pro dotykové ovládání optimalizovanou navigaci Touch optimized navigation will make the HTML page flickable and enlarge the area of touch sensitive elements. If you use a JavaScript framework which optimizes the touch interaction, leave the checkbox unchecked. - Pro dotykové ovládání optimalizované navádění způsobí, že stránku HTML lze obsluhovat pomocí udělání rychlého pohybu (Flick) a zvětší oblast prvků citlivých na dotyk. Nechte nastavení vypnuto, pokud již používáte pro dotykové ovládání vyladěnou soustavu JavaScript. + Pro dotykové ovládání optimalizované navádění způsobí, že stránku HTML lze obsluhovat pomocí udělání rychlého pohybu (Flick) a zvětší oblast prvků citlivých na dotyk. Nechte nastavení vypnuto, pokud již používáte pro dotykové ovládání vyladěnou soustavu JavaScript. MobileAppWizardGenericOptionsPage WizardPage - Stránka průvodce + Stránka průvodce Orientation behavior: - Chování orientace: + Chování orientace: MobileAppWizardHarmattanOptionsPage WizardPage - Stránka průvodce + Stránka průvodce Application icon (80x80): - Ikona programu (80x80): + Ikona programu (80x80): Generate code to speed up the launching on the device. - Vytvořit kód pro zrychlení spouštění na zařízení. + Vytvořit kód pro zrychlení spouštění na zařízení. Make application boostable - Udělat program schopný vzpruhy + Udělat program schopný vzpruhy MobileAppWizardMaemoOptionsPage WizardPage - Stránka průvodce + Stránka průvodce Application icon (64x64): - Ikona programu (64x64): + Ikona programu (64x64): MobileAppWizardSymbianOptionsPage WizardPage - Stránka průvodce + Stránka průvodce Application icon (.svg): - Ikona programu (.svg): + Ikona programu (.svg): Target UID3: - Cílové UID3: + Cílové UID3: Enable network access - Povolit přístup k síti + Povolit přístup k síti MobileLibraryWizardOptionPage WizardPage - Stránka průvodce + Stránka průvodce Target UID3: - Cílové UID3: + Cílové UID3: Plugin's directory name: - Název adresáře přídavného modulu: + Název adresáře přídavného modulu: Enable network access - Povolit přístup k síti + Povolit přístup k síti QtQuickComponentSetOptionsPage Built-in elements only (for all platforms) - Jen vestavěné prvky (pro všechny platformy) + Jen vestavěné prvky (pro všechny platformy) Qt Quick Components for Symbian - Součástky Qt Quick pro Symbian + Součástky Qt Quick pro Symbian Qt Quick Components for Meego/Harmattan @@ -28848,13 +30549,13 @@ Use an existing .qml file - Použít existující soubor .qml + Použít existující soubor .qml The built-in elements in the QtQuick namespace allow you to write cross-platform applications with a custom look and feel. Requires Qt 4.7.1 or newer. - Vestavěné prvky ve jmenném prostoru QtQuick dovolují psát víceplatformní programy s uživatelsky stanoveným vzhledem. + Vestavěné prvky ve jmenném prostoru QtQuick dovolují psát víceplatformní programy s uživatelsky stanoveným vzhledem. Vyžaduje Qt 4.7.1 nebo novější. @@ -28876,17 +30577,17 @@ All files and directories that reside in the same directory as the main QML file are deployed. You can modify the contents of the directory any time before deploying. - Všechny soubory a adresáře, jež se nacházejí v tomtéž adresáři jako hlavní soubor QML, jsou připraveny k nasazení. Obsah adresáře lze před nasazením kdykoli upravit. + Všechny soubory a adresáře, jež se nacházejí v tomtéž adresáři jako hlavní soubor QML, jsou připraveny k nasazení. Obsah adresáře lze před nasazením kdykoli upravit. Qt Quick Components for MeeGo/Harmattan - Součástky Qt Quick pro Meego/Harmattan + Součástky Qt Quick pro Meego/Harmattan The Qt Quick Components for Symbian are a set of ready-made components that are designed with specific native appearance for the Symbian platform. Requires Qt 4.7.4 or newer, and the component set installed for your Qt version. - Součástky Qt Quick pro Symbian jsou souborem předpřipravených součástek, který byly navrženy tak, aby vyjádřily pro Symbian zvláštní vzhled. + Součástky Qt Quick pro Symbian jsou souborem předpřipravených součástek, který byly navrženy tak, aby vyjádřily pro Symbian zvláštní vzhled. Vyžaduje Qt 4.7.4 nebo novější a soubor součástek nainstalovanou pro tutoi verzi Qt. @@ -28894,7 +30595,7 @@ The Qt Quick Components for MeeGo/Harmattan are a set of ready-made components that are designed with specific native appearance for the MeeGo/Harmattan platform. Requires Qt 4.7.4 or newer, and the component set installed for your Qt version. - Součástky Qt Quick pro Meego/Harmattan jsou souborem předpřipravených součástek, který byly navrženy tak, aby vyjádřily pro Meego/Harmattan zvláštní vzhled. + Součástky Qt Quick pro Meego/Harmattan jsou souborem předpřipravených součástek, který byly navrženy tak, aby vyjádřily pro Meego/Harmattan zvláštní vzhled. Vyžaduje Qt 4.7.4 nebo novější, a soubor součástek nainstalovaný pro tuto verzi Qt. @@ -28935,7 +30636,7 @@ Tool Chain: - Řetězec nástrojů: + Řetězec nástrojů: Show compiler output of last build. @@ -28943,7 +30644,7 @@ Show Log - Ukázat zápis + Ukázat záznam Compile debugging helpers that are checked. @@ -28953,6 +30654,10 @@ Build All Sestavit vše + + Tool chain: + Sada nástrojů: + QtSupport::Internal::QtVersionInfo @@ -29000,57 +30705,57 @@ GenericLinuxDeviceConfigurationWizardSetupPage WizardPage - Stránka průvodce + Stránka průvodce The name to identify this configuration: - Název nastavení: + Název nastavení: The device's host name or IP address: - Název hostitelského počítače nebo IP adresa zařízení: + Název hostitelského počítače nebo IP adresa zařízení: The user name to log into the device: - Uživatelské jméno pro přihlášení se k zařízení: + Uživatelské jméno pro přihlášení se k zařízení: The authentication type: - Druh ověření pravosti: + Druh ověření pravosti: Password - Heslo + Heslo Key - Klíč + Klíč The user's password: - Heslo uživatele: + Heslo uživatele: The file containing the user's private key: - Soubor se soukromým klíčem uživatele: + Soubor se soukromým klíčem uživatele: LinuxDeviceFactorySelectionDialog Device Configuration Wizard Selection - Zřízení nového nastavení zařízení + Zřízení nového nastavení zařízení Available device types: - Dostupné typy zařízení: + Dostupné typy zařízení: MaemoDeployConfigurationWidget Form - Formulář + Formulář Device configuration: @@ -29074,11 +30779,11 @@ Add Desktop File - Přidat soubor pro desktop + Přidat soubor pro desktop Add Launcher Icon... - Přidat ikonu spouštěče... + Přidat ikonu spouštěče... @@ -29192,26 +30897,26 @@ MaemoDeviceConfigWizardKeyCreationPage WizardPage - Stránka průvodce + Stránka průvodce Qt Creator will now generate a new pair of keys. Please enter the directory to save the key files in and then press "Create Keys". - Qt Creator nyní vytvoří dvojici klíčů. Zadejte, prosím, adresář, do kterého se klíče mají uložit a potom potvrďte "Vytvořit klíče". + Qt Creator nyní vytvoří dvojici klíčů. Zadejte, prosím, adresář, do kterého se klíče mají uložit a potom potvrďte "Vytvořit klíče". Directory: - Adresář: + Adresář: Create Keys - Vytvořit klíče + Vytvořit klíče MaemoDeviceConfigWizardKeyDeploymentPage WizardPage - Stránka průvodce + Stránka průvodce To deploy the public key to your device, please execute the following steps: @@ -29222,7 +30927,7 @@ <li>In "%%%maddev%%%", press "Developer Password" and enter it in the field below.</li> <li>Click "Deploy Key"</li> - Abyste klíč poslal do mobilního zařízení, proveďte, prosím, následující kroky: + Abyste klíč poslal do mobilního zařízení, proveďte, prosím, následující kroky: <ul> <li>Spojte mobilní zařízení s počítačem (pokud se nepoužívá žádné WLAN).</li> <li>Spusťte program "%%%maddev%%%" na mobilním zařízení.</li> @@ -29233,41 +30938,41 @@ Device address: - Adresa zařízení: + Adresa zařízení: Password: - Heslo: + Heslo: Deploy Key - Poslat klíč + Poslat klíč MaemoDeviceConfigWizardCheckPreviousKeySetupPage WizardPage - Stránka průvodce + Stránka průvodce Has a passwordless (key-based) login already been set up for this device? - Bylo již pro toto mobilní zařízení nastaveno přihlášení bez hesla? + Bylo již pro toto mobilní zařízení nastaveno přihlášení bez hesla? Yes, and the private key is located at - Ano, a soukromý klíč se nachází v + Ano, a soukromý klíč se nachází v No - Ne + Ne MaemoDeviceConfigWizardReuseKeysCheckPage WizardPage - Stránka průvodce + Stránka průvodce Do wou want to re-use an existing pair of keys or should a new one be created? @@ -29275,58 +30980,58 @@ Re-use existing keys - Použít existující klíče znovu + Použít existující klíče znovu File containing the public key: - Soubor obsahující veřejný klíč: + Soubor obsahující veřejný klíč: File containing the private key: - Soubor obsahující soukromý klíč: + Soubor obsahující soukromý klíč: Create new keys - Vytvořit nové klíče + Vytvořit nové klíče Do you want to re-use an existing pair of keys or should a new one be created? - Chcete znovu použít existující dvojici klíčů, nebo se má vytvořit nová? + Chcete znovu použít existující dvojici klíčů, nebo se má vytvořit nová? MaemoDeviceConfigWizardStartPage WizardPage - Stránka průvodce + Stránka průvodce The name to identify this configuration: - Název nastavení: + Název nastavení: The system running on the device: - Na mobilním zařízení běžící systém: + Na mobilním zařízení běžící systém: The kind of device: - Typ zařízení: + Typ zařízení: Emulator - Emulátor + Emulátor Hardware Device - Mobilní zařízení + Mobilní zařízení The device's host name or IP address: - Název hostitele nebo IP adresa zařízení: + Název hostitele nebo IP adresa zařízení: The SSH server port: - Přípojka SSH serveru: + Přípojka SSH serveru: @@ -29352,27 +31057,27 @@ MaemoPublishingWizardPageFremantleFree WizardPage - Stránka průvodce + Stránka průvodce Choose build configuration: - Vybrat nastavení sestavování: + Vybrat nastavení sestavování: Only create source package, do not upload - Pouze vytvořit zdrojový balíček, nenahrávat + Pouze vytvořit zdrojový balíček, nenahrávat MaemoPublishingFileSelectionDialog Choose Package Contents - Vybrat obsah balíčku + Vybrat obsah balíčku <b>Please select the files you want to be included in the source tarball.</b> - <b>Vyberte, prosím, soubory, které mají být zahrnuty do tarballu se zdroji.</b> + <b>Vyberte, prosím, soubory, které mají být zahrnuty do tarballu se zdroji.</b> @@ -29380,69 +31085,69 @@ MaemoPublishingResultPageFremantleFree WizardPage - Stránka průvodce + Stránka průvodce Progress - Průběh + Průběh MaemoPublishingUploadSettingsPageFremantleFree WizardPage - Stránka průvodce + Stránka průvodce Upload Settings - Nahrát nastavení + Nahrát nastavení Garage account name: - Název účtu Garage: + Název účtu Garage: <a href="https://garage.maemo.org/account/register.php">Get an account</a> - <a href="https://garage.maemo.org/account/register.php">Vytvořit účet</a> + <a href="https://garage.maemo.org/account/register.php">Vytvořit účet</a> <a href="https://garage.maemo.org/extras-assistant/index.php">Request upload rights</a> - <a href="https://garage.maemo.org/account/register.php">Požádat o práva k nahrání</a> + <a href="https://garage.maemo.org/account/register.php">Požádat o práva k nahrání</a> Private key file: - Soubor se soukromým klíčem: + Soubor se soukromým klíčem: Server address: - Adresa serveru: + Adresa serveru: Target directory on server: - Cílový adresář na serveru: + Cílový adresář na serveru: MaemoQemuSettingsWidget Form - Formulář + Formulář OpenGL Mode - Režim OpenGL + Režim OpenGL &Hardware acceleration - &Hardwarové zrychlení + &Hardwarové zrychlení &Software rendering - &Softwarové udělání + &Softwarové udělání &Auto-detect - Určit &automaticky + Určit &automaticky @@ -29468,58 +31173,58 @@ BehaviorSettingsPage Form - Formulář + Formulář BehaviorSettingsWidget Cleanup actions which are automatically performed right before the file is saved to disk. - Opravy, které se automaticky provádějí bezprostředně před uložením souboru na disk. + Opravy, které se automaticky provádějí bezprostředně před uložením souboru na disk. Cleanups Upon Saving - Opravy při uložení + Opravy při uložení Removes trailing whitespace upon saving. - Odstraní při ukládání prázdné znaky na konci řádků. + Odstraní při ukládání prázdné znaky na konci řádků. &Clean whitespace - &Vyčistit prázdné znaky + &Vyčistit prázdné znaky Clean whitespace in entire document instead of only for changed parts. - Vyčistí prázdné znaky v celém dokumentu a nejen ve změněných částech. + Vyčistí prázdné znaky v celém dokumentu a nejen ve změněných částech. In entire &document - V celém &dokumentu + V celém &dokumentu Correct leading whitespace according to tab settings. - Opraví prázdné znaky na začátku řádků podle nastavení zarážek. + Opraví prázdné znaky na začátku řádků podle nastavení zarážek. Clean indentation - Opravit odsazení + Opravit odsazení &Ensure newline at end of file - &Doplnit nový řádek na konci souboru + &Doplnit nový řádek na konci souboru File Encodings - Kódování souborů + Kódování souborů Default encoding: - Výchozí kódování: + Výchozí kódování: UTF-8 BOM: - UTF-8 BOM: + UTF-8 BOM: <html><head/><body> @@ -29529,7 +31234,7 @@ <li><i>Always Delete:</i> never write an UTF-8 BOM, possibly deleting a pre-existing one.</li></ul> <p>Note that UTF-8 BOMs are uncommon and treated incorrectly by some editors, so it usually makes little sense to add any.</p> <p>This setting does <b>not</b> influence the use of UTF-16 and UTF-32 BOMs.</p></body></html> - <html><head/><body> + <html><head/><body> <p>Udává, jak se chovají textové editory, co se týče UTF-8 BOM (Byte Order Marks). Na výběr je:</p> <ul ><li><i>Přidat, je-li kódování UTF-8</i> Vždy přidat. BOM při ukládání souboru s kódováním UTF-8. Toto však nebude fungovat, je-li kódování <i>System</i>, neboť Qt Creator v tomto případě kódování nedokáže určit.</li> <li><i>Zachovat, je-li už přítomno: </i>Uložit soubor s BOM, pokud jej mělo již při nahrání, což bylo zjištěno.</li> @@ -29539,39 +31244,39 @@ Add If Encoding Is UTF-8 - Přidat, pokud je kódování UTF-8 + Přidat, pokud je kódování UTF-8 Keep If Already Present - Zachovat, pokud je už přítomno + Zachovat, pokud je už přítomno Always Delete - Smazat vždy + Smazat vždy Mouse - Myš + Myš Enable &mouse navigation - Povolit navádění &myší + Povolit navádění &myší Enable scroll &wheel zooming - Povolit přibližování a oddalování pomocí &kolečka myši + Povolit přibližování a oddalování pomocí &kolečka myši Typing - Při psaní + Při psaní Enable automatic &indentation - Povolit automatické &odsazení + Povolit automatické &odsazení Backspace indentation: - Odsazení při zpětné klávese (Backspace): + Odsazení při zpětné klávese (Backspace): <html><head/><body> @@ -29588,7 +31293,7 @@ </li> </ul></body></html> - <html><head/><body> + <html><head/><body> Určuje, jak se chová zpětná klávesa (backspace) co se týče odsazování. <ul> @@ -29605,134 +31310,134 @@ None - Žádné + Žádné Follows Previous Indents - Sledovat předchozí odsazení + Sledovat předchozí odsazení Unindents - Zrušit odsazení + Zrušit odsazení Tab key performs auto-indent: - Klávesa pro zarážku provede automatické odsazení: + Klávesa pro zarážku provede automatické odsazení: Never - Nikdy + Nikdy Always - Vždy + Vždy In Leading White Space - Pouze v prázdném místu na začátku řádku + Pouze v prázdném místu na začátku řádku Always write a newline character at the end of the file. - Vždy psát znak pro nový řádek na konci souboru. + Vždy psát znak pro nový řádek na konci souboru. Enable &tooltips only when Shift key is down - Ukazovat &nástrojové rady jen při stisknuté klávese Shift + Ukazovat &vysvětlivky jen při stisknuté klávese Shift HighlighterSettingsPage Form - Formulář + Formulář <html><head/><body> <p>Highlight definitions are provided by the <a href="http://kate-editor.org/">Kate Text Editor</a>.</p></body></html> - <html><head/><body> + <html><head/><body> <p>Soubory s vymezeními pro zvýrazňování skladby jsou poskytovány <a href="http://kate-editor.org/">textovým editorem Kate</a>.</p></body></html> Syntax Highlight Definition Files - Soubory s vymezením pro zvýraznění skladby + Soubory s vymezením pro zvýraznění skladby Location: - Umístění: + Umístění: Use fallback location - Použít záložní umístění + Použít záložní umístění Behavior - Chování + Chování Alert when a highlight definition is not found - Zobrazit upozornění, pokud se nepodařilo najít žádný soubor s vymezením + Zobrazit upozornění, pokud se nepodařilo najít žádný soubor s vymezením Ignored file patterns: - Vyloučené soubory (vzor hledání): + Vyloučené soubory (vzor hledání): ManageDefinitionsDialog Dialog - Dialog + Dialog Definitions - Soubory s vymezeními + Soubory s vymezeními Select All - Vybrat vše + Vybrat vše Clear Selection - Smazat výběr + Smazat výběr Invert Selection - Obrátit výběr + Obrátit výběr Download Selected Definitions - Stáhnout vybrané soubory s vymezeními + Stáhnout vybrané soubory s vymezeními SnippetsSettingsPage Form - Formulář + Formulář Group: - Skupina: + Skupina: Add - Přidat + Přidat Remove - Odstranit + Odstranit Revert Built-in - Vrátit zpět vnitřní kousek + Vrátit zpět vnitřní kousek Restore Removed Built-ins - Obnovit všechny vnitřní kousky + Obnovit všechny vnitřní kousky Reset All - Nastavit znovu vše + Nastavit znovu vše @@ -29750,11 +31455,11 @@ TextEditor::TabSettingsWidget Form - Formulář + Formulář Tabs And Indentation - Zarážky a odsazení + Zarážky a odsazení Insert &spaces instead of tabs @@ -29770,11 +31475,11 @@ Ta&b size: - Šířka &zarážky: + Šířka &zarážky: &Indent size: - &Velikost odsazení: + &Velikost odsazení: Enable automatic &indentation @@ -29790,7 +31495,7 @@ Align continuation lines: - Zarovnání navazujících řádků: + Zarovnání navazujících řádků: <html><head/><body> @@ -29818,7 +31523,7 @@ </pre> </li> </ul></body></html> - <html><head/><body> + <html><head/><body> Určuje chování odsazení se zřetelem k navazujícím řádkům. <ul> @@ -29846,15 +31551,15 @@ Not At All - Žádné odsazení + Žádné odsazení With Spaces - Prázdné znaky + Prázdné znaky With Regular Indent - Normální odsazení + Normální odsazení Tab key performs auto-indent: @@ -29874,26 +31579,26 @@ Tab policy: - Chování zarážek: + Chování zarážek: Spaces Only - Pouze mezery + Pouze mezery Tabs Only - Pouze zarážky + Pouze zarážky Mixed - Smíchaně + Smíchaně Valgrind::Internal::SuppressionDialog Dialog - Dialog + Dialog Suppression File: @@ -29924,11 +31629,11 @@ Memory Analysis Options - Nastavení pro rozbor paměti + Volby pro rozbor paměti Backtrace frame count: - Počet snímků zpětné stopy (Backtrace): + Počet položek ve výpisu volání (backtrace): Suppression files: @@ -29952,11 +31657,11 @@ Limits the amount of results the profiler gives you. A lower limit will likely increase performance. - Toto nastavení omezí počet výsledků profileru. Nižší hodnoty zvyšují rychlost (výkon). + Toto nastavení omezí počet výsledků profileru. Nižší hodnoty mohou urychlit běh. Result view: Minimum event cost: - Nejmenší událostní náklady na zobrazení výsledků: + U výsledků: Nejmenší událostní náklady: % @@ -29964,7 +31669,7 @@ Show additional information for events in tooltips - Ukázat dodatečné informace k událostem v nástrojových radách + Ukázat další informace k událostem ve vysvětlivkách <html><head/><body> @@ -29979,19 +31684,19 @@ </body></html> <html><head/><body> -<p>Udává, zda má nastat plná simulace vyrovnávací paměti.</p> -<p>V nastavení předlohy se počítají jen přístupy ke čtení pro příkazy ("Ir").</p> +<p>Udává, zda má nastat plná simulace mezipaměti.</p> +<p>V nastavení předlohy se počítají jen přístupy ke čtení pro instrukce ("Ir").</p> <p> -Při plné simulaci vyrovnávací paměti budou zapnuta další počítadla událostí: -<ul><li>Přístupy k chybám vyrovnávací paměti při čtení příkazů ("I1mr"/"I2mr")</li> -<li>Přístupy ke čtení dat ("Dr") a toho se týkající přístupy k chybám ("D1mr"/"D2mr")</li> -<li>Přístupy k zápisu dat ("Dw") a toho se týkající přístupy k chybám ("D1mw"/"D2mw")</li></ul> +Při plné simulaci mezipaměti budou zapnuta další počítadla událostí: +<ul><li>Minutí mezipaměti při čtení instrukcí ("I1mr"/"I2mr")</li> +<li>Přístupy ke čtení dat ("Dr") a příslušná minutí ("D1mr"/"D2mr")</li> +<li>Přístupy k zápisu dat ("Dw") a příslušná minutí ("D1mw"/"D2mw")</li></ul> </p> </p></body></html> Enable cache simulation - Zapnout simulaci vyrovnávací paměti + Zapnout simulaci mezipaměti <html><head/><body> @@ -30004,22 +31709,22 @@ <html><head/><body> <p>Udává, zda je zapnuta předpověď větvení.</p> <p>Dodatečná počítadla událostí: </p> -<ul><li>Počet provedených podmíněných větví a toho se týkajících chybných předpovědí ( +<ul><li>Počet provedených podmíněných větví a příslušných chybných předpovědí ( "Bc"/"Bcm")</li> <li>Provedené nepřímé skoky a související chybné předpovědi adresy skoku ( "Bi"/"Bim")</li></ul></body></html> Enable branch prediction simulation - Zapnout simulaci předpovídání větvení + Zapnout simulaci předpovědi větvení Collect information for system call times. - Sbírat informace o čase stráveném systémovými voláními. + Sbírat informace o době běhu systémových volání. Collect system call time - Sbírat informace o čase stráveném systémovými voláními + Měřit dobu běhu systémových volání Collect the number of global bus events that are executed. The event type "Ge" is used for these events. @@ -30038,6 +31743,10 @@ Příkaz Valgrind + Valgrind Suppression Files + Soubory vyloučení pro Valgrind + + Valgrind Suppression File (*.supp);;All Files (*) Soubor vyloučení Valgrind (*.supp);;Všechny soubory (*) @@ -30058,7 +31767,7 @@ Configure... - Nastavení... + Nastavení... @@ -30460,15 +32169,15 @@ This property holds whether the item accepts mouse events. - tato vlastnost stanovuje, zda prvek přijímá událost myši. + Tato vlastnost stanovuje, zda prvek přijímá událost myši. Hover Enabled - Hover povoleno + Přejezd (hover) povolen This property holds whether hover events are handled. - tato vlastnost stanovuje, zda prvek přijímá událost Hover. + Tato vlastnost stanovuje, zda prvek přijímá událost přejezdu. @@ -30479,11 +32188,11 @@ Drag margin - Okraj při operaci táhnutí: + Okraj při operaci táhnutí Flick deceleration - Zpomalení udělání rychlého pohybu + Zpomalení při trhnutí Follows current @@ -30491,7 +32200,7 @@ A user cannot drag or flick a PathView that is not interactive. - Při neinteraktivním PathView uživatel nemůže provést žádnou operaci tažení nebo udělání rychlého pohybu. + Při neinteraktivním PathView uživatel nemůže provést žádnou operaci tažení nebo trhnutí. Offset @@ -30507,7 +32216,7 @@ pathItemCount: number of items visible on the path at any one time. - pathItemCount: Počet k danému času viditelných prvků podél cesty. + pathItemCount: Počet najednou viditelných prvků podél cesty. Path View Highlight @@ -30580,7 +32289,7 @@ ExampleDelegate Tags: - Klíčová slova: + Klíčová slova: @@ -30591,59 +32300,59 @@ Latest News - Poslední novinky + Poslední novinky Feedback Feedback - Zpětná vazba + Vaše reakce Help us make Qt Creator even better - Pomozte nám vylepšit Qt Creator + Pomozte nám vylepšit Qt Creator Search in Tutorials, Examples and Demos - Hledat v návodech, příkladech a demech + Hledat v návodech, příkladech a demech Open Project... - Otevřít projekt... + Otevřít projekt... Create Project... - Vytvořit projekt... + Vytvořit projekt... NewsListing Click to read more... - Zde klepněte pro další informace... + Zde klepněte pro další informace... RecentProjects Recently Edited Projects - Naposledy upravované projekty + Naposledy upravované projekty RecentSessions Recently Used Sessions - Naposledy používaná sezení + Naposledy používaná sezení %1 (last session) - %1 (poslední sezení) + %1 (poslední sezení) %1 (current session) - %1 (nynější sezení) + %1 (nynější sezení) @@ -30657,11 +32366,11 @@ QmlEditorWidgets::ContextPaneWidget Hides this toolbar. - Skryje tento nástrojový pruh. + Skryje tento panel nástrojů. Pin Toolbar - Ukotvit nástrojový pruh + Ukotvit panel nástrojů Show Always @@ -30669,11 +32378,11 @@ Unpins the toolbar and moves it to the default position. - Uvolní ukotvení a nástrojový pruh přesune do výchozího umístění. + Uvolní ukotvení a panel nástrojů přesune do výchozího umístění. Hides this toolbar. This toolbar can be permanently disabled in the options page or in the context menu. - Skryje tento nástrojový pruh. Tento nástrojový pruh může být v nastavení nebo v kontextové nabídce trvale zakázán. + Skryje tento panel nástrojů. Tento panel může být v nastavení nebo v kontextové nabídce trvale zakázán. @@ -30773,19 +32482,19 @@ Utils::DebuggerLanguageChooser C++ - C++ + C++ QML - QML + QML Debug port: - Přípojka pro ladič: + Přípojka pro ladič: <a href="qthelp://com.nokia.qtcreator/doc/creator-debugging-qml.html">What are the prerequisites?</a> - <a href="qthelp://com.nokia.qtcreator/doc/creator-debugging-qml.html">Jaké jsou předpoklady?</a> + <a href="qthelp://com.nokia.qtcreator/doc/creator-debugging-qml.html">Jaké jsou předpoklady?</a> @@ -30957,8 +32666,12 @@ Ke vzdálenému souboru nelze připojit žádná data, neboť server nepodporuje vlastnost velikost souboru. + Server could not start session: %1 + Serveru se nepodařilo spustit sezení: %1 + + Server could not start session. - Serveru se nepodařilo spustit žádné sezení. + Serveru se nepodařilo spustit žádné sezení. Error reading local file: %1 @@ -31064,7 +32777,7 @@ The command '%1' did not respond within the timeout limit (%2 ms). - Příkaz '%1' neodpověděl během časového omezení pro překročení času (%2 ms). + Příkaz '%1' neodpověděl do vypršení časového omezení (%2 ms). Process not Responding @@ -31191,11 +32904,11 @@ Analyzer::AnalyzerRunConfigWidget Analyzer Settings - Nastavení rozboru + Nastavení rozboru Available settings: %1 - Dostupná nastavení: %1 + Dostupná nastavení: %1 @@ -31209,7 +32922,7 @@ Analyzer::AnalyzerProjectSettings Analyzer Settings - Nastavení rozboru + Nastavení rozboru @@ -31241,11 +32954,11 @@ Bazaar::Internal::BazaarEditor Annotate %1 - Opatřit vysvětlivkami %1 + Opatřit anotacemi %1 Annotate parent revision %1 - Opatření vysvětlivkou nadřazené revize "%1" + Opatření anotacemi nadřazenou revizi "%1" @@ -31256,19 +32969,19 @@ Annotate Current File - Opatřit nynější soubor vysvětlivkami + Opatřit nynější soubor anotacemi Annotate "%1" - Opatřit vysvětlivkami "%1" + Opatřit anotacemi "%1" Diff Current File - Rozdíly (diff) nynějšího souboru + Rozdíly nynějšího souboru Diff "%1" - Rozdíly (diff) pro "%1" + Rozdíly pro "%1" ALT+Z,Alt+D @@ -31276,11 +32989,11 @@ Log Current File - Zápis pro nynější soubor + Záznamy pro nynější soubor Log "%1" - Zápis pro "%1" + Záznamy pro "%1" ALT+Z,Alt+L @@ -31324,11 +33037,11 @@ Diff - Rozdíly (diff) + Rozdíly Log - Zápis + Záznamy Revert... @@ -31340,19 +33053,19 @@ Pull... - Vytáhnout (pull)... + Přivést (pull)... Push... - Zatlačit (push)... + Odvést (push)... Update... - Obnovit... + Načíst-aktualizovat (update)... Commit... - Odeslat... + Zapsat-odevzdat (commit)... ALT+Z,Alt+C @@ -31364,15 +33077,15 @@ Update - Obnovit + Načíst-aktualizovat Commit - Odeslat + Zapsat-odevzdat (commit) Diff &Selected Files - Rozdíly (diff) pro &vybrané soubory + Rozdíly pro &vybrané soubory &Undo @@ -31384,31 +33097,31 @@ There are no changes to commit. - Nejsou zde žádné změny pro odeslání. + Nejsou zde žádné změny pro zápis. Unable to generate a temporary file for the commit editor. - Nepodařilo se vytvořit žádný dočasný soubor pro editor pro odeslání. + Nepodařilo se vytvořit žádný dočasný soubor pro editor pro zápis. Unable to create an editor for the commit. - Nepodařilo se vytvořit žádný editor pro odeslání. + Nepodařilo se vytvořit žádný editor pro zápis. Unable to create a commit editor. - Nepodařilo se vytvořit žádný editor pro odeslání. + Nepodařilo se vytvořit žádný editor pro zápis. Commit changes for "%1". - Odeslat změny pro "%1". + Zapsat změny pro "%1". Close Commit Editor - Zavřít editor pro odeslání + Zavřít editor pro zápis Do you want to commit the changes? - Chcete odeslat změny? + Chcete zapsat změny? Message check failed. Do you want to proceed? @@ -31434,18 +33147,18 @@ Specify repository URL, clone directory and path. - Zadejte adresu skladiště (URL) a cestu. + Zadejte adresu skladiště (URL), cestu a adresář pro umístění klonu. Clone URL: - Adresa (URL) klonu: + Klonovat z adresy (URL): Bazaar::Internal::CommitEditor Commit Editor - Editor odeslání + Editor zápisu (commit) @@ -31459,11 +33172,11 @@ BinEditorFile Cannot open %1: %2 - Nelze otevřít soubor '%1': %2 + Nelze otevřít soubor '%1': %2 File Error - Chyba souboru + Chyba souboru @@ -31515,10 +33228,11 @@ /c echo Useful text - /c echo užitečný text + /c echo užitečný text Useful text + Sample external tool text Užitečný text @@ -31548,8 +33262,20 @@ Jít vpřed + Split + Rozdělit + + + Split Side by Side + Rozdělit jedno vedle druhého + + + Close Document + Zavřít dokument + + Close - Zavřít + Zavřít Next Open Document in History @@ -31600,7 +33326,7 @@ Core::IFile File was restored from auto-saved copy. Use <i>Save</i> to confirm, or <i>Revert to Saved</i> to discard changes. - Soubor byl obnoven z automaticky uložené záložní kopie. Vyberte <i>Uložit</i> pro potvrzení nebo <i>Obnovit</i> pro zahození změn. + Soubor byl obnoven z automaticky uložené záložní kopie. Vyberte <i>Uložit</i> pro potvrzení nebo <i>Obnovit</i> pro zahození změn. @@ -31617,6 +33343,10 @@ Kouzelné záhlaví + Add Magic Header + Přidat kouzelné záhlaví + + Error Chyba @@ -31624,36 +33354,76 @@ Not a valid byte pattern. Není platným byte vzorem hledání. - - - Core::Internal::MimeTypeSettingsModel - MIME Type - Typ MIME + Dialog + Dialog - Handler - Editor + Value: + Hodnota: - Undefined - Nestanovený + Type + Typ - Invalid MIME Type - Neplatný typ MIME + String + Řetězec - Conflicting pattern(s) will be discarded. - Střetávající se vzory budou odstraněny. + Byte + Byte - - %n pattern(s) already in use. - - Jeden vzor hledání se již používá. - %n vzory hledání se již používají. - %n vzorů hledání se již používá. - + + Use Recommended + Použít doporučené + + + Start range: + Začátek oblasti: + + + End range: + Konec oblasti: + + + Priority: + Přednost: + + + <i>Note: Wide range values might impact Qt Creator's performance when opening files.</i> + <i>Poznámka: Velké oblasti se mohou na rychlosti otevírání souborů odrazit záporně.</i> + + + + Core::Internal::MimeTypeSettingsModel + + MIME Type + Typ MIME + + + Handler + Editor + + + Undefined + Nestanovený + + + Invalid MIME Type + Neplatný typ MIME + + + Conflicting pattern(s) will be discarded. + Střetávající se vzory budou odstraněny. + + + %n pattern(s) already in use. + + Jeden vzor hledání se již používá. + %n vzory hledání se již používají. + %n vzorů hledání se již používá. + @@ -31742,7 +33512,7 @@ <Unknown> Unknown user of paste. - <Neznámý> + <Neznámý> @@ -31782,14 +33552,14 @@ CppEditor::InsertDeclOperation Add %1 Declaration - Přidat prohlášení pro %1 + Přidat deklaraci pro %1 CppEditor::InsertDefOperation Add Definition in %1 - Přidat vymezení v %1 + Přidat definici v %1 @@ -31899,13 +33669,13 @@ Declarations - Prohlášení + Deklarace Scope: %1 Types: %2 Flags: %3 - Oblast: %1 + Obor: %1 Typy: %2 Příznaky: %3 @@ -31942,7 +33712,7 @@ Declarations - Prohlášení + Deklarace Projects only @@ -32023,7 +33793,7 @@ Trace point %1 (%2) in thread %3 triggered. - Stopovací bod %1 (%2) ve vlákně %3 spuštěn. + Trasovací bod %1 (%2) ve vlákně %3 spuštěn. Conditional breakpoint %1 (%2) in thread %3 triggered, examining expression '%4'. @@ -32085,18 +33855,18 @@ Debugger::Internal::Console Clear Contents - Smazat obsah + Smazat obsah Save Contents - Uložit obsah + Uložit obsah Debugger::Internal::ConsoleWindow Console - Konzole + Konzole @@ -32121,6 +33891,26 @@ Spouští se + Setup failed. + Inicializace se nezdařila. + + + Loading finished. + Nahrávání dokončeno. + + + Run failed. + Spuštění se nezdařilo. + + + Running. + Běží. + + + Run requested... + Požadováno spuštění... + + Taking notice of pid %1 Oznámeno PID %1 @@ -32253,8 +34043,12 @@ Volba příkazového řádku %1 vyžaduje parametr. + Only one executable allowed! + Povolen pouze jeden spustitelný soubor! + + The parameter '%1' of option '%2' does not match the pattern <server:port>@<executable>@<architecture>. - Parametr '%1' volby '%2' neodpovídá vzoru <Server:Číslo přípojky>@<Spustitelný soubor>@<Architektura>. + Parametr '%1' volby '%2' neodpovídá vzoru <Server:Číslo přípojky>@<Spustitelný soubor>@<Architektura>. The parameter '%1' of option '%2' does not match the pattern <handle>:<pid>. @@ -32296,22 +34090,30 @@ Warning Varování + + Install &Debug Information + Instalace informací o &ladění + + + This tries to install missing debug information. + Toto se pokusí o instalaci chybějících informací o ladění. + Debugger::Internal::DebuggerPluginPrivate 0x%1 hit Message tracepoint: Address hit. - Adresa 0x%1 nález + Nález na 0x%1 %1:%2 %3() hit Message tracepoint: %1 file, %2 line %3 function hit. - %1:%2 %3() nález + Nález v %1:%2 %3() Add Message Tracepoint - Přidat stopovací bod se zprávou + Přidat trasovací bod se zprávou Message: @@ -32378,6 +34180,18 @@ Ladič připojen k %1 + gdbserver is now listening at %1 + gdbserver nyní naslouchá na %1 + + + Cannot find local executable for remote process "%1". + Nelze najít místní spustitelný soubor pro vzdálený proces "%1". + + + Cannot find ABI for remote process "%1". + Nelze najít ABI pro vzdálený proces "%1". + + Remove Breakpoint %1 Odstranit bod přerušení %1 @@ -32399,15 +34213,15 @@ Set Breakpoint at line %1 - Nastavit bod přerušení u řádku %1 + Nastavit bod přerušení u řádku %1 Set Message Tracepoint at 0x%1... - Nastavit stopovací bod se zprávou při 0x%1... + Nastavit trasovací bod se zprávou při 0x%1... Set Message Tracepoint at line %1... - Nastavit stopovací bod se zprávou u řádku %1... + Nastavit stopovací bod se zprávou u řádku %1... Start '%1' and break at function 'main()' @@ -32415,7 +34229,7 @@ Save Debugger Log - Uložit zápis o ladění + Uložit záznam o ladění User commands are not accepted in the current state. @@ -32439,19 +34253,19 @@ Debugging Helper Missing - Pomocná knihovna pro výstup dat o ladění nebyla nalezena + Pomocný ladicí program nebyl nalezen The debugger could not load the debugging helper library. - Ladicí program nemohl nahrát pomocnou knihovnu pro výstup dat o ladění. + Ladicí program nemohl nahrát pomocnou ladicí knihovnu. The debugging helper is used to nicely format the values of some Qt and Standard Library data types. It must be compiled for each used Qt version separately. On the Qt4 options page, select a Qt installation and click Rebuild. - Pomocná knihovna pro výstup dat o ladění se používá pro výstup některých datových typů z Qt a standardních knihoven. Musí být pro každou verzi Qt sestavena odděleně. To se děje na stránce 'Nastavení Qt' pomocí výběru instalace Qt a klepnutí na 'Sestavit znovu' pro pomocnou knihovnu pro výstup dat o ladění. + Pomocná knihovna pro výstup dat o ladění se používá pro výstup některých datových typů z Qt a standardních knihoven. Musí být pro každou verzi Qt sestavena odděleně. To se děje na stránce 'Nastavení Qt' pomocí výběru instalace Qt a klepnutí na 'Sestavit znovu' pro pomocnou knihovnu pro výstup dat o ladění. Starting debugger '%1' for ABI '%2'... - Spouští se ladicí program '%1' pro ABI '%2'... + Spouští se ladicí program '%1' pro ABI '%2'... Debugger finished. @@ -32459,7 +34273,27 @@ QML Script Console - Skriptovací konzole QML + Skriptovací konzole QML + + + Set Breakpoint at Line %1 + Nastavit bod přerušení u řádku %1 + + + Set Message Tracepoint at Line %1... + Nastavit trasovací bod se zprávou u řádku %1... + + + Disassemble Function "%1" + Rozložit funkci "%1" + + + The debugging helper is used to nicely format the values of some Qt and Standard Library data types. It must be compiled for each used Qt version separately. In the Qt Creator Build and Run preferences page, select a Qt version, expand the Details section and click Build All. + Pomocný ladicí program slouží k výstupu hodnot některých datových typů z Qt a standardních knihoven. Musí být sestaven pro každou používanou verzi Qt. Na stránce nastavení 'Sestavení a spuštění' vyberte verzi Qt, rozbalte Podrobnosti a klepněte na 'Sestavit vše'. + + + Starting debugger "%1" for ABI "%2"... + Spouští se ladicí program '%1' pro ABI '%2'... Continue @@ -32542,6 +34376,10 @@ Spustit ladění + Start Debugging Without Deployment + Spustit ladění bez nasazení + + Start and Debug External Application... Spustit vnější uživatelský program a provádět u něj ladění... @@ -32550,12 +34388,20 @@ Spustit vnější uživatelský program a provádět u něj ladění s vnějším strojem... + Attach to Running Local Application... + Připojit se k běžícímu místnímu uživatelskému programu... + + + Load Core File... + Nahrát soubor core... + + Attach to Running External Application... - Připojit se k běžícímu vnějšímu uživatelskému programu... + Připojit se k běžícímu vnějšímu uživatelskému programu... Attach to Core... - Připojit se k hlavnímu souboru... + Připojit se k hlavnímu souboru... Start and Debug Remote Application... @@ -32566,8 +34412,16 @@ Připojit se ke vzdálenému ladicímu serveru... + Start Remote Debug Server Attached to Process... + Spustit vzdálený ladicí server připojený na proces... + + + Attach to Running Remote Process... + Ladit vzdálený běžící program... + + Attach to QML Port... - Připojit se k přípojce QML... + Připojit se k portu QML... Start and Attach to Remote Application... @@ -32582,6 +34436,14 @@ Odpojit ladicí program + Start Gdbserver + Spustit gdbserver + + + Attach to Remote Process + Připojit k vzdálenému procesu + + Interrupt Debugger Přerušit ladicí program @@ -32673,7 +34535,7 @@ Unable to create a debugger engine of the type '%1' - Nepodařilo se vytvořit žádný stroj ladicího programu typu '%1' + Nepodařilo se vytvořit žádný ladicí stroj typu '%1' @@ -32788,6 +34650,10 @@ Spojení s podřízeným procesem se nepodařilo zřídit: %1 + Warning + Varování + + This does not seem to be a "Debug" build. Setting breakpoints by file name and line number may fail. Zdá se, že toto není sestavení "Debug". @@ -32832,7 +34698,7 @@ Fatal engine shutdown. Consult debugger log for details. - Stroj byl kvůli vážné chybě zastaven. Zápis ladicího programu obsahuje další informace. + Stroj byl kvůli vážné chybě zastaven. Záznam ladicího programu obsahuje další informace. SSH connection error: %1 @@ -32868,7 +34734,7 @@ Debugger::Internal::LogWindow Debugger Log - Zápis ladicího programu + Záznam ladicího programu Command: @@ -32876,7 +34742,7 @@ Log File - Soubor se zápisem + Soubor se záznamem @@ -32947,6 +34813,18 @@ Zavírá se spojení... + Status of '%1' Version: %2 changed to 'unavailable'. + Stav '%1' verze: %2 se změnil na 'nedostupný'. + + + Status of '%1' Version: %2 changed to 'enabled'. + Stav '%1' verze: %2 se změnil na 'povolený'. + + + Status of '%1' Version: %2 changed to 'not connected'. + Stav '%1' verze: %2 se změnil na 'nepřipojený'. + + Status of '%1' changed to 'unavailable'. Stav '%1' se změnil na 'nedostupný'. @@ -32994,11 +34872,11 @@ Debugger::Internal::QmlEngine QML Debugger connected. - Ladič QML spojen. + Ladič QML spojen. QML Debugger connecting... - Vytváří se spojení s ladičem QML... + Vytváří se spojení s ladičem QML... Qt Creator @@ -33011,6 +34889,12 @@ Má se to zkusit ještě jednou? + Could not connect to the in-process QML debugger. +%1 + Nepodařilo se vytvořit žádné spojení se součástkou ladění v procesu: +%1 + + QML Debugger: Remote host closed connection. Ladič QML: Vzdálený hostitel uzavřel spojení. @@ -33019,20 +34903,28 @@ Ladič QML: Nepodařilo se vytvořit žádné spojení se službou '%1'. + JS Source for %1 + Zdroj JS pro %1 + + + Context: + Souvislosti: + + The port seems to be in use. Error message shown after 'Could not connect ... debugger:" - Přípojka se už používá. + Přípojka se už používá. The application is not set up for QML/JS debugging. Error message shown after 'Could not connect ... debugger:" - Tento program není nastaven pro ladění QML/JS. + Tento program není nastaven pro ladění QML/JS. Could not connect to the in-process QML debugger: %1 %1 is detailed error message - Ladič QML: Nepodařilo se vytvořit žádné spojení se součástkou ladění v procesu: + Ladič QML: Nepodařilo se vytvořit žádné spojení se součástkou ladění v procesu: %1 @@ -33048,6 +34940,10 @@ Program se nepodařilo spustit: %1 + Run to line %1 (%2) requested... + Požadováno provedení až po řádek %1 (%2)... + + Stopped. Zastaveno. @@ -33072,16 +34968,16 @@ Debugger::Internal::ScriptConsole <Type expression to evaluate> - <Zadejte výraz pro vyhodnocení> + <Zadejte výraz pro vyhodnocení> Write and evaluate QtScript expressions. - Zapsat a vyhodnotit výrazy QtScriptu. + Zapsat a vyhodnotit výrazy QtScriptu. Script Console - Skriptovací konzole + Skriptovací konzole @@ -33149,12 +35045,28 @@ To: Do: + + Note: + Poznámka: + + + Sources for this frame are available.<br>Double-click on the file name to open an editor. + Zdroje pro tento snímek jsou dostupné.<br>Dvojité klepnutí na název souboru pro otevření editoru. + + + Binary debug information is not accessible for this frame. This either means the core was not compiled with debug information, or the debug information is not accessible. Note that most distributions ship debug information in separate packages. + Informace o ladění spustitelného souboru nejsou pro tento snímek přístupné. To buď znamená, že core nebyl sestaven s informacemi o ladění, nebo že informace o ladění nejsou přístupné. Uvědomte si, že většina distribucí vede informace o ladění v oddělených balíčcích. + + + Binary debug information is accessible for this frame. However, matching sources have not been found. Note that some distributions ship debug sources in in separate packages. + Informace o ladění spustitelného souboru jsou pro tento snímek přístupné. Nicméně odpovídající zdroje nebyly nalezeny. Uvědomte si, že většina distribucí vede informace o ladění v oddělených balíčcích. + FakeVim::Internal::FakeVimUserCommandsModel Action - Krok + Úkon Command @@ -33337,6 +35249,10 @@ currently set to '%1' nyní nastaveno na '%1' + + Git Repository Browser Command + Příkaz pro prohlížeč skladiště Git + GLSLEditor @@ -33354,7 +35270,7 @@ Creates a fragment shader in the OpenGL/ES 2.0 Shading Language (GLSL/ES). Fragment shaders generate the final pixel colors for triangles, points and lines rendered with OpenGL. - Vytvoří stínovač kousku (Fragment Shader) v stínovacím jazyku, OpenGL/ES 2.0 Shading Language (GLSL/ES). Stínovače kousků (Fragment Shader) vytvářejí barvy pixelů pro trojúhelníky, body a čáry udělané OpenGL. + Vytvoří stínovač pixelu (fragment shader) v jazyku OpenGL/ES 2.0 Shading Language (GLSL/ES). Stínovače pixelu vytvářejí barvy pixelů pro trojúhelníky, body a čáry vykreslené OpenGL. Fragment Shader (OpenGL/ES 2.0) @@ -33362,7 +35278,7 @@ Creates a vertex shader in the OpenGL/ES 2.0 Shading Language (GLSL/ES). Vertex shaders transform the positions, normals and texture co-ordinates of triangles, points and lines rendered with OpenGL. - Vytvoří stínovač vrcholu (Vertex Shader) v stínovacím jazyku, OpenGL/ES 2.0 Shading Language (GLSL/ES). Stínovače vrcholů (Vertex Shader) proměňují polohy, kolmice a souřadnice textur OpenGL udělaných trojúhelníků, bodů a čar. + Vytvoří stínovač vrcholu (vertex shader) v jazyku OpenGL/ES 2.0 Shading Language (GLSL/ES). Stínovače vrcholu proměňují polohy, kolmice a souřadnice textur pro trojúhelníky, body a čáry vykreslené OpenGL. Vertex Shader (OpenGL/ES 2.0) @@ -33370,7 +35286,7 @@ Creates a fragment shader in the Desktop OpenGL Shading Language (GLSL). Fragment shaders generate the final pixel colors for triangles, points and lines rendered with OpenGL. - Vytvoří stínovač kousku (Fragment Shader) v stínovacím jazyku, Desktop OpenGL Shading Language (GLSL). Stínovače kousků (Fragment Shader) vytvářejí barvy pixelů pro OpenGL udělané trojúhelníky, body a čáry. + Vytvoří stínovač pixelu (fragment shader) v jazyku Desktop OpenGL Shading Language (GLSL). Stínovače pixelu vytvářejí barvy pixelů pro trojúhelníky, body a čáry vykreslené OpenGL. Fragment Shader (Desktop OpenGL) @@ -33378,7 +35294,7 @@ Creates a vertex shader in the Desktop OpenGL Shading Language (GLSL). Vertex shaders transform the positions, normals and texture co-ordinates of triangles, points and lines rendered with OpenGL. - Vytvoří stínovač vrcholu (Vertex Shader) v stínovacím jazyku, Desktop OpenGL Shading Language (GLSL). Stínovače vrcholů (Vertex Shader) proměňují polohy, kolmice a souřadnice textur OpenGL udělaných trojúhelníků, bodů a čar. + Vytvoří stínovač vrcholu (vertex shader) v jazyku Desktop OpenGL Shading Language (GLSL). Stínovače vrcholu proměňují polohy, kolmice a souřadnice textur pro trojúhelníky, body a čáry vykreslené OpenGL. Vertex Shader (Desktop OpenGL) @@ -33414,14 +35330,18 @@ Help::Internal::RemoteHelpFilter Web Search - &Hledat na internetu + Hledat na internetu ImageViewer::Internal::ImageViewer Cannot open image file %1 - Nelze otevřít soubor s obrázkem %1 + Nelze otevřít soubor s obrázkem %1 + + + Cannot open image file %1. + Nelze otevřít soubor s obrázkem %1. @@ -33618,6 +35538,10 @@ ProjectExplorer::Internal::BuildStepListWidget + Disable + Zakázat + + Move Up Posunout nahoru @@ -33632,11 +35556,11 @@ %1 Steps %1 is the name returned by BuildStepList::displayName - Kroky pro %1 + Postup pro %1 No %1 Steps - Žádné kroky pro %1 + Prázdný postup pro %1 Add %1 Step @@ -33652,7 +35576,7 @@ No Build Steps - Žádné kroky sestavování + Prázdný postup sestavování @@ -33899,6 +35823,18 @@ Autodetect Zjistit + + mkspec: + mkspec: + + + All possible mkspecs separated by a semicolon (';'). + Všechny možné mkspecs oddělené středníkem (';'). + + + Reset + Nastavit znovu + ProjectExplorer::Internal::ToolChainModel @@ -33908,7 +35844,7 @@ Manual - Ruční + Ručně <nobr><b>ABI:</b> %1 @@ -33928,26 +35864,38 @@ Duplicate Tool Chain detected - Byl zjištěn zdvojený řetězec nástrojů + Byla zjištěna zdvojená sada nástrojů The following tool chain was already configured:<br>&nbsp;%1<br>It was not configured again. - Následující řetězec nástrojů je již nastaven:<br>&nbsp;%1<br>Nebyl nastaven ještě jednou znovu. + Následující sada nástrojů již byla nastavena:<br>&nbsp;%1<br>Nebyla nastavena znovu ještě jednou. Duplicate Tool Chains detected - Zdvojené řetězce nástrojů + Zdvojené sady nástrojů The following tool chains were already configured:<br>&nbsp;%1<br>They were not configured again. - Následující řetězce nástrojů jsou již nastaveny:<br>&nbsp;%1<br>Nebyly nastaveny ještě jednou znovu. + Následující sady nástrojů již byly nastaveny:<br>&nbsp;%1<br>Nebyly nastaveny znovu ještě jednou. ProjectExplorer::Internal::ToolChainOptionsPage Tool Chains - Řetězce nástrojů + Sady nástrojů + + + Add + Přidat + + + Clone + Klonovat + + + Remove + Odstranit @@ -33996,7 +35944,11 @@ ProjectExplorer::Internal::VcsAnnotateTaskHandler &Annotate - &Opatřit vysvětlivkami + &Opatřit anotacemi + + + Annotate using version control system + Opatřit anotacemi za použití systému na správu verzí @@ -34211,7 +36163,7 @@ Move Component into '%1.qml' - Přesunout součástku do '%1.qml' + Přesunout součástku do '%1.qml' @@ -34228,6 +36180,26 @@ Invalid path Neplatná cesta + + Move Component into Separate File + Přesunout součástku do odděleného souboru + + + Path: + Cesta: + + + Component name: + Název součástky: + + + Dialog + Dialog + + + Choose... + Vybrat... + QmlJSEditor::QmlJSTextEditorWidget @@ -34260,7 +36232,7 @@ QmlJsEditor QML - QML + QML @@ -34416,6 +36388,10 @@ QmlJSInspector::Internal::QmlJSPropertyInspector + Always Adjust Column Widths to Contents + Vždy přizpůsobit šířku sloupců obsahu + + Enter expression Zadat výraz @@ -34608,12 +36584,24 @@ The port seems to be in use. Error message shown after 'Could not connect ... debugger:" - Přípojka se už používá. + Přípojka se už používá. The application is not set up for QML/JS debugging. Error message shown after 'Could not connect ... debugger:" - Tento program není nastaven pro ladění QML/JS. + Tento program není nastaven pro ladění QML/JS. + + + Unexpected engine stop from state %1 in %2:%3 + Neočekávané zastavení stroje ze stavu %1 v %2:%3 + + + Process died unexpectedly from state %1 in %2:%3 + Proces neočekávaně skončil ze stavu %1 v %2:%3 + + + Unexpected process termination requested with state %1 in %2:%3 + Neočekávané zastavení procesu požadováno se stavem %1 v %2:%3 Qt Creator @@ -34646,23 +36634,23 @@ QmlProfiler::Internal::QmlProfilerEventsView Location - Umístění + Umístění Type - Typ + Typ Time in Percent - Čas v procentu + Čas v procentu Total Time - Celkový čas + Celkový čas Calls - Volání + Volání Time per Call @@ -34670,50 +36658,50 @@ Mean Time - Střední čas + Střední čas Median Time - Střední hodnota času + Střední hodnota času Longest Time - Nejdelší čas + Nejdelší čas Shortest Time - Nejkratší čas + Nejkratší čas Details - Podrobnosti + Podrobnosti Paint - Barva + Barva Compile - Sestavit + Sestavit Create - Vytvořit + Vytvořit Binding - Přiřazená klávesa + Přiřazená klávesa Signal - Signál + Signál QmlProfiler::Internal::QmlProfilerRunControlFactory QML Profiler - Profiler QML + Profiler QML @@ -34728,11 +36716,53 @@ Load QML Trace - Nahrát stopu QML + Nahrát výpis volání QML + + + QML Profiler Options + Nastavení profileru QML Save QML Trace - Uložit stopu QML + Uložit výpis volání QML + + + Extended Event Statistics + Rozšířená statistika událostí + + + Limit Events Pane to Current Range + Omezit tabulku na nynější rozsah + + + Reset Events Pane + Vynulovat tabulku událostí + + + Reset Zoom + Nastavit zvětšení znovu + + + JavaScript + JavaScript + + + %1 s + %1 s + + + Elapsed: %1 + Uplynulo: %1 + + + QML traces (*%1) + Výpisy volání QML (*%1) + + + Application finished before loading profiled data. + Please use the stop button instead. + Program byl před nahráním profilovaných dat ukončen. +Použijte, prosím, místo toho tlačítko Zastavit. Copy Row @@ -34760,11 +36790,11 @@ Callees - Volané funkce + Volané funkce Callers - Vyvolávající + Vyvolávající Discard data @@ -34772,7 +36802,7 @@ Elapsed: 0 s - Uplynulo: 0 s + Uplynulo: 0 s Disable profiling @@ -34784,7 +36814,7 @@ Elapsed: %1 s - Uplynulo: %1 s + Uplynulo: %1 s Qt Creator @@ -34798,7 +36828,7 @@ QML traces (%1) - Stopy QML (%1) + Stopy QML (%1) @@ -34806,7 +36836,7 @@ Not enough free ports on device for analyzing. - Na zařízení není dostatek volných přípojek pro rozbor. + Na zařízení není dostatek volných bran pro rozbor. @@ -34833,31 +36863,59 @@ Jít na další událost + Show zoom slider + Ukázat zvětšovací posuvník + + + Select range + Vybrat rozsah + + + View event information on mouseover + Ukázat informace o události při přejezdu myši + + Zoom in 10% - Přiblížit o 10% + Přiblížit o 10% Zoom out 10% - Oddálit o 10% + Oddálit o 10% QmlProjectManager::QmlProjectPlugin Open Qt4 Options - Otevřít nastavení pro knihovnu Qt4 + Otevřít nastavení pro knihovnu Qt4 + + + Open Qt Versions + Otevřít nastavení pro knihovnu Qt QML Observer Missing Pozorovatel QML chybí + QML Observer could not be found for this Qt version. + Žádného pozorovatele QML se pro tuto verzi Qt nepodařilo najít. + + + QML Observer is used to offer debugging features for Qt Quick UI projects in the Qt 4.7 series. + +To compile QML Observer, go to the Qt Versions page, select the current Qt version, and click Build in the Helpers section. + Pozorovatel QML se používá pro nabídnutí dodatečných funkcí pro projekty Qt Quick UI v řadě Qt 4.7 + +Sestavení pozorovatele QML se děje na stránce pro nastavení Qt pomocí výběru odpovídající verze Qt a klepnutí na 'Sestavit znovu' v části s pomocnými součástmi. + + QML Observer could not be found. - Žádného pozorovatele QML se nepodařilo najít. + Žádného pozorovatele QML se nepodařilo najít. QML Observer is used to offer debugging features for QML applications, such as interactive debugging and inspection tools. It must be compiled for each used Qt version separately. On the Qt4 options page, select the current Qt installation and click Rebuild. - Pozorovatel QML se používá pro nabídnutí dodatečných funkcí pro programy QML, jako je například interaktivní náhled na změny kódu a další nástroje ke zkoumání. Musí být pro každou použitou verzi Qt sestaven odděleně. To se děje na stránce 'Nastavení Qt' pomocí výběru nynější instalace Qt a klepnutí na 'Sestavit znovu'. + Pozorovatel QML se používá pro nabídnutí dodatečných funkcí pro programy QML, jako je například interaktivní náhled na změny kódu a další nástroje ke zkoumání. Musí být pro každou použitou verzi Qt sestaven odděleně. To se děje na stránce 'Nastavení Qt' pomocí výběru nynější instalace Qt a klepnutí na 'Sestavit znovu'. @@ -34884,7 +36942,7 @@ Debugger: - Ladič: + Ladič: Run Environment @@ -35032,7 +37090,7 @@ The following snippet will be added to the<br><b>%1</b> file: - Následující údaj bude přidán do souboru <br><b>%1</b>: + Následující úryvek bude přidán do souboru <br><b>%1</b>: @@ -35104,6 +37162,11 @@ Qt4 Meego target display name Meego + + Android + Qt4 Android target display name + Android + Qt4ProjectManager::CodaRunControl @@ -35197,7 +37260,7 @@ Qt Creator is waiting for the CODA application to connect.<br>Please make sure the application is running on your mobile phone and the right IP address and/or port are configured in the project settings. - Qt Creator čeká na spojení s programem CODA.<br>Spusťte, prosím, program na svém mobilním telefonu a prověřte nastavení IP adresy a přípojky v nastavení projektu. + Qt Creator čeká na spojení s programem CODA.<br>Spusťte, prosím, program na svém mobilním telefonu a prověřte nastavení IP adresy a portu v nastavení projektu. Canceled. @@ -35290,18 +37353,18 @@ The certificate "%1" has already expired and cannot be used. Expiration date: %2. - Osvědčení "%1" již vypršelo a proto je nelze používat. + Certifikát "%1" již vypršel a proto jej nelze používat. Datum vypršení platnosti: %2. The certificate "%1" is not yet valid. Valid from: %2. - Osvědčení "%1" ještě není platné. + Certifikát "%1" ještě není platný. Platí od: %2. The certificate "%1" is not a valid X.509 certificate. - Osvědčení "%1" není platným osvědčením X.509. + Certifikát "%1" není platným osvědčením X.509. Type: @@ -35309,11 +37372,11 @@ Developer certificate - Vývojářské osvědčení + Vývojářský certifikát Self signed certificate - Osobně podepsané osvědčení + Osobně podepsaný certifikát Issued by: @@ -35353,7 +37416,7 @@ %1 package name, %2 will be replaced by a list of patching lines. Osobně podepsaný balíček '%1' byl změněn, aby byla umožněna jeho instalace. %2 -Používejte vývojářské osvědčení nebo jakoukoli jinou podobu podepsání, abyste se tomute vyhnuli. +Použitím vývojářského certifikátu nebo jakékoliv jiné podoby podpisu se těmto změnám vyhnete. Cannot create Smart Installer package as the Smart Installer's base file is missing. Please ensure that it is located in the SDK. @@ -35406,7 +37469,7 @@ Silent installation is an installation mode that does not require user's intervention. In case it fails the non silent installation is launched. - V režimu 'Tichá instalace' se nevyžaduje žádný zásah uživatele na mobilním zařízení. Pokud toto selže, spustí se ne-tichá instalace (běžná). + V režimu 'Tichá instalace' se nevyžaduje žádný zásah uživatele na mobilním zařízení. Pokud toto selže, spustí se běžná, ne-tichá instalace. Installation drive: @@ -35470,7 +37533,7 @@ unknown - Neznámé + Neznámý ROM version: @@ -35526,7 +37589,12 @@ Deploy Qt4 Deploystep display name - Nasazení + Nasazení + + + Deploy SIS Package + Qt4 Deploystep display name + Nasazení balíčku SIS No package has been found. Specify at least one installation package. @@ -35562,7 +37630,7 @@ No such port - Přípojka neexistuje + Port neexistuje Could not open serial device: %1 @@ -35634,14 +37702,14 @@ A timeout while deploying has occurred. CODA might not be responding. Try reconnecting the device. - Při nasazení se vyskytlo jedno překročení času. CODA možná neodpovídá. Pokuste se vytvořit spojení se zařízením znovu. + Při nasazování bylo překročeno časové omezení. CODA možná neodpovídá. Pokuste se znovu vytvořit spojení se zařízením. Qt4ProjectManager::Internal::S60DeployStepWidget Deploy SIS Package - Nasazení balíčku SIS + Nasazení balíčku SIS @@ -35683,7 +37751,7 @@ Qt4ProjectManager::Internal::S60PublisherOvi Clean - Uklidit + Pročistit qmake @@ -35748,7 +37816,25 @@ No valid tool chain has been detected.<br>Define a correct tool chain in "Options > Tool Chains" - Nepodařilo se najít žádný platný řetězec nástrojů.<br> Zadejte, prosím, správný řetězec nástrojů v "Nastavení > Řetězce nástrojů" + Nepodařilo se najít žádnou platnou sadu nástrojů.<br> Zadejte, prosím, správnou sadu nástrojů v "Nastavení > Sady nástrojů" + + + Form + Formulář + + + Choose a build configuration: + Vybrat nastavení sestavování: + + + Choose a tool chain: + Vybrat sadu nástrojů: + + + Only Qt versions above 4.6.3 are made available in this wizard. +Previous Qt versions have limitations in building suitable SIS files. + Tento průvodce ukáže jen verze Qt po verzi 4.6.3. +Předchozí verze Qt mají omezení v sestavování vhodných souborů SIS. @@ -35761,6 +37847,10 @@ Close Zavřít + + Form + Formulář + Qt4ProjectManager::Internal::S60PublishingSisSettingsPageOvi @@ -35774,7 +37864,7 @@ "%1" is a default vendor name used for testing and development. <br>The Vendor_Name field cannot contain the name 'Nokia'. <br>You are advised against using the default names 'Vendor' and 'Vendor-EN'. <br>You should also not leave the entry blank. <br>see <a href="http://www.forum.nokia.com/Distribute/Packaging_and_signing.xhtml">Packaging and Signing</a> for guidelines.<br> - "%1" je výchozí název prodejce používaný pro zkoušení a vývoj. <br>Pole Vendor_Name nesmí obsahovat název 'Nokia'.<br>Radí se vám, abyste nepoužívali jako výchozí názvy 'Vendor' nebo 'Vendor-EN'.<br>Pole by také nemělo zůstat ponecháno prázdné.<br>Podívejte se na <a href="http://www.forum.nokia.com/Distribute/Packaging_and_signing.xhtml">zásady pro balíčkování a podepsání</a>.<br> + "%1" je výchozí název prodejce používaný pro zkoušení a vývoj. <br>Pole Vendor_Name nesmí obsahovat název 'Nokia'.<br>Radí se vám, abyste nepoužívali jako výchozí názvy 'Vendor' nebo 'Vendor-EN'.<br>Pole by také nemělo zůstat ponecháno prázdné.<br>Podívejte se na <a href="http://www.forum.nokia.com/Distribute/Packaging_and_signing.xhtml">zásady pro balíčkování a podepsání</a>.<br> %1 is a default vendor name used for testing and development. @@ -35786,19 +37876,39 @@ %1 <br>The Vendor_Name field cannot contain the name 'Nokia'. <br>You are advised against using the default names 'Vendor' and 'Vendor-EN'. <br>You should also not leave the entry blank. <br>See <a href="http://www.forum.nokia.com/Distribute/Packaging_and_signing.xhtml">Packaging and Signing</a> for guidelines.<br> - %1 <br>Pole Vendor_Name nesmí obsahovat název 'Nokia'.<br>Radí se vám, abyste nepoužívali jako výchozí názvy 'Vendor' nebo 'Vendor-EN'.<br>Pole by také nemělo zůstat ponecháno prázdné.<br>Podívejte se na <a href="http://www.forum.nokia.com/Distribute/Packaging_and_signing.xhtml">zásady pro balíčkování a podepsání</a>.<br> + %1 <br>Pole Vendor_Name nesmí obsahovat název 'Nokia'.<br>Radí se vám, abyste nepoužívali jako výchozí názvy 'Vendor' nebo 'Vendor-EN'.<br>Pole by také nemělo zůstat ponecháno prázdné.<br>Podívejte se na <a href="http://www.forum.nokia.com/Distribute/Packaging_and_signing.xhtml">zásady pro balíčkování a podepsání</a>.<br> The application UID %1 is only for testing and development.<br>SIS packages built with it cannot be distributed via the Ovi Store.<br> - UID programu %1 lze používat jen ke zkoušení a vývoji.<br>S tímto vytvořené balíčky SIS nelze rozšiřovat přes Ovi Store.<br> + UID programu %1 lze používat jen ke zkoušení a vývoji.<br>S tímto vytvořené balíčky SIS nelze rozšiřovat přes Ovi Store.<br> The application UID %1 is a symbiansigned.com UID. <br>Applications with this UID will be rejected by Application Signing Services for Ovi Store.<br>If you want to continue with a symbiansigned.com UID, sign your application on symbiansigned.com and upload the signed application to Publish to Ovi.<br> - UID programu %1 pochází z symbiansigned.com.<br>Programy s tímto UID budou Application Signing Services Ovi Store odmítnuty.<br>Když chcete použít UID pocházející z symbiansigned.com stammende UID, nechte, prosím, svoji aplikaci podepsat od symbiansigned.com a nahrajte podepsanou aplikaci do Publish na Ovi.<br> + UID programu %1 pochází z symbiansigned.com.<br>Programy s tímto UID budou Application Signing Services Ovi Store odmítnuty.<br>Když chcete použít UID pocházející z symbiansigned.com stammende UID, nechte, prosím, svoji aplikaci podepsat od symbiansigned.com a nahrajte podepsanou aplikaci do Publish na Ovi.<br> The application UID %1 is not an acceptable UID.<br>SIS packages built with it cannot be signed by Application Signing Services for Ovi Store.<br> - UID programu %1 je neplatné.<br>S tímto vytvořené balíčky SIS nemohou být podepsány Application Signing Services Ovi Store.<br> + UID programu %1 je neplatné.<br>S tímto vytvořené balíčky SIS nemohou být podepsány Application Signing Services Ovi Store.<br> + + + "%1" is a default vendor name used for testing and development. <br>The Vendor_Name field cannot contain the name 'Nokia'. <br>You are advised against using the default names 'Vendor' and 'Vendor-EN'. <br>You should also not leave the entry blank. <br>see <a href="http://www.developer.nokia.com/Distribute/Packaging_and_signing.xhtml">Packaging and Signing</a> for guidelines.<br> + "%1" je výchozí název prodejce používaný pro zkoušení a vývoj. <br>Pole Vendor_Name nesmí obsahovat název 'Nokia'.<br>Radí se vám, abyste nepoužívali jako výchozí názvy 'Vendor' nebo 'Vendor-EN'.<br>Pole by také nemělo zůstat ponecháno prázdné.<br>Podívejte se na <a href="http://www.developer.nokia.com/Distribute/Packaging_and_signing.xhtml">zásady pro balíčkování a podepsání</a>.<br> + + + %1 <br>The Vendor_Name field cannot contain the name 'Nokia'. <br>You are advised against using the default names 'Vendor' and 'Vendor-EN'. <br>You should also not leave the entry blank. <br>See <a href="http://www.developer.nokia.com/Distribute/Packaging_and_signing.xhtml">Packaging and Signing</a> for guidelines.<br> + %1 <br>Pole Vendor_Name nesmí obsahovat název 'Nokia'.<br>Radí se vám, abyste nepoužívali jako výchozí názvy 'Vendor' nebo 'Vendor-EN'.<br>Pole by také nemělo zůstat ponecháno prázdné.<br>Podívejte se na <a href="http://www.developer.nokia.com/Distribute/Packaging_and_signing.xhtml">zásady pro balíčkování a podepsání</a>.<br> + + + The application UID %1 is only for testing and development.<br>SIS packages built with it cannot be distributed via the Nokia Store.<br> + UID programu %1 lze používat jen ke zkoušení a vývoji.<br>S tímto vytvořené balíčky SIS nelze rozšiřovat přes Nokia Store.<br> + + + The application UID %1 is a symbiansigned.com UID. <br>Applications with this UID will be rejected by Application Signing Services for Nokia Store.<br>If you want to continue with a symbiansigned.com UID, sign your application on symbiansigned.com and upload the signed application to Nokia Publish.<br> + UID programu %1 pochází z symbiansigned.com.<br>Programy s tímto UID budou Application Signing Services Nokia Store odmítnuty.<br>Když chcete použít UID pocházející z symbiansigned.com stammende UID, nechte, prosím, svoji aplikaci podepsat od symbiansigned.com a nahrajte podepsanou aplikaci do Nokia Publish.<br> + + + The application UID %1 is not an acceptable UID.<br>SIS packages built with it cannot be signed by Application Signing Services for Nokia Store.<br> + UID programu %1 je neplatné.<br>S tímto vytvořené balíčky SIS nemohou být podepsány Application Signing Services Nokia Store.<br> The application UID is a global unique indentifier of the SIS package.<br> @@ -35809,8 +37919,12 @@ Abyste dostal jedinečné UID programu pro svůj soubor balíčku,<br>přihlašte se, prosím, u <a href="http://info.publish.ovi.com/">publish.ovi.com</a> + If this UID is from symbiansigned.com, It will be rejected by Application Signing Services for Nokia Store.<br>If you want to continue with a symbiansigned.com UID, sign your application on symbiansigned.com and upload the signed application to Nokia Publish.<br>It is, however, recommended that you obtain a UID from <a href="http://info.publish.ovi.com/">publish.ovi.com</a> + Pokud je toto UID z symbiansigned.com, bude odmítnuto Application Signing Services pro Nokia Store.<br>Pokud chcete pokračovat s symbiansigned.com UID, přihlaste svůj program na symbiansigned.com a podepsaný program nahrajte do Nokia Publish.<br>Nicméně se doporučuje, abyste obdrželi UID z <a href="http://info.publish.ovi.com/">publish.ovi.com</a> + + %1 need(s) to be certified signed. Please go to <a href="symbiansigned.com">symbiansigned.com</a> for guidance. - %1 musí mít osvědčení a být podepsán. Obraťte se, prosím, pro další vedení na <a href="symbiansigned.com">symbiansigned.com</a> . + %1 musí mít certifikát a být podepsán. Obraťte se, prosím, pro další vedení na <a href="symbiansigned.com">symbiansigned.com</a> . <br>%1 need(s) manufacturer approval.<br> @@ -35821,15 +37935,67 @@ Některá oprávnění možná potřebují zvláštní podepsání nebo schválení výrobce.<br> + Please verify that you have a released version of Qt. <br><a href="http://www.developer.nokia.com/Community/Wiki/Nokia_Smart_Installer_for_Symbian">Qt Packages Distributed by Smart Installer</a> has a list of released Qt versions. + Ujistěte se, prosím, že používáte vydanou verzi Qt. <br><a href="http://wiki.developer.nokia.com/index.php/Nokia_Smart_Installer_for_Symbian">Qt Packages Distributed by Smart Installer</a> má seznam uvolněných verzí. + + Please verify that you have a released version of Qt. <br><a href="http://wiki.forum.nokia.com/index.php/Nokia_Smart_Installer_for_Symbian">Qt Packages Distributed by Smart Installer</a> has a list of released Qt versions. - Ujistěte se, prosím, že používáte vydanou verzi Qt. <br><a href="http://wiki.forum.nokia.com/index.php/Nokia_Smart_Installer_for_Symbian">Qt Packages Distributed by Smart Installer</a> má seznam uvolněných verzí. + Ujistěte se, prosím, že používáte vydanou verzi Qt. <br><a href="http://wiki.forum.nokia.com/index.php/Nokia_Smart_Installer_for_Symbian">Qt Packages Distributed by Smart Installer</a> má seznam uvolněných verzí. + + + Form + Formulář + + + Localised Vendor Names + Lokalizovaný název prodejce + + + Current Global Vendor Name + Nynější jednoznačný název prodejce + + + Display name: + Název k zobrazení: + + + Localised vendor names: + Lokalizovaný název prodejce: + + + Capabilities: + Oprávnění: + + + Current UID3 + Nynější UID3 + + + Application UID: + UID programu: + + + Current Qt Version + Nynější verze Qt + + + Qt version used in builds: + Při sestavování používaná verze Qt: + + + Current set of capabilities + Nynější oprávnění + + + Global vendor name: + Jednoznačný název prodejce: Qt4ProjectManager::Internal::S60PublishingWizardFactoryOvi Publish Qt Symbian Applications to Ovi Store - Zveřejnit jako program Qt Symbianv Ovi Store + Zveřejnit jako program Qt Symbianv Ovi Store This wizard checks your project file to make sure it complies with Ovi Store submission criteria. @@ -35842,7 +38008,7 @@ NetworkControl, MultimediaDD, CommDD, DiskAdmin, AllFiles, DRM and TCB. Your application will also be rejected by Ovi QA if you choose an unreleased Qt version on the next page. - Tento průvodce přezkouší váš projektový soubor kvůli ujištění, že odpovídá měřítkům Ovi Store. + Tento průvodce přezkouší váš projektový soubor kvůli ujištění, že odpovídá měřítkům Ovi Store. Tento průvodce vytvoří soubory s balíčky SIS-Paketdateien, které mohou být předloženy ke zveřejnění na Ovi. @@ -35853,12 +38019,42 @@ Váš program bude také odmítnut v případě, že si vyberete nějakou neuvolněnou verzi Qt na další straně. + + Publish Qt Symbian Applications to Nokia Store + Zveřejnit jako program Qt Symbian v Nokia Store + + + This wizard checks your project file to make sure it complies with Nokia Store submission criteria. + +The wizard creates SIS files that can be submitted to Nokia Publish. + +You cannot use it if you use application UIDs from Symbian Signed. + +You cannot use it for the Certified Signed and Manufacturer level capabilities: +NetworkControl, MultimediaDD, CommDD, DiskAdmin, AllFiles, DRM and TCB. + +Your application will also be rejected by Nokia Store QA if you choose an unreleased Qt version on the next page. + Tento průvodce přezkouší váš projektový soubor kvůli ujištění, že odpovídá měřítkům Nokia Store. + +Tento průvodce vytvoří soubory s balíčky SIS, které mohou být odeslány ke zveřejnění na Nokia. + +Nelze jej používat, pokud už používáte UIDs programů od Symbian Signed. + +Taktéž jej nelze používat pro programy s certifikátem a podepsané nebo s oprávněním od výrobce: +NetworkControl, MultimediaDD, CommDD, DiskAdmin, AllFiles, DRM a TCB + +Váš program bude Nokia Store QA také odmítnut v případě, že si vyberete nějakou neuvolněnou verzi Qt na další straně. + Qt4ProjectManager::Internal::S60PublishingWizardOvi Publishing to Ovi Store - Zveřejnit na Ovi Store + Zveřejnit na Ovi Store + + + Publishing to Nokia Store + Zveřejnit na Nokia Store Build Configuration @@ -35912,11 +38108,11 @@ Qt4ProjectManager::SbsV2Parser SBSv2 build log - Zápis o sestavování SBSv2 + Záznam o sestavování SBSv2 The file '%1' is not a SBSv2 log file. - Soubor '%1; není zápis o sestavování SBSv2. + Soubor '%1; není záznam o sestavování SBSv2. Running command: %1 @@ -36008,6 +38204,10 @@ Create Build Configurations: + Vytvořit nastavení sestavování: + + + Create build configurations: Vytvořit nastavení sestavování: @@ -36027,12 +38227,44 @@ Žádný + Shadow build + Stínové sestavování + + + Qt version: + Verze Qt: + + + No Build Found + Žádná sestavení nebyla nalezena + + + Incompatible Build Found + Nalezeno neslučitelné sestavování + + + The build found in %1 is incompatible with this target. + Sestavení nalezené ve složce %1 není slučitelné s tímto cílem. + + + Already Imported Build + Již zavedené sestavení + + + The build found in %1 is already imported. + Sestavení nalezené v %1 je již zavedeno. + + + Import build from %1. + Zavést sestavení z %1. + + Use Shadow Building - Použít stínové sestavování + Použít stínové sestavování Qt Version: - Verze Qt: + Verze Qt: debug @@ -36046,7 +38278,7 @@ No build found - Žádná sestavení nebyla nalezena + Žádná sestavení nebyla nalezena No build found in %1 matching project %2. @@ -36054,15 +38286,15 @@ Incompatible build found - Nalezeno neslučitelné sestavování + Nalezeno neslučitelné sestavování The build found in %1 is incompatible with this target - Sestavení nalezené ve složce %1 není slučitelné s tímto cílem + Sestavení nalezené ve složce %1 není slučitelné s tímto cílem Import build from %1 - Zavést sestavení z %1 + Zavést sestavení z %1 <b>Error:</b> @@ -36175,6 +38407,14 @@ Lock to Portrait Orientation Stanovit formát na výšku + + WizardPage + WizardPage + + + Orientation behavior: + Chování orientace: + Qt4ProjectManager::Internal::PngIconScaler @@ -36229,21 +38469,25 @@ Tento průvodce vytvoří projekt aplikace Qt Quick. + Select existing QML file + Použít existující soubor .qml + + Application Type - Typ aplikace + Typ aplikace Qt4ProjectManager::Internal::QtQuickAppWizard Qt Quick Application - Nový program Qt Quick + Nový program Qt Quick Creates a Qt Quick application project that can contain both QML and C++ code and includes a QDeclarativeView. You can build the application and deploy it on desktop and mobile target platforms. For example, you can create signed Symbian Installation System (SIS) packages for this type of projects. Moreover, you can select to use a set of premade UI components in your Qt Quick application. To utilize the components, Qt 4.7.4 or newer is required. - Vytvoří program Qt Quick, který obsahuje jak kód QML tak C++ a zahrnuje QDeclarativeView. + Vytvoří program Qt Quick, který obsahuje jak kód QML tak C++ a zahrnuje QDeclarativeView. Můžete tuto aplikaci sestavit a nasadit jak na stolním počítači tak na mobilních zařízeních. Tento typ projektu například umožní vytvoření podepsaného balíčku Symbian Installation System (SIS). Dodatečně dovoluje užití předem zhotovených součástek UI ve vaší aplikaci Qt Quick. Toto vyžaduje Qt 4.7.4 nebo novější. @@ -36255,6 +38499,62 @@ Můžete tuto aplikaci sestavit a nasadit jak na stolním počítači tak na mobilních zařízeních. Tento typ projektu například umožní vytvoření podepsaného balíčku Symbian Installation System (SIS). Dodatečně dovoluje užití předem zhotovených součástek UI ve vaší aplikaci Qt Quick. Toto vyžaduje Qt 4.7.3 nebo novější. + + Creates a Qt Quick application project that can contain both QML and C++ code and includes a QDeclarativeView. + + + Vytvoří projekt programu Qt Quick, který může obsahovat jak kód QML tak C++ a zahrnuje QDeclarativeView. + + + + + Qt Quick Application (Built-in Elements) + Program Qt Quick (jen vestavěné prvky) + + + The built-in elements in the QtQuick namespace allow you to write cross-platform applications with a custom look and feel. + +Requires <b>Qt 4.7.0</b> or newer. + Vestavěné prvky ve jmenném prostoru QtQuick dovolují psát víceplatformní programy s uživatelsky stanoveným vzhledem. + +Vyžaduje Qt 4.7.0 nebo novější. + + + Qt Quick Application for Symbian + Program Qt Quick pro Symbian + + + The Qt Quick Components for Symbian are a set of ready-made components that are designed with specific native appearance for the Symbian platform. + +Requires <b>Qt 4.7.4</b> or newer, and the component set installed for your Qt version. + Součástky Qt Quick pro Symbian jsou souborem předpřipravených součástek, který jsou navrženy s původním vzhledem pro Symbian. + +Vyžaduje <b>Qt 4.7.4</b> nebo novější a soubor součástek nainstalovaný pro tuto verzi Qt. + + + Qt Quick Application for MeeGo Harmattan + Program Qt Quick pro Meego Harmattan + + + The Qt Quick Components for MeeGo Harmattan are a set of ready-made components that are designed with specific native appearance for the MeeGo Harmattan platform. + +Requires <b>Qt 4.7.4</b> or newer, and the component set installed for your Qt version. + Součástky Qt Quick pro Meego/Harmattan jsou souborem předpřipravených součástek, který jsou navrženy s původním vzhledem pro Meego Harmattan. + +Vyžaduje <b>Qt 4.7.4</b> nebo novější, a soubor součástek nainstalovaný pro vaši verzi Qt. + + + Qt Quick Application (from Existing QML File) + Program Qt Quick (ze stávajícího souboru QML) + + + Creates a deployable Qt Quick application from existing QML files. All files and directories that reside in the same directory as the main .qml file are deployed. You can modify the contents of the directory any time before deploying. + +Requires <b>Qt 4.7.0</b> or newer. + Vytvoří nasaditelný program Qt Quick ze stávajících souborů QML. Všechny soubory a adresáře ležící ve stejném adresáři jako hlavní soubor .qml budou nasazeny. Obsah adresáře můžete měnit kdykoli před nasazením. + +Vyžaduje <b>Qt 4.7.0</b> nebo novější. + Qt4ProjectManager::Internal::QtQuickComponentSetOptionsPage @@ -36263,39 +38563,47 @@ Vybrat soubor QML + Select Existing QML file + Použít existující soubor QML + + Qt Quick Application Type - Typ programu Qt Quick + Typ programu Qt Quick + + + All files and directories that reside in the same directory as the main QML file are deployed. You can modify the contents of the directory any time before deploying. + Všechny soubory a adresáře nacházející se v tomtéž adresáři jako hlavní soubor QML jsou připraveny k nasazení. Obsah adresáře lze před nasazením kdykoli upravit. Qt4ProjectManager::Internal::SubdirsProjectWizard Subdirs Project - Subdirs projekt + Projekt s podadresáři Creates a qmake-based subdirs project. This allows you to group your projects in a tree structure. - Vytvoří na qmake založený projekt typu subdirs. Toto vám umožní vaše projekty seskupovat ve stromové struktuře. + Vytvoří na qmake založený projekt s podadresáři. Toto vám umožní projekty seskupovat ve stromové struktuře. Done && Add Subproject - Hotovo a přidat podřízený projekt + Hotovo a přidat dílčí projekt Finish && Add Subproject - Dokončit a přidat podřízený projekt + Dokončit a přidat dílčí projekt New Subproject Title of dialog - Nový podřízený projekt + Nový dílčí projekt Qt4ProjectManager::Internal::SubdirsProjectWizardDialog This wizard generates a Qt4 subdirs project. Add subprojects to it later on by using the other wizards. - Tento průvodce vytvoří projekt Qt4 typu subdirs. S pomocí dalších průvodců lze do něj později přidat další podřízené projekty. + Tento průvodce vytvoří projekt Qt4 s podadresáři. Dílčí projekty lze později přidat s pomocí dalších průvodců. @@ -36307,7 +38615,7 @@ Qt Creator can set up the following targets for project <b>%1</b>: %1: Project name - Qt Creator může pro projekt nastavit následující cíle <b>%1</b>: + Qt Creator může pro projekt <b>%1</b> nastavit následující cíle: @@ -36349,43 +38657,43 @@ QtSupport::Internal::GettingStartedWelcomePage Demos and Examples - Dema a příklady + Dema a příklady Copy Project to writable Location? - Má se projekt zkopírovat do zapisovatelného umístění? + Má se projekt zkopírovat do zapisovatelného umístění? <p>The project you are about to open is located in the write-protected location:</p><blockquote>%1</blockquote><p>Please select a writable location below and click "Copy Project and Open" to open a modifiable copy of the project or click "Keep Project and Open" to open the project in location.</p><p><b>Note:</b> You will not be able to alter or compile your project in the current location.</p> - <p>Projekt, který se chystáte otevřít, je umístěn v proti zápisu chráněném umístění:</p><blockquote>%1</blockquote><p>Vyberte, prosím, níže zapisovatelné umístění a klepněte na "Kopírovat projekt a otevřít" kvůli otevření upravovatelné kopie projektu, nebo klepněte na "Zachovat projekt a otevřít" kvůli otevření projektu v umístění.</p><p><b>Poznámka:</b> V současném umístění projekt nelze ani sestavit ani změnit.</p> + <p>Projekt, který se chystáte otevřít, je v místě chráněném před zápisem:</p><blockquote>%1</blockquote><p>Níže vyberte zapisovatelné umístění a klepněte na "Kopírovat projekt a otevřít" abyste otevřeli upravovatelnou kopii projektu, anebo klepněte na "Zachovat projekt a otevřít" abyste otevřeli projektu na místě.</p><p><b>Poznámka:</b> V současném umístění projekt nelze ani sestavit ani změnit.</p> &Location: - &Umístění: + &Umístění: &Copy Project and Open - &Kopírovat projekt a otevřít + &Kopírovat projekt a otevřít &Keep Project and Open - &Zachovat projekt a otevřít + &Zachovat projekt a otevřít Cannot Use Location - Umístění nelze použít + Umístění nelze použít The specified location already exists. Please specify a valid location. - Zadané umístění již existuje. Zadejte, prosím, platné umístění. + Zadané umístění již existuje. Zadejte, prosím, platné umístění. Cannot Copy Project - Chyba při kopírování projektu + Chyba při kopírování projektu Failed to open project - Nepodařilo se otevřít projekt + Nepodařilo se otevřít projekt Getting Started @@ -36415,7 +38723,7 @@ - %1 Reason: %2 - Pomocnou knihovnu pro výstup dat QML se nepodařilo vytvořit v žádném z následujících adresářů: + Ladicí knihovnu QML se nepodařilo sestavit v žádném z následujících adresářů: - %1 Důvod: %2 @@ -36425,7 +38733,7 @@ Qt4ProjectManager::QmlDumpTool Only available for Qt for Desktop and Qt for Qt Simulator. - Dostupné jen pro Qt pro Desktop a Qt pro Simulator. + Dostupné jen pro "Qt pro Desktop" a "Qt pro Qt Simulator". Only available for Qt 4.7.1 or newer. @@ -36451,7 +38759,7 @@ - %1 Reason: %2 - qmldump se nepodařilo vytvořit v žádném z následujících adresářů: + qmldump se nepodařilo sestavit v žádném z následujících adresářů: - %1 Důvod: %2 @@ -36461,14 +38769,14 @@ QmlDumpBuildTask Building helper - Pomocná knihovna pro výstup + Pomocná knihovna pro sestavení Qt4ProjectManager::QmlObserverTool Only available for Qt for Desktop or Qt for Qt Simulator. - Dostupné jen pro Qt pro Desktop a Qt pro Simulator. + Dostupné jen pro "Qt pro Desktop" a "Qt pro Qt Simulator". Only available for Qt 4.7.1 or newer. @@ -36494,7 +38802,7 @@ - %1 Reason: %2 - Pozorovatele QML (QMLObserver) se nepodařilo vytvořit v žádném z následujících adresářů: + Pozorovatele QML (QMLObserver) se nepodařilo sestavit v žádném z následujících adresářů: - %1 Důvod: %2 @@ -36512,7 +38820,7 @@ Manual - Ruční + Ručně Remove invalid Qt Versions @@ -36528,11 +38836,11 @@ No tool chain can produce code for this Qt version. Please define one or more tool chains. - Žádný řetězec nástrojů nemůže vytvářet kód pro tuto verzi Qt. Stanovte, prosím, jeden nebo více řetězců nástrojů. + Žádná sada nástrojů nemůže vytvářet kód pro tuto verzi Qt. Stanovte, prosím, jednu či více sad nástrojů. Not all possible target environments can be supported due to missing tool chains. - Ne všechna cílová prostředí mohou být podporována, neboť chybí některé řetězce nástrojů. + Ne všechna cílová prostředí mohou být podporována, neboť chybí některé sady nástrojů. The following ABIs are currently not supported:<ul><li>%1</li></ul> @@ -36544,7 +38852,7 @@ Debugging Helper Build Log for '%1' - Zápis o vytvoření pomocné knihovny pro výstup dat pro '%1' + Záznam o sestavení pomocného ladicího programu pro '%1' Select a qmake executable @@ -36723,6 +39031,14 @@ Start Wizard Spustit průvodce + + Device Configuration Wizard Selection + Zřízení nového nastavení zařízení + + + Available device types: + Dostupné typy zařízení: + RemoteLinux::Internal::MaddeDeviceConfigurationFactory @@ -38036,7 +40352,7 @@ RemoteLinux::PublicKeyDeploymentDialog Waiting for file name... - Čeká se na název souboru... + Čeká se na název souboru... Choose Public Key File @@ -38204,6 +40520,12 @@ Preparing remote side ... + Připravuje se vzdálená strana... + + + + Preparing remote side... + Připravuje se vzdálená strana... @@ -38236,7 +40558,7 @@ No device configuration set. - Nebylo zadáno žádné nastavení zařízení. + Nebylo zadáno žádné nastavení zařízení. No active build configuration. @@ -38316,20 +40638,28 @@ Argumenty: + <default> + <Výchozí> + + + Working directory: + Pracovní adresář: + + C++ only - jen C++ + jen C++ QML only - jen QML + jen QML C++ and QML - C++ a QML + C++ a QML Debugging type: - Typ ladění: + Typ ladění: Base environment for this run configuration: @@ -38365,6 +40695,12 @@ Starting remote process ... + Spouští se vzdálený proces... + + + + Starting remote process... + Spouští se vzdálený proces... @@ -38449,6 +40785,14 @@ Select Encoding Vybrat kódování + + Delete UTF-8 BOM on Save + Smazat UTF-8 BOM při uložení + + + Add UTF-8 BOM on Save + Přidat UTF-8 BOM při uložení + TextEditor::FunctionHintProposalWidget @@ -38489,7 +40833,7 @@ Download Definitions... - Stáhnout vymezení... + Stáhnout definici... Autodetect @@ -38497,11 +40841,11 @@ Autodetect Definitions - Automaticky zjistit vymezení + Automaticky zjistit definici No pre-installed definitions could be found. - Žádné předem instalované soubory s vymezeními se nepodařilo nalézt. + Žádné předem instalované soubory s definicemi se nepodařilo nalézt. Error connecting to server. @@ -38528,7 +40872,7 @@ Download Definitions - Stáhnout vymezení + Stáhnout definici Download Information @@ -38536,29 +40880,53 @@ There is already one download in progress. Please wait until it is finished. - Již běží jedno stahování. Počkejte, prosím, dokud nebude dokončeno. + Již probíhá jedno stahování. Počkejte, prosím, dokud nebude dokončeno. - - - TextEditor::Internal::Manager - Registering definitions - Zapisují se vymezení + Dialog + Dialog - Downloading definitions - Stahují se vymezení + Definitions + Soubory s definicemi - Error downloading selected definition(s). - Chyba při stahování vybraného(ných) vymezení. + Select All + Vybrat vše - Error downloading one or more definitions. - Chyba při stahování jednoho nebo více vymezení. + Clear Selection + Smazat výběr - + Invert Selection + Obrátit výběr + + + Download Selected Definitions + Stáhnout vybrané soubory s definicemi + + + + TextEditor::Internal::Manager + + Registering definitions + Přihlašují se definice + + + Downloading definitions + Stahují se definice + + + Error downloading selected definition(s). + Chyba při stahování vybrané definice. + + + Error downloading one or more definitions. + Chyba při stahování jedné nebo více definicí. + + + Please check the directory's access rights. Prověřte, prosím, oprávnění pro přístup k adresáři. @@ -38594,11 +40962,11 @@ TextEditor::Internal::PlainTextEditorFactory A highlight definition was not found for this file. Would you like to try to find one? - Pro tento soubor se nepodařilo najít žádné vymezení zvýrazňování skladby. Chcete se pokusit nějaké vymezení najít? + Pro tento soubor se nepodařilo najít žádnou definici zvýrazňování skladby. Chcete se pokusit nějakou definici najít? Show highlighter options... - Ukázat nastavení zvýrazňování skladby... + Ukázat volby zvýrazňování... Show highlighter options @@ -38616,7 +40984,7 @@ TextEditor::Internal::SnippetsCollection Cannot create user snippet directory %1 - Nepodařilo se vytvořit adresář na uživatelské kousky %1 + Nepodařilo se vytvořit adresář na uživatelské úryvky %1 @@ -38639,18 +41007,18 @@ Error reverting snippet. - Chyba při vracení spouštěče. + Chyba při vracení úryvku zpět. TextEditor::Internal::SnippetsSettingsPagePrivate Snippets - Kousky + Úryvky Error While Saving Snippet Collection - Chyba při ukládání sbírky kousků + Chyba při ukládání sbírky úryvků Error @@ -38658,7 +41026,7 @@ No snippet selected. - Nevybrán žádný kousek. + Žádný úryvek nebyl vybrán. @@ -38678,11 +41046,11 @@ Valgrind::Internal::CallgrindTool Valgrind Function Profiler - Profilování funkcí Valgrind + Profilování funkcí pomocí Valgrind Valgrind Profile uses the "callgrind" tool to record function calls when a program runs. - Profilování Valgrind používá nástroj "callgrind" na nahrání volání funkce během spuštění programu. + Profilování Valgrind používá nástroj "callgrind" pro záznam volání funkcí během spuštění programu. Profile Costs of this Function and its Callees @@ -38693,7 +41061,7 @@ Valgrind::Internal::CallgrindToolPrivate Callers - Vyvolávající + Volající Functions @@ -38709,23 +41077,23 @@ Request the dumping of profile information. This will update the callgrind visualization. - Požaduje výpis informací profileru. Tím bude aktualizována vizualizace callgrind. + Žádost o výpis informací profileru. Tím bude aktualizována vizualizace callgrind. Reset all event counters. - Nastavit všechna počítadla událostí znovu. + Vynulovat všechna počítadla událostí. Pause event logging. No events are counted which will speed up program execution during profiling. - Způsobí, že nejsou přijímány žádné události, což zrychlí provádění programu během profilování. + Způsobí, že nejsou zaznamenávány žádné události, což zrychlí provádění programu během profilování. Go back one step in history. This will select the previously selected item. - Jít jeden krok zpět v historii. Předtím vybraný prvek bude znovu vybrán. + Jít o krok zpět v historii. Předtím vybraný prvek bude znovu vybrán. Go forward one step in history. - Jít jeden krok vpřed v historii. + Jít o krok vpřed v historii. Selects which events from the profiling data are shown and visualized. @@ -38749,11 +41117,11 @@ Relative Costs to Parent - Náklady poměrné k nadřazenému prvku (rodiči) + Náklady v poměru k nadřazenému prvku (rodiči) Show costs relative to parent functions inclusive cost. - Ukázat náklady poměrné k zahrnutým nákladům volané funkce. + Ukázat náklady v poměru k zahrnutým nákladům nadřazené funkce. Cost Format @@ -38808,7 +41176,7 @@ Valgrind::Internal::Visualisation All functions with an inclusive cost ratio higher than %1 (%2 are hidden) - Všechny funkce se zahrnutým poměrem nákladů větším než %1 (%2 jsou skryty) + Všechny funkce s poměrem zahrnutým nákladů větším než %1 (%2 jsou skryty) @@ -38866,11 +41234,11 @@ Definite Memory Leaks - Jednoznačná vytečení paměti + Jednoznačné úniky paměti Possible Memory Leaks - Možná vytečení paměti + Možné úniky paměti Use of Uninitialized Memory @@ -38886,7 +41254,7 @@ Valgrind Analyze Memory uses the "memcheck" tool to find memory leaks - Rozbor paměti s Valgrind používá nástroj "memcheck" pro nalezení vytečení paměti + Rozbor paměti s Valgrind používá nástroj "memcheck" pro nalezení úniků paměti Memory Issues @@ -38894,11 +41262,11 @@ Go to previous leak. - Jít na předchozí vytečení paměti. + Jít na předchozí únik. Go to next leak. - Jít na další vytečení paměti. + Jít na další únik. Error Filter @@ -38921,7 +41289,7 @@ Caller - Vyvolávající + Volající Cost @@ -38944,7 +41312,7 @@ Resetting event counters... - Nastavuje se znovu počítadlo událostí... + Vynulování počítadel událostí... Pausing instrumentation... @@ -39093,7 +41461,7 @@ mispredicted - Chybně předpovídáno + Chybně předpověděno executed @@ -39101,7 +41469,7 @@ miss - Neúspěch + Minutí access @@ -39127,7 +41495,7 @@ Valgrind::Memcheck::MemcheckRunner No network interface found for remote analysis. - Nepodařilo se najít žádné síťové rozhraní pro dálkově řízený rozbor. + Nebylo nalezeno žádné síťové rozhraní pro dálkově řízený rozbor. Select Network Interface @@ -39135,14 +41503,18 @@ More than one network interface was found on your machine. Please select which one you want to use for remote analysis. - Bylo nalezeno více síťových rozhraní. Vyberte, prosím, jedno, které chcete používat pro dálkově řízený rozbor. + Bylo nalezeno více síťových rozhraní. Vyberte, prosím, to, které chcete používat pro dálkově řízený rozbor. + + + No Network Interface was chosen for remote analysis + Nebylo vybráno žádné síťové rozhraní pro dálkově řízený rozbor Valgrind::RemoteValgrindProcess Could not determine remote PID. - Nepodařilo se určit ID vzdáleného procesu. + Nepodařilo se určit PID vzdáleného procesu. @@ -39181,11 +41553,11 @@ Leaked Blocks - Vytečené (neuvolněné) bloky + Uniklé (neuvolněné) bloky Leaked Bytes - Vytečené (neuvolněné) byty + Uniklé (neuvolněné) byty Helgrind Thread ID @@ -39227,7 +41599,7 @@ Unexpected token type %1 - Neočekávaný symbol %1 + Neočekávaný typ symbolu %1 Could not parse protocol version from "%1" @@ -39255,7 +41627,7 @@ Could not parse error kind, tool not yet set. - Druh chyby se nepodařilo určit, protože není nastaven žádný nástroj. + Druh chyby se nepodařilo určit, ještě totiž není nastaven žádný nástroj. Unknown state "%1" @@ -39320,8 +41692,8 @@ ** Error: "%1" could not be started: %2 ** - -** Chyba: "%1" se nepodařilo spustit: %2 ** + ** Chyba: "%1" se nepodařilo spustit: %2 ** + ** Error: no valgrind executable set ** @@ -39344,7 +41716,7 @@ Valgrind::Internal::ValgrindRunControlFactory Analyzer - Analyzátor + Analyzátor @@ -39358,7 +41730,7 @@ VcsBase::Internal::CommonSettingsWidget Command used for reverting diff chunks - Příkaz používaný pro vrácení jednotlivých změn v oznámení Diff + Příkaz používaný pro vrácení jednotlivých změn @@ -39369,7 +41741,7 @@ Timed out after %1s waiting for the process %2 to finish. - Překročení časového omezení %1s při čekání na ukončení %2. + Překročení časového omezení %1s při čekání na ukončení %2. Working... @@ -39380,7 +41752,7 @@ VcsBase::VcsBaseEditorWidget Annotate "%1" - Opatřit vysvětlivkami "%1" + Opatřit anotacemi "%1" Copy "%1" @@ -39392,7 +41764,7 @@ Send to CodePaster... - Poslat CodePaster... + Poslat na CodePaster... Apply Chunk... @@ -39488,31 +41860,31 @@ SymbianUtils::VirtualSerialDevice The port %1 could not be opened: %2 (POSIX error %3) - Přípojku %1 se nepodařilo otevřít: %2 (Chyba POSIX %3) + Port %1 se nepodařilo otevřít: %2 (Chyba POSIX %3) Unable to retrieve terminal settings of port %1: %2 (POSIX error %3) - Terminálová nastavení přípojky %1 se nepodařilo vyvolat: %2 (Chyba POSIX %3) + Terminálová nastavení portu %1 se nepodařilo získat: %2 (Chyba POSIX %3) Unable to apply terminal settings to port %1: %2 (POSIX error %3) - Terminálová nastavení se na přípojku %1nepodařilo použít: %2 (Chyba POSIX %3) + Terminálová nastavení se na port %1 nepodařilo použít: %2 (Chyba POSIX %3) Cannot write to port %1: %2 (POSIX error %3) - Nepodařilo se zapisovat do přípojky %1: %2 (Chyba POSIX %3) + Nelze zapisovat na port %1: %2 (Chyba POSIX %3) The function select() returned an error on port %1: %2 (POSIX error %3) - U přípojky %1 vrátila funkce select() chybu: %2 (Chyba POSIX %3) + U portu %1 vrátila funkce select() chybu: %2 (Chyba POSIX %3) Port not found - Přípojka nenalezena + Port nenalezen Port in use - Přípojka se už používá + Port se už používá Timed out @@ -39520,11 +41892,11 @@ Port unreachable - Přípojka nedosažitelná + Port je nedosažitelný The port %1 could not be opened: %2 - Přípojku %1 se nepodařilo otevřít: %2 + Port %1 se nepodařilo otevřít: %2 An error occurred while waiting for read notifications from %1: %2 @@ -39559,7 +41931,7 @@ Synchronizes translator's ts files with the program code - Seřídí soubory ts od překladatele se zdrojovým textem (programovým kódem) + Seřídí soubory ts od překladatele se zdrojovým kódem programu Update Translations (lupdate) @@ -39610,7 +41982,7 @@ ExtensionSystem::Internal::PluginErrorOverview Qt Creator - Plugin loader messages - Qt Creator - Zprávy správy přídavných modulů + Qt Creator - Zprávy zavaděče přídavných modulů The following plugins have errors and cannot be loaded: @@ -39625,26 +41997,26 @@ AttachToQmlPortDialog Start Debugger - Spustit ladicí program + Spustit ladicí program &Host: - &Hostitel: + &Hostitel: &Port: - &Přípojka: + &Přípojka: MainView Painting - Malování + Vykreslení Compiling - Sestavení + Překlad Creating @@ -39652,302 +42024,322 @@ Binding - Svázání + Vázání + + + Handling Signal + Zacházení se signálem Signal Handler - Manipulátor signálu + Manipulátor signálu RangeDetails Duration - Doba trvání + Doba trvání Details - Podrobnosti + Podrobnosti Location - Umístění + Umístění + + + Duration: + Doba trvání: + + + Details: + Podrobnosti: + + + Location: + Umístění: + + + Binding loop detected + Zjištěna nekonečná vázací smyčka LinuxDeviceConfigurationsSettingsWidget Linux Device Configurations - Nastavení linuxového zařízení + Nastavení linuxového zařízení &Configuration: - &Nastavení: + &Nastavení: &Name: - &Název: + &Název: OS type: - Typ operačního systému: + Typ operačního systému: Device type: - Typ zařízení: + Typ zařízení: Authentication type: - Druh ověření pravosti: + Druh ověření pravosti: Password - Heslo + Heslo &Key - &Klíč + &Klíč &Host name: - Název &hostitelského počítače: + Název &hostitelského počítače: IP or host name of the device - IP nebo název hostitelského počítače zařízení + IP nebo název hostitelského počítače zařízení &SSH port: - Přípojka &SSH: + Přípojka &SSH: Free ports: - Volné přípojky: + Volné přípojky: You can enter lists and ranges like this: 1024,1026-1028,1030 - Můžete zadat seznamy a oblasti jako jsou tyto: 1024,1026-1028,1030 + Můžete zadat seznamy a oblasti jako jsou tyto: 1024,1026-1028,1030 Timeout: - Časové omezení: + Časové omezení: s - s + s &Username: - &Uživatelské jméno: + &Uživatelské jméno: &Password: - He&slo: + He&slo: Show password - Ukázat heslo + Ukázat heslo Private key file: - Soubor se soukromým klíčem: + Soubor se soukromým klíčem: Set as Default - Nastavit jako výchozí + Nastavit jako výchozí &Add... - &Přidat... + &Přidat... &Remove - &Odstranit + &Odstranit Set As Default - Nastavit jako výchozí + Nastavit jako výchozí Click here if you do not have an SSH key yet. - Klepněte sem, pokud ještě nemáte nějaký klíč SSH. + Klepněte sem, pokud ještě nemáte nějaký klíč SSH. &Generate SSH Key... - &Vytvořit klíč SSH... + &Vytvořit klíč SSH... LinuxDeviceTestDialog Device Test - Zkouška zařízení + Zkouška zařízení ProFilesUpdateDialog Maemo Deployment Issue - Problém s nasazením Maemo + Problém s nasazením Maemo The project files listed below do not contain deployment information, which means the respective targets cannot be deployed to and/or run on a device. Qt Creator will add the missing information to these files if you check the respective rows below. - Níže uvedené projektové soubory neobsahují požadované informace o nasazení, což znamená, že pro příslušné cíle nelze provést žádné nasazení a/nebo je nelze spustit na nějakém mobilním zařízení. Vyberte, prosím, příslušné projekty, do nichž má Qt Creator přidat chybějící informace. + Níže uvedené projektové soubory neobsahují požadované informace o nasazení, což znamená, že pro příslušné cíle nelze provést žádné nasazení a/nebo je nelze spustit na nějakém mobilním zařízení. Vyberte, prosím, příslušné projekty, do nichž má Qt Creator přidat chybějící informace. &Check all - Označit &vše + Označit &vše &Uncheck All - Odstranit označení u vš&eho + Odstranit označení u vš&eho RemoteLinuxDeployConfigurationWidget Form - Formulář + Formulář Device configuration: - Nastavení zařízení: + Nastavení zařízení: <a href="irrelevant">Manage device configurations</a> - <a href="irrelevant">Spravovat nastavení zařízení</a> + <a href="irrelevant">Spravovat nastavení zařízení</a> These show the INSTALLS settings from the project file(s). - Ukáže nastavení INSTALLS z projektového souboru. + Ukáže nastavení INSTALLS z projektového souboru. Files to install for subproject: - Soubory k instalaci pro podřízený projekt: + Soubory k instalaci pro podřízený projekt: Edit the project file to add or remove entries. - Upravit soubor s projektem pro přidání nebo odstranění záznamů. + Upravit soubor s projektem pro přidání nebo odstranění záznamů. RemoteLinuxProcessesDialog List of Remote Processes - Seznam vzdálených procesů + Seznam vzdálených procesů &Filter by process name: - &Filtrovat podle názvu procesu: + &Filtrovat podle názvu procesu: &Update List - &Obnovit seznam + &Obnovit seznam &Kill Selected Process - &Ukončit vybraný proces + &Ukončit vybraný proces SshKeyCreationDialog SSH Key Configuration - Nastavení klíče SSH + Nastavení klíče SSH Options - Volby + Volby Key &size: - &Velikost klíče: + &Velikost klíče: Key algorithm: - Algoritmus klíče: + Algoritmus klíče: &RSA - &RSA + &RSA &DSA - &DSA + &DSA Key - Klíč + Klíč &Generate SSH Key - &Vytvořit klíč SSH + &Vytvořit klíč SSH Save P&ublic Key... - Uložit v&eřejný klíč... + Uložit v&eřejný klíč... Save Pr&ivate Key... - Uložit &soukromý klíč... + Uložit &soukromý klíč... &Close - &Zavřít + &Zavřít StartGdbServerDialog List of Remote Processes - Seznam vzdálených procesů + Seznam vzdálených procesů Device: - Zařízení: + Zařízení: &Filter by process name: - &Filtrovat podle názvu procesu: + &Filtrovat podle názvu procesu: &Attach to Selected Process - &Připojit vybranému procesu + &Připojit vybranému procesu &Update List - &Obnovit seznam + &Obnovit seznam TextEditor::CodeStyleSelectorWidget Form - Formulář + Formulář Current settings: - Nynější nastavení: + Nynější nastavení: Copy... - Kopírovat... + Kopírovat... Edit... - Upravit... + Upravit... Remove - Odstranit + Odstranit Export... - Vyvést... + Vyvést... Import... - Zavést... + Zavést... Copy Code Style @@ -40023,6 +42415,22 @@ Není dostupný žádný zdrojový kód + <program> + <program> + + + Main Program + Hlavní program + + + Animation Timer Update + Aktualizace časovače animace + + + <Animation Update> + <aktualizace animace> + + No data to save Nejsou přítomna žádná data k uložení @@ -40038,6 +42446,14 @@ Error while parsing %1 Chyba při zpracování %1 + + Invalid version of QML Trace file. + Neplatná verze souboru QML Trace. + + + %1 animations at %2 FPS + %1 animace při %2 FPS + Utils::Ssh @@ -40143,24 +42559,44 @@ Windows Explorer se nepodařilo spustit, protože se v cestě nepodařilo nalézt soubor explorer.exe. + Show in Explorer + Ukázat v průzkumníku + + + Show in Finder + Ukázat v hledáčku + + + Show Containing Folder + Ukázat obsaženou složku + + + Open Command Prompt Here + Otevřít výzvu k příkazu zde + + + Open Terminal Here + Otevřít terminál zde + + Show in Explorer... - Ukázat v průzkumníku... + Ukázat v průzkumníku... Show in Finder... - Ukázat v nálezci... + Ukázat v nálezci... Show Containing Folder... - Ukázat obsaženou složku... + Ukázat obsaženou složku... Open Command Prompt Here... - Otevřít výzvu k příkazu zde... + Otevřít výzvu k příkazu zde... Open Terminal Here... - Otevřít terminál zde... + Otevřít terminál zde... @@ -40178,11 +42614,11 @@ Apply changes to definition - Použít změny vymezení + Použít změny definice Apply changes to declaration - Použít změny prohlášení + Použít změny deklarace Apply function signature changes @@ -40291,10 +42727,14 @@ <p>Vyskytla se nezpracovaná výjimka:</p><p>%1</p> - <p>An uncaught exception occurred in <i>%1</i>:</p><p>%2</p> + <p>An uncaught exception occurred in '%1':</p><p>%2</p> <p>V <i>%1</i> se vyskytla nezpracovaná výjimka:</p><p>%2</p> + <p>An uncaught exception occurred in <i>%1</i>:</p><p>%2</p> + <p>V <i>%1</i> se vyskytla nezpracovaná výjimka:</p><p>%2</p> + + Uncaught Exception Nezpracovaná výjimka @@ -40303,7 +42743,7 @@ Find::IFindFilter Case sensitive - Rozlišování velkých a malých písmen + Rozlišovat velká a malá písmena Whole words @@ -40333,6 +42773,14 @@ Zrušit + Repeat the search with same parameters + Opakovat hledání se stejnými parametry + + + Search again + Hledat ještě jednou + + Replace with: Nahradit: @@ -40591,6 +43039,18 @@ Could not save icon to '%1'. Ikonu se nepodařilo uložit pod '%1'. + + Form + Formulář + + + Add Desktop File + Přidat soubor pro desktop + + + Add Launcher Icon... + Přidat ikonu spouštěče... + Madde::Internal::MaemoDeploymentMounter @@ -40609,6 +43069,38 @@ MeeGo Device Zařízení MeeGo + + WizardPage + WizardPage + + + The name to identify this configuration: + Název nastavení: + + + The system running on the device: + Na mobilním zařízení běžící systém: + + + The kind of device: + Typ zařízení: + + + Emulator + Emulátor + + + Hardware Device + Mobilní zařízení + + + The device's host name or IP address: + Název hostitelského počítače nebo IP adresa zařízení: + + + The SSH server port: + Port SSH serveru: + Madde::Internal::MaemoDeviceConfigWizardPreviousKeySetupCheckPage @@ -40623,6 +43115,30 @@ Existing Keys Check Přezkoušení již existujících klíčů + + WizardPage + WizardPage + + + Do you want to re-use an existing pair of keys or should a new one be created? + Chcete znovu použít existující dvojici klíčů, nebo se má vytvořit nová? + + + Re-use existing keys + Použít existující klíče znovu + + + File containing the public key: + Soubor obsahující veřejný klíč: + + + File containing the private key: + Soubor obsahující soukromý klíč: + + + Create new keys + Vytvořit nové klíče + Madde::Internal::MaemoDeviceConfigWizardKeyCreationPage @@ -40643,10 +43159,14 @@ Zadaný adresář buď neexistuje nebo jej nelze vytvořit. - Creating keys ... + Creating keys... Vytváří se klíče... + Creating keys ... + Vytváří se klíče... + + Key creation failed: %1 Při vytváření klíče se vyskytla chyba: %1 @@ -40658,6 +43178,22 @@ Could Not Save Key File Chyba při ukládání souboru s klíčem + + WizardPage + WizardPage + + + Qt Creator will now generate a new pair of keys. Please enter the directory to save the key files in and then press "Create Keys". + Qt Creator nyní vytvoří dvojici klíčů. Zadejte, prosím, adresář, do kterého se klíče mají uložit a potom potvrďte "Vytvořit klíče". + + + Directory: + Adresář: + + + Create Keys + Vytvořit klíče + Madde::Internal::MaemoDeviceConfigWizardKeyDeploymentPage @@ -40685,6 +43221,40 @@ Done. Hotovo. + + WizardPage + WizardPage + + + To deploy the public key to your device, please execute the following steps: +<ul> +<li>Connect the device to your computer (unless you plan to connect via WLAN).</li> +<li>On the device, start the "%%%maddev%%%" application.</li> +<li>In "%%%maddev%%%", configure the device's IP address to the one shown below (or edit the field below to match the address you have configured).</li> +<li>In "%%%maddev%%%", press "Developer Password" and enter it in the field below.</li> +<li>Click "Deploy Key"</li> + + Abyste klíč poslal do mobilního zařízení, proveďte, prosím, následující kroky: +<ul> +<li>Spojte mobilní zařízení s počítačem (pokud se nepoužívá žádné WLAN).</li> +<li>Spusťte program "%%%maddev%%%" na mobilním zařízení.</li> +<li>nastavte dole zobrazenou adresu IP v programu "%%%maddev%%%" (nebo upravte pole níže, aby odpovídalo adrese, kterou jste nastavil).</li> +<li>Potvrďte "heslo vývojáře" v programu "%%%maddev%%%" a zadejte je v dole zobrazeném poli "Heslo".</li> +<li>Potvrďte "Poslat klíč".</li> + + + + Device address: + Adresa zařízení: + + + Password: + Heslo: + + + Deploy Key + Poslat klíč + Madde::Internal::MaemoDeviceConfigWizardFinalPage @@ -40722,10 +43292,14 @@ Bez verze Qt nelze provést žádnou instalaci balíčku na sysroot. - Installing package to sysroot ... + Installing package to sysroot... Instaluje se balíček na sysroot... + Installing package to sysroot ... + Instaluje se balíček na sysroot... + + Installation to sysroot failed, continuing anyway. Instalace balíčku na sysroot se nezdařila, ale pokračuje se. @@ -40755,10 +43329,14 @@ Bez platné verze Qt nelze na sysroot kopírovat. - Copying files to sysroot ... + Copying files to sysroot... Kopírují se soubory na sysroot... + Copying files to sysroot ... + Kopírují se soubory na sysroot... + + Sysroot installation failed: %1 Continuing anyway. Instalace na sysroot se nezdařila: %1, @@ -40784,7 +43362,7 @@ Creating package file ... - Vytváří se soubor s balíčkem... + Vytváří se soubor s balíčkem... Package created. @@ -40795,6 +43373,14 @@ Chyba při vytváření balíčku: Žádná verze Qt. + No Qt4 build configuration + Žádné nastavení sestavování Qt4 + + + Creating package file... + Vytváří se soubor s balíčkem... + + Package Creation: Running command '%1'. Vytvoření balíčku: Provádí se příkaz '%1'. @@ -40897,7 +43483,11 @@ Choose Image (will be scaled to 48x48 pixels if necessary) - Vyberte ikonu (její velikost bude změněna na 48x48 pixelů, pokud to bude potřeba) + Vyberte ikonu (její velikost bude změněna na 48x48 pixelů, pokud to bude potřeba) + + + Choose Image (will be scaled to %1x%2 pixels if necessary) + Vyberte obrázek (v případě potřeby bude jeho velikost změněna na %1x%2 pixelů) Could Not Set New Icon @@ -40927,6 +43517,50 @@ Could Not Set Version Number Nepodařilo se nastavit číslo verze + + Package name: + Název balíčku: + + + Package version: + Verze balíčku: + + + Major: + Větší: + + + Minor: + Menší: + + + Patch: + Záplata: + + + Short package description: + Krátký popis balíčku: + + + Name to be displayed in Package Manager: + Název k zobrazení ve správci balíčků: + + + Icon to be displayed in Package Manager: + Ikona k zobrazení ve správci balíčků: + + + Adapt Debian file: + Upravit soubor Debianu: + + + Edit... + Upravit... + + + Edit spec file + Upravit soubor spec + Madde::Internal::MaemoDebianPackageInstaller @@ -40970,7 +43604,7 @@ Removing left-over temporary directory ... - Maže se zbývající dočasný adresář... + Maže se zbývající dočasný adresář... Error removing temporary directory: %1 @@ -40982,7 +43616,7 @@ Setting up temporary directory ... - Vytváří se dočasný adresář... + Vytváří se dočasný adresář... Error: Could not create temporary directory. @@ -41002,6 +43636,18 @@ Cleaning up temporary directory ... + Uklízí se dočasný adresář... + + + Removing left-over temporary directory... + Maže se zbývající dočasný adresář... + + + Setting up temporary directory... + Vytváří se dočasný adresář... + + + Cleaning up temporary directory... Uklízí se dočasný adresář... @@ -41009,6 +43655,10 @@ Adresář '%1' se nepodařilo vytvořit. + Could not set execute permissions for rules file: %1 + Nepodařilo se nastavit práva ke spuštění pro soubor s pravidly: %1 + + Could not copy file '%1' to '%2': %3. Soubor '%1' se nepodařilo zkopírovat do '%2': %3. @@ -41043,10 +43693,18 @@ Vytváří se soubor s balíčkem... - Starting scp ... + Starting scp... Spouští se scp... + Uploading file %1... + Nahrává se soubor %1... + + + Starting scp ... + Spouští se scp... + + SSH error: %1 Chyba SSH: %1 @@ -41072,7 +43730,7 @@ Uploading file %1 ... - Nahrává se soubor %1... + Nahrává se soubor %1... Cannot open file for reading: %1. @@ -41109,6 +43767,38 @@ Choose a private key file Vyberte soubor se soukromým klíčem + + WizardPage + WizardPage + + + Upload Settings + Nahrát nastavení + + + Garage account name: + Název účtu Garage: + + + <a href="https://garage.maemo.org/account/register.php">Get an account</a> + <a href="https://garage.maemo.org/account/register.php">Vytvořit účet</a> + + + <a href="https://garage.maemo.org/extras-assistant/index.php">Request upload rights</a> + <a href="https://garage.maemo.org/account/register.php">Požádat o práva k nahrání</a> + + + Private key file: + Soubor se soukromým klíčem: + + + Server address: + Adresa serveru: + + + Target directory on server: + Cílový adresář na serveru: + Madde::Internal::MaemoPublishingWizardFactoryFremantleFree @@ -41289,17 +43979,17 @@ WARNING: You want to mount %1 directories, but your device has only %n free ports.<br>You will not be able to run this configuration. - Varování: Chcete připojit %1 adresář, ale vaše zařízení má jen jednu volnou přípojku.<br>Toto nastavení nebudete moci provést. - Varování: Chcete připojit %1 adresáře, ale vaše zařízení má jen %n volné přípojky.<br>Toto nastavení nebudete moci provést. - Varování: Chcete připojit %1 adresářů, ale vaše zařízení má jen %n volných přípojek.<br>Toto nastavení nebudete moci provést. + Varování: Chcete připojit %1 adresář, ale vaše zařízení má jen jeden volný port.<br>Toto nastavení nebudete moci provést. + Varování: Chcete připojit %1 adresáře, ale vaše zařízení má jen %n volné porty.<br>Toto nastavení nebudete moci provést. + Varování: Chcete připojit %1 adresářů, ale vaše zařízení má jen %n volných portů.<br>Toto nastavení nebudete moci provést. WARNING: You want to mount %1 directories, but only %n ports on the device will be available in debug mode. <br>You will not be able to debug your application with this configuration. - Varování: Chcete připojit %1 adresář, ale vaše zařízení má v režimu ladění dostupnou jen jednu přípojku.<br>S tímto nastavením nebudete moci provést ladění vašeho programu. - Varování: Chcete připojit %1 adresáře, ale vaše zařízení má v režimu ladění dostupné jen %n přípojky.<br>S tímto nastavením nebudete moci provést ladění vašeho programu. - Varování: Chcete připojit %1 adresářů, ale vaše zařízení má v režimu ladění dostupných jen %n přípojek.<br>S tímto nastavením nebudete moci provést ladění vašeho programu. + Varování: Chcete připojit %1 adresář, ale vaše zařízení má v režimu ladění dostupnou jen jeden port.<br>S tímto nastavením nebudete moci provést ladění vašeho programu. + Varování: Chcete připojit %1 adresáře, ale vaše zařízení má v režimu ladění dostupné jen %n porty.<br>S tímto nastavením nebudete moci provést ladění vašeho programu. + Varování: Chcete připojit %1 adresářů, ale vaše zařízení má v režimu ladění dostupných jen %n portů.<br>S tímto nastavením nebudete moci provést ladění vašeho programu. @@ -41427,10 +44117,14 @@ Qt Creator - Do you want to remove the packaging file(s) associated with the target '%1'? + Do you want to remove the packaging files associated with the target '%1'? Chcete smazat soubory s balíčky, které patří k cíli '%1'? + Do you want to remove the packaging file(s) associated with the target '%1'? + Chcete smazat soubory s balíčky, které patří k cíli '%1'? + + Error creating packaging directory '%1'. Chyba při vytváření adresáře s balíčky '%1'. @@ -41443,15 +44137,15 @@ Madde::Internal::AbstractDebBasedQt4MaemoTarget Debian changelog file '%1' has unexpected format. - Soubor se zápisem změn Debian '%1' je v nějakém neznámém formátu. + Soubor se záznamem změn Debian '%1' je v nějakém neznámém formátu. Refusing to update changelog file: Already contains version '%1'. - Odmítnutí aktualizace souboru se zápisem změn: Již obsahuje verzi '%1'. + Odmítnutí aktualizace souboru se záznamem změn: Již obsahuje verzi '%1'. Cannot update changelog: Invalid format (no maintainer entry found). - Soubor se zápisem změn nelze aktualizovat: Neplatný formát (nenalezen žádný záznam o údržbáři). + Soubor se záznamem změn nelze aktualizovat: Neplatný formát (nenalezen žádný záznam o údržbáři). Invalid icon data in Debian control file. @@ -41466,16 +44160,28 @@ Soubor s obrázkem '%1' se nepodařilo vyvést. + Unable to create Debian templates: No Qt version set. + Nepodařilo se vytvořit žádné předlohy pro Debian. Není nastavena žádná verze Qt. + + + Unable to create Debian templates: dh_make failed (%1). + Nepodařilo se vytvořit žádné soubory předloh pro Debian: dh_make selhal (%1). + + + Unable to create debian templates: dh_make failed (%1). + Nepodařilo se vytvořit žádné soubory předloh pro Debian: dh_make selhal (%1). + + Unable to create Debian templates: No Qt version set - Nepodařilo se vytvořit žádné předlohy pro Debian. Není nastavena žádná verze Qt + Nepodařilo se vytvořit žádné předlohy pro Debian. Není nastavena žádná verze Qt Unable to create Debian templates: dh_make failed (%1) - Nepodařilo se vytvořit žádné soubory předloh pro Debian: dh_make selhal (%1) + Nepodařilo se vytvořit žádné soubory předloh pro Debian: dh_make selhal (%1) Unable to create debian templates: dh_make failed (%1) - Nepodařilo se vytvořit žádné soubory předloh pro Debian: dh_make selhal (%1) + Nepodařilo se vytvořit žádné soubory předloh pro Debian: dh_make selhal (%1) Unable to move new debian directory to '%1'. @@ -41649,7 +44355,11 @@ RemoteLinux::AbstractRemoteLinuxDeployStep Deployment failed: %1 - Nasazení se nezdařilo: %1 + Nasazení se nezdařilo: %1 + + + Cannot deploy: %1 + Nasazení nelze provést: %1 User requests deployment to stop; cleaning up. @@ -41687,7 +44397,15 @@ Upload of file '%1' failed: %2 - Nahrání souboru '%1' se nezdařilo: %2 + Nahrání souboru '%1' se nezdařilo: %2 + + + Upload of file '%1' failed. The server said: '%2'. + Nepodařilo se nahrát soubor '%1'. Server řekl: '%2'. + + + If '%1' is currently running on the remote host, you might need to stop it first. + Pokud '%1' nyní běží na vzdáleném hostiteli, budete jej patrně muset nejprve zastavit. Failed to upload file '%1'. @@ -41762,7 +44480,7 @@ RemoteLinux::Internal::LinuxDeviceConfigurationsSettingsWidget You will need at least one port. - Potřebujete alespoň jednu volnou přípojku. + Potřebujete alespoň jednu volnou přípojku. Physical Device @@ -41772,6 +44490,54 @@ Emulator Emulátor + + Linux Device Configurations + Nastavení linuxového zařízení + + + &Configuration: + &Nastavení: + + + General + Obecné + + + &Name: + &Název: + + + OS type: + Typ operačního systému: + + + Device type: + Typ zařízení: + + + OS Type Specific + Zvláštní pro typ operačního systému + + + &Add... + &Přidat... + + + &Remove + &Odstranit + + + Set As Default + Nastavit jako výchozí + + + Click here if you do not have an SSH key yet. + Klepněte sem, pokud ještě nemáte nějaký klíč SSH. + + + &Generate SSH Key... + &Vytvořit klíč SSH... + RemoteLinux::LinuxDeviceTestDialog @@ -41813,7 +44579,7 @@ Checking if specified ports are available... - Ověřuje se, zda jsou zadané přípojky dostupné... + Ověřuje se, zda jsou zadané přípojky (porty) dostupné... Error gathering ports: %1 @@ -41824,7 +44590,7 @@ The following specified ports are currently in use: %1 - Následující zadané přípojky se na zařízení používají: %1 + Následující zadané přípojky (porty) se na zařízení používají: %1 @@ -41857,6 +44623,22 @@ Updateable Project Files Aktualizovatelné projektové soubory + + Maemo Deployment Issue + Problém s nasazením Maemo + + + The project files listed below do not contain deployment information, which means the respective targets cannot be deployed to and/or run on a device. Qt Creator will add the missing information to these files if you check the respective rows below. + Níže uvedené projektové soubory neobsahují požadované informace o nasazení, což znamená, že pro příslušné cíle nelze provést žádné nasazení a/nebo je nelze spustit na nějakém mobilním zařízení. Vyberte, prosím, příslušné projekty, do nichž má Qt Creator přidat chybějící informace. + + + &Check all + Označit &vše + + + &Uncheck All + Odstranit označení u vš&eho + RemoteLinux::AbstractRemoteLinuxApplicationRunner @@ -41964,7 +44746,7 @@ RemoteLinux::RemoteLinuxDeployStepWidget <b>%1 using device</b>: %2 - <b>%1 za použití zařízení</b>: %2 + <b>%1 za použití zařízení</b>: %2 @@ -41999,11 +44781,11 @@ RemoteLinux::Internal::RemoteLinuxPlugin Start Remote Debug Server... - Spustit vzdálený ladicí server... + Spustit vzdálený ladicí server... Start Gdbserver - Spustit gdbserver + Spustit gdbserver @@ -42082,11 +44864,63 @@ RemoteLinux::Internal::SshKeyCreationDialog Save Public Key File - Uložit soubor s veřejným klíčem + Uložit soubor s veřejným klíčem Save Private Key File - Uložit soubor se soukromým klíčem + Uložit soubor se soukromým klíčem + + + SSH Key Configuration + Nastavení klíče SSH + + + Options + Volby + + + Key algorithm: + Algoritmus klíče: + + + &RSA + &RSA + + + &DSA + &DSA + + + Key &size: + &Velikost klíče: + + + Private key file: + Soubor se soukromým klíčem: + + + Public key file: + Soubor s veřejným klíčem: + + + &Generate And Save Key Pair + &Vytvořit a uložit dvojici klíčů + + + &Cancel + &Zrušit + + + Key Generation Failed + Chyba při vytváření klíče + + + Failure To Save Key File + Chyba při ukládání souboru s klíčem + + + Failed to create directory: '%1'. + Adresář '%1' se nepodařilo vytvořit. @@ -42107,10 +44941,46 @@ RemoteLinux::StartGdbServerDialog + Select Sysroot + Vybrat Sysroot + + + &Attach to Selected Process + &Připojit vybranému procesu + + + Close + Zavřít + + + Device: + Zařízení: + + + Sysroot: + Sysroot: + + + &Filter entries: + &Filtrovat záznamy: + + + &Filter by process name: + &Filtrovat podle názvu procesu: + + + List of Remote Processes + Seznam vzdálených procesů + + Remote Error Vzdálená chyba + Process aborted + Proces zrušen + + Could not retrieve list of free ports: Nepodařilo se určit seznam volných přípojek: @@ -42124,12 +44994,20 @@ Port %1 is now accessible. - Přípojka %1 je nyní dostupná. + Port %1 je nyní dostupný. + + + Server started on %1 + Server spuštěn na %1 Process gdbserver finished. Status: %1 Proces gdbserver ukončen. Stav: %1 + + Running command: %1 + Provádí se příkaz: %1 + RemoteLinux::TarPackageCreationStep @@ -42154,6 +45032,10 @@ Chyba: Soubor tar %1 se nepodařilo otevřít (%2). + No remote path specified for file '%1', skipping. + Souboru '%1' není přiřazena žádná vzdálená cesta. Přeskakuje se. + + Error writing tar file '%1': %2. Chyba při zápisu souboru tar %1: %2. @@ -42185,6 +45067,10 @@ (default) (výchozí) + + %1 (default) + %1 (výchozí) + RemoteLinux::UploadAndInstallTarPackageStep @@ -42201,7 +45087,7 @@ TextEditor::CodeStyleEditor Edit preview contents to see how the current settings are applied to custom code snippets. Changes in the preview do not affect the current settings. - Změňte obsah náhledu, abyste viděl, jak se nynější nastavení projevují na uživatelskystanovených kouscích kódu. Změny v náhledu nemají žádný vliv na současná nastavení. + Změnou obsahu náhledu zjistíte, jak se nynější nastavení projeví na uživatelsky stanovených úryvcích kódu. Změny v náhledu nemají žádný vliv na současná nastavení. @@ -42241,6 +45127,7 @@ Path: %1 Filter: %2 %3 + %3 is filled by BaseFileFind::runNewSearch Cesta: %1 Filtr: %2 %3 @@ -42311,4 +45198,5909 @@ Nepodařilo se spustit proces, protože nebyl zadán žádný spustitelný soubor + + Utils::SshKeyCreationDialog + + SSH Key Configuration + Nastavení klíče SSH + + + Options + Volby + + + Key algorithm: + Algoritmus klíče: + + + &RSA + &RSA + + + &DSA + &DSA + + + Key &size: + &Velikost klíče: + + + Private key file: + Soubor se soukromým klíčem: + + + Public key file: + Soubor s veřejným klíčem: + + + &Generate And Save Key Pair + &Vytvořit a uložit dvojici klíčů + + + &Cancel + &Zrušit + + + Key Generation Failed + Chyba při vytváření klíče + + + Failure To Save Key File + Chyba při ukládání souboru s klíčem + + + Failed to create directory: '%1'. + Adresář '%1' se nepodařilo vytvořit. + + + + Analyzer::Internal::StartRemoteDialog + + Start Remote Analysis + Spustit vzdálený rozbor + + + Remote + Vzdálený + + + Host: + Hostitelský počítač: + + + User: + Uživatel: + + + Port: + Port: + + + You need to pass either a password or an SSH key. + Potřebujete buď heslo nebo klíč SSH. + + + Password: + Heslo: + + + Private key: + Soukromý klíč: + + + Target + Cíl + + + Executable: + Spustitelný soubor: + + + Arguments: + Argumenty: + + + Working directory: + Pracovní adresář: + + + + AddNewAVDDialog + + Create new AVD + Vytvořit nový AVD + + + Name: + Název: + + + Target: + Cíl: + + + SD card size: + Velikost karty SD: + + + MiB + MiB + + + + AndroidCreateKeystoreCertificate + + Create a keystore and a cetificate + Vytvořit úložiště pro klíč a certifikát + + + Keystore + Úložiště pro klíč + + + Password: + Heslo: + + + Retype password: + Nypsat heslo znovu: + + + Show password + Ukázat heslo + + + <span style=" color:#ff0000;">Password is too short</span> + <span style=" color:#ff0000;">Heslo je příliš krátké</span> + + + Certificate + Certifikát + + + Alias name: + Název přezdívky: + + + Aaaaaaaa; + Aaaaaaaa; + + + Keysize: + Velikost klíče: + + + Validity (days): + Platnost (dny): + + + Certificate Distinguished Names + Odlišené názvy certifikátu + + + First and last name: + První a poslední název: + + + Organizational unit (e.g. Necessitas): + Organizační jednotka (např. Necessitas): + + + Organization (e.g. KDE): + Organizace (např. KDE): + + + City or Locality: + Město nebo místo: + + + State or Province: + Stát nebo územní jednotka: + + + Two-letter country code for this unit (e.g. RO): + Kód země tvořený dvěma písmeny této jednotky (např. RO): + + + >AA; + >AA; + + + + AndroidDeployStepWidget + + Form + Formulář + + + Use devices qt libs + Použít knihovny qt ze zařízení + + + Push local qt libs to device. +You must have Qt libs compiled for that platform + Odeslat místní knihovny qt do zařízení. +Musíte mít pro ten systém sestaveny knihovny qt + + + Deploy local qt libs + Nasadit místní knihovny qt + + + Check this option to force the application to use local qt libs instead of system libs. + Zaškrtněte tuto volbu pro vynucení místních knihoven qt namísto systémových knihoven v programu. + + + Use local qt libs + Použít místní knihovny qt + + + Edit rules file + Upravit soubor s pravidly + + + Choose and install Ministro system wide qt shared libraries. +This option is useful when you want to try your application on devices which don't have Android Market (e.g. Android Emulator). + Vybrat a nainstalovat systémové sdílené knihovny qt Ministro. +Tato volba je užitečná, když chcete svůj program vyzkoušet na zařízeních, která nemájí Android Market (např. Android Emulator). + + + Install Ministro system wide qt shared libraries installer + Instalátor pro nainstalování systémových sdílených knihoven qt Ministro + + + Choose apk + Vybrat apk + + + + AndroidPackageCreationWidget + + Manifest + Provolání (manifest) + + + <!DOCTYPE HTML PUBLIC "-//W3C//DTD HTML 4.0//EN" "http://www.w3.org/TR/REC-html40/strict.dtd"> +<html><head><meta name="qrichtext" content="1" /><style type="text/css"> +p, li { white-space: pre-wrap; } +</style></head><body style=" font-family:'DejaVu Sans'; font-size:9pt; font-weight:400; font-style:normal;"> +<table border="0" style="-qt-table-type: root; margin-top:4px; margin-bottom:4px; margin-left:4px; margin-right:4px;"> +<tr> +<td style="border: none;"> +<p style=" margin-top:0px; margin-bottom:0px; margin-left:0px; margin-right:0px; -qt-block-indent:0; text-indent:0px;"><span style=" font-weight:600;">Android target SDK:</span></p></td></tr></table></body></html> + <!DOCTYPE HTML PUBLIC "-//W3C//DTD HTML 4.0//EN" "http://www.w3.org/TR/REC-html40/strict.dtd"> +<html><head><meta name="qrichtext" content="1" /><style type="text/css"> +p, li { white-space: pre-wrap; } +</style></head><body style=" font-family:'DejaVu Sans'; font-size:9pt; font-weight:400; font-style:normal;"> +<table border="0" style="-qt-table-type: root; margin-top:4px; margin-bottom:4px; margin-left:4px; margin-right:4px;"> +<tr> +<td style="border: none;"> +<p style=" margin-top:0px; margin-bottom:0px; margin-left:0px; margin-right:0px; -qt-block-indent:0; text-indent:0px;"><span style=" font-weight:600;">SDK cíl Android:</span></p></td></tr></table></body></html> + + + <!DOCTYPE HTML PUBLIC "-//W3C//DTD HTML 4.0//EN" "http://www.w3.org/TR/REC-html40/strict.dtd"> +<html><head><meta name="qrichtext" content="1" /><style type="text/css"> +p, li { white-space: pre-wrap; } +</style></head><body style=" font-family:'DejaVu Sans'; font-size:9pt; font-weight:400; font-style:normal;"> +<table style="-qt-table-type: root; margin-top:4px; margin-bottom:4px; margin-left:4px; margin-right:4px;"> +<tr> +<td style="border: none;"> +<p style=" margin-top:0px; margin-bottom:0px; margin-left:0px; margin-right:0px; -qt-block-indent:0; text-indent:0px;"><span style=" font-weight:600;">Package name:</span></p></td></tr></table></body></html> + <!DOCTYPE HTML PUBLIC "-//W3C//DTD HTML 4.0//EN" "http://www.w3.org/TR/REC-html40/strict.dtd"> +<html><head><meta name="qrichtext" content="1" /><style type="text/css"> +p, li { white-space: pre-wrap; } +</style></head><body style=" font-family:'DejaVu Sans'; font-size:9pt; font-weight:400; font-style:normal;"> +<table style="-qt-table-type: root; margin-top:4px; margin-bottom:4px; margin-left:4px; margin-right:4px;"> +<tr> +<td style="border: none;"> +<p style=" margin-top:0px; margin-bottom:0px; margin-left:0px; margin-right:0px; -qt-block-indent:0; text-indent:0px;"><span style=" font-weight:600;">Název balíčku:</span></p></td></tr></table></body></html> + + + <!DOCTYPE HTML PUBLIC "-//W3C//DTD HTML 4.0//EN" "http://www.w3.org/TR/REC-html40/strict.dtd"> +<html><head><meta name="qrichtext" content="1" /><style type="text/css"> +p, li { white-space: pre-wrap; } +</style></head><body style=" font-family:'DejaVu Sans'; font-size:9pt; font-weight:400; font-style:normal;"> +<table style="-qt-table-type: root; margin-top:4px; margin-bottom:4px; margin-left:4px; margin-right:4px;"> +<tr> +<td style="border: none;"> +<p align="justify" style=" margin-top:0px; margin-bottom:0px; margin-left:0px; margin-right:0px; -qt-block-indent:0; text-indent:0px;">Please choose a valid package name for your application (e.g. &quot;org.example.myapplication&quot;). </p> +<p align="justify" style="-qt-paragraph-type:empty; margin-top:0px; margin-bottom:0px; margin-left:0px; margin-right:0px; -qt-block-indent:0; text-indent:0px;"></p> +<p align="justify" style=" margin-top:0px; margin-bottom:0px; margin-left:0px; margin-right:0px; -qt-block-indent:0; text-indent:0px;">Packages are usually defined using a hierarchical naming pattern, with levels in the hierarchy separated by periods (.) (pronounced &quot;dot&quot;).</p> +<p align="justify" style=" margin-top:0px; margin-bottom:0px; margin-left:0px; margin-right:0px; -qt-block-indent:0; text-indent:0px;">In general, a package name begins with the top level domain name of the organization and then the organization's domain and then any subdomains listed in reverse order. The organization can then choose a specific name for their package. Package names should be all lowercase characters whenever possible.</p> +<p align="justify" style="-qt-paragraph-type:empty; margin-top:0px; margin-bottom:0px; margin-left:0px; margin-right:0px; -qt-block-indent:0; text-indent:0px;"></p> +<p align="justify" style=" margin-top:0px; margin-bottom:0px; margin-left:0px; margin-right:0px; -qt-block-indent:0; text-indent:0px;">Complete conventions for disambiguating package names and rules for naming packages when the Internet domain name cannot be directly used as a package name are described in section 7.7 of the Java Language Specification.</p> +<p align="justify" style="-qt-paragraph-type:empty; margin-top:0px; margin-bottom:0px; margin-left:0px; margin-right:0px; -qt-block-indent:0; text-indent:0px;"></p></td></tr></table></body></html> + <!DOCTYPE HTML PUBLIC "-//W3C//DTD HTML 4.0//EN" "http://www.w3.org/TR/REC-html40/strict.dtd"> +<html><head><meta name="qrichtext" content="1" /><style type="text/css"> +p, li { white-space: pre-wrap; } +</style></head><body style=" font-family:'DejaVu Sans'; font-size:9pt; font-weight:400; font-style:normal;"> +<table style="-qt-table-type: root; margin-top:4px; margin-bottom:4px; margin-left:4px; margin-right:4px;"> +<tr> +<td style="border: none;"> +<p align="justify" style=" margin-top:0px; margin-bottom:0px; margin-left:0px; margin-right:0px; -qt-block-indent:0; text-indent:0px;">Vyberte, prosím, pro svůj program platný název balíčku (např. &quot;org.example.myapplication&quot;). </p> +<p align="justify" style="-qt-paragraph-type:empty; margin-top:0px; margin-bottom:0px; margin-left:0px; margin-right:0px; -qt-block-indent:0; text-indent:0px;"></p> +<p align="justify" style=" margin-top:0px; margin-bottom:0px; margin-left:0px; margin-right:0px; -qt-block-indent:0; text-indent:0px;">balíčky jsou obvykle určeny pomocí hierarchického pojmenovávacího vzoru, s úrovněmi hierarchie oddělenými tečkou (.).</p> +<p align="justify" style=" margin-top:0px; margin-bottom:0px; margin-left:0px; margin-right:0px; -qt-block-indent:0; text-indent:0px;">Obecně začíná název balíčku vrcholovým názvem domény organizace, a potom pokračuje doménou organizace, a pak jakoukoli podřízenou doménou, uvedeno v obráceném pořadí. Organizace pak pro svůj balíček může zvolit zvláštní název. Názvy balíčků by měly být, když je to možné, tvořeny malými písmeny.</p> +<p align="justify" style="-qt-paragraph-type:empty; margin-top:0px; margin-bottom:0px; margin-left:0px; margin-right:0px; -qt-block-indent:0; text-indent:0px;"></p> +<p align="justify" style=" margin-top:0px; margin-bottom:0px; margin-left:0px; margin-right:0px; -qt-block-indent:0; text-indent:0px;">Zvyklosti pro nedvojznačné názvy balíčků a pravidla pro pojmenovávání balíčků, když za název balíčku nelze použít přímo název internetové domény jsou popsány v oddíle 7.7 specifikace jazyka Java (Java Language Specification).</p> +<p align="justify" style="-qt-paragraph-type:empty; margin-top:0px; margin-bottom:0px; margin-left:0px; margin-right:0px; -qt-block-indent:0; text-indent:0px;"></p></td></tr></table></body></html> + + + <!DOCTYPE HTML PUBLIC "-//W3C//DTD HTML 4.0//EN" "http://www.w3.org/TR/REC-html40/strict.dtd"> +<html><head><meta name="qrichtext" content="1" /><style type="text/css"> +p, li { white-space: pre-wrap; } +</style></head><body style=" font-family:'DejaVu Sans'; font-size:9pt; font-weight:600; font-style:normal;"> +<table style="-qt-table-type: root; margin-top:4px; margin-bottom:4px; margin-left:4px; margin-right:4px;"> +<tr> +<td style="border: none;"> +<p style=" margin-top:0px; margin-bottom:0px; margin-left:0px; margin-right:0px; -qt-block-indent:0; text-indent:0px;">Version code:</p></td></tr></table></body></html> + <!DOCTYPE HTML PUBLIC "-//W3C//DTD HTML 4.0//EN" "http://www.w3.org/TR/REC-html40/strict.dtd"> +<html><head><meta name="qrichtext" content="1" /><style type="text/css"> +p, li { white-space: pre-wrap; } +</style></head><body style=" font-family:'DejaVu Sans'; font-size:9pt; font-weight:600; font-style:normal;"> +<table style="-qt-table-type: root; margin-top:4px; margin-bottom:4px; margin-left:4px; margin-right:4px;"> +<tr> +<td style="border: none;"> +<p style=" margin-top:0px; margin-bottom:0px; margin-left:0px; margin-right:0px; -qt-block-indent:0; text-indent:0px;">Kód verze:</p></td></tr></table></body></html> + + + <!DOCTYPE HTML PUBLIC "-//W3C//DTD HTML 4.0//EN" "http://www.w3.org/TR/REC-html40/strict.dtd"> +<html><head><meta name="qrichtext" content="1" /><style type="text/css"> +p, li { white-space: pre-wrap; } +</style></head><body style=" font-family:'DejaVu Sans'; font-size:9pt; font-weight:600; font-style:normal;"> +<table border="0" style="-qt-table-type: root; margin-top:4px; margin-bottom:4px; margin-left:4px; margin-right:4px;"> +<tr> +<td style="border: none;"> +<p style=" margin-top:0px; margin-bottom:0px; margin-left:0px; margin-right:0px; -qt-block-indent:0; text-indent:0px;">Version name:</p></td></tr></table></body></html> + <!DOCTYPE HTML PUBLIC "-//W3C//DTD HTML 4.0//EN" "http://www.w3.org/TR/REC-html40/strict.dtd"> +<html><head><meta name="qrichtext" content="1" /><style type="text/css"> +p, li { white-space: pre-wrap; } +</style></head><body style=" font-family:'DejaVu Sans'; font-size:9pt; font-weight:600; font-style:normal;"> +<table border="0" style="-qt-table-type: root; margin-top:4px; margin-bottom:4px; margin-left:4px; margin-right:4px;"> +<tr> +<td style="border: none;"> +<p style=" margin-top:0px; margin-bottom:0px; margin-left:0px; margin-right:0px; -qt-block-indent:0; text-indent:0px;">Název verze:</p></td></tr></table></body></html> + + + 1.0.0 + 1.0.0 + + + Application + Program + + + <!DOCTYPE HTML PUBLIC "-//W3C//DTD HTML 4.0//EN" "http://www.w3.org/TR/REC-html40/strict.dtd"> +<html><head><meta name="qrichtext" content="1" /><style type="text/css"> +p, li { white-space: pre-wrap; } +</style></head><body style=" font-family:'DejaVu Sans'; font-size:9pt; font-weight:400; font-style:normal;"> +<table border="0" style="-qt-table-type: root; margin-top:4px; margin-bottom:4px; margin-left:4px; margin-right:4px;"> +<tr> +<td style="border: none;"> +<p style=" margin-top:0px; margin-bottom:0px; margin-left:0px; margin-right:0px; -qt-block-indent:0; text-indent:0px;"><span style=" font-weight:600;">Application name:</span></p></td></tr></table></body></html> + <!DOCTYPE HTML PUBLIC "-//W3C//DTD HTML 4.0//EN" "http://www.w3.org/TR/REC-html40/strict.dtd"> +<html><head><meta name="qrichtext" content="1" /><style type="text/css"> +p, li { white-space: pre-wrap; } +</style></head><body style=" font-family:'DejaVu Sans'; font-size:9pt; font-weight:400; font-style:normal;"> +<table border="0" style="-qt-table-type: root; margin-top:4px; margin-bottom:4px; margin-left:4px; margin-right:4px;"> +<tr> +<td style="border: none;"> +<p style=" margin-top:0px; margin-bottom:0px; margin-left:0px; margin-right:0px; -qt-block-indent:0; text-indent:0px;"><span style=" font-weight:600;">Název programu:</span></p></td></tr></table></body></html> + + + <!DOCTYPE HTML PUBLIC "-//W3C//DTD HTML 4.0//EN" "http://www.w3.org/TR/REC-html40/strict.dtd"> +<html><head><meta name="qrichtext" content="1" /><style type="text/css"> +p, li { white-space: pre-wrap; } +</style></head><body style=" font-family:'DejaVu Sans'; font-size:9pt; font-weight:400; font-style:normal;"> +<table border="0" style="-qt-table-type: root; margin-top:4px; margin-bottom:4px; margin-left:4px; margin-right:4px;"> +<tr> +<td style="border: none;"> +<p style=" margin-top:0px; margin-bottom:0px; margin-left:0px; margin-right:0px; -qt-block-indent:0; text-indent:0px;"><span style=" font-weight:600;">Run:</span></p></td></tr></table></body></html> + <!DOCTYPE HTML PUBLIC "-//W3C//DTD HTML 4.0//EN" "http://www.w3.org/TR/REC-html40/strict.dtd"> +<html><head><meta name="qrichtext" content="1" /><style type="text/css"> +p, li { white-space: pre-wrap; } +</style></head><body style=" font-family:'DejaVu Sans'; font-size:9pt; font-weight:400; font-style:normal;"> +<table border="0" style="-qt-table-type: root; margin-top:4px; margin-bottom:4px; margin-left:4px; margin-right:4px;"> +<tr> +<td style="border: none;"> +<p style=" margin-top:0px; margin-bottom:0px; margin-left:0px; margin-right:0px; -qt-block-indent:0; text-indent:0px;"><span style=" font-weight:600;">Spuštění:</span></p></td></tr></table></body></html> + + + <!DOCTYPE HTML PUBLIC "-//W3C//DTD HTML 4.0//EN" "http://www.w3.org/TR/REC-html40/strict.dtd"> +<html><head><meta name="qrichtext" content="1" /><style type="text/css"> +p, li { white-space: pre-wrap; } +</style></head><body style=" font-family:'DejaVu Sans'; font-size:9pt; font-weight:400; font-style:normal;"> +<table style="-qt-table-type: root; margin-top:4px; margin-bottom:4px; margin-left:4px; margin-right:4px;"> +<tr> +<td style="border: none;"> +<p style=" margin-top:0px; margin-bottom:0px; margin-left:0px; margin-right:0px; -qt-block-indent:0; text-indent:0px;"><span style=" font-weight:600;">Application icon:</span></p></td></tr></table></body></html> + <!DOCTYPE HTML PUBLIC "-//W3C//DTD HTML 4.0//EN" "http://www.w3.org/TR/REC-html40/strict.dtd"> +<html><head><meta name="qrichtext" content="1" /><style type="text/css"> +p, li { white-space: pre-wrap; } +</style></head><body style=" font-family:'DejaVu Sans'; font-size:9pt; font-weight:400; font-style:normal;"> +<table border="0" style="-qt-table-type: root; margin-top:4px; margin-bottom:4px; margin-left:4px; margin-right:4px;"> +<tr> +<td style="border: none;"> +<p style=" margin-top:0px; margin-bottom:0px; margin-left:0px; margin-right:0px; -qt-block-indent:0; text-indent:0px;"><span style=" font-weight:600;">Ikona programu:</span></p></td></tr></table></body></html> + + + Select low dpi icon + Vybrat ikonu s nízkým dpi + + + Select medium dpi icon + Vybrat ikonu se středním dpi + + + Select high dpi icon + Vybrat ikonu s vysokým dpi + + + Permissions + Oprávnění + + + Add + Přidat + + + Remove + Odstranit + + + Save + Uložit + + + Discard + Odmítnout + + + Name: + Název: + + + android.permission.ACCESS_CHECKIN_PROPERTIES + android.permission.ACCESS_CHECKIN_PROPERTIES + + + android.permission.ACCESS_COARSE_LOCATION + android.permission.ACCESS_COARSE_LOCATION + + + android.permission.ACCESS_FINE_LOCATION + android.permission.ACCESS_FINE_LOCATION + + + android.permission.ACCESS_LOCATION_EXTRA_COMMANDS + android.permission.ACCESS_LOCATION_EXTRA_COMMANDS + + + android.permission.ACCESS_MOCK_LOCATION + android.permission.ACCESS_MOCK_LOCATION + + + android.permission.ACCESS_NETWORK_STATE + android.permission.ACCESS_NETWORK_STATE + + + android.permission.ACCESS_SURFACE_FLINGER + android.permission.ACCESS_SURFACE_FLINGER + + + android.permission.ACCESS_WIFI_STATE + android.permission.ACCESS_WIFI_STATE + + + android.permission.ACCOUNT_MANAGER + android.permission.ACCOUNT_MANAGER + + + android.permission.AUTHENTICATE_ACCOUNTS + android.permission.AUTHENTICATE_ACCOUNTS + + + android.permission.BATTERY_STATS + android.permission.BATTERY_STATS + + + android.permission.BIND_APPWIDGET + android.permission.BIND_APPWIDGET + + + android.permission.BIND_DEVICE_ADMIN + android.permission.BIND_DEVICE_ADMIN + + + android.permission.BIND_INPUT_METHOD + android.permission.BIND_INPUT_METHOD + + + android.permission.BIND_REMOTEVIEWS + android.permission.BIND_REMOTEVIEWS + + + android.permission.BIND_WALLPAPER + android.permission.BIND_WALLPAPER + + + android.permission.BLUETOOTH + android.permission.BLUETOOTH + + + android.permission.BLUETOOTH_ADMIN + android.permission.BLUETOOTH_ADMIN + + + android.permission.BRICK + android.permission.BRICK + + + android.permission.BROADCAST_PACKAGE_REMOVED + android.permission.BROADCAST_PACKAGE_REMOVED + + + android.permission.BROADCAST_SMS + android.permission.BROADCAST_SMS + + + android.permission.BROADCAST_STICKY + android.permission.BROADCAST_STICKY + + + android.permission.BROADCAST_WAP_PUSH + android.permission.BROADCAST_WAP_PUSH + + + android.permission.CALL_PHONE + android.permission.CALL_PHONE + + + android.permission.CALL_PRIVILEGED + android.permission.CALL_PRIVILEGED + + + android.permission.CAMERA + android.permission.CAMERA + + + android.permission.CHANGE_COMPONENT_ENABLED_STATE + android.permission.CHANGE_COMPONENT_ENABLED_STATE + + + android.permission.CHANGE_CONFIGURATION + android.permission.CHANGE_CONFIGURATION + + + android.permission.CHANGE_NETWORK_STATE + android.permission.CHANGE_NETWORK_STATE + + + android.permission.CHANGE_WIFI_MULTICAST_STATE + android.permission.CHANGE_WIFI_MULTICAST_STATE + + + android.permission.CHANGE_WIFI_STATE + android.permission.CHANGE_WIFI_STATE + + + android.permission.CLEAR_APP_CACHE + android.permission.CLEAR_APP_CACHE + + + android.permission.CLEAR_APP_USER_DATA + android.permission.CLEAR_APP_USER_DATA + + + android.permission.CONTROL_LOCATION_UPDATES + android.permission.CONTROL_LOCATION_UPDATES + + + android.permission.DELETE_CACHE_FILES + android.permission.DELETE_CACHE_FILES + + + android.permission.DELETE_PACKAGES + android.permission.DELETE_PACKAGES + + + android.permission.DEVICE_POWER + android.permission.DEVICE_POWER + + + android.permission.DIAGNOSTIC + android.permission.DIAGNOSTIC + + + android.permission.DISABLE_KEYGUARD + android.permission.DISABLE_KEYGUARD + + + android.permission.DUMP + android.permission.DUMP + + + android.permission.EXPAND_STATUS_BAR + android.permission.EXPAND_STATUS_BAR + + + android.permission.FACTORY_TEST + android.permission.FACTORY_TEST + + + android.permission.FLASHLIGHT + android.permission.FLASHLIGHT + + + android.permission.FORCE_BACK + android.permission.FORCE_BACK + + + android.permission.GET_ACCOUNTS + android.permission.GET_ACCOUNTS + + + android.permission.GET_PACKAGE_SIZE + android.permission.GET_PACKAGE_SIZE + + + android.permission.GET_TASKS + android.permission.GET_TASKS + + + android.permission.GLOBAL_SEARCH + android.permission.GLOBAL_SEARCH + + + android.permission.HARDWARE_TEST + android.permission.HARDWARE_TEST + + + android.permission.INJECT_EVENTS + android.permission.INJECT_EVENTS + + + android.permission.INSTALL_LOCATION_PROVIDER + android.permission.INSTALL_LOCATION_PROVIDER + + + android.permission.INSTALL_PACKAGES + android.permission.INSTALL_PACKAGES + + + android.permission.INTERNAL_SYSTEM_WINDOW + android.permission.INTERNAL_SYSTEM_WINDOW + + + android.permission.INTERNET + android.permission.INTERNET + + + android.permission.KILL_BACKGROUND_PROCESSES + android.permission.KILL_BACKGROUND_PROCESSES + + + android.permission.MANAGE_ACCOUNTS + android.permission.MANAGE_ACCOUNTS + + + android.permission.MANAGE_APP_TOKENS + android.permission.MANAGE_APP_TOKENS + + + android.permission.MASTER_CLEAR + android.permission.MASTER_CLEAR + + + android.permission.MODIFY_AUDIO_SETTINGS + android.permission.MODIFY_AUDIO_SETTINGS + + + android.permission.MODIFY_PHONE_STATE + android.permission.MODIFY_PHONE_STATE + + + android.permission.MOUNT_FORMAT_FILESYSTEMS + android.permission.MOUNT_FORMAT_FILESYSTEMS + + + android.permission.MOUNT_UNMOUNT_FILESYSTEMS + android.permission.MOUNT_UNMOUNT_FILESYSTEMS + + + android.permission.NFC + android.permission.NFC + + + android.permission.PERSISTENT_ACTIVITY + android.permission.PERSISTENT_ACTIVITY + + + android.permission.PROCESS_OUTGOING_CALLS + android.permission.PROCESS_OUTGOING_CALLS + + + android.permission.READ_CALENDAR + android.permission.READ_CALENDAR + + + android.permission.READ_CONTACTS + android.permission.READ_CONTACTS + + + android.permission.READ_FRAME_BUFFER + android.permission.READ_FRAME_BUFFER + + + com.android.browser.permission.READ_HISTORY_BOOKMARKS + com.android.browser.permission.READ_HISTORY_BOOKMARKS + + + android.permission.READ_INPUT_STATE + android.permission.READ_INPUT_STATE + + + android.permission.READ_LOGS + android.permission.READ_LOGS + + + android.permission.READ_OWNER_DATA + android.permission.READ_OWNER_DATA + + + android.permission.READ_PHONE_STATE + android.permission.READ_PHONE_STATE + + + android.permission.READ_SMS + android.permission.READ_SMS + + + android.permission.READ_SYNC_SETTINGS + android.permission.READ_SYNC_SETTINGS + + + android.permission.READ_SYNC_STATS + android.permission.READ_SYNC_STATS + + + android.permission.REBOOT + android.permission.REBOOT + + + android.permission.RECEIVE_BOOT_COMPLETED + android.permission.RECEIVE_BOOT_COMPLETED + + + android.permission.RECEIVE_MMS + android.permission.RECEIVE_MMS + + + android.permission.RECEIVE_SMS + android.permission.RECEIVE_SMS + + + android.permission.RECEIVE_WAP_PUSH + android.permission.RECEIVE_WAP_PUSH + + + android.permission.RECORD_AUDIO + android.permission.RECORD_AUDIO + + + android.permission.REORDER_TASKS + android.permission.REORDER_TASKS + + + android.permission.RESTART_PACKAGES + android.permission.RESTART_PACKAGES + + + android.permission.SEND_SMS + android.permission.SEND_SMS + + + android.permission.SET_ACTIVITY_WATCHER + android.permission.SET_ACTIVITY_WATCHER + + + com.android.alarm.permission.SET_ALARM + com.android.alarm.permission.SET_ALARM + + + android.permission.SET_ALWAYS_FINISH + android.permission.SET_ALWAYS_FINISH + + + android.permission.SET_ANIMATION_SCALE + android.permission.SET_ANIMATION_SCALE + + + android.permission.SET_DEBUG_APP + android.permission.SET_DEBUG_APP + + + android.permission.SET_ORIENTATION + android.permission.SET_ORIENTATION + + + android.permission.SET_PREFERRED_APPLICATIONS + android.permission.SET_PREFERRED_APPLICATIONS + + + android.permission.SET_PROCESS_LIMIT + android.permission.SET_PROCESS_LIMIT + + + android.permission.SET_TIME + android.permission.SET_TIME + + + android.permission.SET_TIME_ZONE + android.permission.SET_TIME_ZONE + + + android.permission.SET_WALLPAPER + android.permission.SET_WALLPAPER + + + android.permission.SET_WALLPAPER_HINTS + android.permission.SET_WALLPAPER_HINTS + + + android.permission.SIGNAL_PERSISTENT_PROCESSES + android.permission.SIGNAL_PERSISTENT_PROCESSES + + + android.permission.STATUS_BAR + android.permission.STATUS_BAR + + + android.permission.SUBSCRIBED_FEEDS_READ + android.permission.SUBSCRIBED_FEEDS_READ + + + android.permission.SUBSCRIBED_FEEDS_WRITE + android.permission.SUBSCRIBED_FEEDS_WRITE + + + android.permission.SYSTEM_ALERT_WINDOW + android.permission.SYSTEM_ALERT_WINDOW + + + android.permission.UPDATE_DEVICE_STATS + android.permission.UPDATE_DEVICE_STATS + + + android.permission.USE_CREDENTIALS + android.permission.USE_CREDENTIALS + + + android.permission.USE_SIP + android.permission.USE_SIP + + + android.permission.VIBRATE + android.permission.VIBRATE + + + android.permission.WAKE_LOCK + android.permission.WAKE_LOCK + + + android.permission.WRITE_APN_SETTINGS + android.permission.WRITE_APN_SETTINGS + + + android.permission.WRITE_CALENDAR + android.permission.WRITE_CALENDAR + + + android.permission.WRITE_CONTACTS + android.permission.WRITE_CONTACTS + + + android.permission.WRITE_EXTERNAL_STORAGE + android.permission.WRITE_EXTERNAL_STORAGE + + + android.permission.WRITE_GSERVICES + android.permission.WRITE_GSERVICES + + + com.android.browser.permission.WRITE_HISTORY_BOOKMARKS + com.android.browser.permission.WRITE_HISTORY_BOOKMARKS + + + android.permission.WRITE_OWNER_DATA + android.permission.WRITE_OWNER_DATA + + + android.permission.WRITE_SECURE_SETTINGS + android.permission.WRITE_SECURE_SETTINGS + + + android.permission.WRITE_SETTINGS + android.permission.WRITE_SETTINGS + + + android.permission.WRITE_SMS + android.permission.WRITE_SMS + + + android.permission.WRITE_SYNC_SETTINGS + android.permission.WRITE_SYNC_SETTINGS + + + Libraries + Knihovny + + + Automatically check required Qt libraries from compiled application + Automaticky ověřit požadované knihovny Qt ze sestaveného programu + + + Read information from application (must be compiled) + Číst informace z programu (musí být sestaven) + + + Required Qt libraries + Požadované knihovny Qt + + + <!DOCTYPE HTML PUBLIC "-//W3C//DTD HTML 4.0//EN" "http://www.w3.org/TR/REC-html40/strict.dtd"> +<html><head><meta name="qrichtext" content="1" /><style type="text/css"> +p, li { white-space: pre-wrap; } +</style></head><body style=" font-family:'DejaVu Sans'; font-size:9pt; font-weight:400; font-style:normal;"> +<table border="0" style="-qt-table-type: root; margin-top:4px; margin-bottom:4px; margin-left:4px; margin-right:4px;"> +<tr> +<td style="border: none;"> +<p align="center" style=" margin-top:0px; margin-bottom:0px; margin-left:0px; margin-right:0px; -qt-block-indent:0; text-indent:0px;">Prebundled libraries</p> +<p style=" margin-top:0px; margin-bottom:0px; margin-left:0px; margin-right:0px; -qt-block-indent:0; text-indent:0px;">Be aware, the order is very important: if library A depends on library B, B MUST be come before A!</p></td></tr></table></body></html> + <!DOCTYPE HTML PUBLIC "-//W3C//DTD HTML 4.0//EN" "http://www.w3.org/TR/REC-html40/strict.dtd"> +<html><head><meta name="qrichtext" content="1" /><style type="text/css"> +p, li { white-space: pre-wrap; } +</style></head><body style=" font-family:'DejaVu Sans'; font-size:9pt; font-weight:400; font-style:normal;"> +<table border="0" style="-qt-table-type: root; margin-top:4px; margin-bottom:4px; margin-left:4px; margin-right:4px;"> +<tr> +<td style="border: none;"> +<p align="center" style=" margin-top:0px; margin-bottom:0px; margin-left:0px; margin-right:0px; -qt-block-indent:0; text-indent:0px;">Předbalené knihovny</p> +<p style=" margin-top:0px; margin-bottom:0px; margin-left:0px; margin-right:0px; -qt-block-indent:0; text-indent:0px;">Buďte opatrní, pořadí je velice důležité: pokud knihovna A závisí na knihovně B, B MUSÍ přijít před A!</p></td></tr></table></body></html> + + + Up + Nahoru + + + Down + Dolů + + + Sign package + Podepsat balíček + + + Keystore: + Úložiště pro klíč: + + + Create + Vytvořit + + + Browse + Procházet + + + Open package location after is complete + Otevřít umístění balíčku, když je hotový + + + Certificate alias: + Osvědčit přezdívku: + + + + AndroidSettingsWidget + + Android Configuration + Nastavení pro Android + + + Android SDK location: + Umístění SDK pro Android: + + + Browse + Procházet + + + Android NDK location: + Umístění NDK pro Android: + + + Android NDK toolchain version: + Verze sady nástrojů NDK pro Android: + + + Ant location: + Umístění Ant: + + + arm GDB location: + Umístění GDB arm: + + + arm GDBserver location: + Umístění GDBserver arm: + + + x86 GDB location: + Umístění GDB x86: + + + x86 GDBserver location: + Umístění GDBserver x86: + + + OpenJDK location: + Umístění OpenJDK: + + + Start + Spustit + + + AVD Manager + Správce AVD + + + System/data partition size: + Velikost oddílu se systémem/daty: + + + Mb + MiB + + + Start Android AVD Manager + Spustit správce AVD Android + + + Add + Přidat + + + Remove + Odstranit + + + + Core::Internal::CommandMappings + + Command Mappings + Přiřazení příkazů + + + Command + Příkaz + + + Label + Popis + + + Target + Cíl + + + Reset all to default + Nastavit vše znovu na výchozí + + + Reset All + Nastavit znovu vše + + + Import... + Zavést... + + + Export... + Vyvést... + + + Target Identifier + Identifikátor cíle + + + Target: + Cíl: + + + Reset to default + Nastavit znovu výchozí + + + Reset + Nastavit znovu + + + + Core::Internal::MimeTypeSettingsPage + + Form + Formulář + + + Registered MIME Types + Přihlášené MIME typy + + + Reset all to default. + Nastavit vše znovu na výchozí. + + + Reset All + Nastavit znovu vše + + + Details + Podrobnosti + + + Patterns: + Vzory: + + + Magic Header + Kouzelné záhlaví + + + Type + Typ + + + Range + Rozsah + + + Priority + Přednost + + + Add... + Přidat... + + + Edit... + Upravit... + + + Remove + Odstranit + + + + Core::Internal::VariableChooser + + Variables + Proměnné + + + Select a variable to insert. + Vyberte proměnnou, která se má vložit. + + + + CodePaster::Internal::FileShareProtocolSettingsWidget + + Form + Formulář + + + The fileshare-based paster protocol allows for sharing code snippets using simple files on a shared network drive. Files are never deleted. + Protokol vložení založený na sdílení souborů umožňuje sdílení úryvků kódu pomocí prostých souborů na sdíleném síťovém disku. Soubory nejsou nikdy mazány. + + + &Path: + &Cesta: + + + &Display: + &Zobrazit: + + + entries + záznamů + + + + CodePaster::Internal::PasteBinComSettingsWidget + + Form + Formulář + + + <a href="http://pastebin.com">pastebin.com</a> allows for sending posts to custom subdomains (eg. creator.pastebin.com). Fill in the desired prefix. + <a href="http://pastebin.com">pastebin.com</a> dovoluje posílání uživatelsky stanovených subdomén (například creator.pastebin.com). Zadejte požadovanou předponu. + + + Server prefix: + Předpona serveru: + + + <i>Note: The plugin will use this for posting as well as fetching.</i> + <i>Poznámka: Přídavný modul toto použije pro posílání a stejně tak natahování.</i> + + + + CodePaster::Internal::PasteSelectDialog + + Protocol: + Protokol: + + + Paste: + Vložení: + + + + CodePaster::Internal::ViewDialog + + Send to Codepaster + Poslat na CodePaster + + + Protocol: + Protokol: + + + &Username: + &Uživatelské jméno: + + + <Username> + <Uživatelské jméno> + + + &Description: + &Popis: + + + <Description> + <Popis> + + + <!DOCTYPE HTML PUBLIC "-//W3C//DTD HTML 4.0//EN" "http://www.w3.org/TR/REC-html40/strict.dtd"> +<html><head><meta name="qrichtext" content="1" /><style type="text/css"> +p, li { white-space: pre-wrap; } +</style></head><body style=" font-family:'Sans Serif'; font-size:9pt; font-weight:400; font-style:normal;"> +<p style=" margin-top:0px; margin-bottom:0px; margin-left:0px; margin-right:0px; -qt-block-indent:0; text-indent:0px;">&lt;Comment&gt;</p></body></html> + <!DOCTYPE HTML PUBLIC "-//W3C//DTD HTML 4.0//EN" "http://www.w3.org/TR/REC-html40/strict.dtd"> +<html><head><meta name="qrichtext" content="1" /><style type="text/css"> +p, li { white-space: pre-wrap; } +</style></head><body style=" font-family:'Sans Serif'; font-size:9pt; font-weight:400; font-style:normal;"> +<p style=" margin-top:0px; margin-bottom:0px; margin-left:0px; margin-right:0px; -qt-block-indent:0; text-indent:0px;">&lt;Poznámka&gt;</p></body></html> + + + Parts to Send to Server + Části pro poslání serveru + + + Patch 1 + Záplata 1 + + + Patch 2 + Záplata 2 + + + + CodePaster::Internal::SettingsPage + + Display Output pane after sending a post + Po odeslání ukázat výstupní tabulku + + + Copy-paste URL to clipboard + Kopírovat URL do schránky + + + Username: + Uživatelské jméno: + + + Default protocol: + Výchozí protokol: + + + + CppTools::Internal::CppFileSettingsPage + + Header suffix: + Přípona hlavičkových souborů: + + + Source suffix: + Přípona zdrojových souborů: + + + Lower case file names + Pro názvy souborů používat malých písmen + + + License template: + Licenční předloha: + + + + Debugger::Internal::AttachToQmlPortDialog + + Start Debugger + Spustit ladicí program + + + &Host: + &Hostitel: + + + &Port: + &Port: + + + Sys&root: + Sys&root: + + + + Debugger::Internal::CommonOptionsPage + + Behavior + Chování + + + Use alternating row colors in debug views + V pohledech na ladění užít střídavých barev řádků + + + Change the font size in the debugger views when the font size in the main editor changes. + Změnit velikost písma v oknech pro ladění, když se velikost písma změní v hlavním editoru. + + + Debugger font size follows main editor + Velikost písma ladiče následuje hlavní editor + + + Use tooltips in main editor while debugging + Při ladění použít vysvětlivky v hlavním editoru + + + Populate the source file view automatically. This might slow down debugger startup considerably. + Aktualizovat pohled na zdrojový soubor automaticky. Toto může spuštění ladiče výrazně zpomalit. + + + Populate source file view automatically + Aktualizovat pohled na zdrojový soubor automaticky + + + Close temporary buffers on debugger exit + Při ukončení ladění zavřít editory + + + Switch to previous mode on debugger exit + Režim činný na začátku při ukončení ladění obnovit + + + Maximum stack depth: + Největší hloubka zásobníku: + + + <unlimited> + <neomezená> + + + Register Qt Creator for debugging crashed applications. + Přihlásit Qt Creator pro ladění spadlých programů. + + + Use Qt Creator for post-mortem debugging + Použít Qt Creator pro ladění - následný rozbor + + + Bring Qt Creator to foreground when application interrupts + Přivést Qt Creator do popředí, když dojde k přerušení programu + + + + Debugger::Internal::LldbOptionsPageWidget + + Enable LLDB + Povolit LLDB + + + Use GDB Python dumpers + Použít pomocné knihovny pro výstup dat Python GDB + + + + Debugger::Internal::StartRemoteEngineDialog + + Start Remote Engine + Spustit vzdálený stroj + + + &Host: + &Hostitel: + + + &Username: + &Uživatelské jméno: + + + &Password: + He&slo: + + + &Engine path: + Cesta ke &stroji: + + + &Inferior path: + &Podřízená cesta: + + + + Madde::Internal::MaemoDeviceConfigWizardCheckPreviousKeySetupPage + + WizardPage + WizardPage + + + Has a passwordless (key-based) login already been set up for this device? + Bylo již pro toto mobilní zařízení nastaveno přihlášení bez hesla? + + + Yes, and the private key is located at + Ano, a soukromý klíč se nachází v + + + No + Ne + + + + Madde::Internal::MaemoPublishingWizardPageFremantleFree + + WizardPage + WizardPage + + + Choose build configuration: + Vybrat nastavení sestavování: + + + Only create source package, do not upload + Pouze vytvořit zdrojový balíček, nenahrávat + + + + Madde::Internal::MaemoPublishingFileSelectionDialog + + Choose Package Contents + Vybrat obsah balíčku + + + <b>Please select the files you want to be included in the source tarball.</b> + + <b>Vyberte, prosím, soubory, které mají být zahrnuty do tarballu se zdroji.</b> + + + + + Madde::Internal::MaemoPublishingResultPageFremantleFree + + WizardPage + WizardPage + + + Progress + Průběh + + + + Madde::Internal::MaemoQemuSettingsWidget + + Form + Formulář + + + OpenGL Mode + Režim OpenGL + + + &Hardware acceleration + &Hardwarové zrychlení + + + &Software rendering + &Softwarové vykreslování + + + &Auto-detect + Určit &automaticky + + + + ProjectExplorer::Internal::CodeStyleSettingsPropertiesPage + + Form + Formulář + + + Language: + Jazyk: + + + + ProjectExplorer::Internal::DeviceFactorySelectionDialog + + Device Configuration Wizard Selection + Zřízení nového nastavení zařízení + + + Available device types: + Dostupné typy zařízení: + + + Start Wizard + Spustit průvodce + + + + ProjectExplorer::Internal::DeviceSettingsWidget + + Linux Device Configurations + Nastavení linuxového zařízení + + + &Device: + &Zařízení: + + + General + Obecné + + + Type: + Typ: + + + &Name: + &Název: + + + Auto-detected: + Automaticky zjištěno: + + + Type Specific + Typově zvláštní + + + &Add... + &Přidat... + + + &Remove + &Odstranit + + + Set As Default + Nastavit jako výchozí + + + Yes (fingerprint is '%1') + Ano (otisk prstu je '%1') + + + No + Ne + + + + QmlDesigner::Internal::BehaviorDialog + + Dialog + Dialog + + + Type: + Typ: + + + ID: + ID: + + + Property name: + Název vlastnosti: + + + Animation + Animace + + + SpringFollow + SpringFollow + + + Settings + Nastavení + + + Duration: + Doba trvání: + + + Curve: + Křivka: + + + easeNone + easeNone + + + Source: + Zdroj: + + + Velocity: + Rychlost: + + + Spring: + Pružina: + + + Damping: + Tlumení: + + + + SelectionRangeDetails + + Selection + Výběr + + + Start + Začátek + + + End + Konec + + + Duration + Doba trvání + + + + Qt4ProjectManager::Internal::MakeStep + + Make arguments: + Argumenty příkazového řádku pro 'make': + + + Override %1: + Přepsat %1: + + + + Qt4ProjectManager::Internal::QMakeStep + + qmake build configuration: + Nastavení sestavování pro qmake: + + + Debug + Ladění + + + Release + Vydání + + + Additional arguments: + Dodatečné argumenty: + + + Link QML debugging library: + Připojit knihovnu pro ladění QML: + + + Effective qmake call: + Výsledné vyvolání qmake: + + + + Qt4ProjectManager::Internal::S60CertificateDetailsDialog + + Details of Certificate + Podrobnosti k certifikátu + + + + Qt4ProjectManager::Internal::Html5AppWizardSourcesPage + + WizardPage + WizardPage + + + Main HTML File + Hlavní soubor HTML + + + Generate an index.html file + Vytvořit soubor index.html + + + Import an existing .html file + Zavést existující soubor .html + + + Load a URL + Nahrát adresu (URL) + + + http:// + http:// + + + Note: Unless you chose to load a URL, all files and directories that reside in the same directory as the main HTML file are deployed. You can modify the contents of the directory any time before deploying. + Poznámka: Když nezadáte žádnou adresu (URL), budou nasazeny všechny soubory, které se nachází v tomtéž adresáři jako hlavní soubor HTML. Obsah adresáře lze před nasazením kdykoli upravit. + + + Touch optimized navigation + Pro dotykové ovládání optimalizovaná navigace + + + Enable touch optimized navigation + Zapnout pro dotykové ovládání optimalizovanou navigaci + + + Touch optimized navigation will make the HTML page flickable and enlarge the area of touch sensitive elements. If you use a JavaScript framework which optimizes the touch interaction, leave the checkbox unchecked. + Pro dotykové ovládání optimalizované navádění způsobí, že stránku HTML lze obsluhovat pomocí udělání rychlého pohybu (Flick) a zvětší oblast prvků citlivých na dotyk. Nechte nastavení vypnuto, pokud již používáte pro dotykové ovládání vyladěnou soustavu JavaScript. + + + + Qt4ProjectManager::Internal::MobileAppWizardHarmattanOptionsPage + + WizardPage + WizardPage + + + Application icon (80x80): + Ikona programu (80x80): + + + Generate code to speed up the launching on the device. + Vytvořit kód pro zrychlení spouštění na zařízení. + + + Make application boostable + Udělat program schopný vzpruhy + + + + Qt4ProjectManager::Internal::MobileAppWizardMaemoOptionsPage + + WizardPage + WizardPage + + + Application icon (64x64): + Ikona programu (64x64): + + + + Qt4ProjectManager::Internal::MobileAppWizardSymbianOptionsPage + + WizardPage + WizardPage + + + Application icon (.svg): + Ikona programu (.svg): + + + Target UID3: + Cílové UID3: + + + Enable network access + Povolit přístup k síti + + + + Qt4ProjectManager::Internal::MobileLibraryWizardOptionPage + + WizardPage + WizardPage + + + Target UID3: + Cílové UID3: + + + Plugin's directory name: + Název adresáře přídavného modulu: + + + Enable network access + Povolit přístup k síti + + + + QtSupport::Internal::ShowBuildLog + + Debugging Helper Build Log + Záznam o sestavení pomocného ladicího programu + + + + RemoteLinux::GenericLinuxDeviceConfigurationWidget + + Form + Formulář + + + Authentication type: + Druh ověření pravosti: + + + Password + Heslo + + + &Key + &Klíč + + + &Host name: + Název &hostitelského počítače: + + + IP or host name of the device + IP nebo název hostitelského počítače zařízení + + + &SSH port: + Port &SSH: + + + Free ports: + Volné přípojky (porty): + + + You can enter lists and ranges like this: 1024,1026-1028,1030 + Můžete zadat seznamy a rozsahy jako jsou tyto: 1024,1026-1028,1030 + + + Timeout: + Časové omezení: + + + s + s + + + &Username: + &Uživatelské jméno: + + + &Password: + He&slo: + + + Show password + Ukázat heslo + + + Private key file: + Soubor se soukromým klíčem: + + + Create New... + Vytvořit nový... + + + Machine type: + Typ stroje: + + + Physical Device + Fyzické zařízení + + + Emulator + Emulátor + + + You will need at least one port. + Potřebujete alespoň jeden volný port. + + + Set as Default + Nastavit jako výchozí + + + + RemoteLinux::Internal::GenericLinuxDeviceConfigurationWizardSetupPage + + WizardPage + WizardPage + + + The name to identify this configuration: + Název nastavení: + + + The device's host name or IP address: + Název hostitelského počítače nebo IP adresa zařízení: + + + The user name to log into the device: + Uživatelské jméno pro přihlášení se k zařízení: + + + The authentication type: + Druh ověření pravosti: + + + Password + Heslo + + + Key + Klíč + + + The user's password: + Heslo uživatele: + + + The file containing the user's private key: + Soubor se soukromým klíčem uživatele: + + + + RemoteLinux::Internal::LinuxDeviceTestDialog + + Device Test + Zkouška zařízení + + + + RemoteLinux::Internal::RemoteLinuxDeployConfigurationWidget + + Form + Formulář + + + Device configuration: + Nastavení zařízení: + + + <a href="irrelevant">Manage device configurations</a> + <a href="irrelevant">Spravovat nastavení zařízení</a> + + + These show the INSTALLS settings from the project file(s). + Ukáže nastavení INSTALLS z projektového souboru. + + + Files to install for subproject: + Soubory k instalaci pro dílčí projekt: + + + + RemoteLinux::Internal::RemoteLinuxProcessesDialog + + List of Remote Processes + Seznam vzdálených procesů + + + &Filter entries: + &Filtrovat záznamy: + + + &Update List + &Obnovit seznam + + + &Kill Selected Process + &Ukončit vybraný proces + + + &Filter by process name: + &Filtrovat podle názvu procesu: + + + + TextEditor::Internal::BehaviorSettingsPage + + Form + Formulář + + + + TextEditor::Internal::BehaviorSettingsWidget + + Typing + Při psaní + + + Enable automatic &indentation + Povolit automatické &odsazení + + + Backspace indentation: + Odsazení při Backspace: + + + <html><head/><body> +Specifies how backspace interacts with indentation. + +<ul> +<li>None: No interaction at all. Regular plain backspace behavior. +</li> + +<li>Follows Previous Indents: In leading white space it will take the cursor back to the nearest indentation level used in previous lines. +</li> + +<li>Unindents: If the character behind the cursor is a space it behaves as a backtab. +</li> +</ul></body></html> + + <html><head/><body> +Určuje, jak se chová zpětná klávesa (backspace) co se týče odsazování. + +<ul> +<li>Žádné: Žádné zvláštní chování. Obvyklé chování zpětné klávesy. +</li> + +<li>Sledovat předchozí odsazení: V prázdném místě na začátku je ukazovátko postaveno zpět na nejbližší úroveň odsazení použitou na předchozích řádcích. +</li> + +<li>Zrušit odsazení: Pokud je znak nacházející se za ukazovátkem mezera, chová se zpětná klávesa jako zpětný tabulátor. +</li> +</ul></body></html> + + + + None + Žádné + + + Follows Previous Indents + Dle předchozího odsazení + + + Unindents + Zrušit odsazení + + + Tab key performs auto-indent: + Tabulátor dělá automatické odsazení: + + + Never + Nikdy + + + Always + Vždy + + + In Leading White Space + Jen při začátku řádku + + + Cleanup actions which are automatically performed right before the file is saved to disk. + Opravy, které se automaticky provádějí bezprostředně před uložením souboru na disk. + + + Cleanups Upon Saving + Opravy při uložení + + + Removes trailing whitespace upon saving. + Odstraní při ukládání prázdné znaky na konci řádků. + + + &Clean whitespace + &Vyčistit prázdné znaky + + + Clean whitespace in entire document instead of only for changed parts. + Vyčistí prázdné znaky v celém dokumentu a nejen ve změněných částech. + + + In entire &document + V celém &dokumentu + + + Correct leading whitespace according to tab settings. + Opraví prázdné znaky na začátku řádků podle nastavení tabulátorů. + + + Clean indentation + Opravit odsazení + + + Always write a newline character at the end of the file. + Vždy psát znak pro nový řádek na konci souboru. + + + &Ensure newline at end of file + &Doplnit nový řádek na konci souboru + + + File Encodings + Kódování souborů + + + Default encoding: + Výchozí kódování: + + + <html><head/><body> +<p>How text editors should deal with UTF-8 Byte Order Marks. The options are:</p> +<ul ><li><i>Add If Encoding Is UTF-8:</i> always add a BOM when saving a file in UTF-8 encoding. Note that this will not work if the encoding is <i>System</i>, as Qt Creator does not know what it actually is.</li> +<li><i>Keep If Already Present: </i>save the file with a BOM if it already had one when it was loaded.</li> +<li><i>Always Delete:</i> never write an UTF-8 BOM, possibly deleting a pre-existing one.</li></ul> +<p>Note that UTF-8 BOMs are uncommon and treated incorrectly by some editors, so it usually makes little sense to add any.</p> +<p>This setting does <b>not</b> influence the use of UTF-16 and UTF-32 BOMs.</p></body></html> + <html><head/><body> +<p>Udává, jak se chovají textové editory, co se týče UTF-8 BOM (Byte Order Marks). Na výběr je:</p> +<ul ><li><i>Přidat, je-li kódování UTF-8</i> Vždy přidat. BOM při ukládání souboru s kódováním UTF-8. Toto však nebude fungovat, je-li kódování <i>System</i>, neboť Qt Creator v tomto případě kódování nedokáže určit.</li> +<li><i>Zachovat, je-li už přítomno: </i>Uložit soubor s BOM, pokud jej mělo již při nahrání, což bylo zjištěno.</li> +<li><i>Smazat vždy:</i> Nikdy nezapisovat UTF-8 BOM, existující podle okolností odstranit.</li></ul> +<p>Dejte pozor na to, že UTF-8 BOMs nejsou běžné a některé editory s nimi nezacházejí správně. Z toho důvodu to nedává obvykle skoro žádný smysl je přidávat.</p> +<p>Toto nastavení <b>neovlivní</b> používání UTF-16 a UTF-32 BOMs.</p></body></html> + + + Add If Encoding Is UTF-8 + Přidat, pokud je kódování UTF-8 + + + Keep If Already Present + Zachovat, pokud je už přítomno + + + Always Delete + Smazat vždy + + + UTF-8 BOM: + UTF-8 BOM: + + + Mouse and Keyboard + Myš a klávesnice + + + Enable &mouse navigation + Povolit navádění &myší + + + Enable scroll &wheel zooming + Povolit změnu velikosti &kolečkem myši + + + Enable built-in camel case &navigation + Povolit vestavěné &navádění pro "CamelCase" + + + Show help tooltips: + Ukázat vysvětlivky s nápovědou: + + + On Mouseover + Při přejezdu myši + + + On Shift+Mouseover + Při přejezdu a stisku Shift + + + Press Alt to display context-sensitive help or type information as tooltips. + Stiskněte klávesu Alt pro zobrazení vysvětlivek s kontextovou nápovědou nebo informací o typu. + + + Using keyboard shortcut (Alt) + Užitím klávesové zkratky (Alt) + + + + TextEditor::Internal::CodeStyleSelectorWidget + + Form + Formulář + + + Current settings: + Nynější nastavení: + + + Copy... + Kopírovat... + + + Edit... + Upravit... + + + Remove + Odstranit + + + Export... + Vyvést... + + + Import... + Zavést... + + + + TextEditor::Internal::DisplaySettingsPage + + Display + Zobrazení + + + Display line &numbers + Zo&brazit čísla řádků + + + Highlight current &line + Zvýraznit nynější řá&dek + + + Display &folding markers + Zobrazit značky &skládání kódu + + + Highlight &blocks + Zvýraznit &bloky + + + Mark &text changes + Vyznačit &textové změny + + + Show tabs and spaces. + Ukázat tabulátory a prázdné znaky (mezery). + + + &Visualize whitespace + Z&viditelnit prázdné znaky + + + &Animate matching parentheses + &Rozhýbat odpovídající závorky + + + Auto-fold first &comment + Automaticky složit první &poznámku + + + Center &cursor on scroll + Při posunu držet &ukazovátko vprostřed + + + Text Wrapping + Zalomení textu + + + Enable text &wrapping + Povolit &zalomení textu + + + Display right &margin at column: + Zobrazit pravý &okraj sloupce: + + + + TextEditor::Internal::HighlighterSettingsPage + + Form + Formulář + + + <html><head/><body> +<p>Highlight definitions are provided by the <a href="http://kate-editor.org/">Kate Text Editor</a>.</p></body></html> + <html><head/><body> +<p>Soubory s definicemi pro zvýrazňování skladby jsou poskytovány <a href="http://kate-editor.org/">textovým editorem Kate</a>.</p></body></html> + + + Syntax Highlight Definition Files + Soubory s definicí pro zvýraznění skladby + + + Location: + Umístění: + + + Use fallback location + Použít záložní umístění + + + Behavior + Chování + + + Alert when a highlight definition is not found + Zobrazit upozornění, pokud se nepodařilo najít žádný soubor s definicí + + + Ignored file patterns: + Vyloučené soubory (vzor hledání): + + + + TextEditor::Internal::SnippetsSettingsPage + + Form + Formulář + + + Group: + Skupina: + + + Add + Přidat + + + Remove + Odstranit + + + Revert Built-in + Vrátit zpět vestavěný + + + Restore Removed Built-ins + Obnovit odstraněné vestavěné + + + Reset All + Nastavit znovu vše + + + + TextEditor::Internal::TabSettingsWidget + + Form + Formulář + + + Tabs And Indentation + Tabulátory a odsazení + + + Tab policy: + Chování tabulátorů: + + + Spaces Only + Pouze mezery + + + Tabs Only + Pouze tabulátory + + + Mixed + Smíchaně + + + Ta&b size: + Šířka &tabulátoru: + + + &Indent size: + &Velikost odsazení: + + + Align continuation lines: + Zarovnání navazujících řádků: + + + <html><head/><body> +Influences the indentation of continuation lines. + +<ul> +<li>Not At All: Do not align at all. Lines will only be indented to the current logical indentation depth. +<pre> +(tab)int i = foo(a, b +(tab)c, d); +</pre> +</li> + +<li>With Spaces: Always use spaces for alignment, regardless of the other indentation settings. +<pre> +(tab)int i = foo(a, b +(tab) c, d); +</pre> +</li> + +<li>With Regular Indent: Use tabs and/or spaces for alignment, as configured above. +<pre> +(tab)int i = foo(a, b +(tab)(tab)(tab) c, d); +</pre> +</li> +</ul></body></html> + <html><head/><body> +Určuje chování odsazení se zřetelem k navazujícím řádkům. + +<ul> +<li>Žádné odsazení: Žádné odsazení dodatečně k logickému odsazení. +<pre> +(tab)int i = foo(a, b +(tab)c, d); +</pre> +</li> + +<li>Prázdné znaky: Dodatečné odsazení pomocí prázdných znaků. +<pre> +(tab)int i = foo(a, b +(tab) c, d); +</pre> +</li> + +<li>Normální odsazení: Používají se tabulátory nebo prázdné znaky odpovídající hořejšímu nastavení. +<pre> +(tab)int i = foo(a, b +(tab)(tab)(tab) c, d); +</pre> +</li> +</ul></body></html> + + + Not At All + Žádné odsazení + + + With Spaces + Prázdné znaky + + + With Regular Indent + Normální odsazení + + + + Todo::Internal::KeywordDialog + + Keyword + Klíčové slovo + + + Icon + Ikona + + + Color + Barva + + + errorLabel + chybaŠtítek + + + Keyword cannot be empty, contain spaces or colons. + Klíčové slovo nemůže být prázdné, nemůže obsahovat mezery nebo dvojtečky. + + + There is already a keyword with this name. + Klíčové slovo s tímto názvem již existuje. + + + + Todo::Internal::OptionsDialog + + Form + Formulář + + + Keywords + Štítky + + + Add + Přidat + + + Edit + Úpravy + + + Remove + Odstranit + + + Reset + Nastavit znovu + + + Scanning scope + Oblast prohledávání + + + Scan in the whole project + Hledat v celém projektu + + + Scan in the current opened file + Hledat v nyní otevřeném souboru + + + + VcsBase::Internal::BaseCheckoutWizardPage + + WizardPage + WizardPage + + + Repository + Skladiště + + + The remote repository to check out. + Vzdálené skladiště ke stažení. + + + Branch: + Větev: + + + The development branch in the remote repository to check out. + Vývojářská větev ve vzdáleném skladišti ke stažení. + + + Retrieve list of branches in repository. + Získat seznam větví ve skladišti. + + + ... + ... + + + Working Copy + Pracovní kopie + + + The path in which the directory containing the checkout will be created. + Cesta, ve které bude vytvořen adresář obsahující stažené. + + + Checkout path: + Cesta ke staženému: + + + The local directory that will contain the code after the checkout. + Místní adresář, který bude po stažení obsahovat kód. + + + Checkout directory: + Adresář se stažením (checkout): + + + + VcsBase::Internal::CleanDialog + + Clean Repository + Uklidit skladiště + + + + VcsBase::Internal::CommonSettingsPage + + Wrap submit message at: + Zalomit popis revize na: + + + characters + znacích + + + An executable which is called with the submit message in a temporary file as first argument. It should return with an exit != 0 and a message on standard error to indicate failure. + Spustitelný soubor, volaný s popisem revize v dočasném souboru jako prvním argumentem. Při neúspěchu má vrátit nenulovou hodnotu a poslat na chybový výstup zprávu o selhání. + + + Submit message &check script: + Skript k &ověření popisu revize: + + + A file listing user names and email addresses in a 4-column mailmap format: +name <email> alias <email> + Soubor, který obsahuje jména uživatelů a e-mailové adresy ve čtyřsloupcovém "mailmap" formátu: +Jméno <E-mail> Přezdívka <E-mail> + + + User/&alias configuration file: + Soubor s nastavením uživatele/&přezdívky: + + + A simple file containing lines with field names like "Reviewed-By:" which will be added below the submit editor. + Soubor, který obsahuje řádky s názvy polí (například "Reviewed-By:"), který bude bude přidán níže v editoru odeslání (submit). + + + User &fields configuration file: + Soubor s nastavením p&olí uživatele: + + + &Patch command: + &Příkaz pro záplatování: + + + Specifies a command that is executed to graphically prompt for a password, +should a repository require SSH-authentication (see documentation on SSH and the environment variable SSH_ASKPASS). + Příkaz pro grafickou výzvu k zadání hesla, které je požadováno při ověření pravosti SSH skladiště +(viz dokumentaci k SSH a k proměnné prostředí SSH-ASKPASS). + + + &SSH prompt command: + &Příkaz pro výzvu o heslo k SSH: + + + + VcsBase::Internal::VcsConfigurationPage + + Configure... + Nastavení... + + + + develop + + Develop + Vývoj + + + Sessions + Sezení + + + Recent Projects + Naposledy otevřené projekty + + + Open Project + Otevřít projekt + + + Create Project + Vytvořit projekt + + + + examples + + Examples + Příklady + + + Search in Examples... + Hledat v příkladech... + + + + gettingstarted + + Getting Started + Jak začít + + + To select a tutorial and learn how to develop applications. + Pro výběr návodů a lekcí k vývoji programů. + + + Start Developing + Začít vyvíjet + + + To check that the Qt SDK installation was successful, open an example application and run it. + Pro ověření, že instalace Qt SDK proběhla úspěšně, otevřete vzorový program a spusťte jej. + + + Building and Running an Example Application + Sestavení a spuštění vzorového programu + + + IDE Overview + Přehled vývojového prostředí + + + To find out what kind of integrated enviroment (IDE) Qt Creator is. + Pro bližší seznámení s jednotným vývojovým prostředím (IDE) Qt Creator. + + + To become familar with the parts of the Qt Creator user interface and to learn how to use them. + Obeznámit se s jednotlivými částmi uživatelského rozhraní Qt Creator a naučit se jak je používat. + + + User Interface + Uživatelské rozhraní + + + User Guide + Průvodce pro uživatele + + + Online Community + Společenství na síti + + + Labs + Laboratoře + + + + tutorials + + Tutorials + Návody + + + Search in Tutorials... + Hledat v návodech... + + + + Delegate + + 2D PAINTING EXAMPLE long description + Delší popis pro příklad dvojrozměrného malování + + + The 2D Painting example shows how QPainter and QGLWidget work together. + Příklad dvojrozměrného malování ukazuje, jak QPainter and QGLWidget pracují dohromady. + + + The 2D Painting example shows how QPainter and QGLWidget. The 2D Painting example shows how QPainter and QGLWidget work together. + Příklad dvojrozměrného malování ukazuje, jak QPainter and QGLWidget pracují dohromady. + + + Tags: + Štítky: + + + + SearchBar + + Search... + Hledat... + + + + SessionItem + + Clone + Klonovat + + + Rename + Přejmenovat + + + Delete + Smazat + + + + Sessions + + %1 (last session) + %1 (poslední sezení) + + + %1 (current session) + %1 (nynější sezení) + + + + QmlDebug::QmlOutputParser + + The port seems to be in use. + Error message shown after 'Could not connect ... debugger:" + Port se už používá. + + + The application is not set up for QML/JS debugging. + Error message shown after 'Could not connect ... debugger:" + Tento program není nastaven pro ladění QML/JS. + + + + StaticAnalysisMessages + + do not use '%1' as a constructor + '%1' se nesmí používat jako konstruktor + + + invalid value for enum + Neplatná hodnota pro 'enum' + + + enum value must be a string or a number + Hodnota 'enum' musí být řetězcem nebo číslem + + + number value expected + Očekávána číselná hodnota + + + boolean value expected + Očekávána logická hodnota + + + string value expected + Očekáván řetězec + + + invalid URL + Neplatná adresa (URL) + + + file or directory does not exist + Soubor nebo adresář neexistuje + + + invalid color + Neplatná barva + + + anchor line expected + Očekáván kotevní řádek + + + duplicate property binding + Dvojitá vazba vlastnosti + + + id expected + Očekáváno ID + + + invalid id + Neplatné ID + + + duplicate id + Zdvojené ID + + + invalid property name '%1' + Neplatný název vlastnosti '%1' + + + '%1' does not have members + ' %1' nemá členy + + + '%1' is not a member of '%2' + ' %1' nepatří k '%2' + + + assignment in condition + Přiřazení v podmínce + + + unterminated non-empty case block + Neprázdný 'case' blok není ukončen + + + do not use 'eval' + Nepoužívat 'eval' + + + unreachable + Nedosažitelná + + + do not use 'with' + Nepoužívat 'with' + + + do not use comma expressions + Nepoužívat výrazy s čárkou + + + '%1' is already a formal parameter + '%1' už je formálním parametrem + + + unnecessary message suppression + Zbytečné potlačení zprávy + + + '%1' is already a function + '%1' už je funkcí + + + var '%1' is used before its declaration + Proměnná '%1' se používá před její deklarací + + + '%1' is already a var + '%1' už je proměnnou + + + '%1' is declared more than once + '%1' má víc než jednu deklaraci + + + function '%1' is used before its declaration + Funkce '%1' se používá před její deklarací + + + the 'function' keyword and the opening parenthesis should be separated by a single space + Klíčové slovo 'function' a úvodní závorka mají být odděleny jednou mezerou + + + do not use stand-alone blocks + Nepoužívat samostatné bloky + + + do not use void expressions + Nepoužívat prázdné výrazy + + + confusing pluses + Matoucí plusy + + + confusing minuses + Matoucí minusy + + + declare all function vars on a single line + Prohlásit všechny proměnné funkce na jednom řádku + + + unnecessary parentheses + Zbytečné závorky + + + == and != may perform type coercion, use === or !== to avoid + == a != může vynutit převod typu, čemuž se vyhnete použitím === nebo !== + + + expression statements should be assignments, calls or delete expressions only + Příkazy s výrazy by měly být jen přiřazení, volání funkcí nebo výrazy 'delete' + + + var declarations should be at the start of a function + Deklarace proměnných mají stát na začátku funkce + + + only use one statement per line + Použít pouze jeden příkaz na řádek + + + unknown component + Neznámá součástka + + + could not resolve the prototype '%1' of '%2' + Prototyp '%1' z '%2' se nepodařilo vyřešit + + + could not resolve the prototype '%1' + Prototyp '%1' se nepodařilo vyřešit + + + prototype cycle, the last non-repeated component is '%1' + Prototypová smyčka, poslední neopakovaná část je %1 + + + invalid property type '%1' + Neplatný typ vlastnosti '%1' + + + == and != perform type coercion, use === or !== to avoid + == a != vynutí převod typu, čemuž se vyhnete použitím === nebo !== + + + calls of functions that start with an uppercase letter should use 'new' + Funkce začínající velkým písmenem by se měly volat s 'new' + + + 'new' should only be used with functions that start with an uppercase letter + 'new' by se měl používat jen s funkcemi, které začínají velkým písmenem + + + use spaces around binary operators + Užívat mezery okolo binárních operátorů + + + unintentional empty block, use ({}) for empty object literal + Nezamýšlený prázdný blok, použijte '({})' jako symbol prázdného objektu + + + use %1 instead of 'var' or 'variant' to improve performance + Použít %1 namísto 'var' nebo 'variant' pro lepší výkon + + + missing property '%1' + Chybí vlastnost '%1' + + + object value expected + Očekávána hodnota objektu + + + array value expected + Očekávána hodnota pole + + + %1 value expected + Očekávána hodnota %1 + + + maximum number value is %1 + Největší hodnota čísla je %1 + + + minimum number value is %1 + Nejmenší hodnota čísla je %1 + + + maximum number value is exclusive + Největší hodnota čísla je vyloučena + + + minimum number value is exclusive + Nejmenší hodnota čísla je vyloučena + + + string value does not match required pattern + Hodnota řetězce neodpovídá požadovanému vzoru + + + minimum string value length is %1 + Nejmenší délka hodnoty řetězce je %1 + + + maximum string value length is %1 + Největší délka hodnoty řetězce je %1 + + + %1 elements expected in array value + Pole potřebuje %1 hodnot + + + + Utils::BaseTreeView + + Adjust Column Widths to Contents + Přizpůsobit šířku sloupců obsahu + + + + Utils::SettingsSelector + + Add + Přidat + + + Remove + Odstranit + + + Rename + Přejmenovat + + + Do you really want to delete the configuration <b>%1</b>? + Opravdu chcete odstranit nastavení <b>%1</b>? + + + New name for configuration <b>%1</b>: + Nový název pro nastavení <b>%1</b>: + + + Rename... + Přejmenovat... + + + + Utils::SftpFileSystemModel + + File Type + Typ souboru + + + File Name + Název souboru + + + Error getting 'stat' info about '%1': %2 + Chyba při získávání 'stat' informací o '%1': %2 + + + Error listing contents of directory '%1': %2 + Chyba při vypisování obsahu adresáře '%1': %2 + + + + Utils::Internal::SshRemoteProcessPrivate + + Process killed by signal + Proces ukončen signálem + + + Server sent invalid signal '%1' + Server poslal neplatný signál '%1' + + + + ZeroConf::Internal::ZConfLib + + AvahiZConfLib could not load the native library '%1': %2 + AvahiZConfLib se nepodařilo nahrát rodilou knihovnu '%1': %2 + + + %1 could not create a client (probably the daemon is not running) + %1 se nepodařilo vytvořit klienta (pravděpodobně neběží démon) + + + cAvahiClient, server collision + cAvahiClient, střet serveru + + + cAvahiClient, some kind of error happened on the client side + cAvahiClient, na klientské straně se přihodil nějaký druh chyby + + + cAvahiClient, still connecting, no server available + cAvahiClient, stále se připojuje, není dostupný žádný server + + + Error: unexpected state %1 in cAvahiClientReply, ignoring it + Chyba: neočekávaný stav %1 v cAvahiClientReply, přehlíží se + + + Error: unexpected state %1 in cAvahiBrowseReply, ignoring it + Chyba: neočekávaný stav %1 v cAvahiBrowseReply, přehlíží se + + + zeroconf failed to kill other daemons with '%1' + zeroconf se nepodařilo ukončit další démony s '%1' + + + %1 failed starting embedded daemon at %2 + %1 se nepodařilo spustit vloženého démona na %2 + + + + ZeroConf + + DnsSdZConfLib could not load native library + DnsSdZConfLib se nepodařilo nahrát rodilou knihovnu + + + DnsSdZConfLib skipping over avahi compatibility lib (or obsolete mdnsd) + DnsSdZConfLib přeskakuje knihovnu kompatibility avahi (nebo zastaralé mdnsd) + + + *WARNING* DnsSdZConfLib detected an obsolete version of Apple Bonjour, either disable/uninstall it or upgrade it. Otherwise zeroconf will fail. + *WARNING* DnsSdZConfLib zjistil zastaralou verzi Apple Bonjour, buď ji zakažte/odinstalujte, nebo ji povyšte na verzi vyšší. Jinak zeroconf will spadne. + + + MainConnection giving up on non Ok lib %1 (%2) + Hlavní spojení bylo vzdáno na základě chybové knihovny %1 (%2) + + + MainConnection has no valid library, aborting connection + Přerušení hlavního spojení na základě chyby platné knihovny + + + MainConnection giving up on lib %1, switching to lib %2 + Hlavní spojení se s knihovnou %1 nepodařilo vytvořit, nový pokus s knihovnou %2 + + + MainConnection giving up on lib %1, no fallback provided, aborting connection + Hlavní spojení se s knihovnou %1 nepodařilo vytvořit. Následuje přerušení spojení, neboť nebyla poskytnuta žádná záloha + + + MainConnection using lib %1 failed the initialization of mainRef with error %2 + Hlavní spojení se s knihovnou %1 nepodařilo vytvořit, protože se nepodařila inicializace mainRef (chyba %2) + + + MainConnection using lib %1 failed because no daemon is running + Hlavní spojení se při použití knihovny %1 nepodařilo vytvořit, protože neběží žádný proces na pozadí (démon) + + + MainConnection using lib %1 daemon starting seem successful, continuing + Hlavní spojení při použití knihovny %1: proces na pozadí (démon) byl spuštěn. Pokračuje se + + + MainConnection using lib %1 failed getProperty call with error %2 + Hlavní spojení se při použití knihovny %1 nepodařilo vytvořit, protože se nepodařilo vyvolat getProperty (chyba %2) + + + MainConnection could successfully create a connection using lib %1 + Hlavní spojení se při použití knihovny %1 podařilo úspěšně vytvořit + + + MainConnection::handleEvents called with m_status != Starting, aborting + MainConnection::handleEvents bylo vyvoláno s m_status != Starting , přerušení + + + MainConnection::handleEvents unexpected return status of handleEvent + MainConnection::handleEvents: neočekávaný vrácený stav handleEvent + + + MainConnection for [%1] accumulated %2 consecutive errors, aborting + Hlavní spojení pro [%1] nahromadilo %2 po sobě jdoucích chyb, přerušení + + + + Analyzer::Internal::AnalyzerToolDetailWidget + + <strong>%1</strong> settings + Nastavení <strong>%1</strong> + + + + Analyzer::Internal::AnalyzerRunConfigWidget + + Analyzer settings: + Nastavení rozboru: + + + Analyzer Settings + Nastavení rozboru + + + + Analyzer::Internal::AnalyzerRunControlFactory + + Analyzer + Rozbor + + + + Analyzer::AnalyzerRunConfigurationAspect + + Analyzer Settings + Nastavení rozboru + + + + Android::Internal::AndroidConfigurations + + Create AVD error + Chyba při vytváření AVD + + + Can't create a new AVD, not enough android SDKs available +Please install one SDK with api version >=%1 + Nelze vytvořit nové AVD, není dostupný dostatek SDK pro Android +Nainstalujte, prosím, jedno SDK s API verze >=%1 + + + + Android + + Android + Android + + + + QtSupport + + Android + Android + + + MeeGo/Harmattan + MeeGo/Harmattan + + + Symbian + Symbian + + + Desktop + Stolní počítač + + + Embedded Linux + Vložený Linux + + + Windows CE + Windows CE + + + + Android::Internal::AndroidCreateKeystoreCertificate + + <span style=" color:#ff0000;">Password is too short</span> + <span style=" color:#ff0000;">Heslo je příliš krátké</span> + + + <span style=" color:#ff0000;">Passwords don't match</span> + <span style=" color:#ff0000;">Heslo neodpovídá</span> + + + <span style=" color:#00ff00;">Password is ok</span> + <span style=" color:#ff0000;">Heslo je OK</span> + + + Keystore file name + Název souboru úložiště klíče + + + Keystore files (*.keystore *.jks) + Soubory úložiště klíče (*.keystore *.jks) + + + Error + Chyba + + + + Android::Internal::AndroidDeployConfiguration + + Deploy to Android device + Poslat na zařízení Android + + + + Android::Internal::AndroidDeployConfigurationFactory + + Deploy on Android + Poslat na Android + + + + Android::Internal::AndroidDeployStep + + Deploy to Android device + AndroidDeployStep default display name + Poslat na zařízení Android + + + Cannot deploy: current target is not android. + Nelze nasadit: nynějšícíl není Android. + + + Please wait, searching for a suitable device for target:%1. + Počkejte, prosím, hledá se vhodné zařízení pro cíl %1. + + + Cannot deploy: no devices or emulators found for your package. + Nelze nasadit: Pro váš balíček nebyla nalezena žádná zařízení nebo emulátory. + + + Package deploy: Running command '%1 %2'. + Nasazení balíčku: Spouští se příkaz '%1 %2'. + + + Packaging error: Could not start command '%1 %2'. Reason: %3 + Chyba při vytváření balíčku: Nepodařilo se spustit příkaz '%1 %2': Důvod %3 + + + Packaging Error: Command '%1 %2' failed. + Chyba při vytváření balíčku: Příkaz '%1 %2' se nepodařilo provést. + + + Reason: %1 + Důvod: %1 + + + Exit code: %1 + Kód ukončení: %1 + + + Clean old qt libs + Uklidit staré knihovny qt + + + Deploy qt libs ... this may take some time, please wait + Nasadit knihovny qt ...To může nějakou dobu trvat, počkejte, prosím + + + Qt Android smart installer instalation failed + Instalace chytrého Qt instalátoru pro Android se nezdařila + + + Installing package onto %1. + Instalace balíčku do %1. + + + Package instalation failed + Instalace balíčku se nezdařila + + + Pulling files necessary for debugging + Vytahují se soubory nezbytné pro ladění + + + + Qt4ProjectManager::Internal::AndroidDeployStepFactory + + Deploy to Android device/emulator + Nasadit na zařízení/emulátor Android + + + + Android::Internal::AndroidDeployStepWidget + + <b>Deploy configurations</b> + <b>Nastavení nasazení</b> + + + Qt Android smart installer + Chytrý instalátor pro Qt Android + + + Android package (*.apk) + Balíček pro Android (*.apk) + + + + Qt4ProjectManager::Internal::AndroidPackageCreationFactory + + Create Android (.apk) Package + Vytvořit balíček pro Android (*.apk) + + + + Android::Internal::AndroidPackageCreationStep + + Packaging for Android + Vytvoření balíčku pro Android + + + Cannot create android package: current target is not android. + Nelze vytvořit balíček pro Android: nynější cíl není Android. + + + Can't find read elf information + Nelze najít informaci pro čtení ELF + + + Can't find '%1'. +Please make sure your application is built successfully and is selected in Application tab ('Run option') + Nelze najít '%1'. +Ujistěte se, prosím, že váš program je sestaven úspěšně a je vybrán na kartě s programy ('Volba pro spuštění') + + + Error + Chyba + + + Failed to run keytool + Nepodařilo se spustit nástroj klíče + + + Invalid password + Neplatné heslo + + + Copy Qt app & libs to Android package ... + Kopírovat programy & knihovny Qt do balíčku Android... + + + Can't copy gdbserver from '%1' to '%2' + Nelze kopírovat gdbserver z '%1' do '%2' + + + Creating package file ... + Vytváří se soubor s balíčkem... + + + Signing package ... + Podepisuje se balíček... + + + Failed, try again + Nepodařilo se. Zkuste to znovu + + + Release signed package created to %1 + Vydat podepsaný vytvořený soubor s balíčkem do %1 + + + Package created. + Soubor s balíčkem byl vytvořen. + + + Package deploy: Running command '%1 %2'. + Nasazení balíčku: Spouští se příkaz '%1 %2'. + + + Packaging failed. + Vytvoření balíčku se nezdařilo. + + + Packaging error: Could not start command '%1 %2'. Reason: %3 + Chyba při vytváření balíčku: Nepodařilo se spustit příkaz '%1 %2': Důvod %3 + + + Packaging Error: Command '%1 %2' failed. + Chyba při vytváření balíčku: Příkaz '%1 %2' se nepodařilo provést. + + + Reason: %1 + Důvod: %1 + + + Exit code: %1 + Kód ukončení: %1 + + + Keystore + Úložiště pro klíč + + + Keystore password: + Heslo pro úložiště klíče: + + + Certificate + Certifikát + + + Certificate password (%1): + Heslo k certifikátu (%1): + + + + Android::Internal::AndroidPackageCreationWidget + + Invalid package name + Neplatný název balíčku + + + The package name '%1' is not valid. +Please choose a valid package name for your application (e.g. "org.example.myapplication"). + Název balíčku '%1' není platný. +Vyberte, prosím, platný název balíčku pro váš program (např. "org.example.myapplication"). + + + Choose High DPI Icon + Vybrat ikonu s vysokým dpi + + + png images (*.png) + Soubory s obrázky PNG (*.png) + + + Choose Medium DPI Icon + Vybrat ikonu se středním dpi + + + Choose Low DPI Icon + Vybrat ikonu s nízkým dpi + + + < type or choose a permission > + < napište nebo vyberte oprávnění > + + + <b>Package configurations</b> + <b>Nastavení balíčků</b> + + + Select keystore file + Vybrat soubor úložiště klíče + + + Keystore files (*.keystore *.jks) + Soubory úložiště klíče (*.keystore *.jks) + + + + Qt4ProjectManager::Internal::AndroidPackageInstallationFactory + + Deploy to device + Nasadit na zařízení + + + + Android::Internal::AndroidPackageInstallationStep + + Copy application data + Kopírovat data programu + + + Current target is not an android target + Nynější cíl není cílem Android + + + + Android::Internal::AndroidRunConfiguration + + Run on Android device + Spustit na zařízení Android + + + + Android::Internal::AndroidRunControl + + Starting remote process ... + Spouští se vzdálený proces... + + + + Android::Internal::AndroidRunControlFactory + + Run on Android device/emulator + Spustit na zařízení/emulátor Android + + + + Android::Internal::AndroidRunner + + + +'%1' died + + +'%1' zemřel + + + Failed to forward debugging ports. Reason: $1 + Nepodařilo se přeposlat porty pro ladění. Důvod: $1 + + + Failed to forward debugging ports + Nepodařilo se přeposlat porty pro ladění. Důvod: $1 + + + Failed to start the activity. Reason: $1 + Nepodařilo se spustit činnost. Důvod: $1 + + + Unable to start '%1' + Nelze spustit '%1' + + + Can't find %1 process + Nelze najít proces %1 + + + + +'%1' killed + + +'%1' ukončen + + + + Android::Internal::AndroidSettingsPage + + Android Configurations + Nastavení pro Android + + + + Android::Internal::AVDModel + + AVD Name + Název AVD + + + AVD Target + Cíl AVD + + + CPU/ABI + CPU/ABI + + + + Android::Internal::AndroidSettingsWidget + + Android SDK Folder + Složka SDK pro Android + + + "%1" doesn't seem to be an Android SDK top folder + "%1" se nezdá být hlavní složkou SDK pro Android + + + "%1" doesn't seem to be an Android NDK top folder + "%1" se nezdá být hlavní složkou NDK pro Android + + + Select Android SDK folder + Vybrat složku SDK pro Android + + + Select Android NDK folder + Vybrat složku NDK pro Android + + + Select ant script + Vybrat skript ant + + + Select gdb executable + Vybrat spustitelný soubor gdb + + + Select gdbserver android executable + Vybrat spustitelný soubor android gdbserver + + + Select OpenJDK path + Vybrat cestu pro OpenJDK + + + + Android::Internal::AndroidTarget + + Error creating Android directory '%1'. + Chyba při vytváření adresáře Android '%1'. + + + No Qt for Android SDKs were found. +Please install at least one SDK. + Nepodařilo se najít žádné Qt pro SDK Android. +Nainstalujte, prosím, alespoň jedno SDK. + + + Warning + Varování + + + Android files have been updated automatically + Soubory Android byly zaktualizovány automaticky + + + Can't parse '%1' + Nelze zpracovat '%1' + + + Can't open '%1' + Nelze otevřít '%1' + + + Error creating Android templates + Chyba při vytváření předloh Android + + + + Android::Internal::AndroidTargetFactory + + Debug + Ladění + + + Release + Vydání + + + + Android::Internal::AndroidToolChainFactory + + Android GCC + GCC Android + + + Android Gcc for %1 + GCC Android pro %1 + + + Android GCC (%1-%2) + GCC Android (%1-%2) + + + + Android::Internal::AndroidToolChainConfigWidget + + NDK Root: %1 + Kořen (root) NDK: %1 + + + + AutotoolsProjectManager::Internal::AutogenStepFactory + + Autogen + Display name for AutotoolsProjectManager::AutogenStep id. + Autogen + + + + AutotoolsProjectManager::Internal::AutogenStep + + Autogen + Autogen + + + Configuration unchanged, skipping autogen step. + Nastavení nezměněno. Přeskakuje se krok autogen. + + + + AutotoolsProjectManager::Internal::AutogenStepConfigWidget + + Arguments: + Argumenty: + + + Autogen + AutotoolsProjectManager::AutogenStepConfigWidget display name. + Autogen + + + + AutotoolsProjectManager::Internal::AutoreconfStepFactory + + Autoreconf + Display name for AutotoolsProjectManager::AutoreconfStep id. + Autoreconf + + + + AutotoolsProjectManager::Internal::AutoreconfStep + + Autoreconf + Autoreconf + + + Configuration unchanged, skipping autoreconf step. + Nastavení nezměněno. Přeskakuje se krok autoreconf. + + + + AutotoolsProjectManager::Internal::AutoreconfStepConfigWidget + + Arguments: + Argumenty: + + + Autoreconf + AutotoolsProjectManager::AutoreconfStepConfigWidget display name. + Autoreconf + + + + AutotoolsProjectManager::Internal::AutotoolsBuildConfigurationFactory + + Build + Sestavování + + + New Configuration + Nové nastavení + + + New configuration name: + Název nového nastavení: + + + + AutotoolsProjectManager::Internal::AutotoolsBuildSettingsWidget + + Build directory: + Adresář pro sestavování: + + + Tool chain: + Sada nástrojů: + + + <Invalid tool chain> + <Neplatná sada nástrojů> + + + + AutotoolsProjectManager::Internal::AutotoolsManager + + Failed opening project '%1': Project file does not exist + Projekt %1 se nepodařil otevřít: Soubor s projektem neexistuje + + + Failed opening project '%1': Project already open + Projekt '%1' se nepodařil otevřít, neboť projekt je již otevřen + + + + AutotoolsProjectManager::Internal::AutotoolsOpenProjectWizard + + Autotools Wizard + Průvodce Autotools + + + + AutotoolsProjectManager::Internal::BuildPathPage + + Please enter the directory in which you want to build your project. Qt Creator recommends to not use the source directory for building. This ensures that the source directory remains clean and enables multiple builds with different settings. + Zadejte, prosím, adresář, ve kterém chcete vytvořit svůj projekt.. Doporučuje se, nepoužívat pro vytvoření projektu zdrojový adresář. Tím se zajistí, že zdrojový adresář zůstane volný, a umožní různá sestavení s rozdílnými nastaveními. + + + Build directory: + Adresář pro sestavování: + + + Build Location + Umístění sestavování + + + + AutotoolsProjectManager::Internal::AutotoolsTarget + + Desktop + Autotools Default target display name + Stolní počítač + + + + AutotoolsProjectManager::Internal::AutotoolsTargetFactory + + Default Build + Výchozí sestavení + + + + AutotoolsProjectManager::Internal::ConfigureStepFactory + + Configure + Display name for AutotoolsProjectManager::ConfigureStep id. + Configure + + + + AutotoolsProjectManager::Internal::ConfigureStep + + Configure + Configure + + + Configuration unchanged, skipping configure step. + Nastavení nezměněno, přeskakuje se krok nastavení. + + + + AutotoolsProjectManager::Internal::ConfigureStepConfigWidget + + Arguments: + Argumenty: + + + Configure + AutotoolsProjectManager::ConfigureStepConfigWidget display name. + Nastavit + + + + AutotoolsProjectManager::Internal::MakefileParser + + Parsing %1 in directory %2 + Zpracovává se %1 v adresáři %2 + + + Parsing directory %1 + Zpracovává se adresář %1 + + + + AutotoolsProjectManager::Internal::MakeStepFactory + + Make + Display name for AutotoolsProjectManager::MakeStep id. + Make + + + + AutotoolsProjectManager::Internal::MakeStep + + Make + Make + + + + AutotoolsProjectManager::Internal::MakeStepConfigWidget + + Arguments: + Argumenty: + + + Make + AutotoolsProjectManager::MakeStepConfigWidget display name. + Make + + + <b>Unknown tool chain</b> + <b>Neznámá sada nástrojů</b> + + + + BinEditorDocument + + Cannot open %1: %2 + Nelze otevřít soubor '%1': %2 + + + File Error + Chyba souboru + + + + CMakeProjectManager::Internal::CMakeLocatorFilter + + Build CMake target + Sestavit cíl CMake + + + + Core::DocumentManager + + File Error + Chyba souboru + + + Error while saving file: %1 + Při ukládání souboru: %1 se vyskytla chyba + + + Overwrite? + Přepsat? + + + An item named '%1' already exists at this location. Do you want to overwrite it? + V tomto umístění již existuje soubor s názvem '%1'. Chcete jej přepsat? + + + Save File As + Uložit soubor jako + + + Open File + Otevřít soubor + + + File Is Read Only + Soubor je pouze pro čtení + + + The file <i>%1</i> is read only. + Soubor <i>%1</i> je pouze pro čtení. + + + Open with VCS (%1) + Otevřít s pomocí systému na ověřování verzí (VCS) (%1) + + + Make Writable + Udělat zapisovatelným + + + Save As... + Uložit jako... + + + Cannot reload %1 + %1 se nahrát znovu nepodařilo + + + + Core::IDocument + + File was restored from auto-saved copy. Use <i>Save</i> to confirm, or <i>Revert to Saved</i> to discard changes. + Soubor byl obnoven z automaticky uložené záložní kopie. Vyberte <i>Uložit</i> pro potvrzení nebo <i>Obnovit</i> pro zahození změn. + + + + QuickFix::ExtractFunction + + Extract Function + Vytáhnout funkci + + + Extract Function Refactoring + Vytáhnout funkci + + + Enter function name + Zadat název funkce + + + Invalid function name + Neplatný název funkce + + + + CppTools::CppClassesFilter + + Classes + Třídy + + + + Debugger::Internal::BreakTreeView + + Delete Breakpoint + Smazat bod přerušení + + + Delete All Breakpoints + Smazat všechny body přerušení + + + Delete Breakpoints of "%1" + Smazat body přerušení v "%1" + + + Delete Breakpoints of File + Smazat body přerušení v souboru + + + Adjust Column Widths to Contents + Přizpůsobit šířku sloupců obsahu + + + Edit Breakpoint... + Upravit bod přerušení... + + + Associate Breakpoint With All Threads + Dát bod přerušení pro všechna vlákna + + + Associate Breakpoint With Thread %1 + Dát bod přerušení pro vlákno %1 + + + Synchronize Breakpoints + Seřídit body přerušení + + + Disable Selected Breakpoints + Vypnout vybrané body přerušení + + + Enable Selected Breakpoints + Zapnout vybrané body přerušení + + + Disable Breakpoint + Vypnout bod přerušení + + + Enable Breakpoint + Zapnout bod přerušení + + + Add Breakpoint... + Přidat bod přerušení... + + + Add Breakpoint + Přidat bod přerušení + + + + Debugger::Internal::TypeFormatsDialog + + Reset + Nastavit znovu + + + Type Formats + Formáty typů + + + Qt Types + Typy Qt + + + Standard Types + Obvyklé typy + + + Misc Types + Různé typy + + + + Debugger::DebuggerEnginePrivate + + Attempting to interrupt. + Pokus o přerušení. + + + Debug Information + Informace o ladění + + + Debugger Test + Zkouška ladění + + + + Debugger::Internal::DebuggerRunConfigWidget + + Debugger Settings + Nastavení ladiče + + + Enable C++ + Povolit C++ + + + Enable QML + Povolit QML + + + Debug port: + Port pro ladič: + + + <a href="qthelp://com.nokia.qtcreator/doc/creator-debugging-qml.html">What are the prerequisites?</a> + <a href="qthelp://com.nokia.qtcreator/doc/creator-debugging-qml.html">Jaké jsou předpoklady?</a> + + + Enable Debugging of Subprocesses + Povolit ladění podřízených procesů + + + + DebuggerPlugin + + Debug + Ladění + + + Debugger + Ladič + + + Unable to create a debugger engine of the type '%1' + Nepodařilo se vytvořit žádný ladicí stroj typu '%1' + + + + Debugger::Internal::ModulesTreeView + + Update Module List + Obnovit seznam modulů + + + Show Source Files for Module "%1" + Ukázat zdrojové soubory k modulu "%1" + + + Load Symbols for All Modules + Nahrát symboly ke všem modulům + + + Examine All Modules + Vyšetřit všechny moduly + + + Load Symbols for Module + Nahrát symboly k modulu + + + Edit File + Upravit soubor + + + Show Symbols + Ukázat symboly + + + Show Dependencies + Ukázat závislosti + + + Load Symbols for Module "%1" + Nahrát symboly k modulu "%1" + + + Edit File "%1" + Upravit soubor "%1" + + + Show Symbols in File "%1" + Ukázat symboly v souboru "%1" + + + Show Dependencies of "%1" + Ukázat závislosti "%1" + + + + Debugger::Internal::QmlV8DebuggerClient + + anonymous function + Anonymní funkce + + + + Debugger::Internal::QtMessageLogEditor + + Cu&t + Vyj&mout + + + &Copy + &Kopírovat + + + &Paste + &Vložit + + + Select &All + Vybrat &vše + + + C&lear + Sma&zat + + + + Debugger::Internal::QtMessageLogView + + &Copy + &Kopírovat + + + &Show in Editor + &Ukázat v editoru + + + C&lear + Sma&zat + + + + Debugger::Internal::QtMessageLogWindow + + Log + Záznamy + + + Show debug, log, and info messages. + Ukázat ladění, záznamy a informační zprávy. + + + Warning + Varování + + + Show warning messages. + Ukázat varovné zprávy. + + + Error + Chyba + + + Show error and fatal messages. + Ukázat chybové a kritické zprávy. + + + Clear Console + Smazat konzoli + + + + Debugger::Internal::RegisterTreeView + + Reload Register Listing + Registry nahrát znovu + + + Open Disassembler... + Otevřít disasembler... + + + Open Memory Editor at 0x%1 + Otevřít editor paměti při 0x%1 + + + Open Memory View at Value of Register %1 0x%2 + Otevřít náhled paměti při hodnotě registru %1 0x%2 + + + Open Disassembler at 0x%1 + Otevřít disasembler při %1 + + + Open Memory Editor + Otevřít editor paměti + + + Open Memory View at Value of Register + Otevřít náhled paměti při hodnotě registru + + + Open Disassembler + Otevřít disasembler + + + Hexadecimal + Šestnáctková + + + Decimal + Desítková + + + Octal + Osmičková + + + Binary + Dvojková + + + + Debugger::Internal::SnapshotTreeView + + Snapshots + Snímky + + + Create Snapshot + Vytvořit snímek + + + Remove Snapshot + Odstranit snímek + + + + Debugger::Internal::SourceFilesTreeView + + Reload Data + Data nahrát znovu + + + Open File + Otevřít soubor + + + Open File "%1"' + Otevřít soubor "%1"' + + + + Debugger::Internal::StackTreeView + + Stack + Zásobník + + + Function: + Funkce: + + + Disassemble Function + Rozložit funkci (disassemble) + + + Copy Contents to Clipboard + Kopírovat obsah do schránky + + + Open Memory Editor + Otevřít editor paměti + + + Open Memory Editor at 0x%1 + Otevřít editor paměti při 0x%1 + + + Open Disassembler at Address... + Otevřít disasembler na adrese... + + + Disassemble Function... + Rozložit funkci (disassemble)... + + + Open Disassembler + Otevřít disasembler + + + Open Disassembler at 0x%1 + Otevřít disasembler při %1 + + + Try to Load Unknown Symbols + Nahrát neznámé symboly + + + Memory at Frame #%1 (%2) 0x%3 + Paměť při snímku #%1 (%2) 0x%3 + + + Frame #%1 (%2) + Snímek #%1 (%2) + + + + Debugger::Internal::WatchTreeView + + <i>%1</i> %2 at #%3 + HTML tooltip of a variable in the memory editor + <i>%1</i> %2 při #%3 + + + <i>%1</i> %2 + HTML tooltip of a variable in the memory editor + <i>%1</i> %2 + + + Register <i>%1</i> + Registr <i>%1</i> + + + Memory Referenced by Pointer "%1" (0x%2) + Paměť odkazovaná ukazatelem "%1" (0x%2) + + + Memory at Variable "%1" (0x%2) + Paměť při proměnné "%1" (0x%2) + + + Cannot Display Stack Layout + Nelze zobrazit rozvržení zásobníku + + + Could not determine a suitable address range. + Nepodařilo se určit žádný vhodný rozsah adresy. + + + Memory Layout of Local Variables at 0x%1 + Rozložení paměti místní proměnné při 0x%1 + + + Locals and Expressions + Místní proměnné a výrazy + + + Evaluate Expression + Vyhodnotit výraz + + + Evaluate Expression "%1" + Vyhodnotit výraz "%1" + + + Remove Evaluated Expression + Odstranit vyhodnocený výraz + + + Remove Evaluated Expression "%1" + Odstranit vyhodnocený výraz "%1" + + + Change Local Display Format... + Změnit místní formát zobrazení... + + + Treat All Characters as Printable + Považovat všechny znaky za tisknutelné + + + Show Unprintable Characters as Escape Sequences + Ukázat netisknutelné znaky jako únikové posloupnosti + + + Show Unprintable Characters as Octal + Ukázat netisknutelné znaky jako osmičkové + + + Show Unprintable Characters as Hexadecimal + Ukázat netisknutelné znaky jako šestnáctkové + + + Change Display for Object Named "%1": + Změnit formát zobrazení pro objekt s názvem "%1": + + + Use Format for Type (Currently %1) + Použít formát zobrazení na typ (nyní %1) + + + Use Display Format Based on Type + Použít formát zobrazení založený na typu + + + Change Display for Type "%1": + Změnit zobrazení pro typ '%1': + + + Automatic + Automaticky + + + Change Display for Type or Item... + Změnit zobrazení pro typ nebo prvek... + + + Add Data Breakpoint... + Přidat bod přerušení dat... + + + Add Data Breakpoint at Object's Address (0x%1) + Přidat bod přerušení dat na adrese objektu (0x%1) + + + Add Data Breakpoint at Referenced Address (0x%1) + Přidat bod přerušení dat na odkazované adrese (0x%1) + + + Add Data Breakpoint + Přidat bod přerušení dat + + + Setting a data breakpoint on an address will cause the program to stop when the data at the address is modified. + Nastavení bodu přerušení dat na určitou adresu způsobí, že program se zastaví, když jsou tam nacházející se data změněna. + + + Add Data Breakpoint at Expression + Přidat bod přerušení dat při výrazu + + + Add Data Breakpoint at Expression "%1" + Přidat bod přerušení dat při výrazu "%1" + + + Setting a data breakpoint on an expression will cause the program to stop when the data at the address given by the expression is modified. + Nastavení bodu přerušení dat na určitý výraz způsobí, že program se zastaví, když jsou změněna data nacházející se na adrese dané oním výrazem. + + + Insert New Evaluated Expression + Vložit nový vyhodnocený výraz + + + Select Widget to Watch + Vybrat prvek za účelem sledování + + + Change Global Display Formats... + Změnit celkové formáty zobrazení... + + + Remove All Evaluated Expressions + Odstranit všechny vyhodnocené výrazy + + + Open Memory Editor... + Otevřít editor paměti... + + + Open Memory Editor at Object's Address (0x%1) + Otevřít editor paměti na adrese objektu (0x%1) + + + Open Memory View at Object's Address (0x%1) + Otevřít zobrazení paměti na adrese objektu (0x%1) + + + Open Memory Editor at Object's Address + Otevřít editor paměti na adrese objektu + + + Open Memory View at Object's Address + Otevřít zobrazení paměti na adrese objektu + + + Open Memory Editor at Referenced Address (0x%1) + Otevřít editor paměti na odkazované adrese (0x%1) + + + Open Memory View at Referenced Address (0x%1) + Otevřít zobrazení paměti na odkazované adrese (0x%1) + + + Open Memory Editor at Referenced Address + Otevřít editor paměti na odkazované adrese + + + Open Memory View at Referenced Address + Otevřít zobrazení paměti na odkazované adrese + + + Open Memory Editor Showing Stack Layout + Otevřít editor paměti s ukázáním rozložení zásobníku + + + Copy Contents to Clipboard + Obsah kopírovat do schránky + + + Copy Value to Clipboard + Kopírovat hodnotu do schránky + + + Refresh Code Model Snapshot + Obnovit stav modelu kódu + + + Show View Contents in Editor + Ukázat obsah pohledu v editoru + + + Close Editor Tooltips + Zavřít vysvětlivky editoru + + + Enter watch expression + Zadat sledovaný výraz + + + Expression: + Výraz: + + + Locals & Watchers + Místní proměnné a sledované výrazy + + + + Git::Internal::CommitData + + untracked + neverzováno + + + staged + + připraveno k zápisu + + + + modified + změněno + + + added + přidáno + + + deleted + smazáno + + + renamed + přejmenováno + + + copied + zkopírováno + + + updated + aktualizováno + + + + Gerrit::Internal::GerritDialog + + Gerrit %1@%2 + Gerrit %1@%2 + + + Open + Otevřít + + + Copy URL + Kopírovat adresu (URL) + + + Diff... + Rozdíly... + + + Apply... + Použít... + + + Checkout... + Načíst (checkout)... + + + Refresh + Obnovit + + + + Gerrit::Internal::GerritModel + + Subject: %1 +Number: %2 Id: %3 +Owner: %4 <%5> +Project: %6 Branch: %7 Patch set: %8 +Status: %9, %10 +URL: %11 +Approvals: %12 + Předmět: %1 +Číslo: %2 Id: %3 +Vlastník: %4 <%5> +Projekt: %6 Větev: %7 Sada změn: %8 +Stav: %9, %10 +Adresa (URL): %11 +Schválení: %12 + + + Title + Název + + + Owner + Vlastník + + + Date + Datum + + + Project + Projekt + + + Approvals + Schválení + + + Status + Stav + + + + Gerrit::Internal::QueryContext + + Gerrit + Gerrit + + + Error running %1: %2 + Chyba při spouštění %1: %2 + + + %1 crashed. + %1 spadl. + + + %1 returned %2. + %1 vrátil %2. + + + + Gerrit::Internal::GerritOptionsPage + + Gerrit + Gerrit + + + + Gerrit::Internal::GerritOptionsWidget + + HTTPS + HTTPS + + + &Host: + &Hostitel: + + + &User: + &Uživatel: + + + &ssh: + &ssh: + + + &Port: + &Port: + + + &Additional queries: + Dodatečné &dotazy: + + + A comma-separated list of additional queries. +For example "status:staged,status:integrating" can be used to show the status of the Continuous Integration +of the Qt project. + Čárkou oddělený seznam dodatečných dotazů. +Například "status:staged,status:integrating" lze použít pro ukázání stavu pokračující integrace projektu Qt. + + + P&rotocol: + &Protokol: + + + Determines the protocol used to form a URL in case +"canonicalWebUrl" is not configured in the file +"gerrit.config". + Udává protokol použitý na vytvoření adresy (URL) v případě že +"canonicalWebUrl"není nastavena v souboru +"gerrit.config". + + + + Gerrit::Internal::FetchContext + + Gerrit Fetch + Gerrit Fetch (přivedení) + + + %1 crashed. + %1 spadl. + + + %1 returned %2. + %1 vrátil %2. + + + Error running %1: %2 + Chyba při spouštění %1: %2 + + + Error writing to temporary file. + Chyba při zápisu do dočasného souboru. + + + Writing %1... + Zapisuje se %1... + + + Cherry-picking %1... + Vybírá se %1... + + + + Gerrit::Internal::GerritPlugin + + Gerrit... + Gerrit... + + + git is not available. + Git je nedostupný. + + + Enter Local Repository for '%1' (%2) + Vstoupit do místního skladiště pro '%1' (%2) + + + + Git::Internal::GitLogArgumentsWidget + + Show Diff + Ukázat rozdíly + + + Show difference. + Ukázat rozdíly. + + + + Git::Internal::ResetDialog + + Sha1 + Sha1 + + + Subject + Předmět + + + Reset to: + Nastavit znovu na: + + + Undo Changes to %1 + Změny pro %1 vrátit zpět + + + + Locator::Internal::ExecuteFilter + + Previous command is still running ('%1'). +Do you want to kill it? + Předchozí příkaz stále běží ('%1'). +Chcete jej ukončit? + + + Kill Previous Process? + Ukončit předchozí proces? + + + finished + Dokončeno + + + failed + Nepodařilo se + + + Could not find executable for '%1' + Nepodařilo se najít spustitelný soubor pro '%1' + + + Starting command '%1' + Spouští se příkaz '%1' + + + Execute Custom Commands + Spustit vlastní příkazy + + + + MaddeDevice + + Test + Zkouška + + + Remote Processes... + Procesy na zařízení... + + + Deploy Public Key... + Poslat veřejný klíč... + + + Maemo5/Fremantle + Maemo 5/Fremantle + + + MeeGo 1.2 Harmattan + MeeGo 1.2 Harmattan + + + Other MeeGo OS + Jiné MeeGo OS + + + + ProjectExplorer::DesktopDevice + + Desktop + Stolní počítač + + + Run locally + Spustit místně + + + + ProjectExplorer::Internal::DesktopDeviceFactory + + Desktop + Stolní počítač + + + + ProjectExplorer::DeviceManagerModel + + %1 (default for %2) + %1 (výchozí pro %2) + + + + ProjectExplorer::Internal::DeviceSettingsPage + + Devices + Zařízení + + + + ProjectExplorer::IDevice + + Device + Title of the connectivity state information in a tool tip +---------- +Title of the connectivity state information in a tool tip + Zařízení + + + not connected + Device is not connected + Nepřipojeno + + + connected + Device is not connected + Připojeno + + + + ProjectExplorer::EnvironmentItemsDialog + + Edit Environment + Upravit prostředí + + + + ProjectExplorer::Internal::ProjectListWidget + + %1 (%2) + %1 (%2) + + + + ProjectExplorer::Internal::SessionModel + + New session name + Název nového sezení + + + + ProjectExplorer::DebuggerRunConfigurationAspect + + Debugger settings + Nastavení ladiče + + + + ProjectExplorer::Internal::WinCEToolChainFactory + + WinCE + WinCE + + + + ProjectExplorer::Internal::WinCEToolChainConfigWidget + + SDK: + SDK: + + + WinCE Version: + Verze WinCE: + + + ABI: + ABI: + + + + QmlJSEditor::AddAnalysisMessageSuppressionComment + + Add a comment to suppress this message + Přidat poznámku pro potlačení této zprávy + + + + QmlJSEditor::Internal::Operation + + Wrap Component in Loader + Zabalit součástku do nahrávače (loader) + + + // TODO: Move position bindings from the component to the Loader. +// Check all uses of 'parent' inside the root element of the component. + + + + + // Rename all outer uses of the id '%1' to '%2.item'. + + // Přejmenovat všechna vnější užití id '%1' na '%2.item'. + + + // Rename all outer uses of the id '%1' to '%2.item.%1'. + + // Přejmenovat všechna vnější užití id '%1' na '%2.item.%1'. + + + + + QmlJSInspector::Internal::QmlJSPropertyInspectorModel + + Name + Název + + + Value + Hodnota + + + Type + Typ + + + + QmlProfiler::Internal::QmlProfilerClientManager + + Qt Creator + Qt Creator + + + Could not connect to the in-process QML profiler. +Do you want to retry? + Qt Creatoru se nepodařilo vytvořit žádné spojení s profilerem QML v procesu. +Má se to zkusit ještě jednou? + + + + QmlProfiler::Internal::QmlProfilerDataModel + + Source code not available + Není dostupný žádný zdrojový kód + + + <bytecode> + <bytecode> + + + Animation Timer Update + Aktualizace časovače animace + + + <Animation Update> + <aktualizace animace> + + + <program> + <program> + + + Main Program + Hlavní program + + + %1 animations at %2 FPS + %1 animace při %2 FPS + + + No data to save + Nejsou přítomna žádná data k uložení + + + Could not open %1 for writing + Soubor '%1' se nepodařilo otevřít pro zápis + + + Could not open %1 for reading + Soubor '%1' se nepodařilo otevřít pro čtení + + + Error while parsing %1 + Chyba při zpracování %1 + + + Invalid version of QML Trace file. + Neplatná verze souboru QML Trace. + + + + QmlProfiler::Internal::QmlProfilerEventsWidget + + Trace information from the v8 JavaScript engine. Available only in Qt5 based applications + Sledovat informace ze stroje v8 JavaScript. Dostupné pouze v programech založených na Qt5 + + + Copy Row + Kopírovat řádek + + + Copy Table + Kopírovat tabulku + + + Extended Event Statistics + Rozšířená statistika událostí + + + Limit Events Pane to Current Range + Omezit tabulku na nynější rozsah + + + Reset Events Pane + Vynulovat tabulku událostí + + + + QmlProfiler::Internal::QmlProfilerEventsMainView + + Location + Umístění + + + Type + Typ + + + Time in Percent + Čas v % + + + Total Time + Celkový čas + + + Self Time in Percent + Vlastní čas v % + + + Self Time + Vlastní čas + + + Calls + Volání + + + Mean Time + Střední čas + + + Median Time + Medián času + + + Longest Time + Nejdelší čas + + + Shortest Time + Nejkratší čas + + + Details + Podrobnosti + + + (v4) + (v4) + + + Binding is evaluated by the optimized v4 engine. + Vázání je vyhodnoceno vyladěným strojem v4. + + + Binding loop detected + Zjištěna smyčka vázaní + + + µs + µs + + + ms + ms + + + s + s + + + Paint + Vykreslení + + + Compile + Překlad + + + Create + Vytvoření + + + Binding + Vázání + + + Signal + Signál + + + + QmlProfiler::Internal::QmlProfilerEventsParentsAndChildrenView + + Part of binding loop + Část smyčky vázaní + + + Callee + Volaná + + + Caller + Volající + + + Type + Typ + + + Total Time + Celkový čas + + + Calls + Volání + + + Callee Description + Popis volané + + + Caller Description + Popis volající + + + + QmlProfiler::Internal::QmlProfilerStateManager + + Switching to unknown state in %1:%2 + Přepnutí do neznámého stavu v %1:%2 + + + + QmlProfiler::Internal::QmlProfilerTraceView + + Jump to previous event + Jít na předchozí událost + + + Jump to next event + Jít na další událost + + + Show zoom slider + Ukázat zvětšovací posuvník + + + Select range + Vybrat rozsah + + + View event information on mouseover + Ukázat informace o události při přejezdu myši + + + Limit Events Pane to Current Range + Omezit tabulku na nynější rozsah + + + Reset Events Pane + Vynulovat tabulku událostí + + + Reset Zoom + Nastavit zvětšení znovu + + + + QmlProfiler::Internal::QmlProfilerViewManager + + Events + Události + + + Timeline + Časová přímka + + + JavaScript + JavaScript + + + + Qt4ProjectManager::SymbianIDevice + + Device + Zařízení + + + Not connected + Žádné spojení + + + %1, %2 + %1 device friendly name, %2 additional information + %1: %2 + + + IP address + Adresa IP + + + %1:%2 + %1:%2 + + + Symbian Device + Zařízení Symbian + + + + Qt4ProjectManager::Internal::SymbianIDeviceConfigurationWidget + + Device: + Zařízení: + + + Serial: + Sériová: + + + WLAN: + WLAN: + + + Queries the device for information + Vyvolává informace ze zařízení + + + Serial port: + Sériová přípojka: + + + Communication Channel + Komunikační protokol + + + Address: + Adresa: + + + Connecting + Připojuje se + + + Unable to create CODA connection. Please try again. + Nepodařilo se vytvořit žádné spojení s CODA. Zkuste to, prosím, znovu. + + + Currently there is no information about the device for this connection type. + Pro tento typ spojení nejsou nyní dostupné žádné informace o zařízení. + + + No device information available + Nejsou dostupné žádné informace o zařízení + + + Qt version: + Verze Qt: + + + Not installed on device + Neinstalováno na zařízení + + + Qt version: + Verze Qt: + + + Unrecognised Symbian version 0x%1 + Nerozpoznaná verze Symbianu 0x%1 + + + Unrecognised S60 version 0x%1 + Nerozpoznaná verze S60 0x%1 + + + OS version: + Verze operačního systému: + + + unknown + Neznámý + + + ROM version: + Verze ROM: + + + Release: + Vydání: + + + CODA version: + Verze CODA: + + + Error reading CODA version + Chyba při čtení verze CODA + + + Qt Mobility version: + Verze Qt Mobility: + + + Error reading Qt Mobility version + Chyba při čtení verze Qt Mobility + + + Qt Quick components version: + Verze součásti Qt Quick: + + + Not installed + Neinstalováno + + + QML Viewer version: + Verze prohlížeče QML: + + + Screen size: + Velikost obrazovky: + + + + Qt4ProjectManager::Internal::SymbianIDeviceFactory + + Symbian Device + Zařízení Symbian + + + + Qt4ProjectManager::Internal::UnconfiguredProjectPanel + + Configure Project + Nastavit projekt + + + + Qt4ProjectManager::Internal::TargetSetupPageWrapper + + Configure Project + Nastavit projekt + + + <p>The project <b>%1</b> is not yet configured.</p><p>Qt Creator uses the Qt version: <b>%2</b> and the tool chain: <b>%3</b> to parse the project. You can edit these in the <b><a href="edit">options.</a></b></p> + <p>Projekt <b>%1</b> ještě není nastaven.</p><p>Qt Creator používá Qt ve verzi: <b>%2</b> a sadu nástrojů: <b>%3</b> pro zpracování projektu. Můžete je upravit v dialogu <b><a href="edit">Volby.</a></b></p> + + + <p>The project <b>%1</b> is not yet configured.</p><p>Qt Creator uses the Qt version: <b>%2</b> and <b>no tool chain</b> to parse the project. You can edit these in the <b><a href="edit">settings</a></b></p> + <p>Projekt <b>%1</b> ještě není nastaven.</p><p>Qt Creator používá Qt ve verzi: <b>%2</b> a žádnou sadu nástrojů</b> pro zpracování projektu. Můžete je upravit v dialogu <b><a href="edit">Nastavení</a></b></p> + + + <p>The project <b>%1</b> is not yet configured.</p><p>Qt Creator uses <b>no Qt version</b> and the tool chain: <b>%2</b> to parse the project. You can edit these in the <b><a href="edit">settings</a></b></p> + <p>Projekt <b>%1</b> ještě není nastaven.</p><p>Qt Creator nepoužívá <b>žádnou verzi Qt</b> a sadu nástrojů: <b>%2</b> pro zpracování projektu. Můžete je upravit v dialogu <b><a href="edit">Nastavení</a></b></p> + + + <p>The project <b>%1</b> is not yet configured.</p><p>Qt Creator uses <b>no Qt version</b> and <b>no tool chain</b> to parse the project. You can edit these in the <b><a href="edit">settings</a></b></p> + <p>Projekt <b>%1</b> ještě není nastaven.</p><p>Qt Creator nepoužívá <b>žádnou verzi Qt</b> a žádnou sadu nástrojů</b> pro zpracování projektu. Můžete je upravit v dialogu <b><a href="edit">Nastavení</a></b></p> + + + + Qt4ProjectManager::Internal::UnConfiguredSettingsWidget + + Qt Creator can open qmake projects without configuring them for building. +The C++ and QML code models need a Qt version and tool chain to offer code completion. + + Qt Creator může otevřít projekty qmake bez jejich nastavení pro sestavení. +Je zapotřebí mít nějakou verzi Qt a sadu nástrojů, aby modely kódu C++ a QML nabídly doplnění kódu. + + + + Qt Version: + Verze Qt: + + + Tool Chain: + Sada nástrojů: + + + + QtSupport::Internal::ExamplesWelcomePage + + Examples + Příklady + + + Tutorials + Návody + + + Copy Project to writable Location? + Má se projekt zkopírovat do zapisovatelného umístění? + + + <p>The project you are about to open is located in the write-protected location:</p><blockquote>%1</blockquote><p>Please select a writable location below and click "Copy Project and Open" to open a modifiable copy of the project or click "Keep Project and Open" to open the project in location.</p><p><b>Note:</b> You will not be able to alter or compile your project in the current location.</p> + <p>Projekt, který se chystáte otevřít, je umístěn v proti zápisu chráněném umístění:</p><blockquote>%1</blockquote><p>Vyberte, prosím, níže zapisovatelné umístění a klepněte na "Kopírovat projekt a otevřít" kvůli otevření upravovatelné kopie projektu, nebo klepněte na "Zachovat projekt a otevřít" kvůli otevření projektu v umístění.</p><p><b>Poznámka:</b> V současném umístění projekt nelze ani sestavit ani změnit.</p> + + + &Location: + &Umístění: + + + &Copy Project and Open + &Kopírovat projekt a otevřít + + + &Keep Project and Open + &Zachovat projekt a otevřít + + + Cannot Use Location + Umístění nelze použít + + + The specified location already exists. Please specify a valid location. + Zadané umístění již existuje. Zadejte, prosím, platné umístění. + + + Cannot Copy Project + Chyba při kopírování projektu + + + Failed to open project + Nepodařilo se otevřít projekt + + + + RemoteLinux::Internal::EmbeddedLinuxTargetFactory + + embedded + Vloženo + + + Embedded Linux + Vložený Linux + + + + RemoteLinux::Internal::GenericEmbeddedLinuxTarget + + Embedded Linux + Vložený Linux + + + + LinuxDeviceConfiguration + + Generic Linux + Obecný Linux + + + Test + Zkouška + + + Remote Processes... + Procesy na zařízení... + + + Deploy Public Key... + Poslat veřejný klíč... + + + + RemoteLinux::RemoteLinuxDeployConfigurationWidget + + Double-click to edit the project file + Dvojité klepnutí pro úpravu souboru s projektem + + + + TextEditor::Internal::CountingLabel + + %1 found + %1 nalezen + + + + TextEditor::Internal::FindInOpenFiles + + Open Documents + Otevřené dokumenty + + + Open Documents: + Otevřené dokumenty: + + + Open Documents +%1 + Otevřené dokumenty +%1 + + + + Todo::Internal::TodoItemsModel + + Description + Popis + + + File + Soubor + + + Line + Řádek + + + + Todo::Internal::TodoOutputPane + + To-Do Entries + Záznamy CO UDĚLAT + + + Scan in the current opened file + Hledat v nyní otevřeném souboru + + + Scan in the whole project + Hledat v celém projektu + + + + Todo::Internal::OptionsPage + + To-Do + CO UDĚLAT + + + + VcsBase::Internal::UrlTextCursorHandler + + Open URL in browser... + Otevřít adresu (URL) v prohlížeči... + + + Copy URL location + Kopírovat umístění adresy (URL) + + + + VcsBase::Internal::EmailTextCursorHandler + + Send email to... + Poslat e-mail... + + + Copy email address + Kopírovat adresu e-mailu + + + + Todo::Internal::AddKeywordDialog + + Keyword + Klíčové slovo + + + Icon + Ikona + + + Color + Barva + + + + QmlJsDebugClient::QDeclarativeOutputParser + + The port seems to be in use. + Error message shown after 'Could not connect ... debugger:" + Port se už používá. + + + The application is not set up for QML/JS debugging. + Error message shown after 'Could not connect ... debugger:" + Tento program není nastaven pro ladění QML/JS. + + diff -Nru qtcreator-2.5.0/share/qtcreator/translations/qtcreator_fr.ts qtcreator-2.5.2/share/qtcreator/translations/qtcreator_fr.ts --- qtcreator-2.5.0/share/qtcreator/translations/qtcreator_fr.ts 2012-05-09 12:13:19.000000000 +0000 +++ qtcreator-2.5.2/share/qtcreator/translations/qtcreator_fr.ts 2012-08-08 13:47:06.000000000 +0000 @@ -32710,17 +32710,20 @@ Debugging starts - Début du débogage + Début du débogage + Debugging has failed - Échec du débogage + Échec du débogage + Debugging has finished - Fin du débogage + Fin du débogage + A debugging session is still in progress. Terminating the session in the current state can leave the target in an inconsistent state. Would you still like to terminate it? diff -Nru qtcreator-2.5.0/share/qtcreator/translations/qtcreator_ru.ts qtcreator-2.5.2/share/qtcreator/translations/qtcreator_ru.ts --- qtcreator-2.5.0/share/qtcreator/translations/qtcreator_ru.ts 2012-05-09 12:13:19.000000000 +0000 +++ qtcreator-2.5.2/share/qtcreator/translations/qtcreator_ru.ts 2012-08-08 13:47:06.000000000 +0000 @@ -12796,7 +12796,7 @@ Locator Locator - Поисковик + Быстрый поиск @@ -12862,8 +12862,8 @@ Specify a short word/abbreviation that can be used to restrict completions to files from this directory tree. To do this, you type this shortcut and a space in the Locator entry field, and then the word to search for. - Укажите сокращение или аббревиатуру, которая будет использоваться для ограничения дополнения до файлов из -данного дерева каталогов. Для этого требуется ввести указанное сокращение, пробел и искомое слово в поле поисковика. + Укажите сокращение или аббревиатуру, которая будет использоваться для выполнения дополнения имён файлов из +данного дерева каталогов. Для этого требуется ввести указанное сокращение, пробел и искомое слово в поле быстрого поиска. Limit to prefix diff -Nru qtcreator-2.5.0/src/app/main.cpp qtcreator-2.5.2/src/app/main.cpp --- qtcreator-2.5.0/src/app/main.cpp 2012-05-09 12:13:19.000000000 +0000 +++ qtcreator-2.5.2/src/app/main.cpp 2012-08-08 13:47:06.000000000 +0000 @@ -283,7 +283,8 @@ uiLanguages.prepend(overrideLanguage); const QString &creatorTrPath = QCoreApplication::applicationDirPath() + QLatin1String(SHARE_PATH "/translations"); - foreach (const QString &locale, uiLanguages) { + foreach (QString locale, uiLanguages) { + locale.replace(QLatin1Char('-'), QLatin1Char('_')); // work around QTBUG-25973 if (translator.load(QLatin1String("qtcreator_") + locale, creatorTrPath)) { const QString &qtTrPath = QLibraryInfo::location(QLibraryInfo::TranslationsPath); const QString &qtTrFile = QLatin1String("qt_") + locale; diff -Nru qtcreator-2.5.0/src/libs/3rdparty/cplusplus/Bind.cpp qtcreator-2.5.2/src/libs/3rdparty/cplusplus/Bind.cpp --- qtcreator-2.5.0/src/libs/3rdparty/cplusplus/Bind.cpp 2012-05-09 12:13:19.000000000 +0000 +++ qtcreator-2.5.2/src/libs/3rdparty/cplusplus/Bind.cpp 2012-08-08 13:47:06.000000000 +0000 @@ -1851,6 +1851,8 @@ name = elabTypeSpec->name->name; } + ensureValidClassName(&name, sourceLocation); + ForwardClassDeclaration *decl = control()->newForwardClassDeclaration(sourceLocation, name); setDeclSpecifiers(decl, type); _scope->addMember(decl); @@ -2882,20 +2884,7 @@ } } - // get the unqualified class name - const QualifiedNameId *q = className->asQualifiedNameId(); - const Name *unqualifiedClassName = q ? q->name() : className; - - if (! unqualifiedClassName) // paranoia check - className = 0; - else if (! (unqualifiedClassName->isNameId() || unqualifiedClassName->isTemplateNameId())) { - translationUnit()->error(sourceLocation, "expected a class-name"); - - className = unqualifiedClassName->identifier(); - - if (q && className) - className = control()->qualifiedNameId(q->base(), className); - } + ensureValidClassName(&className, sourceLocation); } Class *klass = control()->newClass(sourceLocation, className); @@ -3124,6 +3113,24 @@ return false; } +void Bind::ensureValidClassName(const Name **name, unsigned sourceLocation) +{ + if (!*name) + return; + + const QualifiedNameId *qName = (*name)->asQualifiedNameId(); + if (qName) + *name = qName->name(); + + if (!(*name)->isNameId() && !(*name)->isTemplateNameId()) { + translationUnit()->error(sourceLocation, "expected a class-name"); + + *name = (*name)->identifier(); + if (qName) + *name = control()->qualifiedNameId(qName->base(), *name); + } +} + int Bind::visibilityForAccessSpecifier(int tokenKind) { switch (tokenKind) { diff -Nru qtcreator-2.5.0/src/libs/3rdparty/cplusplus/Bind.h qtcreator-2.5.2/src/libs/3rdparty/cplusplus/Bind.h --- qtcreator-2.5.0/src/libs/3rdparty/cplusplus/Bind.h 2012-05-09 12:13:19.000000000 +0000 +++ qtcreator-2.5.2/src/libs/3rdparty/cplusplus/Bind.h 2012-08-08 13:47:06.000000000 +0000 @@ -274,6 +274,8 @@ private: static const int kMaxDepth; + void ensureValidClassName(const Name **name, unsigned sourceLocation); + Scope *_scope; ExpressionTy _expression; const Name *_name; diff -Nru qtcreator-2.5.0/src/libs/cplusplus/CppDocument.h qtcreator-2.5.2/src/libs/cplusplus/CppDocument.h --- qtcreator-2.5.0/src/libs/cplusplus/CppDocument.h 2012-05-09 12:13:19.000000000 +0000 +++ qtcreator-2.5.2/src/libs/cplusplus/CppDocument.h 2012-08-08 13:47:06.000000000 +0000 @@ -42,6 +42,11 @@ #include #include +// in debug mode: make dumpers widely available without an extra include +#ifdef QT_DEBUG +#include "Dumpers.h" +#endif + namespace CPlusPlus { class Macro; diff -Nru qtcreator-2.5.0/src/libs/cplusplus/Dumpers.cpp qtcreator-2.5.2/src/libs/cplusplus/Dumpers.cpp --- qtcreator-2.5.0/src/libs/cplusplus/Dumpers.cpp 1970-01-01 00:00:00.000000000 +0000 +++ qtcreator-2.5.2/src/libs/cplusplus/Dumpers.cpp 2012-08-08 13:47:06.000000000 +0000 @@ -0,0 +1,140 @@ +/************************************************************************** +** +** This file is part of Qt Creator +** +** Copyright (c) 2012 Nokia Corporation and/or its subsidiary(-ies). +** +** Contact: Nokia Corporation (qt-info@nokia.com) +** +** +** GNU Lesser General Public License Usage +** +** This file may be used under the terms of the GNU Lesser General Public +** License version 2.1 as published by the Free Software Foundation and +** appearing in the file LICENSE.LGPL included in the packaging of this file. +** Please review the following information to ensure the GNU Lesser General +** Public License version 2.1 requirements will be met: +** http://www.gnu.org/licenses/old-licenses/lgpl-2.1.html. +** +** In addition, as a special exception, Nokia gives you certain additional +** rights. These rights are described in the Nokia Qt LGPL Exception +** version 1.1, included in the file LGPL_EXCEPTION.txt in this package. +** +** Other Usage +** +** Alternatively, this file may be used in accordance with the terms and +** conditions contained in a signed written agreement between you and Nokia. +** +** If you have questions regarding the use of this file, please contact +** Nokia at qt-info@nokia.com. +** +**************************************************************************/ + +#include "Dumpers.h" + +#include +#include +#include +#include +#include + +#include + +static QString indent(QString s, int level = 2) +{ + QString indentString(level, QLatin1Char(' ')); + QString result; + int last = 0; + for (int i = 0; i < s.length(); ++i) { + if (s.at(i) == QLatin1Char('\n') || i == s.length() - 1) { + result.append(indentString); + result.append(s.midRef(last, i + 1)); + last = i + 1; + } + } + return result; +} + +QString CPlusPlus::toString(const Name *name, QString id) +{ + Overview oo; + return QString("%0: %1").arg(id, name ? oo(name) : QLatin1String("(null)")); +} + +QString CPlusPlus::toString(FullySpecifiedType ty, QString id) +{ + Overview oo; + return QString("%0: %1 (a %2)").arg(id, oo(ty), ty.type() ? typeid(*ty.type()).name() : "(null)"); +} + +QString CPlusPlus::toString(const Symbol *s, QString id) +{ + if (!s) + return QString("%0: (null)").arg(id); + + return QString("%0: %1 (%2) at %3:%4:%5\n%6").arg( + id, + QString::fromLatin1(typeid(*s).name()), + QString::fromUtf8(s->identifier()->chars()), + QString::fromLatin1(s->fileName()), + QString::number(s->line()), + QString::number(s->column()), + indent(toString(s->type()))); +} + +QString CPlusPlus::toString(LookupItem it, QString id) +{ + QString result = QString("%1:").arg(id); + if (it.declaration()) { + result.append(QString("\n%1").arg(indent(toString(it.declaration(), QLatin1String("Decl"))))); + } + if (it.type().isValid()) { + result.append(QString("\n%1").arg(indent(toString(it.type())))); + } + if (it.scope()) { + result.append(QString("\n%1").arg(indent(toString(it.scope(), QLatin1String("Scope"))))); + } + if (it.binding()) { + result.append(QString("\n%1").arg(indent(toString(it.binding(), QLatin1String("Binding"))))); + } + return result; +} + +QString CPlusPlus::toString(const ClassOrNamespace *binding, QString id) +{ + if (!binding) + return QString("%0: (null)").arg(id); + + QString result = QString("%0: %1 symbols").arg( + id, + QString::number(binding->symbols().length())); + if (binding->templateId()) { + result.append(QString("\n%1").arg(indent(toString(binding->templateId(), QLatin1String("Template"))))); + } + return result; +} + +void CPlusPlus::dump(const Name *name) +{ + qDebug() << qPrintable(toString(name)); +} + +void CPlusPlus::dump(FullySpecifiedType ty) +{ + qDebug() << qPrintable(toString(ty)); +} + +void CPlusPlus::dump(const Symbol *s) +{ + qDebug() << qPrintable(toString(s)); +} + +void CPlusPlus::dump(LookupItem it) +{ + qDebug() << qPrintable(toString(it)); +} + +void CPlusPlus::dump(const ClassOrNamespace *binding) +{ + qDebug() << qPrintable(toString(binding)); +} diff -Nru qtcreator-2.5.0/src/libs/cplusplus/Dumpers.h qtcreator-2.5.2/src/libs/cplusplus/Dumpers.h --- qtcreator-2.5.0/src/libs/cplusplus/Dumpers.h 1970-01-01 00:00:00.000000000 +0000 +++ qtcreator-2.5.2/src/libs/cplusplus/Dumpers.h 2012-08-08 13:47:06.000000000 +0000 @@ -0,0 +1,59 @@ +/************************************************************************** +** +** This file is part of Qt Creator +** +** Copyright (c) 2012 Nokia Corporation and/or its subsidiary(-ies). +** +** Contact: Nokia Corporation (qt-info@nokia.com) +** +** +** GNU Lesser General Public License Usage +** +** This file may be used under the terms of the GNU Lesser General Public +** License version 2.1 as published by the Free Software Foundation and +** appearing in the file LICENSE.LGPL included in the packaging of this file. +** Please review the following information to ensure the GNU Lesser General +** Public License version 2.1 requirements will be met: +** http://www.gnu.org/licenses/old-licenses/lgpl-2.1.html. +** +** In addition, as a special exception, Nokia gives you certain additional +** rights. These rights are described in the Nokia Qt LGPL Exception +** version 1.1, included in the file LGPL_EXCEPTION.txt in this package. +** +** Other Usage +** +** Alternatively, this file may be used in accordance with the terms and +** conditions contained in a signed written agreement between you and Nokia. +** +** If you have questions regarding the use of this file, please contact +** Nokia at qt-info@nokia.com. +** +**************************************************************************/ + +#ifndef CPLUSPLUS_DUMPERS_H +#define CPLUSPLUS_DUMPERS_H + +#include +#include +#include +#include + +#include + +namespace CPlusPlus { + +QString CPLUSPLUS_EXPORT toString(const Name *name, QString id = QLatin1String("Name")); +QString CPLUSPLUS_EXPORT toString(FullySpecifiedType ty, QString id = QLatin1String("Type")); +QString CPLUSPLUS_EXPORT toString(const Symbol *s, QString id = QLatin1String("Symbol")); +QString CPLUSPLUS_EXPORT toString(LookupItem it, QString id = QLatin1String("LookupItem")); +QString CPLUSPLUS_EXPORT toString(const ClassOrNamespace *binding, QString id = QLatin1String("ClassOrNamespace")); + +void CPLUSPLUS_EXPORT dump(const Name *name); +void CPLUSPLUS_EXPORT dump(FullySpecifiedType ty); +void CPLUSPLUS_EXPORT dump(const Symbol *s); +void CPLUSPLUS_EXPORT dump(LookupItem it); +void CPLUSPLUS_EXPORT dump(const ClassOrNamespace *binding); + +} // namespace CPlusPlus + +#endif diff -Nru qtcreator-2.5.0/src/libs/cplusplus/cplusplus-lib.pri qtcreator-2.5.2/src/libs/cplusplus/cplusplus-lib.pri --- qtcreator-2.5.0/src/libs/cplusplus/cplusplus-lib.pri 2012-05-09 12:13:19.000000000 +0000 +++ qtcreator-2.5.2/src/libs/cplusplus/cplusplus-lib.pri 2012-08-08 13:47:06.000000000 +0000 @@ -55,7 +55,8 @@ $$PWD/pp-scanner.h \ $$PWD/ModelManagerInterface.h \ $$PWD/findcdbbreakpoint.h \ - $$PWD/TypeHierarchyBuilder.h + $$PWD/TypeHierarchyBuilder.h \ + $$PWD/Dumpers.h SOURCES += \ $$PWD/SimpleLexer.cpp \ @@ -84,6 +85,7 @@ $$PWD/pp-scanner.cpp \ $$PWD/ModelManagerInterface.cpp \ $$PWD/findcdbbreakpoint.cpp \ - $$PWD/TypeHierarchyBuilder.cpp + $$PWD/TypeHierarchyBuilder.cpp \ + $$PWD/Dumpers.cpp RESOURCES += $$PWD/cplusplus.qrc diff -Nru qtcreator-2.5.0/src/libs/qmleditorwidgets/contextpanewidget.cpp qtcreator-2.5.2/src/libs/qmleditorwidgets/contextpanewidget.cpp --- qtcreator-2.5.0/src/libs/qmleditorwidgets/contextpanewidget.cpp 2012-05-09 12:13:19.000000000 +0000 +++ qtcreator-2.5.2/src/libs/qmleditorwidgets/contextpanewidget.cpp 2012-08-08 13:47:06.000000000 +0000 @@ -102,10 +102,14 @@ m_startPos = QPoint(-1, -1); m_pos = QPoint(-1, -1); + // TODO: The following code should be enabled for OSX + // when QTBUG-23205 is fixed +#ifndef Q_OS_MAC m_dropShadowEffect = new QGraphicsDropShadowEffect; m_dropShadowEffect->setBlurRadius(6); m_dropShadowEffect->setOffset(2, 2); setGraphicsEffect(m_dropShadowEffect); +#endif } void DragWidget::mousePressEvent(QMouseEvent * event) @@ -123,10 +127,14 @@ { if (event->button() == Qt::LeftButton) { m_startPos = QPoint(-1, -1); + // TODO: The following code should be enabled for OSX + // when QTBUG-23205 is fixed +#ifndef Q_OS_MAC m_dropShadowEffect = new QGraphicsDropShadowEffect; m_dropShadowEffect->setBlurRadius(6); m_dropShadowEffect->setOffset(2, 2); setGraphicsEffect(m_dropShadowEffect); +#endif } QFrame::mouseReleaseEvent(event); } diff -Nru qtcreator-2.5.0/src/libs/qmleditorwidgets/contextpanewidgetimage.cpp qtcreator-2.5.2/src/libs/qmleditorwidgets/contextpanewidgetimage.cpp --- qtcreator-2.5.0/src/libs/qmleditorwidgets/contextpanewidgetimage.cpp 2012-05-09 12:13:19.000000000 +0000 +++ qtcreator-2.5.2/src/libs/qmleditorwidgets/contextpanewidgetimage.cpp 2012-08-08 13:47:06.000000000 +0000 @@ -611,10 +611,14 @@ m_hooverInfo->setFrameShape(QFrame::StyledPanel); m_hooverInfo->setFrameShadow(QFrame::Sunken); + // TODO: The following code should be enabled for OSX + // when QTBUG-23205 is fixed +#ifndef Q_OS_MAC QGraphicsDropShadowEffect *dropShadowEffect = new QGraphicsDropShadowEffect; dropShadowEffect->setBlurRadius(4); dropShadowEffect->setOffset(2, 2); m_hooverInfo->setGraphicsEffect(dropShadowEffect); +#endif m_hooverInfo->setAutoFillBackground(true); m_hooverInfo->raise(); } diff -Nru qtcreator-2.5.0/src/libs/qmleditorwidgets/customcolordialog.cpp qtcreator-2.5.2/src/libs/qmleditorwidgets/customcolordialog.cpp --- qtcreator-2.5.0/src/libs/qmleditorwidgets/customcolordialog.cpp 2012-05-09 12:13:19.000000000 +0000 +++ qtcreator-2.5.2/src/libs/qmleditorwidgets/customcolordialog.cpp 2012-08-08 13:47:06.000000000 +0000 @@ -52,10 +52,14 @@ setFrameShape(QFrame::StyledPanel); setFrameShadow(QFrame::Sunken); + // TODO: The following code should be enabled for OSX + // when QTBUG-23205 is fixed +#ifndef Q_OS_MAC QGraphicsDropShadowEffect *dropShadowEffect = new QGraphicsDropShadowEffect; dropShadowEffect->setBlurRadius(6); dropShadowEffect->setOffset(2, 2); setGraphicsEffect(dropShadowEffect); +#endif setAutoFillBackground(true); m_hueControl = new HueControl(this); diff -Nru qtcreator-2.5.0/src/libs/utils/process_ctrlc_stub.cpp qtcreator-2.5.2/src/libs/utils/process_ctrlc_stub.cpp --- qtcreator-2.5.0/src/libs/utils/process_ctrlc_stub.cpp 2012-05-09 12:13:19.000000000 +0000 +++ qtcreator-2.5.2/src/libs/utils/process_ctrlc_stub.cpp 2012-08-08 13:47:06.000000000 +0000 @@ -86,7 +86,7 @@ // Get the command line and remove the call to this executable. wchar_t *strCommandLine = _wcsdup(GetCommandLine()); const size_t strCommandLineLength = wcslen(strCommandLine); - size_t pos = 1; + size_t pos = 0; bool quoted = false; while (pos < strCommandLineLength) { if (strCommandLine[pos] == L'"') { diff -Nru qtcreator-2.5.0/src/plugins/cmakeprojectmanager/makestep.cpp qtcreator-2.5.2/src/plugins/cmakeprojectmanager/makestep.cpp --- qtcreator-2.5.0/src/plugins/cmakeprojectmanager/makestep.cpp 2012-05-09 12:13:19.000000000 +0000 +++ qtcreator-2.5.2/src/plugins/cmakeprojectmanager/makestep.cpp 2012-08-08 13:47:06.000000000 +0000 @@ -132,6 +132,8 @@ bool MakeStep::init() { CMakeBuildConfiguration *bc = cmakeBuildConfiguration(); + if (!bc) + bc = static_cast(target()->activeBuildConfiguration()); QString arguments = Utils::QtcProcess::joinArgs(m_buildTargets); Utils::QtcProcess::addArgs(&arguments, additionalArguments()); diff -Nru qtcreator-2.5.0/src/plugins/coreplugin/coreplugin.pro qtcreator-2.5.2/src/plugins/coreplugin/coreplugin.pro --- qtcreator-2.5.0/src/plugins/coreplugin/coreplugin.pro 2012-05-09 12:13:19.000000000 +0000 +++ qtcreator-2.5.2/src/plugins/coreplugin/coreplugin.pro 2012-08-08 13:47:06.000000000 +0000 @@ -210,7 +210,7 @@ win32 { SOURCES += progressmanager/progressmanager_win.cpp - LIBS += -lole32 + LIBS += -lole32 -luser32 } else:macx { OBJECTIVE_SOURCES += progressmanager/progressmanager_mac.mm diff -Nru qtcreator-2.5.0/src/plugins/coreplugin/coreplugin.qbs qtcreator-2.5.2/src/plugins/coreplugin/coreplugin.qbs --- qtcreator-2.5.0/src/plugins/coreplugin/coreplugin.qbs 2012-05-09 12:13:19.000000000 +0000 +++ qtcreator-2.5.2/src/plugins/coreplugin/coreplugin.qbs 2012-08-08 13:47:06.000000000 +0000 @@ -24,8 +24,9 @@ ] cpp.dynamicLibraries: { - if (qbs.targetOS == 'windows') return [ - "ole32" + if (qbs.targetOS == "windows") return [ + "ole32", + "user32" ] } diff -Nru qtcreator-2.5.0/src/plugins/coreplugin/mainwindow.cpp qtcreator-2.5.2/src/plugins/coreplugin/mainwindow.cpp --- qtcreator-2.5.0/src/plugins/coreplugin/mainwindow.cpp 2012-05-09 12:13:19.000000000 +0000 +++ qtcreator-2.5.2/src/plugins/coreplugin/mainwindow.cpp 2012-08-08 13:47:06.000000000 +0000 @@ -229,9 +229,7 @@ setAcceptDrops(true); m_autoSaveSessionTimer = new QTimer(this); - m_autoSaveSessionTimer->setSingleShot(true); m_autoSaveSessionTimer->setInterval(10000); - m_autoSaveSessionTimer->start(); connect(m_autoSaveSessionTimer, SIGNAL(timeout()), m_coreImpl, SIGNAL(saveSettingsRequested())); } @@ -375,6 +373,7 @@ emit m_coreImpl->coreAboutToOpen(); show(); emit m_coreImpl->coreOpened(); + m_autoSaveSessionTimer->start(); } void MainWindow::closeEvent(QCloseEvent *event) diff -Nru qtcreator-2.5.0/src/plugins/coreplugin/progressmanager/progressmanager_win.cpp qtcreator-2.5.2/src/plugins/coreplugin/progressmanager/progressmanager_win.cpp --- qtcreator-2.5.0/src/plugins/coreplugin/progressmanager/progressmanager_win.cpp 2012-05-09 12:13:19.000000000 +0000 +++ qtcreator-2.5.2/src/plugins/coreplugin/progressmanager/progressmanager_win.cpp 2012-08-08 13:47:06.000000000 +0000 @@ -132,7 +132,9 @@ // See pixmaputils.cpp in the Windows plugin. Q_UNIMPLEMENTED(); #else - pITask->SetOverlayIcon(winId, pix.toWinHICON(), (wchar_t*)text.utf16()); + const HICON icon = pix.toWinHICON(); + pITask->SetOverlayIcon(winId, icon, (wchar_t*)text.utf16()); + DestroyIcon(icon); #endif } } diff -Nru qtcreator-2.5.0/src/plugins/cpptools/cppcompletion_test.cpp qtcreator-2.5.2/src/plugins/cpptools/cppcompletion_test.cpp --- qtcreator-2.5.0/src/plugins/cpptools/cppcompletion_test.cpp 1970-01-01 00:00:00.000000000 +0000 +++ qtcreator-2.5.2/src/plugins/cpptools/cppcompletion_test.cpp 2012-08-08 13:47:06.000000000 +0000 @@ -0,0 +1,198 @@ +/************************************************************************** +** +** This file is part of Qt Creator +** +** Copyright (c) 2012 Nokia Corporation and/or its subsidiary(-ies). +** +** Contact: Nokia Corporation (qt-info@nokia.com) +** +** +** GNU Lesser General Public License Usage +** +** This file may be used under the terms of the GNU Lesser General Public +** License version 2.1 as published by the Free Software Foundation and +** appearing in the file LICENSE.LGPL included in the packaging of this file. +** Please review the following information to ensure the GNU Lesser General +** Public License version 2.1 requirements will be met: +** http://www.gnu.org/licenses/old-licenses/lgpl-2.1.html. +** +** In addition, as a special exception, Nokia gives you certain additional +** rights. These rights are described in the Nokia Qt LGPL Exception +** version 1.1, included in the file LGPL_EXCEPTION.txt in this package. +** +** Other Usage +** +** Alternatively, this file may be used in accordance with the terms and +** conditions contained in a signed written agreement between you and Nokia. +** +** If you have questions regarding the use of this file, please contact +** Nokia at qt-info@nokia.com. +** +**************************************************************************/ + +#include "cpptoolsplugin.h" + +#include +#include +#include +#include +#include +#include +#include +#include +#include +#include +#include +#include +#include +#include +#include +#include +#include +#include +#include + +#include +#include +#include +#include + +/*! + Tests for code completion. + */ +using namespace CPlusPlus; +using namespace CppTools; +using namespace CppTools::Internal; +using namespace TextEditor; +using namespace Core; + +struct TestData +{ + QByteArray srcText; + int pos; + Snapshot snapshot; + BaseTextEditorWidget *editor; + QTextDocument *doc; +}; + +static QStringList getCompletions(TestData &data) +{ + QStringList completions; + + CppCompletionAssistInterface *ai = new CppCompletionAssistInterface(data.editor->document(), data.pos, + data.editor->editorDocument(), ExplicitlyInvoked, + data.snapshot, QStringList(), QStringList()); + CppCompletionAssistProcessor processor; + IAssistProposal *proposal = processor.perform(ai); + if (!proposal) + return completions; + IAssistProposalModel *model = proposal->model(); + if (!model) + return completions; + BasicProposalItemListModel *listmodel = dynamic_cast(model); + if (!listmodel) + return completions; + + for (int i = 0; i < listmodel->size(); ++i) + completions << listmodel->text(i); + + return completions; +} + +static void setup(TestData *data) +{ + data->pos = data->srcText.indexOf('@'); + QVERIFY(data->pos != -1); + data->srcText[data->pos] = ' '; + Document::Ptr src = Document::create(QDir::tempPath() + QLatin1String("/file.h")); + Utils::FileSaver srcSaver(src->fileName()); + srcSaver.write(data->srcText); + srcSaver.finalize(); + src->setUtf8Source(data->srcText); + src->parse(); + src->check(); + + data->snapshot.insert(src); + + data->editor = new PlainTextEditorWidget(0); + QString error; + data->editor->open(&error, src->fileName(), src->fileName()); + + data->doc = data->editor->document(); +} + +void CppToolsPlugin::test_completion_basic_1() +{ + TestData data; + data.srcText = "\n" + "class Foo\n" + "{\n" + " void foo();\n" + " int m;\n" + "};\n" + "\n" + "void func() {\n" + " Foo f;\n" + " @\n" + " // padding so we get the scope right\n" + "}"; + + setup(&data); + + QStringList basicCompletions = getCompletions(data); + + QVERIFY(!basicCompletions.contains("foo")); + QVERIFY(!basicCompletions.contains("m")); + QVERIFY(basicCompletions.contains("Foo")); + QVERIFY(basicCompletions.contains("func")); + QVERIFY(basicCompletions.contains("f")); + + Utils::ChangeSet change; + change.insert(data.pos, "f."); + QTextCursor cursor(data.doc); + change.apply(&cursor); + data.pos += 2; + + QStringList memberCompletions = getCompletions(data); + + QVERIFY(memberCompletions.contains("foo")); + QVERIFY(memberCompletions.contains("m")); + QVERIFY(!memberCompletions.contains("func")); + QVERIFY(!memberCompletions.contains("f")); +} + +void CppToolsPlugin::test_completion_template_1() +{ + TestData data; + data.srcText = "\n" + "template \n" + "class Foo\n" + "{\n" + " typedef T Type;\n" + " T foo();\n" + " T m;\n" + "};\n" + "\n" + "void func() {\n" + " Foo f;\n" + " @\n" + " // padding so we get the scope right\n" + "}"; + + setup(&data); + + Utils::ChangeSet change; + change.insert(data.pos, "Foo::"); + QTextCursor cursor(data.doc); + change.apply(&cursor); + data.pos += 5; + + QStringList completions = getCompletions(data); + + QVERIFY(completions.contains("Type")); + QVERIFY(completions.contains("foo")); + QVERIFY(completions.contains("m")); + QVERIFY(!completions.contains("T")); + QVERIFY(!completions.contains("f")); + QVERIFY(!completions.contains("func")); +} diff -Nru qtcreator-2.5.0/src/plugins/cpptools/cppcompletionassist.cpp qtcreator-2.5.2/src/plugins/cpptools/cppcompletionassist.cpp --- qtcreator-2.5.0/src/plugins/cpptools/cppcompletionassist.cpp 2012-05-09 12:13:19.000000000 +0000 +++ qtcreator-2.5.2/src/plugins/cpptools/cppcompletionassist.cpp 2012-08-08 13:47:06.000000000 +0000 @@ -1432,6 +1432,13 @@ break; } + } else if (Template *templ = ty->asTemplateType()) { + if (!result.binding()) + continue; + if (ClassOrNamespace *b = result.binding()->lookupType(templ->name())) { + completeClass(b); + break; + } } } diff -Nru qtcreator-2.5.0/src/plugins/cpptools/cpptools.pro qtcreator-2.5.2/src/plugins/cpptools/cpptools.pro --- qtcreator-2.5.0/src/plugins/cpptools/cpptools.pro 2012-05-09 12:13:19.000000000 +0000 +++ qtcreator-2.5.2/src/plugins/cpptools/cpptools.pro 2012-08-08 13:47:06.000000000 +0000 @@ -90,5 +90,6 @@ equals(TEST, 1) { SOURCES += \ - cppcodegen_test.cpp + cppcodegen_test.cpp \ + cppcompletion_test.cpp } diff -Nru qtcreator-2.5.0/src/plugins/cpptools/cpptoolsplugin.h qtcreator-2.5.2/src/plugins/cpptools/cpptoolsplugin.h --- qtcreator-2.5.0/src/plugins/cpptools/cpptoolsplugin.h 2012-05-09 12:13:19.000000000 +0000 +++ qtcreator-2.5.2/src/plugins/cpptools/cpptoolsplugin.h 2012-08-08 13:47:06.000000000 +0000 @@ -90,6 +90,9 @@ void test_codegen_definition_first_member(); void test_codegen_definition_last_member(); void test_codegen_definition_middle_member(); + + void test_completion_basic_1(); + void test_completion_template_1(); #endif private: diff -Nru qtcreator-2.5.0/src/plugins/help/helpviewer.cpp qtcreator-2.5.2/src/plugins/help/helpviewer.cpp --- qtcreator-2.5.0/src/plugins/help/helpviewer.cpp 2012-05-09 12:13:19.000000000 +0000 +++ qtcreator-2.5.2/src/plugins/help/helpviewer.cpp 2012-08-08 13:47:06.000000000 +0000 @@ -145,6 +145,10 @@ return QDesktopServices::openUrl(QUrl(saver.fileName())); } } + + if (url.scheme() == QLatin1String("mailto")) + return QDesktopServices::openUrl(url); + return false; } diff -Nru qtcreator-2.5.0/src/plugins/mercurial/mercurialplugin.cpp qtcreator-2.5.2/src/plugins/mercurial/mercurialplugin.cpp --- qtcreator-2.5.0/src/plugins/mercurial/mercurialplugin.cpp 2012-05-09 12:13:19.000000000 +0000 +++ qtcreator-2.5.2/src/plugins/mercurial/mercurialplugin.cpp 2012-08-08 13:47:06.000000000 +0000 @@ -552,8 +552,8 @@ m_submitRepository = state.topLevel(); - connect(m_client, SIGNAL(parsedStatus(QList)), - this, SLOT(showCommitWidget(QList))); + connect(m_client, SIGNAL(parsedStatus(QList)), + this, SLOT(showCommitWidget(QList))); m_client->emitParsedStatus(m_submitRepository); } @@ -562,8 +562,8 @@ VcsBaseOutputWindow *outputWindow = VcsBaseOutputWindow::instance(); //Once we receive our data release the connection so it can be reused elsewhere - disconnect(m_client, SIGNAL(parsedStatus(QList)), - this, SLOT(showCommitWidget(QList))); + disconnect(m_client, SIGNAL(parsedStatus(QList)), + this, SLOT(showCommitWidget(QList))); if (status.isEmpty()) { outputWindow->appendError(tr("There are no changes to commit.")); diff -Nru qtcreator-2.5.0/src/plugins/projectexplorer/projectexplorer.cpp qtcreator-2.5.2/src/plugins/projectexplorer/projectexplorer.cpp --- qtcreator-2.5.0/src/plugins/projectexplorer/projectexplorer.cpp 2012-05-09 12:13:19.000000000 +0000 +++ qtcreator-2.5.2/src/plugins/projectexplorer/projectexplorer.cpp 2012-08-08 13:47:06.000000000 +0000 @@ -1232,13 +1232,15 @@ if (d->m_shuttingDown) return; - foreach (Project *pro, d->m_session->projects()) - pro->saveSettings(); + if (!d->m_session->loadingSession()) { + foreach (Project *pro, d->m_session->projects()) + pro->saveSettings(); - if (d->m_session->isDefaultVirgin()) { - // do not save new virgin default sessions - } else { - d->m_session->save(); + if (d->m_session->isDefaultVirgin()) { + // do not save new virgin default sessions + } else { + d->m_session->save(); + } } QSettings *s = Core::ICore::settings(); @@ -1492,7 +1494,6 @@ Core::ICore::openFiles(combinedList, Core::ICore::OpenFilesFlags(Core::ICore::CanContainLineNumbers | Core::ICore::SwitchMode)); updateActions(); - } void ProjectExplorerPlugin::loadSession(const QString &session) diff -Nru qtcreator-2.5.0/src/plugins/projectexplorer/session.cpp qtcreator-2.5.2/src/plugins/projectexplorer/session.cpp --- qtcreator-2.5.0/src/plugins/projectexplorer/session.cpp 2012-05-09 12:13:19.000000000 +0000 +++ qtcreator-2.5.2/src/plugins/projectexplorer/session.cpp 2012-08-08 13:47:06.000000000 +0000 @@ -93,6 +93,7 @@ m_sessionNode(new SessionNode(this)), m_sessionName(QLatin1String("default")), m_virginSession(true), + m_loadingSession(false), m_startupProject(0) { connect(ModeManager::instance(), SIGNAL(currentModeChanged(Core::IMode*)), @@ -291,6 +292,11 @@ removeProjects(QList() << project); } +bool SessionManager::loadingSession() +{ + return m_loadingSession; +} + bool SessionManager::save() { if (debug) @@ -810,17 +816,23 @@ } } + m_loadingSession = true; + // Allow everyone to set something in the session and before saving emit aboutToUnloadSession(m_sessionName); if (!isDefaultVirgin()) { - if (!save()) + if (!save()) { + m_loadingSession = false; return false; + } } // Clean up - if (!ICore::editorManager()->closeAllEditors()) + if (!ICore::editorManager()->closeAllEditors()) { + m_loadingSession = false; return false; + } setStartupProject(0); removeProjects(projects()); @@ -872,6 +884,7 @@ // Starts a event loop, better do that at the very end askUserAboutFailedProjects(); + m_loadingSession = false; return true; } diff -Nru qtcreator-2.5.0/src/plugins/projectexplorer/session.h qtcreator-2.5.2/src/plugins/projectexplorer/session.h --- qtcreator-2.5.0/src/plugins/projectexplorer/session.h 2012-05-09 12:13:19.000000000 +0000 +++ qtcreator-2.5.2/src/plugins/projectexplorer/session.h 2012-08-08 13:47:06.000000000 +0000 @@ -123,6 +123,7 @@ QStringList projectsForSessionName(const QString &session) const; void reportProjectLoadingProgress(); + bool loadingSession(); signals: void projectAdded(ProjectExplorer::Project *project); void singleProjectAdded(ProjectExplorer::Project *project); @@ -167,6 +168,7 @@ mutable QStringList m_sessions; mutable QHash m_projectFileCache; + bool m_loadingSession; Project *m_startupProject; QList m_projects; diff -Nru qtcreator-2.5.0/src/plugins/qt4projectmanager/qt4nodes.cpp qtcreator-2.5.2/src/plugins/qt4projectmanager/qt4nodes.cpp --- qtcreator-2.5.0/src/plugins/qt4projectmanager/qt4nodes.cpp 2012-05-09 12:13:19.000000000 +0000 +++ qtcreator-2.5.2/src/plugins/qt4projectmanager/qt4nodes.cpp 2012-08-08 13:47:06.000000000 +0000 @@ -1478,7 +1478,7 @@ m_parseFutureWatcher.waitForFinished(); if (m_readerExact) { // Oh we need to clean up - applyEvaluate(EvalFail, true); + applyEvaluate(EvalAbort, true); m_project->decrementPendingEvaluateFutures(); } } diff -Nru qtcreator-2.5.0/src/plugins/qt4projectmanager/qt4nodes.h qtcreator-2.5.2/src/plugins/qt4projectmanager/qt4nodes.h --- qtcreator-2.5.0/src/plugins/qt4projectmanager/qt4nodes.h 2012-05-09 12:13:19.000000000 +0000 +++ qtcreator-2.5.2/src/plugins/qt4projectmanager/qt4nodes.h 2012-08-08 13:47:06.000000000 +0000 @@ -382,7 +382,7 @@ private: void setupReader(); - enum EvalResult { EvalFail, EvalPartial, EvalOk }; + enum EvalResult { EvalAbort, EvalFail, EvalPartial, EvalOk }; EvalResult evaluate(); void applyEvaluate(EvalResult parseResult, bool async); diff -Nru qtcreator-2.5.0/src/plugins/subversion/subversionplugin.cpp qtcreator-2.5.2/src/plugins/subversion/subversionplugin.cpp --- qtcreator-2.5.0/src/plugins/subversion/subversionplugin.cpp 2012-05-09 12:13:19.000000000 +0000 +++ qtcreator-2.5.2/src/plugins/subversion/subversionplugin.cpp 2012-08-08 13:47:06.000000000 +0000 @@ -1014,7 +1014,7 @@ { const VcsBase::VcsBasePluginState state = currentState(); QTC_ASSERT(state.hasProject(), return); - svnStatus(state.currentFileTopLevel(), state.relativeCurrentProject()); + svnStatus(state.currentProjectTopLevel(), state.relativeCurrentProject()); } void SubversionPlugin::describe(const QString &source, const QString &changeNr) diff -Nru qtcreator-2.5.0/src/tools/mdnssd/DNSCommon.c qtcreator-2.5.2/src/tools/mdnssd/DNSCommon.c --- qtcreator-2.5.0/src/tools/mdnssd/DNSCommon.c 2012-05-09 12:13:19.000000000 +0000 +++ qtcreator-2.5.2/src/tools/mdnssd/DNSCommon.c 1970-01-01 00:00:00.000000000 +0000 @@ -1,3153 +0,0 @@ -/* -*- Mode: C; tab-width: 4 -*- - * - * Copyright (c) 2002-2003 Apple Computer, Inc. All rights reserved. - * - * Licensed under the Apache License, Version 2.0 (the "License"); - * you may not use this file except in compliance with the License. - * You may obtain a copy of the License at - * - * http://www.apache.org/licenses/LICENSE-2.0 - * - * Unless required by applicable law or agreed to in writing, software - * distributed under the License is distributed on an "AS IS" BASIS, - * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. - * See the License for the specific language governing permissions and - * limitations under the License. - */ - -// Set mDNS_InstantiateInlines to tell mDNSEmbeddedAPI.h to instantiate inline functions, if necessary -#define mDNS_InstantiateInlines 1 -#include "DNSCommon.h" - -// Disable certain benign warnings with Microsoft compilers -#if (defined(_MSC_VER)) - // Disable "conditional expression is constant" warning for debug macros. - // Otherwise, this generates warnings for the perfectly natural construct "while(1)" - // If someone knows a variant way of writing "while(1)" that doesn't generate warning messages, please let us know - #pragma warning(disable:4127) - // Disable "array is too small to include a terminating null character" warning - // -- domain labels have an initial length byte, not a terminating null character - #pragma warning(disable:4295) -#endif - -// *************************************************************************** -#if COMPILER_LIKES_PRAGMA_MARK -#pragma mark - Program Constants -#endif - -mDNSexport const mDNSInterfaceID mDNSInterface_Any = 0; -mDNSexport const mDNSInterfaceID mDNSInterfaceMark = (mDNSInterfaceID)-1; -mDNSexport const mDNSInterfaceID mDNSInterface_LocalOnly = (mDNSInterfaceID)-2; -mDNSexport const mDNSInterfaceID mDNSInterface_Unicast = (mDNSInterfaceID)-3; -mDNSexport const mDNSInterfaceID mDNSInterface_P2P = (mDNSInterfaceID)-4; - -// Note: Microsoft's proposed "Link Local Multicast Name Resolution Protocol" (LLMNR) is essentially a limited version of -// Multicast DNS, using the same packet formats, naming syntax, and record types as Multicast DNS, but on a different UDP -// port and multicast address, which means it won't interoperate with the existing installed base of Multicast DNS responders. -// LLMNR uses IPv4 multicast address 224.0.0.252, IPv6 multicast address FF02::0001:0003, and UDP port 5355. -// Uncomment the appropriate lines below to build a special Multicast DNS responder for testing interoperability -// with Microsoft's LLMNR client code. - -#define DiscardPortAsNumber 9 -#define SSHPortAsNumber 22 -#define UnicastDNSPortAsNumber 53 -#define SSDPPortAsNumber 1900 -#define IPSECPortAsNumber 4500 -#define NSIPCPortAsNumber 5030 // Port used for dnsextd to talk to local nameserver bound to loopback -#define NATPMPAnnouncementPortAsNumber 5350 -#define NATPMPPortAsNumber 5351 -#define DNSEXTPortAsNumber 5352 // Port used for end-to-end DNS operations like LLQ, Updates with Leases, etc. -#define MulticastDNSPortAsNumber 5353 -#define LoopbackIPCPortAsNumber 5354 -//#define MulticastDNSPortAsNumber 5355 // LLMNR -#define PrivateDNSPortAsNumber 5533 - -mDNSexport const mDNSIPPort DiscardPort = { { DiscardPortAsNumber >> 8, DiscardPortAsNumber & 0xFF } }; -mDNSexport const mDNSIPPort SSHPort = { { SSHPortAsNumber >> 8, SSHPortAsNumber & 0xFF } }; -mDNSexport const mDNSIPPort UnicastDNSPort = { { UnicastDNSPortAsNumber >> 8, UnicastDNSPortAsNumber & 0xFF } }; -mDNSexport const mDNSIPPort SSDPPort = { { SSDPPortAsNumber >> 8, SSDPPortAsNumber & 0xFF } }; -mDNSexport const mDNSIPPort IPSECPort = { { IPSECPortAsNumber >> 8, IPSECPortAsNumber & 0xFF } }; -mDNSexport const mDNSIPPort NSIPCPort = { { NSIPCPortAsNumber >> 8, NSIPCPortAsNumber & 0xFF } }; -mDNSexport const mDNSIPPort NATPMPAnnouncementPort = { { NATPMPAnnouncementPortAsNumber >> 8, NATPMPAnnouncementPortAsNumber & 0xFF } }; -mDNSexport const mDNSIPPort NATPMPPort = { { NATPMPPortAsNumber >> 8, NATPMPPortAsNumber & 0xFF } }; -mDNSexport const mDNSIPPort DNSEXTPort = { { DNSEXTPortAsNumber >> 8, DNSEXTPortAsNumber & 0xFF } }; -mDNSexport const mDNSIPPort MulticastDNSPort = { { MulticastDNSPortAsNumber >> 8, MulticastDNSPortAsNumber & 0xFF } }; -mDNSexport const mDNSIPPort LoopbackIPCPort = { { LoopbackIPCPortAsNumber >> 8, LoopbackIPCPortAsNumber & 0xFF } }; -mDNSexport const mDNSIPPort PrivateDNSPort = { { PrivateDNSPortAsNumber >> 8, PrivateDNSPortAsNumber & 0xFF } }; - -mDNSexport const OwnerOptData zeroOwner = { 0, 0, { { 0 } }, { { 0 } }, { { 0 } } }; - -mDNSexport const mDNSIPPort zeroIPPort = { { 0 } }; -mDNSexport const mDNSv4Addr zerov4Addr = { { 0 } }; -mDNSexport const mDNSv6Addr zerov6Addr = { { 0 } }; -mDNSexport const mDNSEthAddr zeroEthAddr = { { 0 } }; -mDNSexport const mDNSv4Addr onesIPv4Addr = { { 255, 255, 255, 255 } }; -mDNSexport const mDNSv6Addr onesIPv6Addr = { { 255, 255, 255, 255, 255, 255, 255, 255, 255, 255, 255, 255, 255, 255, 255, 255 } }; -mDNSexport const mDNSEthAddr onesEthAddr = { { 255, 255, 255, 255, 255, 255 } }; -mDNSexport const mDNSAddr zeroAddr = { mDNSAddrType_None, {{{ 0 }}} }; - -mDNSexport const mDNSv4Addr AllDNSAdminGroup = { { 239, 255, 255, 251 } }; -mDNSexport const mDNSv4Addr AllHosts_v4 = { { 224, 0, 0, 1 } }; // For NAT-PMP Annoucements -mDNSexport const mDNSv6Addr AllHosts_v6 = { { 0xFF,0x02,0x00,0x00, 0x00,0x00,0x00,0x00, 0x00,0x00,0x00,0x00, 0x00,0x00,0x00,0x01 } }; -mDNSexport const mDNSv6Addr NDP_prefix = { { 0xFF,0x02,0x00,0x00, 0x00,0x00,0x00,0x00, 0x00,0x00,0x00,0x01, 0xFF,0x00,0x00,0xFB } }; // FF02:0:0:0:0:1:FF00::/104 -mDNSexport const mDNSEthAddr AllHosts_v6_Eth = { { 0x33, 0x33, 0x00, 0x00, 0x00, 0x01 } }; -mDNSexport const mDNSAddr AllDNSLinkGroup_v4 = { mDNSAddrType_IPv4, { { { 224, 0, 0, 251 } } } }; -//mDNSexport const mDNSAddr AllDNSLinkGroup_v4 = { mDNSAddrType_IPv4, { { { 224, 0, 0, 252 } } } }; // LLMNR -mDNSexport const mDNSAddr AllDNSLinkGroup_v6 = { mDNSAddrType_IPv6, { { { 0xFF,0x02,0x00,0x00, 0x00,0x00,0x00,0x00, 0x00,0x00,0x00,0x00, 0x00,0x00,0x00,0xFB } } } }; -//mDNSexport const mDNSAddr AllDNSLinkGroup_v6 = { mDNSAddrType_IPv6, { { { 0xFF,0x02,0x00,0x00, 0x00,0x00,0x00,0x00, 0x00,0x00,0x00,0x00, 0x00,0x01,0x00,0x03 } } } }; // LLMNR - -mDNSexport const mDNSOpaque16 zeroID = { { 0, 0 } }; -mDNSexport const mDNSOpaque16 onesID = { { 255, 255 } }; -mDNSexport const mDNSOpaque16 QueryFlags = { { kDNSFlag0_QR_Query | kDNSFlag0_OP_StdQuery, 0 } }; -mDNSexport const mDNSOpaque16 uQueryFlags = { { kDNSFlag0_QR_Query | kDNSFlag0_OP_StdQuery | kDNSFlag0_RD, 0 } }; -mDNSexport const mDNSOpaque16 ResponseFlags = { { kDNSFlag0_QR_Response | kDNSFlag0_OP_StdQuery | kDNSFlag0_AA, 0 } }; -mDNSexport const mDNSOpaque16 UpdateReqFlags = { { kDNSFlag0_QR_Query | kDNSFlag0_OP_Update, 0 } }; -mDNSexport const mDNSOpaque16 UpdateRespFlags = { { kDNSFlag0_QR_Response | kDNSFlag0_OP_Update, 0 } }; - -mDNSexport const mDNSOpaque64 zeroOpaque64 = { { 0 } }; - -// *************************************************************************** -#if COMPILER_LIKES_PRAGMA_MARK -#pragma mark - -#pragma mark - General Utility Functions -#endif - -// return true for RFC1918 private addresses -mDNSexport mDNSBool mDNSv4AddrIsRFC1918(mDNSv4Addr *addr) - { - return ((addr->b[0] == 10) || // 10/8 prefix - (addr->b[0] == 172 && (addr->b[1] & 0xF0) == 16) || // 172.16/12 - (addr->b[0] == 192 && addr->b[1] == 168)); // 192.168/16 - } - -mDNSexport NetworkInterfaceInfo *GetFirstActiveInterface(NetworkInterfaceInfo *intf) - { - while (intf && !intf->InterfaceActive) intf = intf->next; - return(intf); - } - -mDNSexport mDNSInterfaceID GetNextActiveInterfaceID(const NetworkInterfaceInfo *intf) - { - const NetworkInterfaceInfo *next = GetFirstActiveInterface(intf->next); - if (next) return(next->InterfaceID); else return(mDNSNULL); - } - -mDNSexport mDNSu32 NumCacheRecordsForInterfaceID(const mDNS *const m, mDNSInterfaceID id) - { - mDNSu32 slot, used = 0; - CacheGroup *cg; - const CacheRecord *rr; - FORALL_CACHERECORDS(slot, cg, rr) - if (rr->resrec.InterfaceID == id) used++; - return(used); - } - -mDNSexport char *DNSTypeName(mDNSu16 rrtype) - { - switch (rrtype) - { - case kDNSType_A: return("Addr"); - case kDNSType_NS: return("NS"); - case kDNSType_CNAME:return("CNAME"); - case kDNSType_SOA: return("SOA"); - case kDNSType_NULL: return("NULL"); - case kDNSType_PTR: return("PTR"); - case kDNSType_HINFO:return("HINFO"); - case kDNSType_TXT: return("TXT"); - case kDNSType_AAAA: return("AAAA"); - case kDNSType_SRV: return("SRV"); - case kDNSType_OPT: return("OPT"); - case kDNSType_NSEC: return("NSEC"); - case kDNSType_TSIG: return("TSIG"); - case kDNSQType_ANY: return("ANY"); - default: { - static char buffer[16]; - mDNS_snprintf(buffer, sizeof(buffer), "(%d)", rrtype); - return(buffer); - } - } - } - -// Note slight bug: this code uses the rdlength from the ResourceRecord object, to display -// the rdata from the RDataBody object. Sometimes this could be the wrong length -- but as -// long as this routine is only used for debugging messages, it probably isn't a big problem. -mDNSexport char *GetRRDisplayString_rdb(const ResourceRecord *const rr, const RDataBody *const rd1, char *const buffer) - { - const RDataBody2 *const rd = (RDataBody2 *)rd1; - #define RemSpc (MaxMsg-1-length) - char *ptr = buffer; - mDNSu32 length = mDNS_snprintf(buffer, MaxMsg-1, "%4d %##s %s ", rr->rdlength, rr->name->c, DNSTypeName(rr->rrtype)); - if (rr->RecordType == kDNSRecordTypePacketNegative) return(buffer); - if (!rr->rdlength) { mDNS_snprintf(buffer+length, RemSpc, "<< ZERO RDATA LENGTH >>"); return(buffer); } - - switch (rr->rrtype) - { - case kDNSType_A: mDNS_snprintf(buffer+length, RemSpc, "%.4a", &rd->ipv4); break; - - case kDNSType_NS: // Same as PTR - case kDNSType_CNAME:// Same as PTR - case kDNSType_PTR: mDNS_snprintf(buffer+length, RemSpc, "%##s", rd->name.c); break; - - case kDNSType_SOA: mDNS_snprintf(buffer+length, RemSpc, "%##s %##s %d %d %d %d %d", - rd->soa.mname.c, rd->soa.rname.c, - rd->soa.serial, rd->soa.refresh, rd->soa.retry, rd->soa.expire, rd->soa.min); - break; - - case kDNSType_HINFO:// Display this the same as TXT (show all constituent strings) - case kDNSType_TXT: { - const mDNSu8 *t = rd->txt.c; - while (t < rd->txt.c + rr->rdlength) - { - length += mDNS_snprintf(buffer+length, RemSpc, "%s%#s", t > rd->txt.c ? "|" : "", t); - t += 1 + t[0]; - } - } break; - - case kDNSType_AAAA: mDNS_snprintf(buffer+length, RemSpc, "%.16a", &rd->ipv6); break; - case kDNSType_SRV: mDNS_snprintf(buffer+length, RemSpc, "%u %u %u %##s", - rd->srv.priority, rd->srv.weight, mDNSVal16(rd->srv.port), rd->srv.target.c); break; - - case kDNSType_OPT: { - const rdataOPT *opt; - const rdataOPT *const end = (const rdataOPT *)&rd->data[rr->rdlength]; - length += mDNS_snprintf(buffer+length, RemSpc, "Max %d", rr->rrclass); - for (opt = &rd->opt[0]; opt < end; opt++) - { - switch(opt->opt) - { - case kDNSOpt_LLQ: - length += mDNS_snprintf(buffer+length, RemSpc, " Vers %d", opt->u.llq.vers); - length += mDNS_snprintf(buffer+length, RemSpc, " Op %d", opt->u.llq.llqOp); - length += mDNS_snprintf(buffer+length, RemSpc, " Err/Port %d", opt->u.llq.err); - length += mDNS_snprintf(buffer+length, RemSpc, " ID %08X%08X", opt->u.llq.id.l[0], opt->u.llq.id.l[1]); - length += mDNS_snprintf(buffer+length, RemSpc, " Lease %d", opt->u.llq.llqlease); - break; - case kDNSOpt_Lease: - length += mDNS_snprintf(buffer+length, RemSpc, " Lease %d", opt->u.updatelease); - break; - case kDNSOpt_Owner: - length += mDNS_snprintf(buffer+length, RemSpc, " Vers %d", opt->u.owner.vers); - length += mDNS_snprintf(buffer+length, RemSpc, " Seq %3d", (mDNSu8)opt->u.owner.seq); // Display as unsigned - length += mDNS_snprintf(buffer+length, RemSpc, " MAC %.6a", opt->u.owner.HMAC.b); - if (opt->optlen >= DNSOpt_OwnerData_ID_Wake_Space-4) - { - length += mDNS_snprintf(buffer+length, RemSpc, " I-MAC %.6a", opt->u.owner.IMAC.b); - if (opt->optlen > DNSOpt_OwnerData_ID_Wake_Space-4) - length += mDNS_snprintf(buffer+length, RemSpc, " Password %.6a", opt->u.owner.password.b); - } - break; - default: - length += mDNS_snprintf(buffer+length, RemSpc, " Unknown %d", opt->opt); - break; - } - } - } - break; - - case kDNSType_NSEC: { - mDNSu16 i; - for (i=0; i<255; i++) - if (rd->nsec.bitmap[i>>3] & (128 >> (i&7))) - length += mDNS_snprintf(buffer+length, RemSpc, "%s ", DNSTypeName(i)); - } - break; - - default: mDNS_snprintf(buffer+length, RemSpc, "RDLen %d: %s", rr->rdlength, rd->data); - // Really should scan buffer to check if text is valid UTF-8 and only replace with dots if not - for (ptr = buffer; *ptr; ptr++) if (*ptr < ' ') *ptr = '.'; - break; - } - return(buffer); - } - -// See comments in mDNSEmbeddedAPI.h -#if _PLATFORM_HAS_STRONG_PRNG_ -#define mDNSRandomNumber mDNSPlatformRandomNumber -#else -mDNSlocal mDNSu32 mDNSRandomFromSeed(mDNSu32 seed) - { - return seed * 21 + 1; - } - -mDNSlocal mDNSu32 mDNSMixRandomSeed(mDNSu32 seed, mDNSu8 iteration) - { - return iteration ? mDNSMixRandomSeed(mDNSRandomFromSeed(seed), --iteration) : seed; - } - -mDNSlocal mDNSu32 mDNSRandomNumber() - { - static mDNSBool seeded = mDNSfalse; - static mDNSu32 seed = 0; - if (!seeded) - { - seed = mDNSMixRandomSeed(mDNSPlatformRandomSeed(), 100); - seeded = mDNStrue; - } - return (seed = mDNSRandomFromSeed(seed)); - } -#endif // ! _PLATFORM_HAS_STRONG_PRNG_ - -mDNSexport mDNSu32 mDNSRandom(mDNSu32 max) // Returns pseudo-random result from zero to max inclusive - { - mDNSu32 ret = 0; - mDNSu32 mask = 1; - - while (mask < max) mask = (mask << 1) | 1; - - do ret = mDNSRandomNumber() & mask; - while (ret > max); - - return ret; - } - -mDNSexport mDNSBool mDNSSameAddress(const mDNSAddr *ip1, const mDNSAddr *ip2) - { - if (ip1->type == ip2->type) - { - switch (ip1->type) - { - case mDNSAddrType_None : return(mDNStrue); // Empty addresses have no data and are therefore always equal - case mDNSAddrType_IPv4 : return(mDNSBool)(mDNSSameIPv4Address(ip1->ip.v4, ip2->ip.v4)); - case mDNSAddrType_IPv6 : return(mDNSBool)(mDNSSameIPv6Address(ip1->ip.v6, ip2->ip.v6)); - } - } - return(mDNSfalse); - } - -mDNSexport mDNSBool mDNSAddrIsDNSMulticast(const mDNSAddr *ip) - { - switch(ip->type) - { - case mDNSAddrType_IPv4: return(mDNSBool)(mDNSSameIPv4Address(ip->ip.v4, AllDNSLinkGroup_v4.ip.v4)); - case mDNSAddrType_IPv6: return(mDNSBool)(mDNSSameIPv6Address(ip->ip.v6, AllDNSLinkGroup_v6.ip.v6)); - default: return(mDNSfalse); - } - } - -// *************************************************************************** -#if COMPILER_LIKES_PRAGMA_MARK -#pragma mark - -#pragma mark - Domain Name Utility Functions -#endif - -mDNSexport mDNSBool SameDomainLabel(const mDNSu8 *a, const mDNSu8 *b) - { - int i; - const int len = *a++; - - if (len > MAX_DOMAIN_LABEL) - { debugf("Malformed label (too long)"); return(mDNSfalse); } - - if (len != *b++) return(mDNSfalse); - for (i=0; ic; - const mDNSu8 * b = d2->c; - const mDNSu8 *const max = d1->c + MAX_DOMAIN_NAME; // Maximum that's valid - - while (*a || *b) - { - if (a + 1 + *a >= max) - { debugf("Malformed domain name (more than 256 characters)"); return(mDNSfalse); } - if (!SameDomainLabel(a, b)) return(mDNSfalse); - a += 1 + *a; - b += 1 + *b; - } - - return(mDNStrue); - } - -mDNSexport mDNSBool SameDomainNameCS(const domainname *const d1, const domainname *const d2) - { - mDNSu16 l1 = DomainNameLength(d1); - mDNSu16 l2 = DomainNameLength(d2); - return(l1 <= MAX_DOMAIN_NAME && l1 == l2 && mDNSPlatformMemSame(d1, d2, l1)); - } - -mDNSexport mDNSBool IsLocalDomain(const domainname *d) - { - // Domains that are defined to be resolved via link-local multicast are: - // local., 254.169.in-addr.arpa., and {8,9,A,B}.E.F.ip6.arpa. - static const domainname *nL = (const domainname*)"\x5" "local"; - static const domainname *nR = (const domainname*)"\x3" "254" "\x3" "169" "\x7" "in-addr" "\x4" "arpa"; - static const domainname *n8 = (const domainname*)"\x1" "8" "\x1" "e" "\x1" "f" "\x3" "ip6" "\x4" "arpa"; - static const domainname *n9 = (const domainname*)"\x1" "9" "\x1" "e" "\x1" "f" "\x3" "ip6" "\x4" "arpa"; - static const domainname *nA = (const domainname*)"\x1" "a" "\x1" "e" "\x1" "f" "\x3" "ip6" "\x4" "arpa"; - static const domainname *nB = (const domainname*)"\x1" "b" "\x1" "e" "\x1" "f" "\x3" "ip6" "\x4" "arpa"; - - const domainname *d1, *d2, *d3, *d4, *d5; // Top-level domain, second-level domain, etc. - d1 = d2 = d3 = d4 = d5 = mDNSNULL; - while (d->c[0]) - { - d5 = d4; d4 = d3; d3 = d2; d2 = d1; d1 = d; - d = (const domainname*)(d->c + 1 + d->c[0]); - } - - if (d1 && SameDomainName(d1, nL)) return(mDNStrue); - if (d4 && SameDomainName(d4, nR)) return(mDNStrue); - if (d5 && SameDomainName(d5, n8)) return(mDNStrue); - if (d5 && SameDomainName(d5, n9)) return(mDNStrue); - if (d5 && SameDomainName(d5, nA)) return(mDNStrue); - if (d5 && SameDomainName(d5, nB)) return(mDNStrue); - return(mDNSfalse); - } - -mDNSexport const mDNSu8 *LastLabel(const domainname *d) - { - const mDNSu8 *p = d->c; - while (d->c[0]) - { - p = d->c; - d = (const domainname*)(d->c + 1 + d->c[0]); - } - return(p); - } - -// Returns length of a domain name INCLUDING the byte for the final null label -// e.g. for the root label "." it returns one -// For the FQDN "com." it returns 5 (length byte, three data bytes, final zero) -// Legal results are 1 (just root label) to 256 (MAX_DOMAIN_NAME) -// If the given domainname is invalid, result is 257 (MAX_DOMAIN_NAME+1) -mDNSexport mDNSu16 DomainNameLengthLimit(const domainname *const name, const mDNSu8 *limit) - { - const mDNSu8 *src = name->c; - while (src < limit && *src <= MAX_DOMAIN_LABEL) - { - if (*src == 0) return((mDNSu16)(src - name->c + 1)); - src += 1 + *src; - } - return(MAX_DOMAIN_NAME+1); - } - -// CompressedDomainNameLength returns the length of a domain name INCLUDING the byte -// for the final null label, e.g. for the root label "." it returns one. -// E.g. for the FQDN "foo.com." it returns 9 -// (length, three data bytes, length, three more data bytes, final zero). -// In the case where a parent domain name is provided, and the given name is a child -// of that parent, CompressedDomainNameLength returns the length of the prefix portion -// of the child name, plus TWO bytes for the compression pointer. -// E.g. for the name "foo.com." with parent "com.", it returns 6 -// (length, three data bytes, two-byte compression pointer). -mDNSexport mDNSu16 CompressedDomainNameLength(const domainname *const name, const domainname *parent) - { - const mDNSu8 *src = name->c; - if (parent && parent->c[0] == 0) parent = mDNSNULL; - while (*src) - { - if (*src > MAX_DOMAIN_LABEL) return(MAX_DOMAIN_NAME+1); - if (parent && SameDomainName((const domainname *)src, parent)) return((mDNSu16)(src - name->c + 2)); - src += 1 + *src; - if (src - name->c >= MAX_DOMAIN_NAME) return(MAX_DOMAIN_NAME+1); - } - return((mDNSu16)(src - name->c + 1)); - } - -// CountLabels() returns number of labels in name, excluding final root label -// (e.g. for "apple.com." CountLabels returns 2.) -mDNSexport int CountLabels(const domainname *d) - { - int count = 0; - const mDNSu8 *ptr; - for (ptr = d->c; *ptr; ptr = ptr + ptr[0] + 1) count++; - return count; - } - -// SkipLeadingLabels skips over the first 'skip' labels in the domainname, -// returning a pointer to the suffix with 'skip' labels removed. -mDNSexport const domainname *SkipLeadingLabels(const domainname *d, int skip) - { - while (skip > 0 && d->c[0]) { d = (const domainname *)(d->c + 1 + d->c[0]); skip--; } - return(d); - } - -// AppendLiteralLabelString appends a single label to an existing (possibly empty) domainname. -// The C string contains the label as-is, with no escaping, etc. -// Any dots in the name are literal dots, not label separators -// If successful, AppendLiteralLabelString returns a pointer to the next unused byte -// in the domainname bufer (i.e. the next byte after the terminating zero). -// If unable to construct a legal domain name (i.e. label more than 63 bytes, or total more than 256 bytes) -// AppendLiteralLabelString returns mDNSNULL. -mDNSexport mDNSu8 *AppendLiteralLabelString(domainname *const name, const char *cstr) - { - mDNSu8 * ptr = name->c + DomainNameLength(name) - 1; // Find end of current name - const mDNSu8 *const lim1 = name->c + MAX_DOMAIN_NAME - 1; // Limit of how much we can add (not counting final zero) - const mDNSu8 *const lim2 = ptr + 1 + MAX_DOMAIN_LABEL; - const mDNSu8 *const lim = (lim1 < lim2) ? lim1 : lim2; - mDNSu8 *lengthbyte = ptr++; // Record where the length is going to go - - while (*cstr && ptr < lim) *ptr++ = (mDNSu8)*cstr++; // Copy the data - *lengthbyte = (mDNSu8)(ptr - lengthbyte - 1); // Fill in the length byte - *ptr++ = 0; // Put the null root label on the end - if (*cstr) return(mDNSNULL); // Failure: We didn't successfully consume all input - else return(ptr); // Success: return new value of ptr - } - -// AppendDNSNameString appends zero or more labels to an existing (possibly empty) domainname. -// The C string is in conventional DNS syntax: -// Textual labels, escaped as necessary using the usual DNS '\' notation, separated by dots. -// If successful, AppendDNSNameString returns a pointer to the next unused byte -// in the domainname bufer (i.e. the next byte after the terminating zero). -// If unable to construct a legal domain name (i.e. label more than 63 bytes, or total more than 256 bytes) -// AppendDNSNameString returns mDNSNULL. -mDNSexport mDNSu8 *AppendDNSNameString(domainname *const name, const char *cstring) - { - const char *cstr = cstring; - mDNSu8 * ptr = name->c + DomainNameLength(name) - 1; // Find end of current name - const mDNSu8 *const lim = name->c + MAX_DOMAIN_NAME - 1; // Limit of how much we can add (not counting final zero) - while (*cstr && ptr < lim) // While more characters, and space to put them... - { - mDNSu8 *lengthbyte = ptr++; // Record where the length is going to go - if (*cstr == '.') { LogMsg("AppendDNSNameString: Illegal empty label in name \"%s\"", cstring); return(mDNSNULL); } - while (*cstr && *cstr != '.' && ptr < lim) // While we have characters in the label... - { - mDNSu8 c = (mDNSu8)*cstr++; // Read the character - if (c == '\\') // If escape character, check next character - { - c = (mDNSu8)*cstr++; // Assume we'll just take the next character - if (mDNSIsDigit(cstr[-1]) && mDNSIsDigit(cstr[0]) && mDNSIsDigit(cstr[1])) - { // If three decimal digits, - int v0 = cstr[-1] - '0'; // then interpret as three-digit decimal - int v1 = cstr[ 0] - '0'; - int v2 = cstr[ 1] - '0'; - int val = v0 * 100 + v1 * 10 + v2; - if (val <= 255) { c = (mDNSu8)val; cstr += 2; } // If valid three-digit decimal value, use it - } - } - *ptr++ = c; // Write the character - } - if (*cstr) cstr++; // Skip over the trailing dot (if present) - if (ptr - lengthbyte - 1 > MAX_DOMAIN_LABEL) // If illegal label, abort - return(mDNSNULL); - *lengthbyte = (mDNSu8)(ptr - lengthbyte - 1); // Fill in the length byte - } - - *ptr++ = 0; // Put the null root label on the end - if (*cstr) return(mDNSNULL); // Failure: We didn't successfully consume all input - else return(ptr); // Success: return new value of ptr - } - -// AppendDomainLabel appends a single label to a name. -// If successful, AppendDomainLabel returns a pointer to the next unused byte -// in the domainname bufer (i.e. the next byte after the terminating zero). -// If unable to construct a legal domain name (i.e. label more than 63 bytes, or total more than 256 bytes) -// AppendDomainLabel returns mDNSNULL. -mDNSexport mDNSu8 *AppendDomainLabel(domainname *const name, const domainlabel *const label) - { - int i; - mDNSu8 *ptr = name->c + DomainNameLength(name) - 1; - - // Check label is legal - if (label->c[0] > MAX_DOMAIN_LABEL) return(mDNSNULL); - - // Check that ptr + length byte + data bytes + final zero does not exceed our limit - if (ptr + 1 + label->c[0] + 1 > name->c + MAX_DOMAIN_NAME) return(mDNSNULL); - - for (i=0; i<=label->c[0]; i++) *ptr++ = label->c[i]; // Copy the label data - *ptr++ = 0; // Put the null root label on the end - return(ptr); - } - -mDNSexport mDNSu8 *AppendDomainName(domainname *const name, const domainname *const append) - { - mDNSu8 * ptr = name->c + DomainNameLength(name) - 1; // Find end of current name - const mDNSu8 *const lim = name->c + MAX_DOMAIN_NAME - 1; // Limit of how much we can add (not counting final zero) - const mDNSu8 * src = append->c; - while (src[0]) - { - int i; - if (ptr + 1 + src[0] > lim) return(mDNSNULL); - for (i=0; i<=src[0]; i++) *ptr++ = src[i]; - *ptr = 0; // Put the null root label on the end - src += i; - } - return(ptr); - } - -// MakeDomainLabelFromLiteralString makes a single domain label from a single literal C string (with no escaping). -// If successful, MakeDomainLabelFromLiteralString returns mDNStrue. -// If unable to convert the whole string to a legal domain label (i.e. because length is more than 63 bytes) then -// MakeDomainLabelFromLiteralString makes a legal domain label from the first 63 bytes of the string and returns mDNSfalse. -// In some cases silently truncated oversized names to 63 bytes is acceptable, so the return result may be ignored. -// In other cases silent truncation may not be acceptable, so in those cases the calling function needs to check the return result. -mDNSexport mDNSBool MakeDomainLabelFromLiteralString(domainlabel *const label, const char *cstr) - { - mDNSu8 * ptr = label->c + 1; // Where we're putting it - const mDNSu8 *const limit = label->c + 1 + MAX_DOMAIN_LABEL; // The maximum we can put - while (*cstr && ptr < limit) *ptr++ = (mDNSu8)*cstr++; // Copy the label - label->c[0] = (mDNSu8)(ptr - label->c - 1); // Set the length byte - return(*cstr == 0); // Return mDNStrue if we successfully consumed all input - } - -// MakeDomainNameFromDNSNameString makes a native DNS-format domainname from a C string. -// The C string is in conventional DNS syntax: -// Textual labels, escaped as necessary using the usual DNS '\' notation, separated by dots. -// If successful, MakeDomainNameFromDNSNameString returns a pointer to the next unused byte -// in the domainname bufer (i.e. the next byte after the terminating zero). -// If unable to construct a legal domain name (i.e. label more than 63 bytes, or total more than 256 bytes) -// MakeDomainNameFromDNSNameString returns mDNSNULL. -mDNSexport mDNSu8 *MakeDomainNameFromDNSNameString(domainname *const name, const char *cstr) - { - name->c[0] = 0; // Make an empty domain name - return(AppendDNSNameString(name, cstr)); // And then add this string to it - } - -mDNSexport char *ConvertDomainLabelToCString_withescape(const domainlabel *const label, char *ptr, char esc) - { - const mDNSu8 * src = label->c; // Domain label we're reading - const mDNSu8 len = *src++; // Read length of this (non-null) label - const mDNSu8 *const end = src + len; // Work out where the label ends - if (len > MAX_DOMAIN_LABEL) return(mDNSNULL); // If illegal label, abort - while (src < end) // While we have characters in the label - { - mDNSu8 c = *src++; - if (esc) - { - if (c == '.' || c == esc) // If character is a dot or the escape character - *ptr++ = esc; // Output escape character - else if (c <= ' ') // If non-printing ascii, - { // Output decimal escape sequence - *ptr++ = esc; - *ptr++ = (char) ('0' + (c / 100) ); - *ptr++ = (char) ('0' + (c / 10) % 10); - c = (mDNSu8)('0' + (c ) % 10); - } - } - *ptr++ = (char)c; // Copy the character - } - *ptr = 0; // Null-terminate the string - return(ptr); // and return - } - -// Note: To guarantee that there will be no possible overrun, cstr must be at least MAX_ESCAPED_DOMAIN_NAME (1009 bytes) -mDNSexport char *ConvertDomainNameToCString_withescape(const domainname *const name, char *ptr, char esc) - { - const mDNSu8 *src = name->c; // Domain name we're reading - const mDNSu8 *const max = name->c + MAX_DOMAIN_NAME; // Maximum that's valid - - if (*src == 0) *ptr++ = '.'; // Special case: For root, just write a dot - - while (*src) // While more characters in the domain name - { - if (src + 1 + *src >= max) return(mDNSNULL); - ptr = ConvertDomainLabelToCString_withescape((const domainlabel *)src, ptr, esc); - if (!ptr) return(mDNSNULL); - src += 1 + *src; - *ptr++ = '.'; // Write the dot after the label - } - - *ptr++ = 0; // Null-terminate the string - return(ptr); // and return - } - -// RFC 1034 rules: -// Host names must start with a letter, end with a letter or digit, -// and have as interior characters only letters, digits, and hyphen. -// This was subsequently modified in RFC 1123 to allow the first character to be either a letter or a digit - -mDNSexport void ConvertUTF8PstringToRFC1034HostLabel(const mDNSu8 UTF8Name[], domainlabel *const hostlabel) - { - const mDNSu8 * src = &UTF8Name[1]; - const mDNSu8 *const end = &UTF8Name[1] + UTF8Name[0]; - mDNSu8 * ptr = &hostlabel->c[1]; - const mDNSu8 *const lim = &hostlabel->c[1] + MAX_DOMAIN_LABEL; - while (src < end) - { - // Delete apostrophes from source name - if (src[0] == '\'') { src++; continue; } // Standard straight single quote - if (src + 2 < end && src[0] == 0xE2 && src[1] == 0x80 && src[2] == 0x99) - { src += 3; continue; } // Unicode curly apostrophe - if (ptr < lim) - { - if (mDNSValidHostChar(*src, (ptr > &hostlabel->c[1]), (src < end-1))) *ptr++ = *src; - else if (ptr > &hostlabel->c[1] && ptr[-1] != '-') *ptr++ = '-'; - } - src++; - } - while (ptr > &hostlabel->c[1] && ptr[-1] == '-') ptr--; // Truncate trailing '-' marks - hostlabel->c[0] = (mDNSu8)(ptr - &hostlabel->c[1]); - } - -#define ValidTransportProtocol(X) ( (X)[0] == 4 && (X)[1] == '_' && \ - ((((X)[2] | 0x20) == 'u' && ((X)[3] | 0x20) == 'd') || (((X)[2] | 0x20) == 't' && ((X)[3] | 0x20) == 'c')) && \ - ((X)[4] | 0x20) == 'p') - -mDNSexport mDNSu8 *ConstructServiceName(domainname *const fqdn, - const domainlabel *name, const domainname *type, const domainname *const domain) - { - int i, len; - mDNSu8 *dst = fqdn->c; - const mDNSu8 *src; - const char *errormsg; -#if APPLE_OSX_mDNSResponder - mDNSBool loggedUnderscore = mDNSfalse; - static char typeBuf[MAX_ESCAPED_DOMAIN_NAME]; -#endif - - // In the case where there is no name (and ONLY in that case), - // a single-label subtype is allowed as the first label of a three-part "type" - if (!name && type) - { - const mDNSu8 *s0 = type->c; - if (s0[0] && s0[0] < 0x40) // If legal first label (at least one character, and no more than 63) - { - const mDNSu8 * s1 = s0 + 1 + s0[0]; - if (s1[0] && s1[0] < 0x40) // and legal second label (at least one character, and no more than 63) - { - const mDNSu8 *s2 = s1 + 1 + s1[0]; - if (s2[0] && s2[0] < 0x40 && s2[1+s2[0]] == 0) // and we have three and only three labels - { - static const mDNSu8 SubTypeLabel[5] = "\x04_sub"; - src = s0; // Copy the first label - len = *src; - for (i=0; i <= len; i++) *dst++ = *src++; - for (i=0; i < (int)sizeof(SubTypeLabel); i++) *dst++ = SubTypeLabel[i]; - type = (const domainname *)s1; - - // Special support to enable the DNSServiceBrowse call made by Bonjour Browser - // For these queries, we retract the "._sub" we just added between the subtype and the main type - // Remove after Bonjour Browser is updated to use DNSServiceQueryRecord instead of DNSServiceBrowse - if (SameDomainName((domainname*)s0, (const domainname*)"\x09_services\x07_dns-sd\x04_udp")) - dst -= sizeof(SubTypeLabel); - } - } - } - } - - if (name && name->c[0]) - { - src = name->c; // Put the service name into the domain name - len = *src; - if (len >= 0x40) { errormsg = "Service instance name too long"; goto fail; } - for (i=0; i<=len; i++) *dst++ = *src++; - } - else - name = (domainlabel*)""; // Set this up to be non-null, to avoid errors if we have to call LogMsg() below - - src = type->c; // Put the service type into the domain name - len = *src; - if (len < 2 || len > 16) - { - LogMsg("Bad service type in %#s.%##s%##s Application protocol name must be underscore plus 1-15 characters. " - "See ", name->c, type->c, domain->c); -#if APPLE_OSX_mDNSResponder - ConvertDomainNameToCString(type, typeBuf); - mDNSASLLog(mDNSNULL, "serviceType.nameTooLong", "noop", typeBuf, ""); -#endif - } - if (len < 2 || len >= 0x40 || (len > 16 && !SameDomainName(domain, &localdomain))) return(mDNSNULL); - if (src[1] != '_') { errormsg = "Application protocol name must begin with underscore"; goto fail; } - for (i=2; i<=len; i++) - { - // Letters and digits are allowed anywhere - if (mDNSIsLetter(src[i]) || mDNSIsDigit(src[i])) continue; - // Hyphens are only allowed as interior characters - // Underscores are not supposed to be allowed at all, but for backwards compatibility with some old products we do allow them, - // with the same rule as hyphens - if ((src[i] == '-' || src[i] == '_') && i > 2 && i < len) - { -#if APPLE_OSX_mDNSResponder - if (src[i] == '_' && loggedUnderscore == mDNSfalse) - { - ConvertDomainNameToCString(type, typeBuf); - mDNSASLLog(mDNSNULL, "serviceType.nameWithUnderscore", "noop", typeBuf, ""); - loggedUnderscore = mDNStrue; - } -#endif - continue; - } - errormsg = "Application protocol name must contain only letters, digits, and hyphens"; -#if APPLE_OSX_mDNSResponder - { - ConvertDomainNameToCString(type, typeBuf); - mDNSASLLog(mDNSNULL, "serviceType.nameWithIllegalCharacters", "noop", typeBuf, ""); - } -#endif - goto fail; - } - for (i=0; i<=len; i++) *dst++ = *src++; - - len = *src; - if (!ValidTransportProtocol(src)) { errormsg = "Transport protocol name must be _udp or _tcp"; goto fail; } - for (i=0; i<=len; i++) *dst++ = *src++; - - if (*src) { errormsg = "Service type must have only two labels"; goto fail; } - - *dst = 0; - if (!domain->c[0]) { errormsg = "Service domain must be non-empty"; goto fail; } - if (SameDomainName(domain, (const domainname*)"\x05" "local" "\x04" "arpa")) - { errormsg = "Illegal domain \"local.arpa.\" Use \"local.\" (or empty string)"; goto fail; } - dst = AppendDomainName(fqdn, domain); - if (!dst) { errormsg = "Service domain too long"; goto fail; } - return(dst); - -fail: - LogMsg("ConstructServiceName: %s: %#s.%##s%##s", errormsg, name->c, type->c, domain->c); - return(mDNSNULL); - } - -// A service name has the form: instance.application-protocol.transport-protocol.domain -// DeconstructServiceName is currently fairly forgiving: It doesn't try to enforce character -// set or length limits for the protocol names, and the final domain is allowed to be empty. -// However, if the given FQDN doesn't contain at least three labels, -// DeconstructServiceName will reject it and return mDNSfalse. -mDNSexport mDNSBool DeconstructServiceName(const domainname *const fqdn, - domainlabel *const name, domainname *const type, domainname *const domain) - { - int i, len; - const mDNSu8 *src = fqdn->c; - const mDNSu8 *max = fqdn->c + MAX_DOMAIN_NAME; - mDNSu8 *dst; - - dst = name->c; // Extract the service name - len = *src; - if (!len) { debugf("DeconstructServiceName: FQDN empty!"); return(mDNSfalse); } - if (len >= 0x40) { debugf("DeconstructServiceName: Instance name too long"); return(mDNSfalse); } - for (i=0; i<=len; i++) *dst++ = *src++; - - dst = type->c; // Extract the service type - len = *src; - if (!len) { debugf("DeconstructServiceName: FQDN contains only one label!"); return(mDNSfalse); } - if (len >= 0x40) { debugf("DeconstructServiceName: Application protocol name too long"); return(mDNSfalse); } - if (src[1] != '_'){ debugf("DeconstructServiceName: No _ at start of application protocol"); return(mDNSfalse); } - for (i=0; i<=len; i++) *dst++ = *src++; - - len = *src; - if (!len) { debugf("DeconstructServiceName: FQDN contains only two labels!"); return(mDNSfalse); } - if (!ValidTransportProtocol(src)) - { debugf("DeconstructServiceName: Transport protocol must be _udp or _tcp"); return(mDNSfalse); } - for (i=0; i<=len; i++) *dst++ = *src++; - *dst++ = 0; // Put terminator on the end of service type - - dst = domain->c; // Extract the service domain - while (*src) - { - len = *src; - if (len >= 0x40) - { debugf("DeconstructServiceName: Label in service domain too long"); return(mDNSfalse); } - if (src + 1 + len + 1 >= max) - { debugf("DeconstructServiceName: Total service domain too long"); return(mDNSfalse); } - for (i=0; i<=len; i++) *dst++ = *src++; - } - *dst++ = 0; // Put the null root label on the end - - return(mDNStrue); - } - -// Notes on UTF-8: -// 0xxxxxxx represents a 7-bit ASCII value from 0x00 to 0x7F -// 10xxxxxx is a continuation byte of a multi-byte character -// 110xxxxx is the first byte of a 2-byte character (11 effective bits; values 0x 80 - 0x 800-1) -// 1110xxxx is the first byte of a 3-byte character (16 effective bits; values 0x 800 - 0x 10000-1) -// 11110xxx is the first byte of a 4-byte character (21 effective bits; values 0x 10000 - 0x 200000-1) -// 111110xx is the first byte of a 5-byte character (26 effective bits; values 0x 200000 - 0x 4000000-1) -// 1111110x is the first byte of a 6-byte character (31 effective bits; values 0x4000000 - 0x80000000-1) -// -// UTF-16 surrogate pairs are used in UTF-16 to encode values larger than 0xFFFF. -// Although UTF-16 surrogate pairs are not supposed to appear in legal UTF-8, we want to be defensive -// about that too. (See , "What are surrogates?") -// The first of pair is a UTF-16 value in the range 0xD800-0xDBFF (11101101 1010xxxx 10xxxxxx in UTF-8), -// and the second is a UTF-16 value in the range 0xDC00-0xDFFF (11101101 1011xxxx 10xxxxxx in UTF-8). - -mDNSexport mDNSu32 TruncateUTF8ToLength(mDNSu8 *string, mDNSu32 length, mDNSu32 max) - { - if (length > max) - { - mDNSu8 c1 = string[max]; // First byte after cut point - mDNSu8 c2 = (max+1 < length) ? string[max+1] : (mDNSu8)0xB0; // Second byte after cut point - length = max; // Trim length down - while (length > 0) - { - // Check if the byte right after the chop point is a UTF-8 continuation byte, - // or if the character right after the chop point is the second of a UTF-16 surrogate pair. - // If so, then we continue to chop more bytes until we get to a legal chop point. - mDNSBool continuation = ((c1 & 0xC0) == 0x80); - mDNSBool secondsurrogate = (c1 == 0xED && (c2 & 0xF0) == 0xB0); - if (!continuation && !secondsurrogate) break; - c2 = c1; - c1 = string[--length]; - } - // Having truncated characters off the end of our string, also cut off any residual white space - while (length > 0 && string[length-1] <= ' ') length--; - } - return(length); - } - -// Returns true if a rich text label ends in " (nnn)", or if an RFC 1034 -// name ends in "-nnn", where n is some decimal number. -mDNSexport mDNSBool LabelContainsSuffix(const domainlabel *const name, const mDNSBool RichText) - { - mDNSu16 l = name->c[0]; - - if (RichText) - { - if (l < 4) return mDNSfalse; // Need at least " (2)" - if (name->c[l--] != ')') return mDNSfalse; // Last char must be ')' - if (!mDNSIsDigit(name->c[l])) return mDNSfalse; // Preceeded by a digit - l--; - while (l > 2 && mDNSIsDigit(name->c[l])) l--; // Strip off digits - return (name->c[l] == '(' && name->c[l - 1] == ' '); - } - else - { - if (l < 2) return mDNSfalse; // Need at least "-2" - if (!mDNSIsDigit(name->c[l])) return mDNSfalse; // Last char must be a digit - l--; - while (l > 2 && mDNSIsDigit(name->c[l])) l--; // Strip off digits - return (name->c[l] == '-'); - } - } - -// removes an auto-generated suffix (appended on a name collision) from a label. caller is -// responsible for ensuring that the label does indeed contain a suffix. returns the number -// from the suffix that was removed. -mDNSexport mDNSu32 RemoveLabelSuffix(domainlabel *name, mDNSBool RichText) - { - mDNSu32 val = 0, multiplier = 1; - - // Chop closing parentheses from RichText suffix - if (RichText && name->c[0] >= 1 && name->c[name->c[0]] == ')') name->c[0]--; - - // Get any existing numerical suffix off the name - while (mDNSIsDigit(name->c[name->c[0]])) - { val += (name->c[name->c[0]] - '0') * multiplier; multiplier *= 10; name->c[0]--; } - - // Chop opening parentheses or dash from suffix - if (RichText) - { - if (name->c[0] >= 2 && name->c[name->c[0]] == '(' && name->c[name->c[0]-1] == ' ') name->c[0] -= 2; - } - else - { - if (name->c[0] >= 1 && name->c[name->c[0]] == '-') name->c[0] -= 1; - } - - return(val); - } - -// appends a numerical suffix to a label, with the number following a whitespace and enclosed -// in parentheses (rich text) or following two consecutive hyphens (RFC 1034 domain label). -mDNSexport void AppendLabelSuffix(domainlabel *const name, mDNSu32 val, const mDNSBool RichText) - { - mDNSu32 divisor = 1, chars = 2; // Shortest possible RFC1034 name suffix is 2 characters ("-2") - if (RichText) chars = 4; // Shortest possible RichText suffix is 4 characters (" (2)") - - // Truncate trailing spaces from RichText names - if (RichText) while (name->c[name->c[0]] == ' ') name->c[0]--; - - while (divisor < 0xFFFFFFFFUL/10 && val >= divisor * 10) { divisor *= 10; chars++; } - - name->c[0] = (mDNSu8) TruncateUTF8ToLength(name->c+1, name->c[0], MAX_DOMAIN_LABEL - chars); - - if (RichText) { name->c[++name->c[0]] = ' '; name->c[++name->c[0]] = '('; } - else { name->c[++name->c[0]] = '-'; } - - while (divisor) - { - name->c[++name->c[0]] = (mDNSu8)('0' + val / divisor); - val %= divisor; - divisor /= 10; - } - - if (RichText) name->c[++name->c[0]] = ')'; - } - -mDNSexport void IncrementLabelSuffix(domainlabel *name, mDNSBool RichText) - { - mDNSu32 val = 0; - - if (LabelContainsSuffix(name, RichText)) - val = RemoveLabelSuffix(name, RichText); - - // If no existing suffix, start by renaming "Foo" as "Foo (2)" or "Foo-2" as appropriate. - // If existing suffix in the range 2-9, increment it. - // If we've had ten conflicts already, there are probably too many hosts trying to use the same name, - // so add a random increment to improve the chances of finding an available name next time. - if (val == 0) val = 2; - else if (val < 10) val++; - else val += 1 + mDNSRandom(99); - - AppendLabelSuffix(name, val, RichText); - } - -// *************************************************************************** -#if COMPILER_LIKES_PRAGMA_MARK -#pragma mark - -#pragma mark - Resource Record Utility Functions -#endif - -// Set up a AuthRecord with sensible default values. -// These defaults may be overwritten with new values before mDNS_Register is called -mDNSexport void mDNS_SetupResourceRecord(AuthRecord *rr, RData *RDataStorage, mDNSInterfaceID InterfaceID, - mDNSu16 rrtype, mDNSu32 ttl, mDNSu8 RecordType, AuthRecType artype, mDNSRecordCallback Callback, void *Context) - { - // - // LocalOnly auth record can be created with LocalOnly InterfaceID or a valid InterfaceID. - // Most of the applications normally create with LocalOnly InterfaceID and we store them as - // such, so that we can deliver the response to questions that specify LocalOnly InterfaceID. - // LocalOnly resource records can also be created with valid InterfaceID which happens today - // when we create LocalOnly records for /etc/hosts. - - if (InterfaceID == mDNSInterface_LocalOnly && artype != AuthRecordLocalOnly) - { - LogMsg("mDNS_SetupResourceRecord: ERROR!! Mismatch LocalOnly record InterfaceID %p called with artype %d", InterfaceID, artype); - return; - } - else if (InterfaceID == mDNSInterface_P2P && artype != AuthRecordP2P) - { - LogMsg("mDNS_SetupResourceRecord: ERROR!! Mismatch P2P record InterfaceID %p called with artype %d", InterfaceID, artype); - return; - } - else if (!InterfaceID && (artype == AuthRecordP2P || artype == AuthRecordLocalOnly)) - { - LogMsg("mDNS_SetupResourceRecord: ERROR!! Mismatch InterfaceAny record InterfaceID %p called with artype %d", InterfaceID, artype); - return; - } - - // Don't try to store a TTL bigger than we can represent in platform time units - if (ttl > 0x7FFFFFFFUL / mDNSPlatformOneSecond) - ttl = 0x7FFFFFFFUL / mDNSPlatformOneSecond; - else if (ttl == 0) // And Zero TTL is illegal - ttl = DefaultTTLforRRType(rrtype); - - // Field Group 1: The actual information pertaining to this resource record - rr->resrec.RecordType = RecordType; - rr->resrec.InterfaceID = InterfaceID; - rr->resrec.name = &rr->namestorage; - rr->resrec.rrtype = rrtype; - rr->resrec.rrclass = kDNSClass_IN; - rr->resrec.rroriginalttl = ttl; - rr->resrec.rDNSServer = mDNSNULL; -// rr->resrec.rdlength = MUST set by client and/or in mDNS_Register_internal -// rr->resrec.rdestimate = set in mDNS_Register_internal -// rr->resrec.rdata = MUST be set by client - - if (RDataStorage) - rr->resrec.rdata = RDataStorage; - else - { - rr->resrec.rdata = &rr->rdatastorage; - rr->resrec.rdata->MaxRDLength = sizeof(RDataBody); - } - - // Field Group 2: Persistent metadata for Authoritative Records - rr->Additional1 = mDNSNULL; - rr->Additional2 = mDNSNULL; - rr->DependentOn = mDNSNULL; - rr->RRSet = mDNSNULL; - rr->RecordCallback = Callback; - rr->RecordContext = Context; - - rr->AutoTarget = Target_Manual; - rr->AllowRemoteQuery = mDNSfalse; - rr->ForceMCast = mDNSfalse; - - rr->WakeUp = zeroOwner; - rr->AddressProxy = zeroAddr; - rr->TimeRcvd = 0; - rr->TimeExpire = 0; - rr->ARType = artype; - - // Field Group 3: Transient state for Authoritative Records (set in mDNS_Register_internal) - // Field Group 4: Transient uDNS state for Authoritative Records (set in mDNS_Register_internal) - - // For now, until the uDNS code is fully integrated, it's helpful to zero the uDNS state fields here too, just in case - // (e.g. uDNS_RegisterService short-circuits the usual mDNS_Register_internal record registration calls, so a bunch - // of fields don't get set up properly. In particular, if we don't zero rr->QueuedRData then the uDNS code crashes.) - rr->state = regState_Zero; - rr->uselease = 0; - rr->expire = 0; - rr->Private = 0; - rr->updateid = zeroID; - rr->zone = rr->resrec.name; - rr->nta = mDNSNULL; - rr->tcp = mDNSNULL; - rr->OrigRData = 0; - rr->OrigRDLen = 0; - rr->InFlightRData = 0; - rr->InFlightRDLen = 0; - rr->QueuedRData = 0; - rr->QueuedRDLen = 0; - mDNSPlatformMemZero(&rr->NATinfo, sizeof(rr->NATinfo)); - rr->SRVChanged = mDNSfalse; - rr->mState = mergeState_Zero; - - rr->namestorage.c[0] = 0; // MUST be set by client before calling mDNS_Register() - } - -mDNSexport void mDNS_SetupQuestion(DNSQuestion *const q, const mDNSInterfaceID InterfaceID, const domainname *const name, - const mDNSu16 qtype, mDNSQuestionCallback *const callback, void *const context) - { - q->InterfaceID = InterfaceID; - q->Target = zeroAddr; - AssignDomainName(&q->qname, name); - q->qtype = qtype; - q->qclass = kDNSClass_IN; - q->LongLived = (qtype == kDNSType_PTR); - q->ExpectUnique = (qtype != kDNSType_PTR); - q->ForceMCast = mDNSfalse; - q->ReturnIntermed = mDNSfalse; - q->SuppressUnusable = mDNSfalse; - q->SearchListIndex = 0; - q->AppendSearchDomains = 0; - q->RetryWithSearchDomains = mDNSfalse; - q->TimeoutQuestion = 0; - q->WakeOnResolve = 0; - q->qnameOrig = mDNSNULL; - q->QuestionCallback = callback; - q->QuestionContext = context; - } - -mDNSexport mDNSu32 RDataHashValue(const ResourceRecord *const rr) - { - int len = rr->rdlength; - const RDataBody2 *const rdb = (RDataBody2 *)rr->rdata->u.data; - switch(rr->rrtype) - { - case kDNSType_NS: - case kDNSType_CNAME: - case kDNSType_PTR: - case kDNSType_DNAME: return DomainNameHashValue(&rdb->name); - - case kDNSType_SOA: return rdb->soa.serial + - rdb->soa.refresh + - rdb->soa.retry + - rdb->soa.expire + - rdb->soa.min + - DomainNameHashValue(&rdb->soa.mname) + - DomainNameHashValue(&rdb->soa.rname); - - case kDNSType_MX: - case kDNSType_AFSDB: - case kDNSType_RT: - case kDNSType_KX: return DomainNameHashValue(&rdb->mx.exchange); - - case kDNSType_RP: return DomainNameHashValue(&rdb->rp.mbox) + DomainNameHashValue(&rdb->rp.txt); - - case kDNSType_PX: return DomainNameHashValue(&rdb->px.map822) + DomainNameHashValue(&rdb->px.mapx400); - - case kDNSType_SRV: return DomainNameHashValue(&rdb->srv.target); - - case kDNSType_OPT: return 0; // OPT is a pseudo-RR container structure; makes no sense to compare - - case kDNSType_NSEC: len = sizeof(rdataNSEC); // Use in-memory length of 32, and fall through default checksum computation below - - default: - { - mDNSu32 sum = 0; - int i; - for (i=0; i+1 < len; i+=2) - { - sum += (((mDNSu32)(rdb->data[i])) << 8) | rdb->data[i+1]; - sum = (sum<<3) | (sum>>29); - } - if (i < len) - { - sum += ((mDNSu32)(rdb->data[i])) << 8; - } - return(sum); - } - } - } - -// r1 has to be a full ResourceRecord including rrtype and rdlength -// r2 is just a bare RDataBody, which MUST be the same rrtype and rdlength as r1 -mDNSexport mDNSBool SameRDataBody(const ResourceRecord *const r1, const RDataBody *const r2, DomainNameComparisonFn *samename) - { - const RDataBody2 *const b1 = (RDataBody2 *)r1->rdata->u.data; - const RDataBody2 *const b2 = (RDataBody2 *)r2; - switch(r1->rrtype) - { - case kDNSType_NS: - case kDNSType_CNAME: - case kDNSType_PTR: - case kDNSType_DNAME:return(SameDomainName(&b1->name, &b2->name)); - - case kDNSType_SOA: return(mDNSBool)( b1->soa.serial == b2->soa.serial && - b1->soa.refresh == b2->soa.refresh && - b1->soa.retry == b2->soa.retry && - b1->soa.expire == b2->soa.expire && - b1->soa.min == b2->soa.min && - samename(&b1->soa.mname, &b2->soa.mname) && - samename(&b1->soa.rname, &b2->soa.rname)); - - case kDNSType_MX: - case kDNSType_AFSDB: - case kDNSType_RT: - case kDNSType_KX: return(mDNSBool)( b1->mx.preference == b2->mx.preference && - samename(&b1->mx.exchange, &b2->mx.exchange)); - - case kDNSType_RP: return(mDNSBool)( samename(&b1->rp.mbox, &b2->rp.mbox) && - samename(&b1->rp.txt, &b2->rp.txt)); - - case kDNSType_PX: return(mDNSBool)( b1->px.preference == b2->px.preference && - samename(&b1->px.map822, &b2->px.map822) && - samename(&b1->px.mapx400, &b2->px.mapx400)); - - case kDNSType_SRV: return(mDNSBool)( b1->srv.priority == b2->srv.priority && - b1->srv.weight == b2->srv.weight && - mDNSSameIPPort(b1->srv.port, b2->srv.port) && - samename(&b1->srv.target, &b2->srv.target)); - - case kDNSType_OPT: return mDNSfalse; // OPT is a pseudo-RR container structure; makes no sense to compare - - case kDNSType_NSEC: return(mDNSPlatformMemSame(b1->data, b2->data, sizeof(rdataNSEC))); - - default: return(mDNSPlatformMemSame(b1->data, b2->data, r1->rdlength)); - } - } - -// ResourceRecordAnswersQuestion returns mDNStrue if the given resource record is a valid answer to the given question. -// SameNameRecordAnswersQuestion is the same, except it skips the expensive SameDomainName() call. -// SameDomainName() is generally cheap when the names don't match, but expensive when they do match, -// because it has to check all the way to the end of the names to be sure. -// In cases where we know in advance that the names match it's especially advantageous to skip the -// SameDomainName() call because that's precisely the time when it's most expensive and least useful. - -mDNSexport mDNSBool SameNameRecordAnswersQuestion(const ResourceRecord *const rr, const DNSQuestion *const q) - { - // LocalOnly/P2P questions can be answered with AuthRecordAny in this function. LocalOnly/P2P records - // are handled in LocalOnlyRecordAnswersQuestion - if ((rr->InterfaceID == mDNSInterface_LocalOnly) || (rr->InterfaceID == mDNSInterface_P2P)) - { - LogMsg("SameNameRecordAnswersQuestion: ERROR!! called with LocalOnly ResourceRecord %p, Question %p", rr->InterfaceID, q->InterfaceID); - return mDNSfalse; - } - if (rr->InterfaceID && - q ->InterfaceID && q->InterfaceID != mDNSInterface_LocalOnly && - rr->InterfaceID != q->InterfaceID) return(mDNSfalse); - - // Resource record received via unicast, the DNSServer entries should match ? - if (!rr->InterfaceID && rr->rDNSServer != q->qDNSServer) return(mDNSfalse); - - // If ResourceRecord received via multicast, but question was unicast, then shouldn't use record to answer this question - if (rr->InterfaceID && !mDNSOpaque16IsZero(q->TargetQID)) return(mDNSfalse); - - // RR type CNAME matches any query type. QTYPE ANY matches any RR type. QCLASS ANY matches any RR class. - if (!RRTypeAnswersQuestionType(rr,q->qtype)) return(mDNSfalse); - if (rr->rrclass != q->qclass && q->qclass != kDNSQClass_ANY) return(mDNSfalse); - - return(mDNStrue); - } - -mDNSexport mDNSBool ResourceRecordAnswersQuestion(const ResourceRecord *const rr, const DNSQuestion *const q) - { - // LocalOnly/P2P questions can be answered with AuthRecordAny in this function. LocalOnly/P2P records - // are handled in LocalOnlyRecordAnswersQuestion - if ((rr->InterfaceID == mDNSInterface_LocalOnly) || (rr->InterfaceID == mDNSInterface_P2P)) - { - LogMsg("ResourceRecordAnswersQuestion: ERROR!! called with LocalOnly/P2P ResourceRecord %p, Question %p", rr->InterfaceID, q->InterfaceID); - return mDNSfalse; - } - - if (rr->InterfaceID && - q ->InterfaceID && q->InterfaceID != mDNSInterface_LocalOnly && - rr->InterfaceID != q->InterfaceID) return(mDNSfalse); - - // Resource record received via unicast, the DNSServer entries should match ? - if (!rr->InterfaceID && rr->rDNSServer != q->qDNSServer) return(mDNSfalse); - - // If ResourceRecord received via multicast, but question was unicast, then shouldn't use record to answer this question. - if (rr->InterfaceID && !mDNSOpaque16IsZero(q->TargetQID)) return(mDNSfalse); - - // RR type CNAME matches any query type. QTYPE ANY matches any RR type. QCLASS ANY matches any RR class. - if (!RRTypeAnswersQuestionType(rr,q->qtype)) return(mDNSfalse); - if (rr->rrclass != q->qclass && q->qclass != kDNSQClass_ANY) return(mDNSfalse); - - return(rr->namehash == q->qnamehash && SameDomainName(rr->name, &q->qname)); - } - -// We have a separate function to handle LocalOnly AuthRecords because they can be created with -// a valid InterfaceID (e.g., scoped /etc/hosts) and can be used to answer unicast questions unlike -// multicast resource records (which has a valid InterfaceID) which can't be used to answer -// unicast questions. ResourceRecordAnswersQuestion/SameNameRecordAnswersQuestion can't tell whether -// a resource record is multicast or LocalOnly by just looking at the ResourceRecord because -// LocalOnly records are truly identified by ARType in the AuthRecord. As P2P and LocalOnly record -// are kept in the same hash table, we use the same function to make it easy for the callers when -// they walk the hash table to answer LocalOnly/P2P questions -// -mDNSexport mDNSBool LocalOnlyRecordAnswersQuestion(AuthRecord *const ar, const DNSQuestion *const q) - { - ResourceRecord *rr = &ar->resrec; - - // mDNSInterface_Any questions can be answered with LocalOnly/P2P records in this function. AuthRecord_Any - // records are handled in ResourceRecordAnswersQuestion/SameNameRecordAnswersQuestion - if (RRAny(ar)) - { - LogMsg("LocalOnlyRecordAnswersQuestion: ERROR!! called with regular AuthRecordAny %##s", rr->name->c); - return mDNSfalse; - } - - // Questions with mDNSInterface_LocalOnly InterfaceID should be answered with all resource records that are - // *local* to the machine. These include resource records that have InterfaceID set to mDNSInterface_LocalOnly, - // mDNSInterface_Any and any other real InterfaceID. Hence, LocalOnly questions should not be checked against - // the InterfaceID in the resource record. - // - // mDNSInterface_Unicast does not indicate any scope and hence treat them like mDNSInterface_Any. - - if (rr->InterfaceID && - q->InterfaceID && q->InterfaceID != mDNSInterface_LocalOnly && q->InterfaceID != mDNSInterface_Unicast && - rr->InterfaceID != q->InterfaceID) return(mDNSfalse); - - // Entries in /etc/hosts are added as LocalOnly resource records. The LocalOnly resource records - // may have a scope e.g., fe80::1%en0. The question may be scoped or not: the InterfaceID may be set - // to mDNSInterface_Any, mDNSInterface_LocalOnly or a real InterfaceID (scoped). - // - // 1) Question: Any, LocalOnly Record: no scope. This question should be answered with this record. - // - // 2) Question: Any, LocalOnly Record: scoped. This question should be answered with the record because - // traditionally applications never specify scope e.g., getaddrinfo, but need to be able - // to get to /etc/hosts entries. - // - // 3) Question: Scoped (LocalOnly or InterfaceID), LocalOnly Record: no scope. This is the inverse of (2). - // If we register a LocalOnly record, we need to answer a LocalOnly question. If the /etc/hosts has a - // non scoped entry, it may not make sense to answer a scoped question. But we can't tell these two - // cases apart. As we currently answer LocalOnly question with LocalOnly record, we continue to do so. - // - // 4) Question: Scoped (LocalOnly or InterfaceID), LocalOnly Record: scoped. LocalOnly questions should be - // answered with any resource record where as if it has a valid InterfaceID, the scope should match. - // - // (1) and (2) is bypassed because we check for a non-NULL InterfaceID above. For (3), the InterfaceID is NULL - // and hence bypassed above. For (4) we bypassed LocalOnly questions and checked the scope of the record - // against the question. - // - // For P2P, InterfaceIDs of the question and the record should match. - - // If ResourceRecord received via multicast, but question was unicast, then shouldn't use record to answer this question. - // LocalOnly authoritative answers are exempt. LocalOnly authoritative answers are used for /etc/host entries. - // We don't want a local process to be able to create a fake LocalOnly address record for "www.bigbank.com" which would then - // cause other applications (e.g. Safari) to connect to the wrong address. The rpc to register records filters out records - // with names that don't end in local and have mDNSInterface_LocalOnly set. - // - // Note: The check is bypassed for LocalOnly and for P2P it is not needed as only .local records are registered and for - // a question to match its names, it also has to end in .local and that question can't be a unicast question (See - // Question_uDNS macro and its usage). As P2P does not enforce .local only registrations we still make this check - // and also makes it future proof. - - if (ar->ARType != AuthRecordLocalOnly && rr->InterfaceID && !mDNSOpaque16IsZero(q->TargetQID)) return(mDNSfalse); - - // RR type CNAME matches any query type. QTYPE ANY matches any RR type. QCLASS ANY matches any RR class. - if (!RRTypeAnswersQuestionType(rr,q->qtype)) return(mDNSfalse); - if (rr->rrclass != q->qclass && q->qclass != kDNSQClass_ANY) return(mDNSfalse); - - return(rr->namehash == q->qnamehash && SameDomainName(rr->name, &q->qname)); - } - -mDNSexport mDNSBool AnyTypeRecordAnswersQuestion(const ResourceRecord *const rr, const DNSQuestion *const q) - { - // LocalOnly/P2P questions can be answered with AuthRecordAny in this function. LocalOnly/P2P records - // are handled in LocalOnlyRecordAnswersQuestion - if ((rr->InterfaceID == mDNSInterface_LocalOnly) || (rr->InterfaceID == mDNSInterface_P2P)) - { - LogMsg("AnyTypeRecordAnswersQuestion: ERROR!! called with LocalOnly ResourceRecord %p, Question %p", rr->InterfaceID, q->InterfaceID); - return mDNSfalse; - } - if (rr->InterfaceID && - q ->InterfaceID && q->InterfaceID != mDNSInterface_LocalOnly && - rr->InterfaceID != q->InterfaceID) return(mDNSfalse); - - // Resource record received via unicast, the DNSServer entries should match ? - // Note that Auth Records are normally setup with NULL InterfaceID and - // both the DNSServers are assumed to be NULL in that case - if (!rr->InterfaceID && rr->rDNSServer != q->qDNSServer) return(mDNSfalse); - - // If ResourceRecord received via multicast, but question was unicast, then shouldn't use record to answer this question - if (rr->InterfaceID && !mDNSOpaque16IsZero(q->TargetQID)) return(mDNSfalse); - - if (rr->rrclass != q->qclass && q->qclass != kDNSQClass_ANY) return(mDNSfalse); - - return(rr->namehash == q->qnamehash && SameDomainName(rr->name, &q->qname)); - } - -// This is called with both unicast resource record and multicast resource record. The question that -// received the unicast response could be the regular unicast response from a DNS server or a response -// to a mDNS QU query. The main reason we need this function is that we can't compare DNSServers between the -// question and the resource record because the resource record is not completely initialized in -// mDNSCoreReceiveResponse when this function is called. -mDNSexport mDNSBool ResourceRecordAnswersUnicastResponse(const ResourceRecord *const rr, const DNSQuestion *const q) - { - // For resource records created using multicast, the InterfaceIDs have to match - if (rr->InterfaceID && - q->InterfaceID && rr->InterfaceID != q->InterfaceID) return(mDNSfalse); - - // If ResourceRecord received via multicast, but question was unicast, then shouldn't use record to answer this question. - if (rr->InterfaceID && !mDNSOpaque16IsZero(q->TargetQID)) return(mDNSfalse); - - // RR type CNAME matches any query type. QTYPE ANY matches any RR type. QCLASS ANY matches any RR class. - if (!RRTypeAnswersQuestionType(rr,q->qtype)) return(mDNSfalse); - - if (rr->rrclass != q->qclass && q->qclass != kDNSQClass_ANY) return(mDNSfalse); - - return(rr->namehash == q->qnamehash && SameDomainName(rr->name, &q->qname)); - } - -mDNSexport mDNSu16 GetRDLength(const ResourceRecord *const rr, mDNSBool estimate) - { - const RDataBody2 *const rd = (RDataBody2 *)rr->rdata->u.data; - const domainname *const name = estimate ? rr->name : mDNSNULL; - if (rr->rrclass == kDNSQClass_ANY) return(rr->rdlength); // Used in update packets to mean "Delete An RRset" (RFC 2136) - else switch (rr->rrtype) - { - case kDNSType_A: return(sizeof(rd->ipv4)); - - case kDNSType_NS: - case kDNSType_CNAME: - case kDNSType_PTR: - case kDNSType_DNAME:return(CompressedDomainNameLength(&rd->name, name)); - - case kDNSType_SOA: return(mDNSu16)(CompressedDomainNameLength(&rd->soa.mname, name) + - CompressedDomainNameLength(&rd->soa.rname, name) + - 5 * sizeof(mDNSOpaque32)); - - case kDNSType_NULL: - case kDNSType_TSIG: - case kDNSType_TXT: - case kDNSType_X25: - case kDNSType_ISDN: - case kDNSType_LOC: - case kDNSType_DHCID:return(rr->rdlength); // Not self-describing, so have to just trust rdlength - - case kDNSType_HINFO:return(mDNSu16)(2 + (int)rd->data[0] + (int)rd->data[1 + (int)rd->data[0]]); - - case kDNSType_MX: - case kDNSType_AFSDB: - case kDNSType_RT: - case kDNSType_KX: return(mDNSu16)(2 + CompressedDomainNameLength(&rd->mx.exchange, name)); - - case kDNSType_RP: return(mDNSu16)(CompressedDomainNameLength(&rd->rp.mbox, name) + - CompressedDomainNameLength(&rd->rp.txt, name)); - - case kDNSType_PX: return(mDNSu16)(2 + CompressedDomainNameLength(&rd->px.map822, name) + - CompressedDomainNameLength(&rd->px.mapx400, name)); - - case kDNSType_AAAA: return(sizeof(rd->ipv6)); - - case kDNSType_SRV: return(mDNSu16)(6 + CompressedDomainNameLength(&rd->srv.target, name)); - - case kDNSType_OPT: return(rr->rdlength); - - case kDNSType_NSEC: { - int i; - for (i=sizeof(rdataNSEC); i>0; i--) if (rd->nsec.bitmap[i-1]) break; - // For our simplified use of NSEC synthetic records: - // nextname is always the record's own name, - // and if we have at least one record type that exists, - // - the block number is always 0, - // - the count byte is a value in the range 1-32, - // - followed by the 1-32 data bytes - return(mDNSu16)((estimate ? 2 : DomainNameLength(rr->name)) + (i ? (2 + i) : 0)); - } - - default: debugf("Warning! Don't know how to get length of resource type %d", rr->rrtype); - return(rr->rdlength); - } - } - -// When a local client registers (or updates) a record, we use this routine to do some simple validation checks -// to help reduce the risk of bogus malformed data on the network -mDNSexport mDNSBool ValidateRData(const mDNSu16 rrtype, const mDNSu16 rdlength, const RData *const rd) - { - mDNSu16 len; - - switch(rrtype) - { - case kDNSType_A: return(rdlength == sizeof(mDNSv4Addr)); - - case kDNSType_NS: // Same as PTR - case kDNSType_MD: // Same as PTR - case kDNSType_MF: // Same as PTR - case kDNSType_CNAME:// Same as PTR - //case kDNSType_SOA not checked - case kDNSType_MB: // Same as PTR - case kDNSType_MG: // Same as PTR - case kDNSType_MR: // Same as PTR - //case kDNSType_NULL not checked (no specified format, so always valid) - //case kDNSType_WKS not checked - case kDNSType_PTR: len = DomainNameLengthLimit(&rd->u.name, rd->u.data + rdlength); - return(len <= MAX_DOMAIN_NAME && rdlength == len); - - case kDNSType_HINFO:// Same as TXT (roughly) - case kDNSType_MINFO:// Same as TXT (roughly) - case kDNSType_TXT: if (!rdlength) return(mDNSfalse); // TXT record has to be at least one byte (RFC 1035) - { - const mDNSu8 *ptr = rd->u.txt.c; - const mDNSu8 *end = rd->u.txt.c + rdlength; - while (ptr < end) ptr += 1 + ptr[0]; - return (ptr == end); - } - - case kDNSType_AAAA: return(rdlength == sizeof(mDNSv6Addr)); - - case kDNSType_MX: // Must be at least two-byte preference, plus domainname - // Call to DomainNameLengthLimit() implicitly enforces both requirements for us - len = DomainNameLengthLimit(&rd->u.mx.exchange, rd->u.data + rdlength); - return(len <= MAX_DOMAIN_NAME && rdlength == 2+len); - - case kDNSType_SRV: // Must be at least priority+weight+port, plus domainname - // Call to DomainNameLengthLimit() implicitly enforces both requirements for us - len = DomainNameLengthLimit(&rd->u.srv.target, rd->u.data + rdlength); - return(len <= MAX_DOMAIN_NAME && rdlength == 6+len); - - //case kDNSType_NSEC not checked - - default: return(mDNStrue); // Allow all other types without checking - } - } - -// *************************************************************************** -#if COMPILER_LIKES_PRAGMA_MARK -#pragma mark - -#pragma mark - DNS Message Creation Functions -#endif - -mDNSexport void InitializeDNSMessage(DNSMessageHeader *h, mDNSOpaque16 id, mDNSOpaque16 flags) - { - h->id = id; - h->flags = flags; - h->numQuestions = 0; - h->numAnswers = 0; - h->numAuthorities = 0; - h->numAdditionals = 0; - } - -mDNSexport const mDNSu8 *FindCompressionPointer(const mDNSu8 *const base, const mDNSu8 *const end, const mDNSu8 *const domname) - { - const mDNSu8 *result = end - *domname - 1; - - if (*domname == 0) return(mDNSNULL); // There's no point trying to match just the root label - - // This loop examines each possible starting position in packet, starting end of the packet and working backwards - while (result >= base) - { - // If the length byte and first character of the label match, then check further to see - // if this location in the packet will yield a useful name compression pointer. - if (result[0] == domname[0] && result[1] == domname[1]) - { - const mDNSu8 *name = domname; - const mDNSu8 *targ = result; - while (targ + *name < end) - { - // First see if this label matches - int i; - const mDNSu8 *pointertarget; - for (i=0; i <= *name; i++) if (targ[i] != name[i]) break; - if (i <= *name) break; // If label did not match, bail out - targ += 1 + *name; // Else, did match, so advance target pointer - name += 1 + *name; // and proceed to check next label - if (*name == 0 && *targ == 0) return(result); // If no more labels, we found a match! - if (*name == 0) break; // If no more labels to match, we failed, so bail out - - // The label matched, so now follow the pointer (if appropriate) and then see if the next label matches - if (targ[0] < 0x40) continue; // If length value, continue to check next label - if (targ[0] < 0xC0) break; // If 40-BF, not valid - if (targ+1 >= end) break; // Second byte not present! - pointertarget = base + (((mDNSu16)(targ[0] & 0x3F)) << 8) + targ[1]; - if (targ < pointertarget) break; // Pointertarget must point *backwards* in the packet - if (pointertarget[0] >= 0x40) break; // Pointertarget must point to a valid length byte - targ = pointertarget; - } - } - result--; // We failed to match at this search position, so back up the tentative result pointer and try again - } - return(mDNSNULL); - } - -// Put a string of dot-separated labels as length-prefixed labels -// domainname is a fully-qualified name (i.e. assumed to be ending in a dot, even if it doesn't) -// msg points to the message we're building (pass mDNSNULL if we don't want to use compression pointers) -// end points to the end of the message so far -// ptr points to where we want to put the name -// limit points to one byte past the end of the buffer that we must not overrun -// domainname is the name to put -mDNSexport mDNSu8 *putDomainNameAsLabels(const DNSMessage *const msg, - mDNSu8 *ptr, const mDNSu8 *const limit, const domainname *const name) - { - const mDNSu8 *const base = (const mDNSu8 *)msg; - const mDNSu8 * np = name->c; - const mDNSu8 *const max = name->c + MAX_DOMAIN_NAME; // Maximum that's valid - const mDNSu8 * pointer = mDNSNULL; - const mDNSu8 *const searchlimit = ptr; - - if (!ptr) { LogMsg("putDomainNameAsLabels %##s ptr is null", name->c); return(mDNSNULL); } - - if (!*np) // If just writing one-byte root label, make sure we have space for that - { - if (ptr >= limit) return(mDNSNULL); - } - else // else, loop through writing labels and/or a compression offset - { - do { - if (*np > MAX_DOMAIN_LABEL) - { LogMsg("Malformed domain name %##s (label more than 63 bytes)", name->c); return(mDNSNULL); } - - // This check correctly allows for the final trailing root label: - // e.g. - // Suppose our domain name is exactly 256 bytes long, including the final trailing root label. - // Suppose np is now at name->c[249], and we're about to write our last non-null label ("local"). - // We know that max will be at name->c[256] - // That means that np + 1 + 5 == max - 1, so we (just) pass the "if" test below, write our - // six bytes, then exit the loop, write the final terminating root label, and the domain - // name we've written is exactly 256 bytes long, exactly at the correct legal limit. - // If the name is one byte longer, then we fail the "if" test below, and correctly bail out. - if (np + 1 + *np >= max) - { LogMsg("Malformed domain name %##s (more than 256 bytes)", name->c); return(mDNSNULL); } - - if (base) pointer = FindCompressionPointer(base, searchlimit, np); - if (pointer) // Use a compression pointer if we can - { - const mDNSu16 offset = (mDNSu16)(pointer - base); - if (ptr+2 > limit) return(mDNSNULL); // If we don't have two bytes of space left, give up - *ptr++ = (mDNSu8)(0xC0 | (offset >> 8)); - *ptr++ = (mDNSu8)( offset & 0xFF); - return(ptr); - } - else // Else copy one label and try again - { - int i; - mDNSu8 len = *np++; - // If we don't at least have enough space for this label *plus* a terminating zero on the end, give up - if (ptr + 1 + len >= limit) return(mDNSNULL); - *ptr++ = len; - for (i=0; i> 8 ) & 0xFF); - ptr[1] = (mDNSu8)((val ) & 0xFF); - return ptr + sizeof(mDNSOpaque16); - } - -mDNSlocal mDNSu8 *putVal32(mDNSu8 *ptr, mDNSu32 val) - { - ptr[0] = (mDNSu8)((val >> 24) & 0xFF); - ptr[1] = (mDNSu8)((val >> 16) & 0xFF); - ptr[2] = (mDNSu8)((val >> 8) & 0xFF); - ptr[3] = (mDNSu8)((val ) & 0xFF); - return ptr + sizeof(mDNSu32); - } - -// msg points to the message we're building (pass mDNSNULL if we don't want to use compression pointers) -mDNSexport mDNSu8 *putRData(const DNSMessage *const msg, mDNSu8 *ptr, const mDNSu8 *const limit, const ResourceRecord *const rr) - { - const RDataBody2 *const rdb = (RDataBody2 *)rr->rdata->u.data; - switch (rr->rrtype) - { - case kDNSType_A: if (rr->rdlength != 4) - { debugf("putRData: Illegal length %d for kDNSType_A", rr->rdlength); return(mDNSNULL); } - if (ptr + 4 > limit) return(mDNSNULL); - *ptr++ = rdb->ipv4.b[0]; - *ptr++ = rdb->ipv4.b[1]; - *ptr++ = rdb->ipv4.b[2]; - *ptr++ = rdb->ipv4.b[3]; - return(ptr); - - case kDNSType_NS: - case kDNSType_CNAME: - case kDNSType_PTR: - case kDNSType_DNAME:return(putDomainNameAsLabels(msg, ptr, limit, &rdb->name)); - - case kDNSType_SOA: ptr = putDomainNameAsLabels(msg, ptr, limit, &rdb->soa.mname); - if (!ptr) return(mDNSNULL); - ptr = putDomainNameAsLabels(msg, ptr, limit, &rdb->soa.rname); - if (!ptr || ptr + 20 > limit) return(mDNSNULL); - ptr = putVal32(ptr, rdb->soa.serial); - ptr = putVal32(ptr, rdb->soa.refresh); - ptr = putVal32(ptr, rdb->soa.retry); - ptr = putVal32(ptr, rdb->soa.expire); - ptr = putVal32(ptr, rdb->soa.min); - return(ptr); - - case kDNSType_NULL: - case kDNSType_HINFO: - case kDNSType_TSIG: - case kDNSType_TXT: - case kDNSType_X25: - case kDNSType_ISDN: - case kDNSType_LOC: - case kDNSType_DHCID:if (ptr + rr->rdlength > limit) return(mDNSNULL); - mDNSPlatformMemCopy(ptr, rdb->data, rr->rdlength); - return(ptr + rr->rdlength); - - case kDNSType_MX: - case kDNSType_AFSDB: - case kDNSType_RT: - case kDNSType_KX: if (ptr + 3 > limit) return(mDNSNULL); - ptr = putVal16(ptr, rdb->mx.preference); - return(putDomainNameAsLabels(msg, ptr, limit, &rdb->mx.exchange)); - - case kDNSType_RP: ptr = putDomainNameAsLabels(msg, ptr, limit, &rdb->rp.mbox); - if (!ptr) return(mDNSNULL); - ptr = putDomainNameAsLabels(msg, ptr, limit, &rdb->rp.txt); - return(ptr); - - case kDNSType_PX: if (ptr + 5 > limit) return(mDNSNULL); - ptr = putVal16(ptr, rdb->px.preference); - ptr = putDomainNameAsLabels(msg, ptr, limit, &rdb->px.map822); - if (!ptr) return(mDNSNULL); - ptr = putDomainNameAsLabels(msg, ptr, limit, &rdb->px.mapx400); - return(ptr); - - case kDNSType_AAAA: if (rr->rdlength != sizeof(rdb->ipv6)) - { debugf("putRData: Illegal length %d for kDNSType_AAAA", rr->rdlength); return(mDNSNULL); } - if (ptr + sizeof(rdb->ipv6) > limit) return(mDNSNULL); - mDNSPlatformMemCopy(ptr, &rdb->ipv6, sizeof(rdb->ipv6)); - return(ptr + sizeof(rdb->ipv6)); - - case kDNSType_SRV: if (ptr + 7 > limit) return(mDNSNULL); - *ptr++ = (mDNSu8)(rdb->srv.priority >> 8); - *ptr++ = (mDNSu8)(rdb->srv.priority & 0xFF); - *ptr++ = (mDNSu8)(rdb->srv.weight >> 8); - *ptr++ = (mDNSu8)(rdb->srv.weight & 0xFF); - *ptr++ = rdb->srv.port.b[0]; - *ptr++ = rdb->srv.port.b[1]; - return(putDomainNameAsLabels(msg, ptr, limit, &rdb->srv.target)); - - case kDNSType_OPT: { - int len = 0; - const rdataOPT *opt; - const rdataOPT *const end = (const rdataOPT *)&rr->rdata->u.data[rr->rdlength]; - for (opt = &rr->rdata->u.opt[0]; opt < end; opt++) len += DNSOpt_Data_Space(opt); - if (ptr + len > limit) { LogMsg("ERROR: putOptRData - out of space"); return mDNSNULL; } - - for (opt = &rr->rdata->u.opt[0]; opt < end; opt++) - { - const int space = DNSOpt_Data_Space(opt); - ptr = putVal16(ptr, opt->opt); - ptr = putVal16(ptr, (mDNSu16)space - 4); - switch (opt->opt) - { - case kDNSOpt_LLQ: - ptr = putVal16(ptr, opt->u.llq.vers); - ptr = putVal16(ptr, opt->u.llq.llqOp); - ptr = putVal16(ptr, opt->u.llq.err); - mDNSPlatformMemCopy(ptr, opt->u.llq.id.b, 8); // 8-byte id - ptr += 8; - ptr = putVal32(ptr, opt->u.llq.llqlease); - break; - case kDNSOpt_Lease: - ptr = putVal32(ptr, opt->u.updatelease); - break; - case kDNSOpt_Owner: - *ptr++ = opt->u.owner.vers; - *ptr++ = opt->u.owner.seq; - mDNSPlatformMemCopy(ptr, opt->u.owner.HMAC.b, 6); // 6-byte Host identifier - ptr += 6; - if (space >= DNSOpt_OwnerData_ID_Wake_Space) - { - mDNSPlatformMemCopy(ptr, opt->u.owner.IMAC.b, 6); // 6-byte interface MAC - ptr += 6; - if (space > DNSOpt_OwnerData_ID_Wake_Space) - { - mDNSPlatformMemCopy(ptr, opt->u.owner.password.b, space - DNSOpt_OwnerData_ID_Wake_Space); - ptr += space - DNSOpt_OwnerData_ID_Wake_Space; - } - } - break; - } - } - return ptr; - } - - case kDNSType_NSEC: { - // For our simplified use of NSEC synthetic records: - // nextname is always the record's own name, - // the block number is always 0, - // the count byte is a value in the range 1-32, - // followed by the 1-32 data bytes - int i, j; - for (i=sizeof(rdataNSEC); i>0; i--) if (rdb->nsec.bitmap[i-1]) break; - ptr = putDomainNameAsLabels(msg, ptr, limit, rr->name); - if (!ptr) return(mDNSNULL); - if (i) // Only put a block if at least one type exists for this name - { - if (ptr + 2 + i > limit) return(mDNSNULL); - *ptr++ = 0; - *ptr++ = (mDNSu8)i; - for (j=0; jnsec.bitmap[j]; - } - return ptr; - } - - default: debugf("putRData: Warning! Writing unknown resource type %d as raw data", rr->rrtype); - if (ptr + rr->rdlength > limit) return(mDNSNULL); - mDNSPlatformMemCopy(ptr, rdb->data, rr->rdlength); - return(ptr + rr->rdlength); - } - } - -#define IsUnicastUpdate(X) (!mDNSOpaque16IsZero((X)->h.id) && ((X)->h.flags.b[0] & kDNSFlag0_OP_Mask) == kDNSFlag0_OP_Update) - -mDNSexport mDNSu8 *PutResourceRecordTTLWithLimit(DNSMessage *const msg, mDNSu8 *ptr, mDNSu16 *count, ResourceRecord *rr, mDNSu32 ttl, const mDNSu8 *limit) - { - mDNSu8 *endofrdata; - mDNSu16 actualLength; - // When sending SRV to conventional DNS server (i.e. in DNS update requests) we should not do name compression on the rdata (RFC 2782) - const DNSMessage *const rdatacompressionbase = (IsUnicastUpdate(msg) && rr->rrtype == kDNSType_SRV) ? mDNSNULL : msg; - - if (rr->RecordType == kDNSRecordTypeUnregistered) - { - LogMsg("PutResourceRecord ERROR! Attempt to put kDNSRecordTypeUnregistered %##s (%s)", rr->name->c, DNSTypeName(rr->rrtype)); - return(ptr); - } - - if (!ptr) { LogMsg("PutResourceRecordTTLWithLimit ptr is null"); return(mDNSNULL); } - - ptr = putDomainNameAsLabels(msg, ptr, limit, rr->name); - if (!ptr || ptr + 10 >= limit) return(mDNSNULL); // If we're out-of-space, return mDNSNULL - ptr[0] = (mDNSu8)(rr->rrtype >> 8); - ptr[1] = (mDNSu8)(rr->rrtype & 0xFF); - ptr[2] = (mDNSu8)(rr->rrclass >> 8); - ptr[3] = (mDNSu8)(rr->rrclass & 0xFF); - ptr[4] = (mDNSu8)((ttl >> 24) & 0xFF); - ptr[5] = (mDNSu8)((ttl >> 16) & 0xFF); - ptr[6] = (mDNSu8)((ttl >> 8) & 0xFF); - ptr[7] = (mDNSu8)( ttl & 0xFF); - // ptr[8] and ptr[9] filled in *after* we find out how much space the rdata takes - - endofrdata = putRData(rdatacompressionbase, ptr+10, limit, rr); - if (!endofrdata) { verbosedebugf("Ran out of space in PutResourceRecord for %##s (%s)", rr->name->c, DNSTypeName(rr->rrtype)); return(mDNSNULL); } - - // Go back and fill in the actual number of data bytes we wrote - // (actualLength can be less than rdlength when domain name compression is used) - actualLength = (mDNSu16)(endofrdata - ptr - 10); - ptr[8] = (mDNSu8)(actualLength >> 8); - ptr[9] = (mDNSu8)(actualLength & 0xFF); - - if (count) (*count)++; - else LogMsg("PutResourceRecordTTL: ERROR: No target count to update for %##s (%s)", rr->name->c, DNSTypeName(rr->rrtype)); - return(endofrdata); - } - -mDNSlocal mDNSu8 *putEmptyResourceRecord(DNSMessage *const msg, mDNSu8 *ptr, const mDNSu8 *const limit, mDNSu16 *count, const AuthRecord *rr) - { - ptr = putDomainNameAsLabels(msg, ptr, limit, rr->resrec.name); - if (!ptr || ptr + 10 > limit) return(mDNSNULL); // If we're out-of-space, return mDNSNULL - ptr[0] = (mDNSu8)(rr->resrec.rrtype >> 8); // Put type - ptr[1] = (mDNSu8)(rr->resrec.rrtype & 0xFF); - ptr[2] = (mDNSu8)(rr->resrec.rrclass >> 8); // Put class - ptr[3] = (mDNSu8)(rr->resrec.rrclass & 0xFF); - ptr[4] = ptr[5] = ptr[6] = ptr[7] = 0; // TTL is zero - ptr[8] = ptr[9] = 0; // RDATA length is zero - (*count)++; - return(ptr + 10); - } - -mDNSexport mDNSu8 *putQuestion(DNSMessage *const msg, mDNSu8 *ptr, const mDNSu8 *const limit, const domainname *const name, mDNSu16 rrtype, mDNSu16 rrclass) - { - ptr = putDomainNameAsLabels(msg, ptr, limit, name); - if (!ptr || ptr+4 >= limit) return(mDNSNULL); // If we're out-of-space, return mDNSNULL - ptr[0] = (mDNSu8)(rrtype >> 8); - ptr[1] = (mDNSu8)(rrtype & 0xFF); - ptr[2] = (mDNSu8)(rrclass >> 8); - ptr[3] = (mDNSu8)(rrclass & 0xFF); - msg->h.numQuestions++; - return(ptr+4); - } - -// for dynamic updates -mDNSexport mDNSu8 *putZone(DNSMessage *const msg, mDNSu8 *ptr, mDNSu8 *limit, const domainname *zone, mDNSOpaque16 zoneClass) - { - ptr = putDomainNameAsLabels(msg, ptr, limit, zone); - if (!ptr || ptr + 4 > limit) return mDNSNULL; // If we're out-of-space, return NULL - *ptr++ = (mDNSu8)(kDNSType_SOA >> 8); - *ptr++ = (mDNSu8)(kDNSType_SOA & 0xFF); - *ptr++ = zoneClass.b[0]; - *ptr++ = zoneClass.b[1]; - msg->h.mDNS_numZones++; - return ptr; - } - -// for dynamic updates -mDNSexport mDNSu8 *putPrereqNameNotInUse(const domainname *const name, DNSMessage *const msg, mDNSu8 *const ptr, mDNSu8 *const end) - { - AuthRecord prereq; - mDNS_SetupResourceRecord(&prereq, mDNSNULL, mDNSInterface_Any, kDNSQType_ANY, kStandardTTL, 0, AuthRecordAny, mDNSNULL, mDNSNULL); - AssignDomainName(&prereq.namestorage, name); - prereq.resrec.rrtype = kDNSQType_ANY; - prereq.resrec.rrclass = kDNSClass_NONE; - return putEmptyResourceRecord(msg, ptr, end, &msg->h.mDNS_numPrereqs, &prereq); - } - -// for dynamic updates -mDNSexport mDNSu8 *putDeletionRecord(DNSMessage *msg, mDNSu8 *ptr, ResourceRecord *rr) - { - // deletion: specify record w/ TTL 0, class NONE - const mDNSu16 origclass = rr->rrclass; - rr->rrclass = kDNSClass_NONE; - ptr = PutResourceRecordTTLJumbo(msg, ptr, &msg->h.mDNS_numUpdates, rr, 0); - rr->rrclass = origclass; - return ptr; - } - -// for dynamic updates -mDNSexport mDNSu8 *putDeletionRecordWithLimit(DNSMessage *msg, mDNSu8 *ptr, ResourceRecord *rr, mDNSu8 *limit) - { - // deletion: specify record w/ TTL 0, class NONE - const mDNSu16 origclass = rr->rrclass; - rr->rrclass = kDNSClass_NONE; - ptr = PutResourceRecordTTLWithLimit(msg, ptr, &msg->h.mDNS_numUpdates, rr, 0, limit); - rr->rrclass = origclass; - return ptr; - } - -mDNSexport mDNSu8 *putDeleteRRSetWithLimit(DNSMessage *msg, mDNSu8 *ptr, const domainname *name, mDNSu16 rrtype, mDNSu8 *limit) - { - mDNSu16 class = kDNSQClass_ANY; - - ptr = putDomainNameAsLabels(msg, ptr, limit, name); - if (!ptr || ptr + 10 >= limit) return mDNSNULL; // If we're out-of-space, return mDNSNULL - ptr[0] = (mDNSu8)(rrtype >> 8); - ptr[1] = (mDNSu8)(rrtype & 0xFF); - ptr[2] = (mDNSu8)(class >> 8); - ptr[3] = (mDNSu8)(class & 0xFF); - ptr[4] = ptr[5] = ptr[6] = ptr[7] = 0; // zero ttl - ptr[8] = ptr[9] = 0; // zero rdlength/rdata - - msg->h.mDNS_numUpdates++; - return ptr + 10; - } - -// for dynamic updates -mDNSexport mDNSu8 *putDeleteAllRRSets(DNSMessage *msg, mDNSu8 *ptr, const domainname *name) - { - const mDNSu8 *limit = msg->data + AbsoluteMaxDNSMessageData; - mDNSu16 class = kDNSQClass_ANY; - mDNSu16 rrtype = kDNSQType_ANY; - - ptr = putDomainNameAsLabels(msg, ptr, limit, name); - if (!ptr || ptr + 10 >= limit) return mDNSNULL; // If we're out-of-space, return mDNSNULL - ptr[0] = (mDNSu8)(rrtype >> 8); - ptr[1] = (mDNSu8)(rrtype & 0xFF); - ptr[2] = (mDNSu8)(class >> 8); - ptr[3] = (mDNSu8)(class & 0xFF); - ptr[4] = ptr[5] = ptr[6] = ptr[7] = 0; // zero ttl - ptr[8] = ptr[9] = 0; // zero rdlength/rdata - - msg->h.mDNS_numUpdates++; - return ptr + 10; - } - -// for dynamic updates -mDNSexport mDNSu8 *putUpdateLease(DNSMessage *msg, mDNSu8 *end, mDNSu32 lease) - { - AuthRecord rr; - mDNS_SetupResourceRecord(&rr, mDNSNULL, mDNSInterface_Any, kDNSType_OPT, kStandardTTL, kDNSRecordTypeKnownUnique, AuthRecordAny, mDNSNULL, mDNSNULL); - rr.resrec.rrclass = NormalMaxDNSMessageData; - rr.resrec.rdlength = sizeof(rdataOPT); // One option in this OPT record - rr.resrec.rdestimate = sizeof(rdataOPT); - rr.resrec.rdata->u.opt[0].opt = kDNSOpt_Lease; - rr.resrec.rdata->u.opt[0].u.updatelease = lease; - end = PutResourceRecordTTLJumbo(msg, end, &msg->h.numAdditionals, &rr.resrec, 0); - if (!end) { LogMsg("ERROR: putUpdateLease - PutResourceRecordTTL"); return mDNSNULL; } - return end; - } - -// for dynamic updates -mDNSexport mDNSu8 *putUpdateLeaseWithLimit(DNSMessage *msg, mDNSu8 *end, mDNSu32 lease, mDNSu8 *limit) - { - AuthRecord rr; - mDNS_SetupResourceRecord(&rr, mDNSNULL, mDNSInterface_Any, kDNSType_OPT, kStandardTTL, kDNSRecordTypeKnownUnique, AuthRecordAny, mDNSNULL, mDNSNULL); - rr.resrec.rrclass = NormalMaxDNSMessageData; - rr.resrec.rdlength = sizeof(rdataOPT); // One option in this OPT record - rr.resrec.rdestimate = sizeof(rdataOPT); - rr.resrec.rdata->u.opt[0].opt = kDNSOpt_Lease; - rr.resrec.rdata->u.opt[0].u.updatelease = lease; - end = PutResourceRecordTTLWithLimit(msg, end, &msg->h.numAdditionals, &rr.resrec, 0, limit); - if (!end) { LogMsg("ERROR: putUpdateLease - PutResourceRecordTTLWithLimit"); return mDNSNULL; } - return end; - } - -mDNSexport mDNSu8 *putHINFO(const mDNS *const m, DNSMessage *const msg, mDNSu8 *end, DomainAuthInfo *authInfo, mDNSu8 *limit) - { - if (authInfo && authInfo->AutoTunnel) - { - AuthRecord hinfo; - mDNSu8 *h = hinfo.rdatastorage.u.data; - mDNSu16 len = 2 + m->HIHardware.c[0] + m->HISoftware.c[0]; - mDNSu8 *newptr; - mDNS_SetupResourceRecord(&hinfo, mDNSNULL, mDNSInterface_Any, kDNSType_HINFO, 0, kDNSRecordTypeUnique, AuthRecordAny, mDNSNULL, mDNSNULL); - AppendDomainLabel(&hinfo.namestorage, &m->hostlabel); - AppendDomainName (&hinfo.namestorage, &authInfo->domain); - hinfo.resrec.rroriginalttl = 0; - mDNSPlatformMemCopy(h, &m->HIHardware, 1 + (mDNSu32)m->HIHardware.c[0]); - h += 1 + (int)h[0]; - mDNSPlatformMemCopy(h, &m->HISoftware, 1 + (mDNSu32)m->HISoftware.c[0]); - hinfo.resrec.rdlength = len; - hinfo.resrec.rdestimate = len; - newptr = PutResourceRecordTTLWithLimit(msg, end, &msg->h.numAdditionals, &hinfo.resrec, 0, limit); - return newptr; - } - else - return end; - } - -// *************************************************************************** -#if COMPILER_LIKES_PRAGMA_MARK -#pragma mark - -#pragma mark - DNS Message Parsing Functions -#endif - -mDNSexport mDNSu32 DomainNameHashValue(const domainname *const name) - { - mDNSu32 sum = 0; - const mDNSu8 *c; - - for (c = name->c; c[0] != 0 && c[1] != 0; c += 2) - { - sum += ((mDNSIsUpperCase(c[0]) ? c[0] + 'a' - 'A' : c[0]) << 8) | - (mDNSIsUpperCase(c[1]) ? c[1] + 'a' - 'A' : c[1]); - sum = (sum<<3) | (sum>>29); - } - if (c[0]) sum += ((mDNSIsUpperCase(c[0]) ? c[0] + 'a' - 'A' : c[0]) << 8); - return(sum); - } - -mDNSexport void SetNewRData(ResourceRecord *const rr, RData *NewRData, mDNSu16 rdlength) - { - domainname *target; - if (NewRData) - { - rr->rdata = NewRData; - rr->rdlength = rdlength; - } - // Must not try to get target pointer until after updating rr->rdata - target = GetRRDomainNameTarget(rr); - rr->rdlength = GetRDLength(rr, mDNSfalse); - rr->rdestimate = GetRDLength(rr, mDNStrue); - rr->rdatahash = target ? DomainNameHashValue(target) : RDataHashValue(rr); - } - -mDNSexport const mDNSu8 *skipDomainName(const DNSMessage *const msg, const mDNSu8 *ptr, const mDNSu8 *const end) - { - mDNSu16 total = 0; - - if (ptr < (mDNSu8*)msg || ptr >= end) - { debugf("skipDomainName: Illegal ptr not within packet boundaries"); return(mDNSNULL); } - - while (1) // Read sequence of labels - { - const mDNSu8 len = *ptr++; // Read length of this label - if (len == 0) return(ptr); // If length is zero, that means this name is complete - switch (len & 0xC0) - { - case 0x00: if (ptr + len >= end) // Remember: expect at least one more byte for the root label - { debugf("skipDomainName: Malformed domain name (overruns packet end)"); return(mDNSNULL); } - if (total + 1 + len >= MAX_DOMAIN_NAME) // Remember: expect at least one more byte for the root label - { debugf("skipDomainName: Malformed domain name (more than 256 characters)"); return(mDNSNULL); } - ptr += len; - total += 1 + len; - break; - - case 0x40: debugf("skipDomainName: Extended EDNS0 label types 0x%X not supported", len); return(mDNSNULL); - case 0x80: debugf("skipDomainName: Illegal label length 0x%X", len); return(mDNSNULL); - case 0xC0: return(ptr+1); - } - } - } - -// Routine to fetch an FQDN from the DNS message, following compression pointers if necessary. -mDNSexport const mDNSu8 *getDomainName(const DNSMessage *const msg, const mDNSu8 *ptr, const mDNSu8 *const end, - domainname *const name) - { - const mDNSu8 *nextbyte = mDNSNULL; // Record where we got to before we started following pointers - mDNSu8 *np = name->c; // Name pointer - const mDNSu8 *const limit = np + MAX_DOMAIN_NAME; // Limit so we don't overrun buffer - - if (ptr < (mDNSu8*)msg || ptr >= end) - { debugf("getDomainName: Illegal ptr not within packet boundaries"); return(mDNSNULL); } - - *np = 0; // Tentatively place the root label here (may be overwritten if we have more labels) - - while (1) // Read sequence of labels - { - const mDNSu8 len = *ptr++; // Read length of this label - if (len == 0) break; // If length is zero, that means this name is complete - switch (len & 0xC0) - { - int i; - mDNSu16 offset; - - case 0x00: if (ptr + len >= end) // Remember: expect at least one more byte for the root label - { debugf("getDomainName: Malformed domain name (overruns packet end)"); return(mDNSNULL); } - if (np + 1 + len >= limit) // Remember: expect at least one more byte for the root label - { debugf("getDomainName: Malformed domain name (more than 256 characters)"); return(mDNSNULL); } - *np++ = len; - for (i=0; ic); - return(mDNSNULL); - - case 0x80: debugf("getDomainName: Illegal label length 0x%X in domain name %##s", len, name->c); return(mDNSNULL); - - case 0xC0: offset = (mDNSu16)((((mDNSu16)(len & 0x3F)) << 8) | *ptr++); - if (!nextbyte) nextbyte = ptr; // Record where we got to before we started following pointers - ptr = (mDNSu8 *)msg + offset; - if (ptr < (mDNSu8*)msg || ptr >= end) - { debugf("getDomainName: Illegal compression pointer not within packet boundaries"); return(mDNSNULL); } - if (*ptr & 0xC0) - { debugf("getDomainName: Compression pointer must point to real label"); return(mDNSNULL); } - break; - } - } - - if (nextbyte) return(nextbyte); - else return(ptr); - } - -mDNSexport const mDNSu8 *skipResourceRecord(const DNSMessage *msg, const mDNSu8 *ptr, const mDNSu8 *end) - { - mDNSu16 pktrdlength; - - ptr = skipDomainName(msg, ptr, end); - if (!ptr) { debugf("skipResourceRecord: Malformed RR name"); return(mDNSNULL); } - - if (ptr + 10 > end) { debugf("skipResourceRecord: Malformed RR -- no type/class/ttl/len!"); return(mDNSNULL); } - pktrdlength = (mDNSu16)((mDNSu16)ptr[8] << 8 | ptr[9]); - ptr += 10; - if (ptr + pktrdlength > end) { debugf("skipResourceRecord: RDATA exceeds end of packet"); return(mDNSNULL); } - - return(ptr + pktrdlength); - } - -mDNSexport const mDNSu8 *GetLargeResourceRecord(mDNS *const m, const DNSMessage *const msg, const mDNSu8 *ptr, - const mDNSu8 *end, const mDNSInterfaceID InterfaceID, mDNSu8 RecordType, LargeCacheRecord *const largecr) - { - CacheRecord *const rr = &largecr->r; - RDataBody2 *const rdb = (RDataBody2 *)rr->smallrdatastorage.data; - mDNSu16 pktrdlength; - - if (largecr == &m->rec && m->rec.r.resrec.RecordType) - { - LogMsg("GetLargeResourceRecord: m->rec appears to be already in use for %s", CRDisplayString(m, &m->rec.r)); -#if ForceAlerts - *(long*)0 = 0; -#endif - } - - rr->next = mDNSNULL; - rr->resrec.name = &largecr->namestorage; - - rr->NextInKAList = mDNSNULL; - rr->TimeRcvd = m ? m->timenow : 0; - rr->DelayDelivery = 0; - rr->NextRequiredQuery = m ? m->timenow : 0; // Will be updated to the real value when we call SetNextCacheCheckTimeForRecord() - rr->LastUsed = m ? m->timenow : 0; - rr->CRActiveQuestion = mDNSNULL; - rr->UnansweredQueries = 0; - rr->LastUnansweredTime= 0; -#if ENABLE_MULTI_PACKET_QUERY_SNOOPING - rr->MPUnansweredQ = 0; - rr->MPLastUnansweredQT= 0; - rr->MPUnansweredKA = 0; - rr->MPExpectingKA = mDNSfalse; -#endif - rr->NextInCFList = mDNSNULL; - - rr->resrec.InterfaceID = InterfaceID; - rr->resrec.rDNSServer = mDNSNULL; - - ptr = getDomainName(msg, ptr, end, &largecr->namestorage); // Will bail out correctly if ptr is NULL - if (!ptr) { debugf("GetLargeResourceRecord: Malformed RR name"); return(mDNSNULL); } - rr->resrec.namehash = DomainNameHashValue(rr->resrec.name); - - if (ptr + 10 > end) { debugf("GetLargeResourceRecord: Malformed RR -- no type/class/ttl/len!"); return(mDNSNULL); } - - rr->resrec.rrtype = (mDNSu16) ((mDNSu16)ptr[0] << 8 | ptr[1]); - rr->resrec.rrclass = (mDNSu16)(((mDNSu16)ptr[2] << 8 | ptr[3]) & kDNSClass_Mask); - rr->resrec.rroriginalttl = (mDNSu32) ((mDNSu32)ptr[4] << 24 | (mDNSu32)ptr[5] << 16 | (mDNSu32)ptr[6] << 8 | ptr[7]); - if (rr->resrec.rroriginalttl > 0x70000000UL / mDNSPlatformOneSecond && (mDNSs32)rr->resrec.rroriginalttl != -1) - rr->resrec.rroriginalttl = 0x70000000UL / mDNSPlatformOneSecond; - // Note: We don't have to adjust m->NextCacheCheck here -- this is just getting a record into memory for - // us to look at. If we decide to copy it into the cache, then we'll update m->NextCacheCheck accordingly. - pktrdlength = (mDNSu16)((mDNSu16)ptr[8] << 8 | ptr[9]); - - // If mDNS record has cache-flush bit set, we mark it unique - // For uDNS records, all are implicitly deemed unique (a single DNS server is always - // authoritative for the entire RRSet), unless this is a truncated response - if (ptr[2] & (kDNSClass_UniqueRRSet >> 8) || (!InterfaceID && !(msg->h.flags.b[0] & kDNSFlag0_TC))) - RecordType |= kDNSRecordTypePacketUniqueMask; - ptr += 10; - if (ptr + pktrdlength > end) { debugf("GetLargeResourceRecord: RDATA exceeds end of packet"); return(mDNSNULL); } - end = ptr + pktrdlength; // Adjust end to indicate the end of the rdata for this resource record - - rr->resrec.rdata = (RData*)&rr->smallrdatastorage; - rr->resrec.rdata->MaxRDLength = MaximumRDSize; - - if (!RecordType) LogMsg("GetLargeResourceRecord: No RecordType for %##s", rr->resrec.name->c); - - // IMPORTANT: Any record type we understand and unpack into a structure containing domainnames needs to have corresponding - // cases in SameRDataBody() and RDataHashValue() to do a semantic comparison (or checksum) of the structure instead of a blind - // bitwise memory compare (or sum). This is because a domainname is a fixed size structure holding variable-length data. - // Any bytes past the logical end of the name are undefined, and a blind bitwise memory compare may indicate that - // two domainnames are different when semantically they are the same name and it's only the unused bytes that differ. - if (rr->resrec.rrclass == kDNSQClass_ANY && pktrdlength == 0) // Used in update packets to mean "Delete An RRset" (RFC 2136) - rr->resrec.rdlength = 0; - else switch (rr->resrec.rrtype) - { - case kDNSType_A: if (pktrdlength != sizeof(mDNSv4Addr)) goto fail; - rdb->ipv4.b[0] = ptr[0]; - rdb->ipv4.b[1] = ptr[1]; - rdb->ipv4.b[2] = ptr[2]; - rdb->ipv4.b[3] = ptr[3]; - break; - - case kDNSType_NS: - case kDNSType_CNAME: - case kDNSType_PTR: - case kDNSType_DNAME:ptr = getDomainName(msg, ptr, end, &rdb->name); - if (ptr != end) { debugf("GetLargeResourceRecord: Malformed CNAME/PTR RDATA name"); goto fail; } - //debugf("%##s PTR %##s rdlen %d", rr->resrec.name.c, rdb->name.c, pktrdlength); - break; - - case kDNSType_SOA: ptr = getDomainName(msg, ptr, end, &rdb->soa.mname); - if (!ptr) { debugf("GetLargeResourceRecord: Malformed SOA RDATA mname"); goto fail; } - ptr = getDomainName(msg, ptr, end, &rdb->soa.rname); - if (!ptr) { debugf("GetLargeResourceRecord: Malformed SOA RDATA rname"); goto fail; } - if (ptr + 0x14 != end) { debugf("GetLargeResourceRecord: Malformed SOA RDATA"); goto fail; } - rdb->soa.serial = (mDNSs32) ((mDNSs32)ptr[0x00] << 24 | (mDNSs32)ptr[0x01] << 16 | (mDNSs32)ptr[0x02] << 8 | ptr[0x03]); - rdb->soa.refresh = (mDNSu32) ((mDNSu32)ptr[0x04] << 24 | (mDNSu32)ptr[0x05] << 16 | (mDNSu32)ptr[0x06] << 8 | ptr[0x07]); - rdb->soa.retry = (mDNSu32) ((mDNSu32)ptr[0x08] << 24 | (mDNSu32)ptr[0x09] << 16 | (mDNSu32)ptr[0x0A] << 8 | ptr[0x0B]); - rdb->soa.expire = (mDNSu32) ((mDNSu32)ptr[0x0C] << 24 | (mDNSu32)ptr[0x0D] << 16 | (mDNSu32)ptr[0x0E] << 8 | ptr[0x0F]); - rdb->soa.min = (mDNSu32) ((mDNSu32)ptr[0x10] << 24 | (mDNSu32)ptr[0x11] << 16 | (mDNSu32)ptr[0x12] << 8 | ptr[0x13]); - break; - - case kDNSType_NULL: - case kDNSType_HINFO: - case kDNSType_TSIG: - case kDNSType_TXT: - case kDNSType_X25: - case kDNSType_ISDN: - case kDNSType_LOC: - case kDNSType_DHCID:if (pktrdlength > rr->resrec.rdata->MaxRDLength) - { - debugf("GetLargeResourceRecord: %s rdata size (%d) exceeds storage (%d)", - DNSTypeName(rr->resrec.rrtype), pktrdlength, rr->resrec.rdata->MaxRDLength); - goto fail; - } - rr->resrec.rdlength = pktrdlength; - mDNSPlatformMemCopy(rdb->data, ptr, pktrdlength); - break; - - case kDNSType_MX: - case kDNSType_AFSDB: - case kDNSType_RT: - case kDNSType_KX: if (pktrdlength < 3) goto fail; // Preference + domainname - rdb->mx.preference = (mDNSu16)((mDNSu16)ptr[0] << 8 | ptr[1]); - ptr = getDomainName(msg, ptr+2, end, &rdb->mx.exchange); - if (ptr != end) { debugf("GetLargeResourceRecord: Malformed MX name"); goto fail; } - //debugf("%##s SRV %##s rdlen %d", rr->resrec.name.c, rdb->srv.target.c, pktrdlength); - break; - - case kDNSType_RP: ptr = getDomainName(msg, ptr, end, &rdb->rp.mbox); // Domainname + domainname - if (!ptr) { debugf("GetLargeResourceRecord: Malformed RP mbox"); goto fail; } - ptr = getDomainName(msg, ptr, end, &rdb->rp.txt); - if (ptr != end) { debugf("GetLargeResourceRecord: Malformed RP txt"); goto fail; } - break; - - case kDNSType_PX: if (pktrdlength < 4) goto fail; // Preference + domainname + domainname - rdb->px.preference = (mDNSu16)((mDNSu16)ptr[0] << 8 | ptr[1]); - ptr = getDomainName(msg, ptr, end, &rdb->px.map822); - if (!ptr) { debugf("GetLargeResourceRecord: Malformed PX map822"); goto fail; } - ptr = getDomainName(msg, ptr, end, &rdb->px.mapx400); - if (ptr != end) { debugf("GetLargeResourceRecord: Malformed PX mapx400"); goto fail; } - break; - - case kDNSType_AAAA: if (pktrdlength != sizeof(mDNSv6Addr)) goto fail; - mDNSPlatformMemCopy(&rdb->ipv6, ptr, sizeof(rdb->ipv6)); - break; - - case kDNSType_SRV: if (pktrdlength < 7) goto fail; // Priority + weight + port + domainname - rdb->srv.priority = (mDNSu16)((mDNSu16)ptr[0] << 8 | ptr[1]); - rdb->srv.weight = (mDNSu16)((mDNSu16)ptr[2] << 8 | ptr[3]); - rdb->srv.port.b[0] = ptr[4]; - rdb->srv.port.b[1] = ptr[5]; - ptr = getDomainName(msg, ptr+6, end, &rdb->srv.target); - if (ptr != end) { debugf("GetLargeResourceRecord: Malformed SRV RDATA name"); goto fail; } - //debugf("%##s SRV %##s rdlen %d", rr->resrec.name.c, rdb->srv.target.c, pktrdlength); - break; - - case kDNSType_OPT: { - rdataOPT *opt = rr->resrec.rdata->u.opt; - rr->resrec.rdlength = 0; - while (ptr < end && (mDNSu8 *)(opt+1) < &rr->resrec.rdata->u.data[MaximumRDSize]) - { - const rdataOPT *const currentopt = opt; - if (ptr + 4 > end) { LogInfo("GetLargeResourceRecord: OPT RDATA ptr + 4 > end"); goto fail; } - opt->opt = (mDNSu16)((mDNSu16)ptr[0] << 8 | ptr[1]); - opt->optlen = (mDNSu16)((mDNSu16)ptr[2] << 8 | ptr[3]); - ptr += 4; - if (ptr + opt->optlen > end) { LogInfo("GetLargeResourceRecord: ptr + opt->optlen > end"); goto fail; } - switch (opt->opt) - { - case kDNSOpt_LLQ: - if (opt->optlen == DNSOpt_LLQData_Space - 4) - { - opt->u.llq.vers = (mDNSu16)((mDNSu16)ptr[0] << 8 | ptr[1]); - opt->u.llq.llqOp = (mDNSu16)((mDNSu16)ptr[2] << 8 | ptr[3]); - opt->u.llq.err = (mDNSu16)((mDNSu16)ptr[4] << 8 | ptr[5]); - mDNSPlatformMemCopy(opt->u.llq.id.b, ptr+6, 8); - opt->u.llq.llqlease = (mDNSu32) ((mDNSu32)ptr[14] << 24 | (mDNSu32)ptr[15] << 16 | (mDNSu32)ptr[16] << 8 | ptr[17]); - if (opt->u.llq.llqlease > 0x70000000UL / mDNSPlatformOneSecond) - opt->u.llq.llqlease = 0x70000000UL / mDNSPlatformOneSecond; - opt++; - } - break; - case kDNSOpt_Lease: - if (opt->optlen == DNSOpt_LeaseData_Space - 4) - { - opt->u.updatelease = (mDNSu32) ((mDNSu32)ptr[0] << 24 | (mDNSu32)ptr[1] << 16 | (mDNSu32)ptr[2] << 8 | ptr[3]); - if (opt->u.updatelease > 0x70000000UL / mDNSPlatformOneSecond) - opt->u.updatelease = 0x70000000UL / mDNSPlatformOneSecond; - opt++; - } - break; - case kDNSOpt_Owner: - if (ValidOwnerLength(opt->optlen)) - { - opt->u.owner.vers = ptr[0]; - opt->u.owner.seq = ptr[1]; - mDNSPlatformMemCopy(opt->u.owner.HMAC.b, ptr+2, 6); // 6-byte MAC address - mDNSPlatformMemCopy(opt->u.owner.IMAC.b, ptr+2, 6); // 6-byte MAC address - opt->u.owner.password = zeroEthAddr; - if (opt->optlen >= DNSOpt_OwnerData_ID_Wake_Space-4) - { - mDNSPlatformMemCopy(opt->u.owner.IMAC.b, ptr+8, 6); // 6-byte MAC address - // This mDNSPlatformMemCopy is safe because the ValidOwnerLength(opt->optlen) check above - // ensures that opt->optlen is no more than DNSOpt_OwnerData_ID_Wake_PW6_Space - 4 - if (opt->optlen > DNSOpt_OwnerData_ID_Wake_Space-4) - mDNSPlatformMemCopy(opt->u.owner.password.b, ptr+14, opt->optlen - (DNSOpt_OwnerData_ID_Wake_Space-4)); - } - opt++; - } - break; - } - ptr += currentopt->optlen; - } - rr->resrec.rdlength = (mDNSu16)((mDNSu8*)opt - rr->resrec.rdata->u.data); - if (ptr != end) { LogInfo("GetLargeResourceRecord: Malformed OptRdata"); goto fail; } - break; - } - - case kDNSType_NSEC: { - unsigned int i, j; - domainname d; - ptr = getDomainName(msg, ptr, end, &d); // Ignored for our simplified use of NSEC synthetic records - if (!ptr) { LogInfo("GetLargeResourceRecord: Malformed NSEC nextname"); goto fail; } - mDNSPlatformMemZero(rdb->nsec.bitmap, sizeof(rdb->nsec.bitmap)); - if (ptr < end) - { - if (*ptr++ != 0) { debugf("GetLargeResourceRecord: We only handle block zero NSECs"); goto fail; } - i = *ptr++; - if (i > sizeof(rdataNSEC)) { debugf("GetLargeResourceRecord: invalid block length %d", i); goto fail; } - for (j=0; jnsec.bitmap[j] = *ptr++; - } - if (ptr != end) { debugf("GetLargeResourceRecord: Malformed NSEC"); goto fail; } - break; - } - - default: if (pktrdlength > rr->resrec.rdata->MaxRDLength) - { - debugf("GetLargeResourceRecord: rdata %d (%s) size (%d) exceeds storage (%d)", - rr->resrec.rrtype, DNSTypeName(rr->resrec.rrtype), pktrdlength, rr->resrec.rdata->MaxRDLength); - goto fail; - } - debugf("GetLargeResourceRecord: Warning! Reading resource type %d (%s) as opaque data", - rr->resrec.rrtype, DNSTypeName(rr->resrec.rrtype)); - // Note: Just because we don't understand the record type, that doesn't - // mean we fail. The DNS protocol specifies rdlength, so we can - // safely skip over unknown records and ignore them. - // We also grab a binary copy of the rdata anyway, since the caller - // might know how to interpret it even if we don't. - rr->resrec.rdlength = pktrdlength; - mDNSPlatformMemCopy(rdb->data, ptr, pktrdlength); - break; - } - - SetNewRData(&rr->resrec, mDNSNULL, 0); // Sets rdlength, rdestimate, rdatahash for us - - // Success! Now fill in RecordType to show this record contains valid data - rr->resrec.RecordType = RecordType; - return(end); - -fail: - // If we were unable to parse the rdata in this record, we indicate that by - // returing a 'kDNSRecordTypePacketNegative' record with rdlength set to zero - rr->resrec.RecordType = kDNSRecordTypePacketNegative; - rr->resrec.rdlength = 0; - rr->resrec.rdestimate = 0; - rr->resrec.rdatahash = 0; - return(end); - } - -mDNSexport const mDNSu8 *skipQuestion(const DNSMessage *msg, const mDNSu8 *ptr, const mDNSu8 *end) - { - ptr = skipDomainName(msg, ptr, end); - if (!ptr) { debugf("skipQuestion: Malformed domain name in DNS question section"); return(mDNSNULL); } - if (ptr+4 > end) { debugf("skipQuestion: Malformed DNS question section -- no query type and class!"); return(mDNSNULL); } - return(ptr+4); - } - -mDNSexport const mDNSu8 *getQuestion(const DNSMessage *msg, const mDNSu8 *ptr, const mDNSu8 *end, const mDNSInterfaceID InterfaceID, - DNSQuestion *question) - { - mDNSPlatformMemZero(question, sizeof(*question)); - question->InterfaceID = InterfaceID; - if (!InterfaceID) question->TargetQID = onesID; // In DNSQuestions we use TargetQID as the indicator of whether it's unicast or multicast - ptr = getDomainName(msg, ptr, end, &question->qname); - if (!ptr) { debugf("Malformed domain name in DNS question section"); return(mDNSNULL); } - if (ptr+4 > end) { debugf("Malformed DNS question section -- no query type and class!"); return(mDNSNULL); } - - question->qnamehash = DomainNameHashValue(&question->qname); - question->qtype = (mDNSu16)((mDNSu16)ptr[0] << 8 | ptr[1]); // Get type - question->qclass = (mDNSu16)((mDNSu16)ptr[2] << 8 | ptr[3]); // and class - return(ptr+4); - } - -mDNSexport const mDNSu8 *LocateAnswers(const DNSMessage *const msg, const mDNSu8 *const end) - { - int i; - const mDNSu8 *ptr = msg->data; - for (i = 0; i < msg->h.numQuestions && ptr; i++) ptr = skipQuestion(msg, ptr, end); - return(ptr); - } - -mDNSexport const mDNSu8 *LocateAuthorities(const DNSMessage *const msg, const mDNSu8 *const end) - { - int i; - const mDNSu8 *ptr = LocateAnswers(msg, end); - for (i = 0; i < msg->h.numAnswers && ptr; i++) ptr = skipResourceRecord(msg, ptr, end); - return(ptr); - } - -mDNSexport const mDNSu8 *LocateAdditionals(const DNSMessage *const msg, const mDNSu8 *const end) - { - int i; - const mDNSu8 *ptr = LocateAuthorities(msg, end); - for (i = 0; i < msg->h.numAuthorities; i++) ptr = skipResourceRecord(msg, ptr, end); - return (ptr); - } - -mDNSexport const mDNSu8 *LocateOptRR(const DNSMessage *const msg, const mDNSu8 *const end, int minsize) - { - int i; - const mDNSu8 *ptr = LocateAdditionals(msg, end); - - // Locate the OPT record. - // According to RFC 2671, "One OPT pseudo-RR can be added to the additional data section of either a request or a response." - // This implies that there may be *at most* one OPT record per DNS message, in the Additional Section, - // but not necessarily the *last* entry in the Additional Section. - for (i = 0; ptr && i < msg->h.numAdditionals; i++) - { - if (ptr + DNSOpt_Header_Space + minsize <= end && // Make sure we have 11+minsize bytes of data - ptr[0] == 0 && // Name must be root label - ptr[1] == (kDNSType_OPT >> 8 ) && // rrtype OPT - ptr[2] == (kDNSType_OPT & 0xFF) && - ((mDNSu16)ptr[9] << 8 | (mDNSu16)ptr[10]) >= (mDNSu16)minsize) - return(ptr); - else - ptr = skipResourceRecord(msg, ptr, end); - } - return(mDNSNULL); - } - -// On success, GetLLQOptData returns pointer to storage within shared "m->rec"; -// it is caller's responsibilty to clear m->rec.r.resrec.RecordType after use -// Note: An OPT RDataBody actually contains one or more variable-length rdataOPT objects packed together -// The code that currently calls this assumes there's only one, instead of iterating through the set -mDNSexport const rdataOPT *GetLLQOptData(mDNS *const m, const DNSMessage *const msg, const mDNSu8 *const end) - { - const mDNSu8 *ptr = LocateOptRR(msg, end, DNSOpt_LLQData_Space); - if (ptr) - { - ptr = GetLargeResourceRecord(m, msg, ptr, end, 0, kDNSRecordTypePacketAdd, &m->rec); - if (ptr && m->rec.r.resrec.RecordType != kDNSRecordTypePacketNegative) return(&m->rec.r.resrec.rdata->u.opt[0]); - } - return(mDNSNULL); - } - -// Get the lease life of records in a dynamic update -// returns 0 on error or if no lease present -mDNSexport mDNSu32 GetPktLease(mDNS *m, DNSMessage *msg, const mDNSu8 *end) - { - mDNSu32 result = 0; - const mDNSu8 *ptr = LocateOptRR(msg, end, DNSOpt_LeaseData_Space); - if (ptr) ptr = GetLargeResourceRecord(m, msg, ptr, end, 0, kDNSRecordTypePacketAdd, &m->rec); - if (ptr && m->rec.r.resrec.rdlength >= DNSOpt_LeaseData_Space && m->rec.r.resrec.rdata->u.opt[0].opt == kDNSOpt_Lease) - result = m->rec.r.resrec.rdata->u.opt[0].u.updatelease; - m->rec.r.resrec.RecordType = 0; // Clear RecordType to show we're not still using it - return(result); - } - -mDNSlocal const mDNSu8 *DumpRecords(mDNS *const m, const DNSMessage *const msg, const mDNSu8 *ptr, const mDNSu8 *const end, int count, char *label) - { - int i; - LogMsg("%2d %s", count, label); - for (i = 0; i < count && ptr; i++) - { - // This puts a LargeCacheRecord on the stack instead of using the shared m->rec storage, - // but since it's only used for debugging (and probably only on OS X, not on - // embedded systems) putting a 9kB object on the stack isn't a big problem. - LargeCacheRecord largecr; - ptr = GetLargeResourceRecord(m, msg, ptr, end, mDNSInterface_Any, kDNSRecordTypePacketAns, &largecr); - if (ptr) LogMsg("%2d TTL%8d %s", i, largecr.r.resrec.rroriginalttl, CRDisplayString(m, &largecr.r)); - } - if (!ptr) LogMsg("ERROR: Premature end of packet data"); - return(ptr); - } - -#define DNS_OP_Name(X) ( \ - (X) == kDNSFlag0_OP_StdQuery ? "" : \ - (X) == kDNSFlag0_OP_Iquery ? "Iquery " : \ - (X) == kDNSFlag0_OP_Status ? "Status " : \ - (X) == kDNSFlag0_OP_Unused3 ? "Unused3 " : \ - (X) == kDNSFlag0_OP_Notify ? "Notify " : \ - (X) == kDNSFlag0_OP_Update ? "Update " : "?? " ) - -#define DNS_RC_Name(X) ( \ - (X) == kDNSFlag1_RC_NoErr ? "NoErr" : \ - (X) == kDNSFlag1_RC_FormErr ? "FormErr" : \ - (X) == kDNSFlag1_RC_ServFail ? "ServFail" : \ - (X) == kDNSFlag1_RC_NXDomain ? "NXDomain" : \ - (X) == kDNSFlag1_RC_NotImpl ? "NotImpl" : \ - (X) == kDNSFlag1_RC_Refused ? "Refused" : \ - (X) == kDNSFlag1_RC_YXDomain ? "YXDomain" : \ - (X) == kDNSFlag1_RC_YXRRSet ? "YXRRSet" : \ - (X) == kDNSFlag1_RC_NXRRSet ? "NXRRSet" : \ - (X) == kDNSFlag1_RC_NotAuth ? "NotAuth" : \ - (X) == kDNSFlag1_RC_NotZone ? "NotZone" : "??" ) - -// Note: DumpPacket expects the packet header fields in host byte order, not network byte order -mDNSexport void DumpPacket(mDNS *const m, mStatus status, mDNSBool sent, char *transport, - const mDNSAddr *srcaddr, mDNSIPPort srcport, - const mDNSAddr *dstaddr, mDNSIPPort dstport, const DNSMessage *const msg, const mDNSu8 *const end) - { - mDNSBool IsUpdate = ((msg->h.flags.b[0] & kDNSFlag0_OP_Mask) == kDNSFlag0_OP_Update); - const mDNSu8 *ptr = msg->data; - int i; - DNSQuestion q; - char tbuffer[64], sbuffer[64], dbuffer[64] = ""; - if (!status) tbuffer[mDNS_snprintf(tbuffer, sizeof(tbuffer), sent ? "Sent" : "Received" )] = 0; - else tbuffer[mDNS_snprintf(tbuffer, sizeof(tbuffer), "ERROR %d %sing", status, sent ? "Send" : "Receiv")] = 0; - if (sent) sbuffer[mDNS_snprintf(sbuffer, sizeof(sbuffer), "port " )] = 0; - else sbuffer[mDNS_snprintf(sbuffer, sizeof(sbuffer), "%#a:", srcaddr)] = 0; - if (dstaddr || !mDNSIPPortIsZero(dstport)) - dbuffer[mDNS_snprintf(dbuffer, sizeof(dbuffer), " to %#a:%d", dstaddr, mDNSVal16(dstport))] = 0; - - LogMsg("-- %s %s DNS %s%s (flags %02X%02X) RCODE: %s (%d) %s%s%s%s%s%sID: %d %d bytes from %s%d%s%s --", - tbuffer, transport, - DNS_OP_Name(msg->h.flags.b[0] & kDNSFlag0_OP_Mask), - msg->h.flags.b[0] & kDNSFlag0_QR_Response ? "Response" : "Query", - msg->h.flags.b[0], msg->h.flags.b[1], - DNS_RC_Name(msg->h.flags.b[1] & kDNSFlag1_RC_Mask), - msg->h.flags.b[1] & kDNSFlag1_RC_Mask, - msg->h.flags.b[0] & kDNSFlag0_AA ? "AA " : "", - msg->h.flags.b[0] & kDNSFlag0_TC ? "TC " : "", - msg->h.flags.b[0] & kDNSFlag0_RD ? "RD " : "", - msg->h.flags.b[1] & kDNSFlag1_RA ? "RA " : "", - msg->h.flags.b[1] & kDNSFlag1_AD ? "AD " : "", - msg->h.flags.b[1] & kDNSFlag1_CD ? "CD " : "", - mDNSVal16(msg->h.id), - end - msg->data, - sbuffer, mDNSVal16(srcport), dbuffer, - (msg->h.flags.b[0] & kDNSFlag0_TC) ? " (truncated)" : "" - ); - - LogMsg("%2d %s", msg->h.numQuestions, IsUpdate ? "Zone" : "Questions"); - for (i = 0; i < msg->h.numQuestions && ptr; i++) - { - ptr = getQuestion(msg, ptr, end, mDNSInterface_Any, &q); - if (ptr) LogMsg("%2d %##s %s", i, q.qname.c, DNSTypeName(q.qtype)); - } - ptr = DumpRecords(m, msg, ptr, end, msg->h.numAnswers, IsUpdate ? "Prerequisites" : "Answers"); - ptr = DumpRecords(m, msg, ptr, end, msg->h.numAuthorities, IsUpdate ? "Updates" : "Authorities"); - ptr = DumpRecords(m, msg, ptr, end, msg->h.numAdditionals, "Additionals"); - LogMsg("--------------"); - } - -// *************************************************************************** -#if COMPILER_LIKES_PRAGMA_MARK -#pragma mark - -#pragma mark - Packet Sending Functions -#endif - -// Stub definition of TCPSocket_struct so we can access flags field. (Rest of TCPSocket_struct is platform-dependent.) -struct TCPSocket_struct { TCPSocketFlags flags; /* ... */ }; - -struct UDPSocket_struct - { - mDNSIPPort port; // MUST BE FIRST FIELD -- mDNSCoreReceive expects every UDPSocket_struct to begin with mDNSIPPort port - }; - -// Note: When we sign a DNS message using DNSDigest_SignMessage(), the current real-time clock value is used, which -// is why we generally defer signing until we send the message, to ensure the signature is as fresh as possible. -mDNSexport mStatus mDNSSendDNSMessage(mDNS *const m, DNSMessage *const msg, mDNSu8 *end, - mDNSInterfaceID InterfaceID, UDPSocket *src, const mDNSAddr *dst, mDNSIPPort dstport, TCPSocket *sock, DomainAuthInfo *authInfo) - { - mStatus status = mStatus_NoError; - const mDNSu16 numAdditionals = msg->h.numAdditionals; - mDNSu8 *newend; - mDNSu8 *limit = msg->data + AbsoluteMaxDNSMessageData; - - // Zero-length message data is okay (e.g. for a DNS Update ack, where all we need is an ID and an error code - if (end < msg->data || end - msg->data > AbsoluteMaxDNSMessageData) - { - LogMsg("mDNSSendDNSMessage: invalid message %p %p %d", msg->data, end, end - msg->data); - return mStatus_BadParamErr; - } - - newend = putHINFO(m, msg, end, authInfo, limit); - if (!newend) LogMsg("mDNSSendDNSMessage: putHINFO failed msg %p end %p, limit %p", msg->data, end, limit); // Not fatal - else end = newend; - - // Put all the integer values in IETF byte-order (MSB first, LSB second) - SwapDNSHeaderBytes(msg); - - if (authInfo) DNSDigest_SignMessage(msg, &end, authInfo, 0); // DNSDigest_SignMessage operates on message in network byte order - if (!end) { LogMsg("mDNSSendDNSMessage: DNSDigest_SignMessage failed"); status = mStatus_NoMemoryErr; } - else - { - // Send the packet on the wire - if (!sock) - status = mDNSPlatformSendUDP(m, msg, end, InterfaceID, src, dst, dstport); - else - { - mDNSu16 msglen = (mDNSu16)(end - (mDNSu8 *)msg); - mDNSu8 lenbuf[2] = { (mDNSu8)(msglen >> 8), (mDNSu8)(msglen & 0xFF) }; - long nsent = mDNSPlatformWriteTCP(sock, (char*)lenbuf, 2); // Should do scatter/gather here -- this is probably going out as two packets - if (nsent != 2) { LogMsg("mDNSSendDNSMessage: write msg length failed %d/%d", nsent, 2); status = mStatus_ConnFailed; } - else - { - nsent = mDNSPlatformWriteTCP(sock, (char *)msg, msglen); - if (nsent != msglen) { LogMsg("mDNSSendDNSMessage: write msg body failed %d/%d", nsent, msglen); status = mStatus_ConnFailed; } - } - } - } - - // Swap the integer values back the way they were (remember that numAdditionals may have been changed by putHINFO and/or SignMessage) - SwapDNSHeaderBytes(msg); - - // Dump the packet with the HINFO and TSIG - if (mDNS_PacketLoggingEnabled && !mDNSOpaque16IsZero(msg->h.id)) - DumpPacket(m, status, mDNStrue, sock && (sock->flags & kTCPSocketFlags_UseTLS) ? "TLS" : sock ? "TCP" : "UDP", mDNSNULL, src ? src->port : MulticastDNSPort, dst, dstport, msg, end); - - // put the number of additionals back the way it was - msg->h.numAdditionals = numAdditionals; - - return(status); - } - -// *************************************************************************** -#if COMPILER_LIKES_PRAGMA_MARK -#pragma mark - -#pragma mark - RR List Management & Task Management -#endif - -mDNSexport void mDNS_Lock_(mDNS *const m, const char * const functionname) - { - // MUST grab the platform lock FIRST! - mDNSPlatformLock(m); - - // Normally, mDNS_reentrancy is zero and so is mDNS_busy - // However, when we call a client callback mDNS_busy is one, and we increment mDNS_reentrancy too - // If that client callback does mDNS API calls, mDNS_reentrancy and mDNS_busy will both be one - // If mDNS_busy != mDNS_reentrancy that's a bad sign - if (m->mDNS_busy != m->mDNS_reentrancy) - { - LogMsg("%s: mDNS_Lock: Locking failure! mDNS_busy (%ld) != mDNS_reentrancy (%ld)", functionname, m->mDNS_busy, m->mDNS_reentrancy); -#if ForceAlerts - *(long*)0 = 0; -#endif - } - - // If this is an initial entry into the mDNSCore code, set m->timenow - // else, if this is a re-entrant entry into the mDNSCore code, m->timenow should already be set - if (m->mDNS_busy == 0) - { - if (m->timenow) - LogMsg("%s: mDNS_Lock: m->timenow already set (%ld/%ld)", functionname, m->timenow, mDNS_TimeNow_NoLock(m)); - m->timenow = mDNS_TimeNow_NoLock(m); - if (m->timenow == 0) m->timenow = 1; - } - else if (m->timenow == 0) - { - LogMsg("%s: mDNS_Lock: m->mDNS_busy is %ld but m->timenow not set", functionname, m->mDNS_busy); - m->timenow = mDNS_TimeNow_NoLock(m); - if (m->timenow == 0) m->timenow = 1; - } - - if (m->timenow_last - m->timenow > 0) - { - m->timenow_adjust += m->timenow_last - m->timenow; - LogMsg("%s: mDNSPlatformRawTime went backwards by %ld ticks; setting correction factor to %ld", functionname, m->timenow_last - m->timenow, m->timenow_adjust); - m->timenow = m->timenow_last; - } - m->timenow_last = m->timenow; - - // Increment mDNS_busy so we'll recognise re-entrant calls - m->mDNS_busy++; - } - -mDNSlocal AuthRecord *AnyLocalRecordReady(const mDNS *const m) - { - AuthRecord *rr; - for (rr = m->NewLocalRecords; rr; rr = rr->next) - if (LocalRecordReady(rr)) return rr; - return mDNSNULL; - } - -mDNSlocal mDNSs32 GetNextScheduledEvent(const mDNS *const m) - { - mDNSs32 e = m->timenow + 0x78000000; - if (m->mDNSPlatformStatus != mStatus_NoError) return(e); - if (m->NewQuestions) - { - if (m->NewQuestions->DelayAnswering) e = m->NewQuestions->DelayAnswering; - else return(m->timenow); - } - if (m->NewLocalOnlyQuestions) return(m->timenow); - if (m->NewLocalRecords && AnyLocalRecordReady(m)) return(m->timenow); - if (m->NewLocalOnlyRecords) return(m->timenow); - if (m->SPSProxyListChanged) return(m->timenow); - if (m->LocalRemoveEvents) return(m->timenow); - -#ifndef UNICAST_DISABLED - if (e - m->NextuDNSEvent > 0) e = m->NextuDNSEvent; - if (e - m->NextScheduledNATOp > 0) e = m->NextScheduledNATOp; - if (m->NextSRVUpdate && e - m->NextSRVUpdate > 0) e = m->NextSRVUpdate; -#endif - - if (e - m->NextCacheCheck > 0) e = m->NextCacheCheck; - if (e - m->NextScheduledSPS > 0) e = m->NextScheduledSPS; - // NextScheduledSPRetry only valid when DelaySleep not set - if (!m->DelaySleep && m->SleepLimit && e - m->NextScheduledSPRetry > 0) e = m->NextScheduledSPRetry; - if (m->DelaySleep && e - m->DelaySleep > 0) e = m->DelaySleep; - - if (m->SuppressSending) - { - if (e - m->SuppressSending > 0) e = m->SuppressSending; - } - else - { - if (e - m->NextScheduledQuery > 0) e = m->NextScheduledQuery; - if (e - m->NextScheduledProbe > 0) e = m->NextScheduledProbe; - if (e - m->NextScheduledResponse > 0) e = m->NextScheduledResponse; - } - if (e - m->NextScheduledStopTime > 0) e = m->NextScheduledStopTime; - return(e); - } - -mDNSexport void ShowTaskSchedulingError(mDNS *const m) - { - AuthRecord *rr; - mDNS_Lock(m); - - LogMsg("Task Scheduling Error: Continuously busy for more than a second"); - - // Note: To accurately diagnose *why* we're busy, the debugging code here needs to mirror the logic in GetNextScheduledEvent above - - if (m->NewQuestions && (!m->NewQuestions->DelayAnswering || m->timenow - m->NewQuestions->DelayAnswering >= 0)) - LogMsg("Task Scheduling Error: NewQuestion %##s (%s)", - m->NewQuestions->qname.c, DNSTypeName(m->NewQuestions->qtype)); - - if (m->NewLocalOnlyQuestions) - LogMsg("Task Scheduling Error: NewLocalOnlyQuestions %##s (%s)", - m->NewLocalOnlyQuestions->qname.c, DNSTypeName(m->NewLocalOnlyQuestions->qtype)); - - if (m->NewLocalRecords) - { - rr = AnyLocalRecordReady(m); - if (rr) LogMsg("Task Scheduling Error: NewLocalRecords %s", ARDisplayString(m, rr)); - } - - if (m->NewLocalOnlyRecords) LogMsg("Task Scheduling Error: NewLocalOnlyRecords"); - - if (m->SPSProxyListChanged) LogMsg("Task Scheduling Error: SPSProxyListChanged"); - if (m->LocalRemoveEvents) LogMsg("Task Scheduling Error: LocalRemoveEvents"); - - if (m->timenow - m->NextScheduledEvent >= 0) - LogMsg("Task Scheduling Error: m->NextScheduledEvent %d", m->timenow - m->NextScheduledEvent); - -#ifndef UNICAST_DISABLED - if (m->timenow - m->NextuDNSEvent >= 0) - LogMsg("Task Scheduling Error: m->NextuDNSEvent %d", m->timenow - m->NextuDNSEvent); - if (m->timenow - m->NextScheduledNATOp >= 0) - LogMsg("Task Scheduling Error: m->NextScheduledNATOp %d", m->timenow - m->NextScheduledNATOp); - if (m->NextSRVUpdate && m->timenow - m->NextSRVUpdate >= 0) - LogMsg("Task Scheduling Error: m->NextSRVUpdate %d", m->timenow - m->NextSRVUpdate); -#endif - - if (m->timenow - m->NextCacheCheck >= 0) - LogMsg("Task Scheduling Error: m->NextCacheCheck %d", m->timenow - m->NextCacheCheck); - if (m->timenow - m->NextScheduledSPS >= 0) - LogMsg("Task Scheduling Error: m->NextScheduledSPS %d", m->timenow - m->NextScheduledSPS); - if (!m->DelaySleep && m->SleepLimit && m->timenow - m->NextScheduledSPRetry >= 0) - LogMsg("Task Scheduling Error: m->NextScheduledSPRetry %d", m->timenow - m->NextScheduledSPRetry); - if (m->DelaySleep && m->timenow - m->DelaySleep >= 0) - LogMsg("Task Scheduling Error: m->DelaySleep %d", m->timenow - m->DelaySleep); - - if (m->SuppressSending && m->timenow - m->SuppressSending >= 0) - LogMsg("Task Scheduling Error: m->SuppressSending %d", m->timenow - m->SuppressSending); - if (m->timenow - m->NextScheduledQuery >= 0) - LogMsg("Task Scheduling Error: m->NextScheduledQuery %d", m->timenow - m->NextScheduledQuery); - if (m->timenow - m->NextScheduledProbe >= 0) - LogMsg("Task Scheduling Error: m->NextScheduledProbe %d", m->timenow - m->NextScheduledProbe); - if (m->timenow - m->NextScheduledResponse >= 0) - LogMsg("Task Scheduling Error: m->NextScheduledResponse %d", m->timenow - m->NextScheduledResponse); - - mDNS_Unlock(m); - } - -mDNSexport void mDNS_Unlock_(mDNS *const m, const char * const functionname) - { - // Decrement mDNS_busy - m->mDNS_busy--; - - // Check for locking failures - if (m->mDNS_busy != m->mDNS_reentrancy) - { - LogMsg("%s: mDNS_Unlock: Locking failure! mDNS_busy (%ld) != mDNS_reentrancy (%ld)", functionname, m->mDNS_busy, m->mDNS_reentrancy); -#if ForceAlerts - *(long*)0 = 0; -#endif - } - - // If this is a final exit from the mDNSCore code, set m->NextScheduledEvent and clear m->timenow - if (m->mDNS_busy == 0) - { - m->NextScheduledEvent = GetNextScheduledEvent(m); - if (m->timenow == 0) LogMsg("%s: mDNS_Unlock: ERROR! m->timenow aready zero", functionname); - m->timenow = 0; - } - - // MUST release the platform lock LAST! - mDNSPlatformUnlock(m); - } - -// *************************************************************************** -#if COMPILER_LIKES_PRAGMA_MARK -#pragma mark - -#pragma mark - Specialized mDNS version of vsnprintf -#endif - -static const struct mDNSprintf_format - { - unsigned leftJustify : 1; - unsigned forceSign : 1; - unsigned zeroPad : 1; - unsigned havePrecision : 1; - unsigned hSize : 1; - unsigned lSize : 1; - char altForm; - char sign; // +, - or space - unsigned int fieldWidth; - unsigned int precision; - } mDNSprintf_format_default = { 0, 0, 0, 0, 0, 0, 0, 0, 0, 0 }; - -mDNSexport mDNSu32 mDNS_vsnprintf(char *sbuffer, mDNSu32 buflen, const char *fmt, va_list arg) - { - mDNSu32 nwritten = 0; - int c; - if (buflen == 0) return(0); - buflen--; // Pre-reserve one space in the buffer for the terminating null - if (buflen == 0) goto exit; - - for (c = *fmt; c != 0; c = *++fmt) - { - if (c != '%') - { - *sbuffer++ = (char)c; - if (++nwritten >= buflen) goto exit; - } - else - { - unsigned int i=0, j; - // The mDNS Vsprintf Argument Conversion Buffer is used as a temporary holding area for - // generating decimal numbers, hexdecimal numbers, IP addresses, domain name strings, etc. - // The size needs to be enough for a 256-byte domain name plus some error text. - #define mDNS_VACB_Size 300 - char mDNS_VACB[mDNS_VACB_Size]; - #define mDNS_VACB_Lim (&mDNS_VACB[mDNS_VACB_Size]) - #define mDNS_VACB_Remain(s) ((mDNSu32)(mDNS_VACB_Lim - s)) - char *s = mDNS_VACB_Lim, *digits; - struct mDNSprintf_format F = mDNSprintf_format_default; - - while (1) // decode flags - { - c = *++fmt; - if (c == '-') F.leftJustify = 1; - else if (c == '+') F.forceSign = 1; - else if (c == ' ') F.sign = ' '; - else if (c == '#') F.altForm++; - else if (c == '0') F.zeroPad = 1; - else break; - } - - if (c == '*') // decode field width - { - int f = va_arg(arg, int); - if (f < 0) { f = -f; F.leftJustify = 1; } - F.fieldWidth = (unsigned int)f; - c = *++fmt; - } - else - { - for (; c >= '0' && c <= '9'; c = *++fmt) - F.fieldWidth = (10 * F.fieldWidth) + (c - '0'); - } - - if (c == '.') // decode precision - { - if ((c = *++fmt) == '*') - { F.precision = va_arg(arg, unsigned int); c = *++fmt; } - else for (; c >= '0' && c <= '9'; c = *++fmt) - F.precision = (10 * F.precision) + (c - '0'); - F.havePrecision = 1; - } - - if (F.leftJustify) F.zeroPad = 0; - - conv: - switch (c) // perform appropriate conversion - { - unsigned long n; - case 'h' : F.hSize = 1; c = *++fmt; goto conv; - case 'l' : // fall through - case 'L' : F.lSize = 1; c = *++fmt; goto conv; - case 'd' : - case 'i' : if (F.lSize) n = (unsigned long)va_arg(arg, long); - else n = (unsigned long)va_arg(arg, int); - if (F.hSize) n = (short) n; - if ((long) n < 0) { n = (unsigned long)-(long)n; F.sign = '-'; } - else if (F.forceSign) F.sign = '+'; - goto decimal; - case 'u' : if (F.lSize) n = va_arg(arg, unsigned long); - else n = va_arg(arg, unsigned int); - if (F.hSize) n = (unsigned short) n; - F.sign = 0; - goto decimal; - decimal: if (!F.havePrecision) - { - if (F.zeroPad) - { - F.precision = F.fieldWidth; - if (F.sign) --F.precision; - } - if (F.precision < 1) F.precision = 1; - } - if (F.precision > mDNS_VACB_Size - 1) - F.precision = mDNS_VACB_Size - 1; - for (i = 0; n; n /= 10, i++) *--s = (char)(n % 10 + '0'); - for (; i < F.precision; i++) *--s = '0'; - if (F.sign) { *--s = F.sign; i++; } - break; - - case 'o' : if (F.lSize) n = va_arg(arg, unsigned long); - else n = va_arg(arg, unsigned int); - if (F.hSize) n = (unsigned short) n; - if (!F.havePrecision) - { - if (F.zeroPad) F.precision = F.fieldWidth; - if (F.precision < 1) F.precision = 1; - } - if (F.precision > mDNS_VACB_Size - 1) - F.precision = mDNS_VACB_Size - 1; - for (i = 0; n; n /= 8, i++) *--s = (char)(n % 8 + '0'); - if (F.altForm && i && *s != '0') { *--s = '0'; i++; } - for (; i < F.precision; i++) *--s = '0'; - break; - - case 'a' : { - unsigned char *a = va_arg(arg, unsigned char *); - if (!a) { static char emsg[] = "<>"; s = emsg; i = sizeof(emsg)-1; } - else - { - s = mDNS_VACB; // Adjust s to point to the start of the buffer, not the end - if (F.altForm) - { - mDNSAddr *ip = (mDNSAddr*)a; - switch (ip->type) - { - case mDNSAddrType_IPv4: F.precision = 4; a = (unsigned char *)&ip->ip.v4; break; - case mDNSAddrType_IPv6: F.precision = 16; a = (unsigned char *)&ip->ip.v6; break; - default: F.precision = 0; break; - } - } - if (F.altForm && !F.precision) - i = mDNS_snprintf(mDNS_VACB, sizeof(mDNS_VACB), ""); - else switch (F.precision) - { - case 4: i = mDNS_snprintf(mDNS_VACB, sizeof(mDNS_VACB), "%d.%d.%d.%d", - a[0], a[1], a[2], a[3]); break; - case 6: i = mDNS_snprintf(mDNS_VACB, sizeof(mDNS_VACB), "%02X:%02X:%02X:%02X:%02X:%02X", - a[0], a[1], a[2], a[3], a[4], a[5]); break; - case 16: i = mDNS_snprintf(mDNS_VACB, sizeof(mDNS_VACB), - "%02X%02X:%02X%02X:%02X%02X:%02X%02X:%02X%02X:%02X%02X:%02X%02X:%02X%02X", - a[0x0], a[0x1], a[0x2], a[0x3], a[0x4], a[0x5], a[0x6], a[0x7], - a[0x8], a[0x9], a[0xA], a[0xB], a[0xC], a[0xD], a[0xE], a[0xF]); break; - default: i = mDNS_snprintf(mDNS_VACB, sizeof(mDNS_VACB), "%s", "<< ERROR: Must specify" - " address size (i.e. %.4a=IPv4, %.6a=Ethernet, %.16a=IPv6) >>"); break; - } - } - } - break; - - case 'p' : F.havePrecision = F.lSize = 1; - F.precision = sizeof(void*) * 2; // 8 characters on 32-bit; 16 characters on 64-bit - case 'X' : digits = "0123456789ABCDEF"; - goto hexadecimal; - case 'x' : digits = "0123456789abcdef"; - hexadecimal:if (F.lSize) n = va_arg(arg, unsigned long); - else n = va_arg(arg, unsigned int); - if (F.hSize) n = (unsigned short) n; - if (!F.havePrecision) - { - if (F.zeroPad) - { - F.precision = F.fieldWidth; - if (F.altForm) F.precision -= 2; - } - if (F.precision < 1) F.precision = 1; - } - if (F.precision > mDNS_VACB_Size - 1) - F.precision = mDNS_VACB_Size - 1; - for (i = 0; n; n /= 16, i++) *--s = digits[n % 16]; - for (; i < F.precision; i++) *--s = '0'; - if (F.altForm) { *--s = (char)c; *--s = '0'; i += 2; } - break; - - case 'c' : *--s = (char)va_arg(arg, int); i = 1; break; - - case 's' : s = va_arg(arg, char *); - if (!s) { static char emsg[] = "<>"; s = emsg; i = sizeof(emsg)-1; } - else switch (F.altForm) - { - case 0: i=0; - if (!F.havePrecision) // C string - while (s[i]) i++; - else - { - while ((i < F.precision) && s[i]) i++; - // Make sure we don't truncate in the middle of a UTF-8 character - // If last character we got was any kind of UTF-8 multi-byte character, - // then see if we have to back up. - // This is not as easy as the similar checks below, because - // here we can't assume it's safe to examine the *next* byte, so we - // have to confine ourselves to working only backwards in the string. - j = i; // Record where we got to - // Now, back up until we find first non-continuation-char - while (i>0 && (s[i-1] & 0xC0) == 0x80) i--; - // Now s[i-1] is the first non-continuation-char - // and (j-i) is the number of continuation-chars we found - if (i>0 && (s[i-1] & 0xC0) == 0xC0) // If we found a start-char - { - i--; // Tentatively eliminate this start-char as well - // Now (j-i) is the number of characters we're considering eliminating. - // To be legal UTF-8, the start-char must contain (j-i) one-bits, - // followed by a zero bit. If we shift it right by (7-(j-i)) bits - // (with sign extension) then the result has to be 0xFE. - // If this is right, then we reinstate the tentatively eliminated bytes. - if (((j-i) < 7) && (((s[i] >> (7-(j-i))) & 0xFF) == 0xFE)) i = j; - } - } - break; - case 1: i = (unsigned char) *s++; break; // Pascal string - case 2: { // DNS label-sequence name - unsigned char *a = (unsigned char *)s; - s = mDNS_VACB; // Adjust s to point to the start of the buffer, not the end - if (*a == 0) *s++ = '.'; // Special case for root DNS name - while (*a) - { - char buf[63*4+1]; - if (*a > 63) - { s += mDNS_snprintf(s, mDNS_VACB_Remain(s), "<>", *a); break; } - if (s + *a >= &mDNS_VACB[254]) - { s += mDNS_snprintf(s, mDNS_VACB_Remain(s), "<>"); break; } - // Need to use ConvertDomainLabelToCString to do proper escaping here, - // so it's clear what's a literal dot and what's a label separator - ConvertDomainLabelToCString((domainlabel*)a, buf); - s += mDNS_snprintf(s, mDNS_VACB_Remain(s), "%s.", buf); - a += 1 + *a; - } - i = (mDNSu32)(s - mDNS_VACB); - s = mDNS_VACB; // Reset s back to the start of the buffer - break; - } - } - // Make sure we don't truncate in the middle of a UTF-8 character (see similar comment below) - if (F.havePrecision && i > F.precision) - { i = F.precision; while (i>0 && (s[i] & 0xC0) == 0x80) i--; } - break; - - case 'n' : s = va_arg(arg, char *); - if (F.hSize) * (short *) s = (short)nwritten; - else if (F.lSize) * (long *) s = (long)nwritten; - else * (int *) s = (int)nwritten; - continue; - - default: s = mDNS_VACB; - i = mDNS_snprintf(mDNS_VACB, sizeof(mDNS_VACB), "<>", c); - - case '%' : *sbuffer++ = (char)c; - if (++nwritten >= buflen) goto exit; - break; - } - - if (i < F.fieldWidth && !F.leftJustify) // Pad on the left - do { - *sbuffer++ = ' '; - if (++nwritten >= buflen) goto exit; - } while (i < --F.fieldWidth); - - // Make sure we don't truncate in the middle of a UTF-8 character. - // Note: s[i] is the first eliminated character; i.e. the next character *after* the last character of the - // allowed output. If s[i] is a UTF-8 continuation character, then we've cut a unicode character in half, - // so back up 'i' until s[i] is no longer a UTF-8 continuation character. (if the input was proprly - // formed, s[i] will now be the UTF-8 start character of the multi-byte character we just eliminated). - if (i > buflen - nwritten) - { i = buflen - nwritten; while (i>0 && (s[i] & 0xC0) == 0x80) i--; } - for (j=0; j= buflen) goto exit; - - for (; i < F.fieldWidth; i++) // Pad on the right - { - *sbuffer++ = ' '; - if (++nwritten >= buflen) goto exit; - } - } - } - exit: - *sbuffer++ = 0; - return(nwritten); - } - -mDNSexport mDNSu32 mDNS_snprintf(char *sbuffer, mDNSu32 buflen, const char *fmt, ...) - { - mDNSu32 length; - - va_list ptr; - va_start(ptr,fmt); - length = mDNS_vsnprintf(sbuffer, buflen, fmt, ptr); - va_end(ptr); - - return(length); - } diff -Nru qtcreator-2.5.0/src/tools/mdnssd/DNSCommon.h qtcreator-2.5.2/src/tools/mdnssd/DNSCommon.h --- qtcreator-2.5.0/src/tools/mdnssd/DNSCommon.h 2012-05-09 12:13:19.000000000 +0000 +++ qtcreator-2.5.2/src/tools/mdnssd/DNSCommon.h 1970-01-01 00:00:00.000000000 +0000 @@ -1,292 +0,0 @@ -/* -*- Mode: C; tab-width: 4 -*- - * - * Copyright (c) 2002-2003 Apple Computer, Inc. All rights reserved. - * - * Licensed under the Apache License, Version 2.0 (the "License"); - * you may not use this file except in compliance with the License. - * You may obtain a copy of the License at - * - * http://www.apache.org/licenses/LICENSE-2.0 - * - * Unless required by applicable law or agreed to in writing, software - * distributed under the License is distributed on an "AS IS" BASIS, - * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. - * See the License for the specific language governing permissions and - * limitations under the License. - */ - -#ifndef __DNSCOMMON_H_ -#define __DNSCOMMON_H_ - -#include "mDNSEmbeddedAPI.h" - -#ifdef __cplusplus - extern "C" { -#endif - -//************************************************************************************************************* -// Macros - -// Note: The C preprocessor stringify operator ('#') makes a string from its argument, without macro expansion -// e.g. If "version" is #define'd to be "4", then STRINGIFY_AWE(version) will return the string "version", not "4" -// To expand "version" to its value before making the string, use STRINGIFY(version) instead -#define STRINGIFY_ARGUMENT_WITHOUT_EXPANSION(s) #s -#define STRINGIFY(s) STRINGIFY_ARGUMENT_WITHOUT_EXPANSION(s) - -// *************************************************************************** -#if COMPILER_LIKES_PRAGMA_MARK -#pragma mark - DNS Protocol Constants -#endif - -typedef enum - { - kDNSFlag0_QR_Mask = 0x80, // Query or response? - kDNSFlag0_QR_Query = 0x00, - kDNSFlag0_QR_Response = 0x80, - - kDNSFlag0_OP_Mask = 0x78, // Operation type - kDNSFlag0_OP_StdQuery = 0x00, - kDNSFlag0_OP_Iquery = 0x08, - kDNSFlag0_OP_Status = 0x10, - kDNSFlag0_OP_Unused3 = 0x18, - kDNSFlag0_OP_Notify = 0x20, - kDNSFlag0_OP_Update = 0x28, - - kDNSFlag0_QROP_Mask = kDNSFlag0_QR_Mask | kDNSFlag0_OP_Mask, - - kDNSFlag0_AA = 0x04, // Authoritative Answer? - kDNSFlag0_TC = 0x02, // Truncated? - kDNSFlag0_RD = 0x01, // Recursion Desired? - kDNSFlag1_RA = 0x80, // Recursion Available? - - kDNSFlag1_Zero = 0x40, // Reserved; must be zero - kDNSFlag1_AD = 0x20, // Authentic Data [RFC 2535] - kDNSFlag1_CD = 0x10, // Checking Disabled [RFC 2535] - - kDNSFlag1_RC_Mask = 0x0F, // Response code - kDNSFlag1_RC_NoErr = 0x00, - kDNSFlag1_RC_FormErr = 0x01, - kDNSFlag1_RC_ServFail = 0x02, - kDNSFlag1_RC_NXDomain = 0x03, - kDNSFlag1_RC_NotImpl = 0x04, - kDNSFlag1_RC_Refused = 0x05, - kDNSFlag1_RC_YXDomain = 0x06, - kDNSFlag1_RC_YXRRSet = 0x07, - kDNSFlag1_RC_NXRRSet = 0x08, - kDNSFlag1_RC_NotAuth = 0x09, - kDNSFlag1_RC_NotZone = 0x0A - } DNS_Flags; - -typedef enum - { - TSIG_ErrBadSig = 16, - TSIG_ErrBadKey = 17, - TSIG_ErrBadTime = 18 - } TSIG_ErrorCode; - -// *************************************************************************** -#if COMPILER_LIKES_PRAGMA_MARK -#pragma mark - -#pragma mark - General Utility Functions -#endif - -extern NetworkInterfaceInfo *GetFirstActiveInterface(NetworkInterfaceInfo *intf); -extern mDNSInterfaceID GetNextActiveInterfaceID(const NetworkInterfaceInfo *intf); - -extern mDNSu32 mDNSRandom(mDNSu32 max); // Returns pseudo-random result from zero to max inclusive - -// *************************************************************************** -#if COMPILER_LIKES_PRAGMA_MARK -#pragma mark - -#pragma mark - Domain Name Utility Functions -#endif - -#define mDNSIsDigit(X) ((X) >= '0' && (X) <= '9') -#define mDNSIsUpperCase(X) ((X) >= 'A' && (X) <= 'Z') -#define mDNSIsLowerCase(X) ((X) >= 'a' && (X) <= 'z') -#define mDNSIsLetter(X) (mDNSIsUpperCase(X) || mDNSIsLowerCase(X)) - -#define mDNSValidHostChar(X, notfirst, notlast) (mDNSIsLetter(X) || mDNSIsDigit(X) || ((notfirst) && (notlast) && (X) == '-') ) - -extern mDNSu16 CompressedDomainNameLength(const domainname *const name, const domainname *parent); -extern int CountLabels(const domainname *d); -extern const domainname *SkipLeadingLabels(const domainname *d, int skip); - -extern mDNSu32 TruncateUTF8ToLength(mDNSu8 *string, mDNSu32 length, mDNSu32 max); -extern mDNSBool LabelContainsSuffix(const domainlabel *const name, const mDNSBool RichText); -extern mDNSu32 RemoveLabelSuffix(domainlabel *name, mDNSBool RichText); -extern void AppendLabelSuffix(domainlabel *const name, mDNSu32 val, const mDNSBool RichText); -#define ValidateDomainName(N) (DomainNameLength(N) <= MAX_DOMAIN_NAME) - -// *************************************************************************** -#if COMPILER_LIKES_PRAGMA_MARK -#pragma mark - -#pragma mark - Resource Record Utility Functions -#endif - -// IdenticalResourceRecord returns true if two resources records have -// the same name, type, class, and identical rdata (InterfaceID and TTL may differ) - -// IdenticalSameNameRecord is the same, except it skips the expensive SameDomainName() check, -// which is at its most expensive and least useful in cases where we know in advance that the names match - -// Note: The dominant use of IdenticalResourceRecord is from ProcessQuery(), handling known-answer lists. In this case -// it's common to have a whole bunch or records with exactly the same name (e.g. "_http._tcp.local") but different RDATA. -// The SameDomainName() check is expensive when the names match, and in this case *all* the names match, so we -// used to waste a lot of CPU time verifying that the names match, only then to find that the RDATA is different. -// We observed mDNSResponder spending 30% of its total CPU time on this single task alone. -// By swapping the checks so that we check the RDATA first, we can quickly detect when it's different -// (99% of the time) and then bail out before we waste time on the expensive SameDomainName() check. - -#define IdenticalResourceRecord(r1,r2) ( \ - (r1)->rrtype == (r2)->rrtype && \ - (r1)->rrclass == (r2)->rrclass && \ - (r1)->namehash == (r2)->namehash && \ - (r1)->rdlength == (r2)->rdlength && \ - (r1)->rdatahash == (r2)->rdatahash && \ - SameRDataBody((r1), &(r2)->rdata->u, SameDomainName) && \ - SameDomainName((r1)->name, (r2)->name)) - -#define IdenticalSameNameRecord(r1,r2) ( \ - (r1)->rrtype == (r2)->rrtype && \ - (r1)->rrclass == (r2)->rrclass && \ - (r1)->rdlength == (r2)->rdlength && \ - (r1)->rdatahash == (r2)->rdatahash && \ - SameRDataBody((r1), &(r2)->rdata->u, SameDomainName)) - -// A given RRType answers a QuestionType if RRType is CNAME, or types match, or QuestionType is ANY, -// or the RRType is NSEC and positively asserts the nonexistence of the type being requested -#define RRTypeAnswersQuestionType(R,Q) ((R)->rrtype == kDNSType_CNAME || (R)->rrtype == (Q) || (Q) == kDNSQType_ANY || RRAssertsNonexistence((R),(Q))) -#define RRAssertsNonexistence(R,T) ((R)->rrtype == kDNSType_NSEC && (T) < kDNSQType_ANY && !((R)->rdata->u.nsec.bitmap[(T)>>3] & (128 >> ((T)&7)))) - -extern mDNSu32 RDataHashValue(const ResourceRecord *const rr); -extern mDNSBool SameRDataBody(const ResourceRecord *const r1, const RDataBody *const r2, DomainNameComparisonFn *samename); -extern mDNSBool SameNameRecordAnswersQuestion(const ResourceRecord *const rr, const DNSQuestion *const q); -extern mDNSBool ResourceRecordAnswersQuestion(const ResourceRecord *const rr, const DNSQuestion *const q); -extern mDNSBool AnyTypeRecordAnswersQuestion (const ResourceRecord *const rr, const DNSQuestion *const q); -extern mDNSBool ResourceRecordAnswersUnicastResponse(const ResourceRecord *const rr, const DNSQuestion *const q); -extern mDNSBool LocalOnlyRecordAnswersQuestion(AuthRecord *const rr, const DNSQuestion *const q); -extern mDNSu16 GetRDLength(const ResourceRecord *const rr, mDNSBool estimate); -extern mDNSBool ValidateRData(const mDNSu16 rrtype, const mDNSu16 rdlength, const RData *const rd); - -#define GetRRDomainNameTarget(RR) ( \ - ((RR)->rrtype == kDNSType_NS || (RR)->rrtype == kDNSType_CNAME || (RR)->rrtype == kDNSType_PTR || (RR)->rrtype == kDNSType_DNAME) ? &(RR)->rdata->u.name : \ - ((RR)->rrtype == kDNSType_MX || (RR)->rrtype == kDNSType_AFSDB || (RR)->rrtype == kDNSType_RT || (RR)->rrtype == kDNSType_KX ) ? &(RR)->rdata->u.mx.exchange : \ - ((RR)->rrtype == kDNSType_SRV ) ? &(RR)->rdata->u.srv.target : mDNSNULL ) - -#define LocalRecordReady(X) ((X)->resrec.RecordType != kDNSRecordTypeUnique) - -// *************************************************************************** -#if COMPILER_LIKES_PRAGMA_MARK -#pragma mark - -#pragma mark - DNS Message Creation Functions -#endif - -extern void InitializeDNSMessage(DNSMessageHeader *h, mDNSOpaque16 id, mDNSOpaque16 flags); -extern const mDNSu8 *FindCompressionPointer(const mDNSu8 *const base, const mDNSu8 *const end, const mDNSu8 *const domname); -extern mDNSu8 *putDomainNameAsLabels(const DNSMessage *const msg, mDNSu8 *ptr, const mDNSu8 *const limit, const domainname *const name); -extern mDNSu8 *putRData(const DNSMessage *const msg, mDNSu8 *ptr, const mDNSu8 *const limit, const ResourceRecord *const rr); - -// If we have a single large record to put in the packet, then we allow the packet to be up to 9K bytes, -// but in the normal case we try to keep the packets below 1500 to avoid IP fragmentation on standard Ethernet - -#define AllowedRRSpace(msg) (((msg)->h.numAnswers || (msg)->h.numAuthorities || (msg)->h.numAdditionals) ? NormalMaxDNSMessageData : AbsoluteMaxDNSMessageData) - -extern mDNSu8 *PutResourceRecordTTLWithLimit(DNSMessage *const msg, mDNSu8 *ptr, mDNSu16 *count, ResourceRecord *rr, mDNSu32 ttl, const mDNSu8 *limit); - -#define PutResourceRecordTTL(msg, ptr, count, rr, ttl) \ - PutResourceRecordTTLWithLimit((msg), (ptr), (count), (rr), (ttl), (msg)->data + AllowedRRSpace(msg)) - -#define PutResourceRecordTTLJumbo(msg, ptr, count, rr, ttl) \ - PutResourceRecordTTLWithLimit((msg), (ptr), (count), (rr), (ttl), (msg)->data + AbsoluteMaxDNSMessageData) - -#define PutResourceRecord(MSG, P, C, RR) PutResourceRecordTTL((MSG), (P), (C), (RR), (RR)->rroriginalttl) - -// The PutRR_OS variants assume a local variable 'm', put build the packet at m->omsg, -// and assume a local variable 'OwnerRecordSpace' indicating how many bytes (if any) to reserve to add an OWNER option at the end -#define PutRR_OS_TTL(ptr, count, rr, ttl) \ - PutResourceRecordTTLWithLimit(&m->omsg, (ptr), (count), (rr), (ttl), m->omsg.data + AllowedRRSpace(&m->omsg) - OwnerRecordSpace) - -#define PutRR_OS(P, C, RR) PutRR_OS_TTL((P), (C), (RR), (RR)->rroriginalttl) - -extern mDNSu8 *putQuestion(DNSMessage *const msg, mDNSu8 *ptr, const mDNSu8 *const limit, const domainname *const name, mDNSu16 rrtype, mDNSu16 rrclass); -extern mDNSu8 *putZone(DNSMessage *const msg, mDNSu8 *ptr, mDNSu8 *limit, const domainname *zone, mDNSOpaque16 zoneClass); -extern mDNSu8 *putPrereqNameNotInUse(const domainname *const name, DNSMessage *const msg, mDNSu8 *const ptr, mDNSu8 *const end); -extern mDNSu8 *putDeletionRecord(DNSMessage *msg, mDNSu8 *ptr, ResourceRecord *rr); -extern mDNSu8 *putDeletionRecordWithLimit(DNSMessage *msg, mDNSu8 *ptr, ResourceRecord *rr, mDNSu8 *limit); -extern mDNSu8 *putDeleteRRSetWithLimit(DNSMessage *msg, mDNSu8 *ptr, const domainname *name, mDNSu16 rrtype, mDNSu8 *limit); -extern mDNSu8 *putDeleteAllRRSets(DNSMessage *msg, mDNSu8 *ptr, const domainname *name); -extern mDNSu8 *putUpdateLease(DNSMessage *msg, mDNSu8 *end, mDNSu32 lease); -extern mDNSu8 *putUpdateLeaseWithLimit(DNSMessage *msg, mDNSu8 *ptr, mDNSu32 lease, mDNSu8 *limit); - -extern mDNSu8 *putHINFO(const mDNS *const m, DNSMessage *const msg, mDNSu8 *ptr, DomainAuthInfo *authInfo, mDNSu8 *limit); - -// *************************************************************************** -#if COMPILER_LIKES_PRAGMA_MARK -#pragma mark - -#pragma mark - DNS Message Parsing Functions -#endif - -#define AuthHashSlot(X) (DomainNameHashValue(X) % AUTH_HASH_SLOTS) -#define HashSlot(X) (DomainNameHashValue(X) % CACHE_HASH_SLOTS) -extern mDNSu32 DomainNameHashValue(const domainname *const name); -extern void SetNewRData(ResourceRecord *const rr, RData *NewRData, mDNSu16 rdlength); -extern const mDNSu8 *skipDomainName(const DNSMessage *const msg, const mDNSu8 *ptr, const mDNSu8 *const end); -extern const mDNSu8 *getDomainName(const DNSMessage *const msg, const mDNSu8 *ptr, const mDNSu8 *const end, - domainname *const name); -extern const mDNSu8 *skipResourceRecord(const DNSMessage *msg, const mDNSu8 *ptr, const mDNSu8 *end); -extern const mDNSu8 *GetLargeResourceRecord(mDNS *const m, const DNSMessage * const msg, const mDNSu8 *ptr, - const mDNSu8 * end, const mDNSInterfaceID InterfaceID, mDNSu8 RecordType, LargeCacheRecord *const largecr); -extern const mDNSu8 *skipQuestion(const DNSMessage *msg, const mDNSu8 *ptr, const mDNSu8 *end); -extern const mDNSu8 *getQuestion(const DNSMessage *msg, const mDNSu8 *ptr, const mDNSu8 *end, const mDNSInterfaceID InterfaceID, - DNSQuestion *question); -extern const mDNSu8 *LocateAnswers(const DNSMessage *const msg, const mDNSu8 *const end); -extern const mDNSu8 *LocateAuthorities(const DNSMessage *const msg, const mDNSu8 *const end); -extern const mDNSu8 *LocateAdditionals(const DNSMessage *const msg, const mDNSu8 *const end); -extern const mDNSu8 *LocateOptRR(const DNSMessage *const msg, const mDNSu8 *const end, int minsize); -extern const rdataOPT *GetLLQOptData(mDNS *const m, const DNSMessage *const msg, const mDNSu8 *const end); -extern mDNSu32 GetPktLease(mDNS *m, DNSMessage *msg, const mDNSu8 *end); -extern void DumpPacket(mDNS *const m, mStatus status, mDNSBool sent, char *transport, - const mDNSAddr *srcaddr, mDNSIPPort srcport, - const mDNSAddr *dstaddr, mDNSIPPort dstport, const DNSMessage *const msg, const mDNSu8 *const end); - -// *************************************************************************** -#if COMPILER_LIKES_PRAGMA_MARK -#pragma mark - -#pragma mark - Packet Sending Functions -#endif - -extern mStatus mDNSSendDNSMessage(mDNS *const m, DNSMessage *const msg, mDNSu8 *end, - mDNSInterfaceID InterfaceID, UDPSocket *src, const mDNSAddr *dst, mDNSIPPort dstport, TCPSocket *sock, DomainAuthInfo *authInfo); - -// *************************************************************************** -#if COMPILER_LIKES_PRAGMA_MARK -#pragma mark - -#pragma mark - RR List Management & Task Management -#endif - -extern void ShowTaskSchedulingError(mDNS *const m); -extern void mDNS_Lock_(mDNS *const m, const char * const functionname); -extern void mDNS_Unlock_(mDNS *const m, const char * const functionname); - -#if defined(_WIN32) - #define __func__ __FUNCTION__ -#endif - -#define mDNS_Lock(X) mDNS_Lock_((X), __func__) - -#define mDNS_Unlock(X) mDNS_Unlock_((X), __func__) - -#define mDNS_DropLockBeforeCallback() do { m->mDNS_reentrancy++; \ - if (m->mDNS_busy != m->mDNS_reentrancy) LogMsg("%s: Locking Failure! mDNS_busy (%ld) != mDNS_reentrancy (%ld)", __func__, m->mDNS_busy, m->mDNS_reentrancy); \ - } while (0) - -#define mDNS_ReclaimLockAfterCallback() do { \ - if (m->mDNS_busy != m->mDNS_reentrancy) LogMsg("%s: Unlocking Failure! mDNS_busy (%ld) != mDNS_reentrancy (%ld)", __func__, m->mDNS_busy, m->mDNS_reentrancy); \ - m->mDNS_reentrancy--; } while (0) - -#ifdef __cplusplus - } -#endif - -#endif // __DNSCOMMON_H_ diff -Nru qtcreator-2.5.0/src/tools/mdnssd/DNSDigest.c qtcreator-2.5.2/src/tools/mdnssd/DNSDigest.c --- qtcreator-2.5.0/src/tools/mdnssd/DNSDigest.c 2012-05-09 12:13:19.000000000 +0000 +++ qtcreator-2.5.2/src/tools/mdnssd/DNSDigest.c 1970-01-01 00:00:00.000000000 +0000 @@ -1,1584 +0,0 @@ -/* -*- Mode: C; tab-width: 4 -*- - * - * Copyright (c) 2002-2011 Apple Computer, Inc. All rights reserved. - * - * Licensed under the Apache License, Version 2.0 (the "License"); - * you may not use this file except in compliance with the License. - * You may obtain a copy of the License at - * - * http://www.apache.org/licenses/LICENSE-2.0 - * - * Unless required by applicable law or agreed to in writing, software - * distributed under the License is distributed on an "AS IS" BASIS, - * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. - * See the License for the specific language governing permissions and - * limitations under the License. - */ - - -#ifdef __cplusplus -extern "C" { -#endif - -#include "mDNSEmbeddedAPI.h" -#include "DNSCommon.h" - -// Disable certain benign warnings with Microsoft compilers -#if(defined(_MSC_VER)) - // Disable "conditional expression is constant" warning for debug macros. - // Otherwise, this generates warnings for the perfectly natural construct "while(1)" - // If someone knows a variant way of writing "while(1)" that doesn't generate warning messages, please let us know - #pragma warning(disable:4127) -#endif - - - // *************************************************************************** -#if COMPILER_LIKES_PRAGMA_MARK -#pragma mark - Byte Swapping Functions -#endif - -mDNSlocal mDNSu16 NToH16(mDNSu8 * bytes) - { - return (mDNSu16)((mDNSu16)bytes[0] << 8 | (mDNSu16)bytes[1]); - } - -mDNSlocal mDNSu32 NToH32(mDNSu8 * bytes) - { - return (mDNSu32)((mDNSu32) bytes[0] << 24 | (mDNSu32) bytes[1] << 16 | (mDNSu32) bytes[2] << 8 | (mDNSu32)bytes[3]); - } - - // *************************************************************************** -#if COMPILER_LIKES_PRAGMA_MARK -#pragma mark - MD5 Hash Functions -#endif - - -/* The source for the has is derived CommonCrypto files CommonDigest.h, md32_common.h, md5_locl.h, md5_locl.h, and openssl/md5.h. - * The following changes have been made to the original sources: - * replaced CC_LONG w/ mDNSu32 - * replaced CC_MD5* with MD5* - * replaced CC_LONG w/ mDNSu32, removed conditional #defines from md5.h - * removed extern decls for MD5_Init/Update/Final from CommonDigest.h - * removed APPLE_COMMON_DIGEST specific #defines from md5_locl.h - * - * Note: machine archetecure specific conditionals from the original sources are turned off, but are left in the code - * to aid in platform-specific optimizations and debugging. - * Sources originally distributed under the following license headers: - * CommonDigest.h - APSL - * - * md32_Common.h - * ==================================================================== - * Copyright (c) 1999-2002 The OpenSSL Project. All rights reserved. - * - * Redistribution and use in source and binary forms, with or without - * modification, are permitted provided that the following conditions - * are met: - * - * 1. Redistributions of source code must retain the above copyright - * notice, this list of conditions and the following disclaimer. - * - * 2. Redistributions in binary form must reproduce the above copyright - * notice, this list of conditions and the following disclaimer in - * the documentation and/or other materials provided with the - * distribution. - * - * 3. All advertising materials mentioning features or use of this - * software must display the following acknowledgment: - * "This product includes software developed by the OpenSSL Project - * for use in the OpenSSL Toolkit. (http://www.OpenSSL.org/)" - * - * 4. The names "OpenSSL Toolkit" and "OpenSSL Project" must not be used to - * endorse or promote products derived from this software without - * prior written permission. For written permission, please contact - * licensing@OpenSSL.org. - * - * 5. Products derived from this software may not be called "OpenSSL" - * nor may "OpenSSL" appear in their names without prior written - * permission of the OpenSSL Project. - * - * 6. Redistributions of any form whatsoever must retain the following - * acknowledgment: - * "This product includes software developed by the OpenSSL Project - * for use in the OpenSSL Toolkit (http://www.OpenSSL.org/)" - * - * THIS SOFTWARE IS PROVIDED BY THE OpenSSL PROJECT ``AS IS'' AND ANY - * EXPRESSED OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE - * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR - * PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE OpenSSL PROJECT OR - * ITS CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, - * SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT - * NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; - * LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) - * HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, - * STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) - * ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED - * OF THE POSSIBILITY OF SUCH DAMAGE. - * - * - * md5_dgst.c, md5_locl.h - * ==================================================================== - * - * This product includes cryptographic software written by Eric Young - * (eay@cryptsoft.com). This product includes software written by Tim - * Hudson (tjh@cryptsoft.com). - * - * Copyright (C) 1995-1998 Eric Young (eay@cryptsoft.com) - * All rights reserved. - * - * This package is an SSL implementation written - * by Eric Young (eay@cryptsoft.com). - * The implementation was written so as to conform with Netscapes SSL. - * - * This library is free for commercial and non-commercial use as long as - * the following conditions are aheared to. The following conditions - * apply to all code found in this distribution, be it the RC4, RSA, - * lhash, DES, etc., code; not just the SSL code. The SSL documentation - * included with this distribution is covered by the same copyright terms - * except that the holder is Tim Hudson (tjh@cryptsoft.com). - * - * Copyright remains Eric Young's, and as such any Copyright notices in - * the code are not to be removed. - * If this package is used in a product, Eric Young should be given attribution - * as the author of the parts of the library used. - * This can be in the form of a textual message at program startup or - * in documentation (online or textual) provided with the package. - * - * Redistribution and use in source and binary forms, with or without - * modification, are permitted provided that the following conditions - * are met: - * 1. Redistributions of source code must retain the copyright - * notice, this list of conditions and the following disclaimer. - * 2. Redistributions in binary form must reproduce the above copyright - * notice, this list of conditions and the following disclaimer in the - * documentation and/or other materials provided with the distribution. - * 3. All advertising materials mentioning features or use of this software - * must display the following acknowledgement: - * "This product includes cryptographic software written by - * Eric Young (eay@cryptsoft.com)" - * The word 'cryptographic' can be left out if the rouines from the library - * being used are not cryptographic related :-). - * 4. If you include any Windows specific code (or a derivative thereof) from - * the apps directory (application code) you must include an acknowledgement: - * "This product includes software written by Tim Hudson (tjh@cryptsoft.com)" - * - * THIS SOFTWARE IS PROVIDED BY ERIC YOUNG ``AS IS'' AND - * ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE - * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE - * ARE DISCLAIMED. IN NO EVENT SHALL THE AUTHOR OR CONTRIBUTORS BE LIABLE - * FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL - * DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS - * OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) - * HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT - * LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY - * OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF - * SUCH DAMAGE. - * - * The licence and distribution terms for any publically available version or - * derivative of this code cannot be changed. i.e. this code cannot simply be - * copied and put under another distribution licence - * [including the GNU Public Licence.] - * - */ - -//from CommonDigest.h - -#define MD5_DIGEST_LENGTH 16 /* digest length in bytes */ -#define MD5_BLOCK_BYTES 64 /* block size in bytes */ -#define MD5_BLOCK_LONG (MD5_BLOCK_BYTES / sizeof(mDNSu32)) - -typedef struct MD5state_st -{ - mDNSu32 A,B,C,D; - mDNSu32 Nl,Nh; - mDNSu32 data[MD5_BLOCK_LONG]; - int num; -} MD5_CTX; - - -// from openssl/md5.h - -#define MD5_CBLOCK 64 -#define MD5_LBLOCK (MD5_CBLOCK/4) -#define MD5_DIGEST_LENGTH 16 - -int MD5_Init(MD5_CTX *c); -int MD5_Update(MD5_CTX *c, const void *data, unsigned long len); -int MD5_Final(unsigned char *md, MD5_CTX *c); -void MD5_Transform(MD5_CTX *c, const unsigned char *b); - -// From md5_locl.h - -#ifndef MD5_LONG_LOG2 -#define MD5_LONG_LOG2 2 /* default to 32 bits */ -#endif - -#ifdef MD5_ASM -# if defined(__i386) || defined(__i386__) || defined(_M_IX86) || defined(__INTEL__) -# define md5_block_host_order md5_block_asm_host_order -# elif defined(__sparc) && defined(OPENSSL_SYS_ULTRASPARC) - void md5_block_asm_data_order_aligned (MD5_CTX *c, const mDNSu32 *p,int num); -# define HASH_BLOCK_DATA_ORDER_ALIGNED md5_block_asm_data_order_aligned -# endif -#endif - -void md5_block_host_order (MD5_CTX *c, const void *p,int num); -void md5_block_data_order (MD5_CTX *c, const void *p,int num); - -#if defined(__i386) || defined(__i386__) || defined(_M_IX86) || defined(__INTEL__) -/* - * *_block_host_order is expected to handle aligned data while - * *_block_data_order - unaligned. As algorithm and host (x86) - * are in this case of the same "endianness" these two are - * otherwise indistinguishable. But normally you don't want to - * call the same function because unaligned access in places - * where alignment is expected is usually a "Bad Thing". Indeed, - * on RISCs you get punished with BUS ERROR signal or *severe* - * performance degradation. Intel CPUs are in turn perfectly - * capable of loading unaligned data without such drastic side - * effect. Yes, they say it's slower than aligned load, but no - * exception is generated and therefore performance degradation - * is *incomparable* with RISCs. What we should weight here is - * costs of unaligned access against costs of aligning data. - * According to my measurements allowing unaligned access results - * in ~9% performance improvement on Pentium II operating at - * 266MHz. I won't be surprised if the difference will be higher - * on faster systems:-) - * - * - */ -#define md5_block_data_order md5_block_host_order -#endif - -#define DATA_ORDER_IS_LITTLE_ENDIAN - -#define HASH_LONG mDNSu32 -#define HASH_LONG_LOG2 MD5_LONG_LOG2 -#define HASH_CTX MD5_CTX -#define HASH_CBLOCK MD5_CBLOCK -#define HASH_LBLOCK MD5_LBLOCK - -#define HASH_UPDATE MD5_Update -#define HASH_TRANSFORM MD5_Transform -#define HASH_FINAL MD5_Final - -#define HASH_MAKE_STRING(c,s) do { \ - unsigned long ll; \ - ll=(c)->A; HOST_l2c(ll,(s)); \ - ll=(c)->B; HOST_l2c(ll,(s)); \ - ll=(c)->C; HOST_l2c(ll,(s)); \ - ll=(c)->D; HOST_l2c(ll,(s)); \ - } while (0) -#define HASH_BLOCK_HOST_ORDER md5_block_host_order -#if !defined(L_ENDIAN) || defined(md5_block_data_order) -#define HASH_BLOCK_DATA_ORDER md5_block_data_order -/* - * Little-endians (Intel and Alpha) feel better without this. - * It looks like memcpy does better job than generic - * md5_block_data_order on copying-n-aligning input data. - * But frankly speaking I didn't expect such result on Alpha. - * On the other hand I've got this with egcs-1.0.2 and if - * program is compiled with another (better?) compiler it - * might turn out other way around. - * - * - */ -#endif - - -// from md32_common.h - -/* - * This is a generic 32 bit "collector" for message digest algorithms. - * Whenever needed it collects input character stream into chunks of - * 32 bit values and invokes a block function that performs actual hash - * calculations. - * - * Porting guide. - * - * Obligatory macros: - * - * DATA_ORDER_IS_BIG_ENDIAN or DATA_ORDER_IS_LITTLE_ENDIAN - * this macro defines byte order of input stream. - * HASH_CBLOCK - * size of a unit chunk HASH_BLOCK operates on. - * HASH_LONG - * has to be at lest 32 bit wide, if it's wider, then - * HASH_LONG_LOG2 *has to* be defined along - * HASH_CTX - * context structure that at least contains following - * members: - * typedef struct { - * ... - * HASH_LONG Nl,Nh; - * HASH_LONG data[HASH_LBLOCK]; - * int num; - * ... - * } HASH_CTX; - * HASH_UPDATE - * name of "Update" function, implemented here. - * HASH_TRANSFORM - * name of "Transform" function, implemented here. - * HASH_FINAL - * name of "Final" function, implemented here. - * HASH_BLOCK_HOST_ORDER - * name of "block" function treating *aligned* input message - * in host byte order, implemented externally. - * HASH_BLOCK_DATA_ORDER - * name of "block" function treating *unaligned* input message - * in original (data) byte order, implemented externally (it - * actually is optional if data and host are of the same - * "endianess"). - * HASH_MAKE_STRING - * macro convering context variables to an ASCII hash string. - * - * Optional macros: - * - * B_ENDIAN or L_ENDIAN - * defines host byte-order. - * HASH_LONG_LOG2 - * defaults to 2 if not states otherwise. - * HASH_LBLOCK - * assumed to be HASH_CBLOCK/4 if not stated otherwise. - * HASH_BLOCK_DATA_ORDER_ALIGNED - * alternative "block" function capable of treating - * aligned input message in original (data) order, - * implemented externally. - * - * MD5 example: - * - * #define DATA_ORDER_IS_LITTLE_ENDIAN - * - * #define HASH_LONG mDNSu32 - * #define HASH_LONG_LOG2 mDNSu32_LOG2 - * #define HASH_CTX MD5_CTX - * #define HASH_CBLOCK MD5_CBLOCK - * #define HASH_LBLOCK MD5_LBLOCK - * #define HASH_UPDATE MD5_Update - * #define HASH_TRANSFORM MD5_Transform - * #define HASH_FINAL MD5_Final - * #define HASH_BLOCK_HOST_ORDER md5_block_host_order - * #define HASH_BLOCK_DATA_ORDER md5_block_data_order - * - * - */ - -#if !defined(DATA_ORDER_IS_BIG_ENDIAN) && !defined(DATA_ORDER_IS_LITTLE_ENDIAN) -#error "DATA_ORDER must be defined!" -#endif - -#ifndef HASH_CBLOCK -#error "HASH_CBLOCK must be defined!" -#endif -#ifndef HASH_LONG -#error "HASH_LONG must be defined!" -#endif -#ifndef HASH_CTX -#error "HASH_CTX must be defined!" -#endif - -#ifndef HASH_UPDATE -#error "HASH_UPDATE must be defined!" -#endif -#ifndef HASH_TRANSFORM -#error "HASH_TRANSFORM must be defined!" -#endif -#ifndef HASH_FINAL -#error "HASH_FINAL must be defined!" -#endif - -#ifndef HASH_BLOCK_HOST_ORDER -#error "HASH_BLOCK_HOST_ORDER must be defined!" -#endif - -#if 0 -/* - * Moved below as it's required only if HASH_BLOCK_DATA_ORDER_ALIGNED - * isn't defined. - */ -#ifndef HASH_BLOCK_DATA_ORDER -#error "HASH_BLOCK_DATA_ORDER must be defined!" -#endif -#endif - -#ifndef HASH_LBLOCK -#define HASH_LBLOCK (HASH_CBLOCK/4) -#endif - -#ifndef HASH_LONG_LOG2 -#define HASH_LONG_LOG2 2 -#endif - -/* - * Engage compiler specific rotate intrinsic function if available. - */ -#undef ROTATE -#ifndef PEDANTIC -# if 0 /* defined(_MSC_VER) */ -# define ROTATE(a,n) _lrotl(a,n) -# elif defined(__MWERKS__) -# if defined(__POWERPC__) -# define ROTATE(a,n) (unsigned MD32_REG_T)__rlwinm((int)a,n,0,31) -# elif defined(__MC68K__) - /* Motorola specific tweak. */ -# define ROTATE(a,n) (n<24 ? __rol(a,n) : __ror(a,32-n)) -# else -# define ROTATE(a,n) __rol(a,n) -# endif -# elif defined(__GNUC__) && __GNUC__>=2 && !defined(OPENSSL_NO_ASM) && !defined(OPENSSL_NO_INLINE_ASM) - /* - * Some GNU C inline assembler templates. Note that these are - * rotates by *constant* number of bits! But that's exactly - * what we need here... - * - * - */ - /* - * LLVM is more strict about compatibility of types between input & output constraints, - * but we want these to be rotations of 32 bits, not 64, so we explicitly drop the - * most significant bytes by casting to an unsigned int. - */ -# if defined(__i386) || defined(__i386__) || defined(__x86_64) || defined(__x86_64__) -# define ROTATE(a,n) ({ register unsigned int ret; \ - asm ( \ - "roll %1,%0" \ - : "=r"(ret) \ - : "I"(n), "0"((unsigned int)a) \ - : "cc"); \ - ret; \ - }) -# elif defined(__powerpc) || defined(__ppc) -# define ROTATE(a,n) ({ register unsigned int ret; \ - asm ( \ - "rlwinm %0,%1,%2,0,31" \ - : "=r"(ret) \ - : "r"(a), "I"(n)); \ - ret; \ - }) -# endif -# endif - -/* - * Engage compiler specific "fetch in reverse byte order" - * intrinsic function if available. - */ -# if defined(__GNUC__) && __GNUC__>=2 && !defined(OPENSSL_NO_ASM) && !defined(OPENSSL_NO_INLINE_ASM) - /* some GNU C inline assembler templates by */ -# if (defined(__i386) || defined(__i386__) || defined(__x86_64) || defined(__x86_64__)) && !defined(I386_ONLY) -# define BE_FETCH32(a) ({ register unsigned int l=(a);\ - asm ( \ - "bswapl %0" \ - : "=r"(l) : "0"(l)); \ - l; \ - }) -# elif defined(__powerpc) -# define LE_FETCH32(a) ({ register unsigned int l; \ - asm ( \ - "lwbrx %0,0,%1" \ - : "=r"(l) \ - : "r"(a)); \ - l; \ - }) - -# elif defined(__sparc) && defined(OPENSSL_SYS_ULTRASPARC) -# define LE_FETCH32(a) ({ register unsigned int l; \ - asm ( \ - "lda [%1]#ASI_PRIMARY_LITTLE,%0"\ - : "=r"(l) \ - : "r"(a)); \ - l; \ - }) -# endif -# endif -#endif /* PEDANTIC */ - -#if HASH_LONG_LOG2==2 /* Engage only if sizeof(HASH_LONG)== 4 */ -/* A nice byte order reversal from Wei Dai */ -#ifdef ROTATE -/* 5 instructions with rotate instruction, else 9 */ -#define REVERSE_FETCH32(a,l) ( \ - l=*(const HASH_LONG *)(a), \ - ((ROTATE(l,8)&0x00FF00FF)|(ROTATE((l&0x00FF00FF),24))) \ - ) -#else -/* 6 instructions with rotate instruction, else 8 */ -#define REVERSE_FETCH32(a,l) ( \ - l=*(const HASH_LONG *)(a), \ - l=(((l>>8)&0x00FF00FF)|((l&0x00FF00FF)<<8)), \ - ROTATE(l,16) \ - ) -/* - * Originally the middle line started with l=(((l&0xFF00FF00)>>8)|... - * It's rewritten as above for two reasons: - * - RISCs aren't good at long constants and have to explicitely - * compose 'em with several (well, usually 2) instructions in a - * register before performing the actual operation and (as you - * already realized:-) having same constant should inspire the - * compiler to permanently allocate the only register for it; - * - most modern CPUs have two ALUs, but usually only one has - * circuitry for shifts:-( this minor tweak inspires compiler - * to schedule shift instructions in a better way... - * - * - */ -#endif -#endif - -#ifndef ROTATE -#define ROTATE(a,n) (((a)<<(n))|(((a)&0xffffffff)>>(32-(n)))) -#endif - -/* - * Make some obvious choices. E.g., HASH_BLOCK_DATA_ORDER_ALIGNED - * and HASH_BLOCK_HOST_ORDER ought to be the same if input data - * and host are of the same "endianess". It's possible to mask - * this with blank #define HASH_BLOCK_DATA_ORDER though... - * - * - */ -#if defined(B_ENDIAN) -# if defined(DATA_ORDER_IS_BIG_ENDIAN) -# if !defined(HASH_BLOCK_DATA_ORDER_ALIGNED) && HASH_LONG_LOG2==2 -# define HASH_BLOCK_DATA_ORDER_ALIGNED HASH_BLOCK_HOST_ORDER -# endif -# elif defined(DATA_ORDER_IS_LITTLE_ENDIAN) -# ifndef HOST_FETCH32 -# ifdef LE_FETCH32 -# define HOST_FETCH32(p,l) LE_FETCH32(p) -# elif defined(REVERSE_FETCH32) -# define HOST_FETCH32(p,l) REVERSE_FETCH32(p,l) -# endif -# endif -# endif -#elif defined(L_ENDIAN) -# if defined(DATA_ORDER_IS_LITTLE_ENDIAN) -# if !defined(HASH_BLOCK_DATA_ORDER_ALIGNED) && HASH_LONG_LOG2==2 -# define HASH_BLOCK_DATA_ORDER_ALIGNED HASH_BLOCK_HOST_ORDER -# endif -# elif defined(DATA_ORDER_IS_BIG_ENDIAN) -# ifndef HOST_FETCH32 -# ifdef BE_FETCH32 -# define HOST_FETCH32(p,l) BE_FETCH32(p) -# elif defined(REVERSE_FETCH32) -# define HOST_FETCH32(p,l) REVERSE_FETCH32(p,l) -# endif -# endif -# endif -#endif - -#if !defined(HASH_BLOCK_DATA_ORDER_ALIGNED) -#ifndef HASH_BLOCK_DATA_ORDER -#error "HASH_BLOCK_DATA_ORDER must be defined!" -#endif -#endif - -// None of the invocations of the following macros actually use the result, -// so cast them to void to avoid any compiler warnings/errors about not using -// the result (e.g. when using clang). -// If the resultant values need to be used at some point, these must be changed. -#define HOST_c2l(c,l) ((void)_HOST_c2l(c,l)) -#define HOST_l2c(l,c) ((void)_HOST_l2c(l,c)) - -#if defined(DATA_ORDER_IS_BIG_ENDIAN) - -#define _HOST_c2l(c,l) (l =(((unsigned long)(*((c)++)))<<24), \ - l|=(((unsigned long)(*((c)++)))<<16), \ - l|=(((unsigned long)(*((c)++)))<< 8), \ - l|=(((unsigned long)(*((c)++))) ), \ - l) -#define HOST_p_c2l(c,l,n) { \ - switch (n) { \ - case 0: l =((unsigned long)(*((c)++)))<<24; \ - case 1: l|=((unsigned long)(*((c)++)))<<16; \ - case 2: l|=((unsigned long)(*((c)++)))<< 8; \ - case 3: l|=((unsigned long)(*((c)++))); \ - } } -#define HOST_p_c2l_p(c,l,sc,len) { \ - switch (sc) { \ - case 0: l =((unsigned long)(*((c)++)))<<24; \ - if (--len == 0) break; \ - case 1: l|=((unsigned long)(*((c)++)))<<16; \ - if (--len == 0) break; \ - case 2: l|=((unsigned long)(*((c)++)))<< 8; \ - } } -/* NOTE the pointer is not incremented at the end of this */ -#define HOST_c2l_p(c,l,n) { \ - l=0; (c)+=n; \ - switch (n) { \ - case 3: l =((unsigned long)(*(--(c))))<< 8; \ - case 2: l|=((unsigned long)(*(--(c))))<<16; \ - case 1: l|=((unsigned long)(*(--(c))))<<24; \ - } } -#define _HOST_l2c(l,c) (*((c)++)=(unsigned char)(((l)>>24)&0xff), \ - *((c)++)=(unsigned char)(((l)>>16)&0xff), \ - *((c)++)=(unsigned char)(((l)>> 8)&0xff), \ - *((c)++)=(unsigned char)(((l) )&0xff), \ - l) - -#elif defined(DATA_ORDER_IS_LITTLE_ENDIAN) - -#define _HOST_c2l(c,l) (l =(((unsigned long)(*((c)++))) ), \ - l|=(((unsigned long)(*((c)++)))<< 8), \ - l|=(((unsigned long)(*((c)++)))<<16), \ - l|=(((unsigned long)(*((c)++)))<<24), \ - l) -#define HOST_p_c2l(c,l,n) { \ - switch (n) { \ - case 0: l =((unsigned long)(*((c)++))); \ - case 1: l|=((unsigned long)(*((c)++)))<< 8; \ - case 2: l|=((unsigned long)(*((c)++)))<<16; \ - case 3: l|=((unsigned long)(*((c)++)))<<24; \ - } } -#define HOST_p_c2l_p(c,l,sc,len) { \ - switch (sc) { \ - case 0: l =((unsigned long)(*((c)++))); \ - if (--len == 0) break; \ - case 1: l|=((unsigned long)(*((c)++)))<< 8; \ - if (--len == 0) break; \ - case 2: l|=((unsigned long)(*((c)++)))<<16; \ - } } -/* NOTE the pointer is not incremented at the end of this */ -#define HOST_c2l_p(c,l,n) { \ - l=0; (c)+=n; \ - switch (n) { \ - case 3: l =((unsigned long)(*(--(c))))<<16; \ - case 2: l|=((unsigned long)(*(--(c))))<< 8; \ - case 1: l|=((unsigned long)(*(--(c)))); \ - } } -#define _HOST_l2c(l,c) (*((c)++)=(unsigned char)(((l) )&0xff), \ - *((c)++)=(unsigned char)(((l)>> 8)&0xff), \ - *((c)++)=(unsigned char)(((l)>>16)&0xff), \ - *((c)++)=(unsigned char)(((l)>>24)&0xff), \ - l) - -#endif - -/* - * Time for some action:-) - */ - -int HASH_UPDATE (HASH_CTX *c, const void *data_, unsigned long len) - { - const unsigned char *data=(const unsigned char *)data_; - register HASH_LONG * p; - register unsigned long l; - int sw,sc,ew,ec; - - if (len==0) return 1; - - l=(c->Nl+(len<<3))&0xffffffffL; - /* 95-05-24 eay Fixed a bug with the overflow handling, thanks to - * Wei Dai for pointing it out. */ - if (l < c->Nl) /* overflow */ - c->Nh++; - c->Nh+=(len>>29); - c->Nl=l; - - if (c->num != 0) - { - p=c->data; - sw=c->num>>2; - sc=c->num&0x03; - - if ((c->num+len) >= HASH_CBLOCK) - { - l=p[sw]; HOST_p_c2l(data,l,sc); p[sw++]=l; - for (; swnum); - c->num=0; - /* drop through and do the rest */ - } - else - { - c->num+=len; - if ((sc+len) < 4) /* ugly, add char's to a word */ - { - l=p[sw]; HOST_p_c2l_p(data,l,sc,len); p[sw]=l; - } - else - { - ew=(c->num>>2); - ec=(c->num&0x03); - if (sc) - l=p[sw]; - HOST_p_c2l(data,l,sc); - p[sw++]=l; - for (; sw < ew; sw++) - { - HOST_c2l(data,l); p[sw]=l; - } - if (ec) - { - HOST_c2l_p(data,l,ec); p[sw]=l; - } - } - return 1; - } - } - - sw=(int)(len/HASH_CBLOCK); - if (sw > 0) - { -#if defined(HASH_BLOCK_DATA_ORDER_ALIGNED) - /* - * Note that HASH_BLOCK_DATA_ORDER_ALIGNED gets defined - * only if sizeof(HASH_LONG)==4. - */ - if ((((unsigned long)data)%4) == 0) - { - /* data is properly aligned so that we can cast it: */ - HASH_BLOCK_DATA_ORDER_ALIGNED (c,(HASH_LONG *)data,sw); - sw*=HASH_CBLOCK; - data+=sw; - len-=sw; - } - else -#if !defined(HASH_BLOCK_DATA_ORDER) - while (sw--) - { - mDNSPlatformMemCopy(p=c->data,data,HASH_CBLOCK); - HASH_BLOCK_DATA_ORDER_ALIGNED(c,p,1); - data+=HASH_CBLOCK; - len-=HASH_CBLOCK; - } -#endif -#endif -#if defined(HASH_BLOCK_DATA_ORDER) - { - HASH_BLOCK_DATA_ORDER(c,data,sw); - sw*=HASH_CBLOCK; - data+=sw; - len-=sw; - } -#endif - } - - if (len!=0) - { - p = c->data; - c->num = (int)len; - ew=(int)(len>>2); /* words to copy */ - ec=(int)(len&0x03); - for (; ew; ew--,p++) - { - HOST_c2l(data,l); *p=l; - } - HOST_c2l_p(data,l,ec); - *p=l; - } - return 1; - } - - -void HASH_TRANSFORM (HASH_CTX *c, const unsigned char *data) - { -#if defined(HASH_BLOCK_DATA_ORDER_ALIGNED) - if ((((unsigned long)data)%4) == 0) - /* data is properly aligned so that we can cast it: */ - HASH_BLOCK_DATA_ORDER_ALIGNED (c,(HASH_LONG *)data,1); - else -#if !defined(HASH_BLOCK_DATA_ORDER) - { - mDNSPlatformMemCopy(c->data,data,HASH_CBLOCK); - HASH_BLOCK_DATA_ORDER_ALIGNED (c,c->data,1); - } -#endif -#endif -#if defined(HASH_BLOCK_DATA_ORDER) - HASH_BLOCK_DATA_ORDER (c,data,1); -#endif - } - - -int HASH_FINAL (unsigned char *md, HASH_CTX *c) - { - register HASH_LONG *p; - register unsigned long l; - register int i,j; - static const unsigned char end[4]={0x80,0x00,0x00,0x00}; - const unsigned char *cp=end; - - /* c->num should definitly have room for at least one more byte. */ - p=c->data; - i=c->num>>2; - j=c->num&0x03; - -#if 0 - /* purify often complains about the following line as an - * Uninitialized Memory Read. While this can be true, the - * following p_c2l macro will reset l when that case is true. - * This is because j&0x03 contains the number of 'valid' bytes - * already in p[i]. If and only if j&0x03 == 0, the UMR will - * occur but this is also the only time p_c2l will do - * l= *(cp++) instead of l|= *(cp++) - * Many thanks to Alex Tang for pickup this - * 'potential bug' */ -#ifdef PURIFY - if (j==0) p[i]=0; /* Yeah, but that's not the way to fix it:-) */ -#endif - l=p[i]; -#else - l = (j==0) ? 0 : p[i]; -#endif - HOST_p_c2l(cp,l,j); p[i++]=l; /* i is the next 'undefined word' */ - - if (i>(HASH_LBLOCK-2)) /* save room for Nl and Nh */ - { - if (iNh; - p[HASH_LBLOCK-1]=c->Nl; -#elif defined(DATA_ORDER_IS_LITTLE_ENDIAN) - p[HASH_LBLOCK-2]=c->Nl; - p[HASH_LBLOCK-1]=c->Nh; -#endif - HASH_BLOCK_HOST_ORDER (c,p,1); - -#ifndef HASH_MAKE_STRING -#error "HASH_MAKE_STRING must be defined!" -#else - HASH_MAKE_STRING(c,md); -#endif - - c->num=0; - /* clear stuff, HASH_BLOCK may be leaving some stuff on the stack - * but I'm not worried :-) - OPENSSL_cleanse((void *)c,sizeof(HASH_CTX)); - */ - return 1; - } - -#ifndef MD32_REG_T -#define MD32_REG_T long -/* - * This comment was originaly written for MD5, which is why it - * discusses A-D. But it basically applies to all 32-bit digests, - * which is why it was moved to common header file. - * - * In case you wonder why A-D are declared as long and not - * as mDNSu32. Doing so results in slight performance - * boost on LP64 architectures. The catch is we don't - * really care if 32 MSBs of a 64-bit register get polluted - * with eventual overflows as we *save* only 32 LSBs in - * *either* case. Now declaring 'em long excuses the compiler - * from keeping 32 MSBs zeroed resulting in 13% performance - * improvement under SPARC Solaris7/64 and 5% under AlphaLinux. - * Well, to be honest it should say that this *prevents* - * performance degradation. - * - * Apparently there're LP64 compilers that generate better - * code if A-D are declared int. Most notably GCC-x86_64 - * generates better code. - * - */ -#endif - - -// from md5_locl.h (continued) - -/* -#define F(x,y,z) (((x) & (y)) | ((~(x)) & (z))) -#define G(x,y,z) (((x) & (z)) | ((y) & (~(z)))) -*/ - -/* As pointed out by Wei Dai , the above can be - * simplified to the code below. Wei attributes these optimizations - * to Peter Gutmann's SHS code, and he attributes it to Rich Schroeppel. - */ -#define F(b,c,d) ((((c) ^ (d)) & (b)) ^ (d)) -#define G(b,c,d) ((((b) ^ (c)) & (d)) ^ (c)) -#define H(b,c,d) ((b) ^ (c) ^ (d)) -#define I(b,c,d) (((~(d)) | (b)) ^ (c)) - -#define R0(a,b,c,d,k,s,t) { \ - a+=((k)+(t)+F((b),(c),(d))); \ - a=ROTATE(a,s); \ - a+=b; };\ - -#define R1(a,b,c,d,k,s,t) { \ - a+=((k)+(t)+G((b),(c),(d))); \ - a=ROTATE(a,s); \ - a+=b; }; - -#define R2(a,b,c,d,k,s,t) { \ - a+=((k)+(t)+H((b),(c),(d))); \ - a=ROTATE(a,s); \ - a+=b; }; - -#define R3(a,b,c,d,k,s,t) { \ - a+=((k)+(t)+I((b),(c),(d))); \ - a=ROTATE(a,s); \ - a+=b; }; - -// from md5_dgst.c - - -/* Implemented from RFC1321 The MD5 Message-Digest Algorithm - */ - -#define INIT_DATA_A (unsigned long)0x67452301L -#define INIT_DATA_B (unsigned long)0xefcdab89L -#define INIT_DATA_C (unsigned long)0x98badcfeL -#define INIT_DATA_D (unsigned long)0x10325476L - -int MD5_Init(MD5_CTX *c) - { - c->A=INIT_DATA_A; - c->B=INIT_DATA_B; - c->C=INIT_DATA_C; - c->D=INIT_DATA_D; - c->Nl=0; - c->Nh=0; - c->num=0; - return 1; - } - -#ifndef md5_block_host_order -void md5_block_host_order (MD5_CTX *c, const void *data, int num) - { - const mDNSu32 *X=(const mDNSu32 *)data; - register unsigned MD32_REG_T A,B,C,D; - - A=c->A; - B=c->B; - C=c->C; - D=c->D; - - for (;num--;X+=HASH_LBLOCK) - { - /* Round 0 */ - R0(A,B,C,D,X[ 0], 7,0xd76aa478L); - R0(D,A,B,C,X[ 1],12,0xe8c7b756L); - R0(C,D,A,B,X[ 2],17,0x242070dbL); - R0(B,C,D,A,X[ 3],22,0xc1bdceeeL); - R0(A,B,C,D,X[ 4], 7,0xf57c0fafL); - R0(D,A,B,C,X[ 5],12,0x4787c62aL); - R0(C,D,A,B,X[ 6],17,0xa8304613L); - R0(B,C,D,A,X[ 7],22,0xfd469501L); - R0(A,B,C,D,X[ 8], 7,0x698098d8L); - R0(D,A,B,C,X[ 9],12,0x8b44f7afL); - R0(C,D,A,B,X[10],17,0xffff5bb1L); - R0(B,C,D,A,X[11],22,0x895cd7beL); - R0(A,B,C,D,X[12], 7,0x6b901122L); - R0(D,A,B,C,X[13],12,0xfd987193L); - R0(C,D,A,B,X[14],17,0xa679438eL); - R0(B,C,D,A,X[15],22,0x49b40821L); - /* Round 1 */ - R1(A,B,C,D,X[ 1], 5,0xf61e2562L); - R1(D,A,B,C,X[ 6], 9,0xc040b340L); - R1(C,D,A,B,X[11],14,0x265e5a51L); - R1(B,C,D,A,X[ 0],20,0xe9b6c7aaL); - R1(A,B,C,D,X[ 5], 5,0xd62f105dL); - R1(D,A,B,C,X[10], 9,0x02441453L); - R1(C,D,A,B,X[15],14,0xd8a1e681L); - R1(B,C,D,A,X[ 4],20,0xe7d3fbc8L); - R1(A,B,C,D,X[ 9], 5,0x21e1cde6L); - R1(D,A,B,C,X[14], 9,0xc33707d6L); - R1(C,D,A,B,X[ 3],14,0xf4d50d87L); - R1(B,C,D,A,X[ 8],20,0x455a14edL); - R1(A,B,C,D,X[13], 5,0xa9e3e905L); - R1(D,A,B,C,X[ 2], 9,0xfcefa3f8L); - R1(C,D,A,B,X[ 7],14,0x676f02d9L); - R1(B,C,D,A,X[12],20,0x8d2a4c8aL); - /* Round 2 */ - R2(A,B,C,D,X[ 5], 4,0xfffa3942L); - R2(D,A,B,C,X[ 8],11,0x8771f681L); - R2(C,D,A,B,X[11],16,0x6d9d6122L); - R2(B,C,D,A,X[14],23,0xfde5380cL); - R2(A,B,C,D,X[ 1], 4,0xa4beea44L); - R2(D,A,B,C,X[ 4],11,0x4bdecfa9L); - R2(C,D,A,B,X[ 7],16,0xf6bb4b60L); - R2(B,C,D,A,X[10],23,0xbebfbc70L); - R2(A,B,C,D,X[13], 4,0x289b7ec6L); - R2(D,A,B,C,X[ 0],11,0xeaa127faL); - R2(C,D,A,B,X[ 3],16,0xd4ef3085L); - R2(B,C,D,A,X[ 6],23,0x04881d05L); - R2(A,B,C,D,X[ 9], 4,0xd9d4d039L); - R2(D,A,B,C,X[12],11,0xe6db99e5L); - R2(C,D,A,B,X[15],16,0x1fa27cf8L); - R2(B,C,D,A,X[ 2],23,0xc4ac5665L); - /* Round 3 */ - R3(A,B,C,D,X[ 0], 6,0xf4292244L); - R3(D,A,B,C,X[ 7],10,0x432aff97L); - R3(C,D,A,B,X[14],15,0xab9423a7L); - R3(B,C,D,A,X[ 5],21,0xfc93a039L); - R3(A,B,C,D,X[12], 6,0x655b59c3L); - R3(D,A,B,C,X[ 3],10,0x8f0ccc92L); - R3(C,D,A,B,X[10],15,0xffeff47dL); - R3(B,C,D,A,X[ 1],21,0x85845dd1L); - R3(A,B,C,D,X[ 8], 6,0x6fa87e4fL); - R3(D,A,B,C,X[15],10,0xfe2ce6e0L); - R3(C,D,A,B,X[ 6],15,0xa3014314L); - R3(B,C,D,A,X[13],21,0x4e0811a1L); - R3(A,B,C,D,X[ 4], 6,0xf7537e82L); - R3(D,A,B,C,X[11],10,0xbd3af235L); - R3(C,D,A,B,X[ 2],15,0x2ad7d2bbL); - R3(B,C,D,A,X[ 9],21,0xeb86d391L); - - A = c->A += A; - B = c->B += B; - C = c->C += C; - D = c->D += D; - } - } -#endif - -#ifndef md5_block_data_order -#ifdef X -#undef X -#endif -void md5_block_data_order (MD5_CTX *c, const void *data_, int num) - { - const unsigned char *data=data_; - register unsigned MD32_REG_T A,B,C,D,l; -#ifndef MD32_XARRAY - /* See comment in crypto/sha/sha_locl.h for details. */ - unsigned MD32_REG_T XX0, XX1, XX2, XX3, XX4, XX5, XX6, XX7, - XX8, XX9,XX10,XX11,XX12,XX13,XX14,XX15; -# define X(i) XX##i -#else - mDNSu32 XX[MD5_LBLOCK]; -# define X(i) XX[i] -#endif - - A=c->A; - B=c->B; - C=c->C; - D=c->D; - - for (;num--;) - { - HOST_c2l(data,l); X( 0)=l; HOST_c2l(data,l); X( 1)=l; - /* Round 0 */ - R0(A,B,C,D,X( 0), 7,0xd76aa478L); HOST_c2l(data,l); X( 2)=l; - R0(D,A,B,C,X( 1),12,0xe8c7b756L); HOST_c2l(data,l); X( 3)=l; - R0(C,D,A,B,X( 2),17,0x242070dbL); HOST_c2l(data,l); X( 4)=l; - R0(B,C,D,A,X( 3),22,0xc1bdceeeL); HOST_c2l(data,l); X( 5)=l; - R0(A,B,C,D,X( 4), 7,0xf57c0fafL); HOST_c2l(data,l); X( 6)=l; - R0(D,A,B,C,X( 5),12,0x4787c62aL); HOST_c2l(data,l); X( 7)=l; - R0(C,D,A,B,X( 6),17,0xa8304613L); HOST_c2l(data,l); X( 8)=l; - R0(B,C,D,A,X( 7),22,0xfd469501L); HOST_c2l(data,l); X( 9)=l; - R0(A,B,C,D,X( 8), 7,0x698098d8L); HOST_c2l(data,l); X(10)=l; - R0(D,A,B,C,X( 9),12,0x8b44f7afL); HOST_c2l(data,l); X(11)=l; - R0(C,D,A,B,X(10),17,0xffff5bb1L); HOST_c2l(data,l); X(12)=l; - R0(B,C,D,A,X(11),22,0x895cd7beL); HOST_c2l(data,l); X(13)=l; - R0(A,B,C,D,X(12), 7,0x6b901122L); HOST_c2l(data,l); X(14)=l; - R0(D,A,B,C,X(13),12,0xfd987193L); HOST_c2l(data,l); X(15)=l; - R0(C,D,A,B,X(14),17,0xa679438eL); - R0(B,C,D,A,X(15),22,0x49b40821L); - /* Round 1 */ - R1(A,B,C,D,X( 1), 5,0xf61e2562L); - R1(D,A,B,C,X( 6), 9,0xc040b340L); - R1(C,D,A,B,X(11),14,0x265e5a51L); - R1(B,C,D,A,X( 0),20,0xe9b6c7aaL); - R1(A,B,C,D,X( 5), 5,0xd62f105dL); - R1(D,A,B,C,X(10), 9,0x02441453L); - R1(C,D,A,B,X(15),14,0xd8a1e681L); - R1(B,C,D,A,X( 4),20,0xe7d3fbc8L); - R1(A,B,C,D,X( 9), 5,0x21e1cde6L); - R1(D,A,B,C,X(14), 9,0xc33707d6L); - R1(C,D,A,B,X( 3),14,0xf4d50d87L); - R1(B,C,D,A,X( 8),20,0x455a14edL); - R1(A,B,C,D,X(13), 5,0xa9e3e905L); - R1(D,A,B,C,X( 2), 9,0xfcefa3f8L); - R1(C,D,A,B,X( 7),14,0x676f02d9L); - R1(B,C,D,A,X(12),20,0x8d2a4c8aL); - /* Round 2 */ - R2(A,B,C,D,X( 5), 4,0xfffa3942L); - R2(D,A,B,C,X( 8),11,0x8771f681L); - R2(C,D,A,B,X(11),16,0x6d9d6122L); - R2(B,C,D,A,X(14),23,0xfde5380cL); - R2(A,B,C,D,X( 1), 4,0xa4beea44L); - R2(D,A,B,C,X( 4),11,0x4bdecfa9L); - R2(C,D,A,B,X( 7),16,0xf6bb4b60L); - R2(B,C,D,A,X(10),23,0xbebfbc70L); - R2(A,B,C,D,X(13), 4,0x289b7ec6L); - R2(D,A,B,C,X( 0),11,0xeaa127faL); - R2(C,D,A,B,X( 3),16,0xd4ef3085L); - R2(B,C,D,A,X( 6),23,0x04881d05L); - R2(A,B,C,D,X( 9), 4,0xd9d4d039L); - R2(D,A,B,C,X(12),11,0xe6db99e5L); - R2(C,D,A,B,X(15),16,0x1fa27cf8L); - R2(B,C,D,A,X( 2),23,0xc4ac5665L); - /* Round 3 */ - R3(A,B,C,D,X( 0), 6,0xf4292244L); - R3(D,A,B,C,X( 7),10,0x432aff97L); - R3(C,D,A,B,X(14),15,0xab9423a7L); - R3(B,C,D,A,X( 5),21,0xfc93a039L); - R3(A,B,C,D,X(12), 6,0x655b59c3L); - R3(D,A,B,C,X( 3),10,0x8f0ccc92L); - R3(C,D,A,B,X(10),15,0xffeff47dL); - R3(B,C,D,A,X( 1),21,0x85845dd1L); - R3(A,B,C,D,X( 8), 6,0x6fa87e4fL); - R3(D,A,B,C,X(15),10,0xfe2ce6e0L); - R3(C,D,A,B,X( 6),15,0xa3014314L); - R3(B,C,D,A,X(13),21,0x4e0811a1L); - R3(A,B,C,D,X( 4), 6,0xf7537e82L); - R3(D,A,B,C,X(11),10,0xbd3af235L); - R3(C,D,A,B,X( 2),15,0x2ad7d2bbL); - R3(B,C,D,A,X( 9),21,0xeb86d391L); - - A = c->A += A; - B = c->B += B; - C = c->C += C; - D = c->D += D; - } - } -#endif - - - // *************************************************************************** -#if COMPILER_LIKES_PRAGMA_MARK -#pragma mark - base64 -> binary conversion -#endif - -static const char Base64[] = "ABCDEFGHIJKLMNOPQRSTUVWXYZabcdefghijklmnopqrstuvwxyz0123456789+/"; -static const char Pad64 = '='; - - -#define mDNSisspace(x) (x == '\t' || x == '\n' || x == '\v' || x == '\f' || x == '\r' || x == ' ') - -mDNSlocal const char *mDNSstrchr(const char *s, int c) - { - while (1) - { - if (c == *s) return s; - if (!*s) return mDNSNULL; - s++; - } - } - -// skips all whitespace anywhere. -// converts characters, four at a time, starting at (or after) -// src from base - 64 numbers into three 8 bit bytes in the target area. -// it returns the number of data bytes stored at the target, or -1 on error. -// adapted from BIND sources - -mDNSlocal mDNSs32 DNSDigest_Base64ToBin(const char *src, mDNSu8 *target, mDNSu32 targsize) - { - int tarindex, state, ch; - const char *pos; - - state = 0; - tarindex = 0; - - while ((ch = *src++) != '\0') { - if (mDNSisspace(ch)) /* Skip whitespace anywhere. */ - continue; - - if (ch == Pad64) - break; - - pos = mDNSstrchr(Base64, ch); - if (pos == 0) /* A non-base64 character. */ - return (-1); - - switch (state) { - case 0: - if (target) { - if ((mDNSu32)tarindex >= targsize) - return (-1); - target[tarindex] = (mDNSu8)((pos - Base64) << 2); - } - state = 1; - break; - case 1: - if (target) { - if ((mDNSu32)tarindex + 1 >= targsize) - return (-1); - target[tarindex] |= (pos - Base64) >> 4; - target[tarindex+1] = (mDNSu8)(((pos - Base64) & 0x0f) << 4); - } - tarindex++; - state = 2; - break; - case 2: - if (target) { - if ((mDNSu32)tarindex + 1 >= targsize) - return (-1); - target[tarindex] |= (pos - Base64) >> 2; - target[tarindex+1] = (mDNSu8)(((pos - Base64) & 0x03) << 6); - } - tarindex++; - state = 3; - break; - case 3: - if (target) { - if ((mDNSu32)tarindex >= targsize) - return (-1); - target[tarindex] |= (pos - Base64); - } - tarindex++; - state = 0; - break; - default: - return -1; - } - } - - /* - * We are done decoding Base-64 chars. Let's see if we ended - * on a byte boundary, and/or with erroneous trailing characters. - */ - - if (ch == Pad64) { /* We got a pad char. */ - ch = *src++; /* Skip it, get next. */ - switch (state) { - case 0: /* Invalid = in first position */ - case 1: /* Invalid = in second position */ - return (-1); - - case 2: /* Valid, means one byte of info */ - /* Skip any number of spaces. */ - for ((void)mDNSNULL; ch != '\0'; ch = *src++) - if (!mDNSisspace(ch)) - break; - /* Make sure there is another trailing = sign. */ - if (ch != Pad64) - return (-1); - ch = *src++; /* Skip the = */ - /* Fall through to "single trailing =" case. */ - /* FALLTHROUGH */ - - case 3: /* Valid, means two bytes of info */ - /* - * We know this char is an =. Is there anything but - * whitespace after it? - */ - for ((void)mDNSNULL; ch != '\0'; ch = *src++) - if (!mDNSisspace(ch)) - return (-1); - - /* - * Now make sure for cases 2 and 3 that the "extra" - * bits that slopped past the last full byte were - * zeros. If we don't check them, they become a - * subliminal channel. - */ - if (target && target[tarindex] != 0) - return (-1); - } - } else { - /* - * We ended by seeing the end of the string. Make sure we - * have no partial bytes lying around. - */ - if (state != 0) - return (-1); - } - - return (tarindex); - } - - - // *************************************************************************** -#if COMPILER_LIKES_PRAGMA_MARK -#pragma mark - API exported to mDNS Core -#endif - -// Constants -#define HMAC_IPAD 0x36 -#define HMAC_OPAD 0x5c -#define MD5_LEN 16 - -#define HMAC_MD5_AlgName (*(const domainname*) "\010" "hmac-md5" "\007" "sig-alg" "\003" "reg" "\003" "int") - -// Adapted from Appendix, RFC 2104 -mDNSlocal void DNSDigest_ConstructHMACKey(DomainAuthInfo *info, const mDNSu8 *key, mDNSu32 len) - { - MD5_CTX k; - mDNSu8 buf[MD5_LEN]; - int i; - - // If key is longer than HMAC_LEN reset it to MD5(key) - if (len > HMAC_LEN) - { - MD5_Init(&k); - MD5_Update(&k, key, len); - MD5_Final(buf, &k); - key = buf; - len = MD5_LEN; - } - - // store key in pads - mDNSPlatformMemZero(info->keydata_ipad, HMAC_LEN); - mDNSPlatformMemZero(info->keydata_opad, HMAC_LEN); - mDNSPlatformMemCopy(info->keydata_ipad, key, len); - mDNSPlatformMemCopy(info->keydata_opad, key, len); - - // XOR key with ipad and opad values - for (i = 0; i < HMAC_LEN; i++) - { - info->keydata_ipad[i] ^= HMAC_IPAD; - info->keydata_opad[i] ^= HMAC_OPAD; - } - - } - -mDNSexport mDNSs32 DNSDigest_ConstructHMACKeyfromBase64(DomainAuthInfo *info, const char *b64key) - { - mDNSu8 keybuf[1024]; - mDNSs32 keylen = DNSDigest_Base64ToBin(b64key, keybuf, sizeof(keybuf)); - if (keylen < 0) return(keylen); - DNSDigest_ConstructHMACKey(info, keybuf, (mDNSu32)keylen); - return(keylen); - } - -mDNSexport void DNSDigest_SignMessage(DNSMessage *msg, mDNSu8 **end, DomainAuthInfo *info, mDNSu16 tcode) - { - AuthRecord tsig; - mDNSu8 *rdata, *const countPtr = (mDNSu8 *)&msg->h.numAdditionals; // Get existing numAdditionals value - mDNSu32 utc32; - mDNSu8 utc48[6]; - mDNSu8 digest[MD5_LEN]; - mDNSu8 *ptr = *end; - mDNSu32 len; - mDNSOpaque16 buf; - MD5_CTX c; - mDNSu16 numAdditionals = (mDNSu16)((mDNSu16)countPtr[0] << 8 | countPtr[1]); - - // Init MD5 context, digest inner key pad and message - MD5_Init(&c); - MD5_Update(&c, info->keydata_ipad, HMAC_LEN); - MD5_Update(&c, (mDNSu8 *)msg, (unsigned long)(*end - (mDNSu8 *)msg)); - - // Construct TSIG RR, digesting variables as apporpriate - mDNS_SetupResourceRecord(&tsig, mDNSNULL, 0, kDNSType_TSIG, 0, kDNSRecordTypeKnownUnique, AuthRecordAny, mDNSNULL, mDNSNULL); - - // key name - AssignDomainName(&tsig.namestorage, &info->keyname); - MD5_Update(&c, info->keyname.c, DomainNameLength(&info->keyname)); - - // class - tsig.resrec.rrclass = kDNSQClass_ANY; - buf = mDNSOpaque16fromIntVal(kDNSQClass_ANY); - MD5_Update(&c, buf.b, sizeof(mDNSOpaque16)); - - // ttl - tsig.resrec.rroriginalttl = 0; - MD5_Update(&c, (mDNSu8 *)&tsig.resrec.rroriginalttl, sizeof(tsig.resrec.rroriginalttl)); - - // alg name - AssignDomainName(&tsig.resrec.rdata->u.name, &HMAC_MD5_AlgName); - len = DomainNameLength(&HMAC_MD5_AlgName); - rdata = tsig.resrec.rdata->u.data + len; - MD5_Update(&c, HMAC_MD5_AlgName.c, len); - - // time - // get UTC (universal time), convert to 48-bit unsigned in network byte order - utc32 = (mDNSu32)mDNSPlatformUTC(); - if (utc32 == (unsigned)-1) { LogMsg("ERROR: DNSDigest_SignMessage - mDNSPlatformUTC returned bad time -1"); *end = mDNSNULL; } - utc48[0] = 0; - utc48[1] = 0; - utc48[2] = (mDNSu8)((utc32 >> 24) & 0xff); - utc48[3] = (mDNSu8)((utc32 >> 16) & 0xff); - utc48[4] = (mDNSu8)((utc32 >> 8) & 0xff); - utc48[5] = (mDNSu8)( utc32 & 0xff); - - mDNSPlatformMemCopy(rdata, utc48, 6); - rdata += 6; - MD5_Update(&c, utc48, 6); - - // 300 sec is fudge recommended in RFC 2485 - rdata[0] = (mDNSu8)((300 >> 8) & 0xff); - rdata[1] = (mDNSu8)( 300 & 0xff); - MD5_Update(&c, rdata, sizeof(mDNSOpaque16)); - rdata += sizeof(mDNSOpaque16); - - // digest error (tcode) and other data len (zero) - we'll add them to the rdata later - buf.b[0] = (mDNSu8)((tcode >> 8) & 0xff); - buf.b[1] = (mDNSu8)( tcode & 0xff); - MD5_Update(&c, buf.b, sizeof(mDNSOpaque16)); // error - buf.NotAnInteger = 0; - MD5_Update(&c, buf.b, sizeof(mDNSOpaque16)); // other data len - - // finish the message & tsig var hash - MD5_Final(digest, &c); - - // perform outer MD5 (outer key pad, inner digest) - MD5_Init(&c); - MD5_Update(&c, info->keydata_opad, HMAC_LEN); - MD5_Update(&c, digest, MD5_LEN); - MD5_Final(digest, &c); - - // set remaining rdata fields - rdata[0] = (mDNSu8)((MD5_LEN >> 8) & 0xff); - rdata[1] = (mDNSu8)( MD5_LEN & 0xff); - rdata += sizeof(mDNSOpaque16); - mDNSPlatformMemCopy(rdata, digest, MD5_LEN); // MAC - rdata += MD5_LEN; - rdata[0] = msg->h.id.b[0]; // original ID - rdata[1] = msg->h.id.b[1]; - rdata[2] = (mDNSu8)((tcode >> 8) & 0xff); - rdata[3] = (mDNSu8)( tcode & 0xff); - rdata[4] = 0; // other data len - rdata[5] = 0; - rdata += 6; - - tsig.resrec.rdlength = (mDNSu16)(rdata - tsig.resrec.rdata->u.data); - *end = PutResourceRecordTTLJumbo(msg, ptr, &numAdditionals, &tsig.resrec, 0); - if (!*end) { LogMsg("ERROR: DNSDigest_SignMessage - could not put TSIG"); *end = mDNSNULL; return; } - - // Write back updated numAdditionals value - countPtr[0] = (mDNSu8)(numAdditionals >> 8); - countPtr[1] = (mDNSu8)(numAdditionals & 0xFF); - } - -mDNSexport mDNSBool DNSDigest_VerifyMessage(DNSMessage *msg, mDNSu8 *end, LargeCacheRecord * lcr, DomainAuthInfo *info, mDNSu16 * rcode, mDNSu16 * tcode) - { - mDNSu8 * ptr = (mDNSu8*) &lcr->r.resrec.rdata->u.data; - mDNSs32 now; - mDNSs32 then; - mDNSu8 thisDigest[MD5_LEN]; - mDNSu8 thatDigest[MD5_LEN]; - mDNSu32 macsize; - mDNSOpaque16 buf; - mDNSu8 utc48[6]; - mDNSs32 delta; - mDNSu16 fudge; - domainname * algo; - MD5_CTX c; - mDNSBool ok = mDNSfalse; - - // We only support HMAC-MD5 for now - - algo = (domainname*) ptr; - - if (!SameDomainName(algo, &HMAC_MD5_AlgName)) - { - LogMsg("ERROR: DNSDigest_VerifyMessage - TSIG algorithm not supported: %##s", algo->c); - *rcode = kDNSFlag1_RC_NotAuth; - *tcode = TSIG_ErrBadKey; - ok = mDNSfalse; - goto exit; - } - - ptr += DomainNameLength(algo); - - // Check the times - - now = mDNSPlatformUTC(); - if (now == -1) - { - LogMsg("ERROR: DNSDigest_VerifyMessage - mDNSPlatformUTC returned bad time -1"); - *rcode = kDNSFlag1_RC_NotAuth; - *tcode = TSIG_ErrBadTime; - ok = mDNSfalse; - goto exit; - } - - // Get the 48 bit time field, skipping over the first word - - utc48[0] = *ptr++; - utc48[1] = *ptr++; - utc48[2] = *ptr++; - utc48[3] = *ptr++; - utc48[4] = *ptr++; - utc48[5] = *ptr++; - - then = (mDNSs32)NToH32(utc48 + sizeof(mDNSu16)); - - fudge = NToH16(ptr); - - ptr += sizeof(mDNSu16); - - delta = (now > then) ? now - then : then - now; - - if (delta > fudge) - { - LogMsg("ERROR: DNSDigest_VerifyMessage - time skew > %d", fudge); - *rcode = kDNSFlag1_RC_NotAuth; - *tcode = TSIG_ErrBadTime; - ok = mDNSfalse; - goto exit; - } - - // MAC size - - macsize = (mDNSu32) NToH16(ptr); - - ptr += sizeof(mDNSu16); - - // MAC - - mDNSPlatformMemCopy(thatDigest, ptr, MD5_LEN); - - // Init MD5 context, digest inner key pad and message - - MD5_Init(&c); - MD5_Update(&c, info->keydata_ipad, HMAC_LEN); - MD5_Update(&c, (mDNSu8*) msg, (unsigned long)(end - (mDNSu8*) msg)); - - // Key name - - MD5_Update(&c, lcr->r.resrec.name->c, DomainNameLength(lcr->r.resrec.name)); - - // Class name - - buf = mDNSOpaque16fromIntVal(lcr->r.resrec.rrclass); - MD5_Update(&c, buf.b, sizeof(mDNSOpaque16)); - - // TTL - - MD5_Update(&c, (mDNSu8*) &lcr->r.resrec.rroriginalttl, sizeof(lcr->r.resrec.rroriginalttl)); - - // Algorithm - - MD5_Update(&c, algo->c, DomainNameLength(algo)); - - // Time - - MD5_Update(&c, utc48, 6); - - // Fudge - - buf = mDNSOpaque16fromIntVal(fudge); - MD5_Update(&c, buf.b, sizeof(mDNSOpaque16)); - - // Digest error and other data len (both zero) - we'll add them to the rdata later - - buf.NotAnInteger = 0; - MD5_Update(&c, buf.b, sizeof(mDNSOpaque16)); // error - MD5_Update(&c, buf.b, sizeof(mDNSOpaque16)); // other data len - - // Finish the message & tsig var hash - - MD5_Final(thisDigest, &c); - - // perform outer MD5 (outer key pad, inner digest) - - MD5_Init(&c); - MD5_Update(&c, info->keydata_opad, HMAC_LEN); - MD5_Update(&c, thisDigest, MD5_LEN); - MD5_Final(thisDigest, &c); - - if (!mDNSPlatformMemSame(thisDigest, thatDigest, MD5_LEN)) - { - LogMsg("ERROR: DNSDigest_VerifyMessage - bad signature"); - *rcode = kDNSFlag1_RC_NotAuth; - *tcode = TSIG_ErrBadSig; - ok = mDNSfalse; - goto exit; - } - - // set remaining rdata fields - ok = mDNStrue; - -exit: - - return ok; - } - - -#ifdef __cplusplus -} -#endif diff -Nru qtcreator-2.5.0/src/tools/mdnssd/DebugServices.h qtcreator-2.5.2/src/tools/mdnssd/DebugServices.h --- qtcreator-2.5.0/src/tools/mdnssd/DebugServices.h 2012-05-09 12:13:19.000000000 +0000 +++ qtcreator-2.5.2/src/tools/mdnssd/DebugServices.h 1970-01-01 00:00:00.000000000 +0000 @@ -1,1607 +0,0 @@ -/* -*- Mode: C; tab-width: 4 -*- - * - * Copyright (c) 1997-2004 Apple Computer, Inc. All rights reserved. - * - * Licensed under the Apache License, Version 2.0 (the "License"); - * you may not use this file except in compliance with the License. - * You may obtain a copy of the License at - * - * http://www.apache.org/licenses/LICENSE-2.0 - * - * Unless required by applicable law or agreed to in writing, software - * distributed under the License is distributed on an "AS IS" BASIS, - * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. - * See the License for the specific language governing permissions and - * limitations under the License. - */ - -//--------------------------------------------------------------------------------------------------------------------------- -/*! @header DebugServices - - Debugging Library -*/ - -#ifndef __DEBUG_SERVICES__ -#define __DEBUG_SERVICES__ - -#include - -#include "CommonServices.h" - -#if( TARGET_OS_VXWORKS ) - #include "logLib.h" -#endif - -#if 0 -#pragma mark == Settings == -#endif - -//=========================================================================================================================== -// Settings -//=========================================================================================================================== - -// General - -#if( !defined( DEBUG ) ) - #define DEBUG 0 -#endif - -#if( defined( NDEBUG ) && DEBUG ) - #error NDEBUG defined and DEBUG is also enabled...they need to be in-sync -#endif - -// AssertMacros.h/Debugging.h overrides. - -#if( !defined( DEBUG_OVERRIDE_APPLE_MACROS ) ) - #define DEBUG_OVERRIDE_APPLE_MACROS 1 -#endif - -// Routine name. Uses ISO __func__ where possible. Otherwise, uses the best thing that is available (if anything). - -#if( defined( __MWERKS__ ) || ( __GNUC__ > 2 ) || ( ( __GNUC__ == 2 ) && ( __GNUC_MINOR__ >= 9 ) ) ) - #define __ROUTINE__ __func__ -#elif( defined( __GNUC__ ) ) - #define __ROUTINE__ __PRETTY_FUNCTION__ -#elif( defined( _MSC_VER ) && !defined( _WIN32_WCE ) ) - #define __ROUTINE__ __FUNCTION__ -#else - #define __ROUTINE__ "" -#endif - -// Variable argument macro support. Use ANSI C99 __VA_ARGS__ where possible. Otherwise, use the next best thing. - -#if( defined( __GNUC__ ) ) - #if( ( __GNUC__ > 3 ) || ( ( __GNUC__ == 3 ) && ( __GNUC_MINOR__ >= 3) ) ) - #define DEBUG_C99_VA_ARGS 1 - #define DEBUG_GNU_VA_ARGS 0 - #else - #define DEBUG_C99_VA_ARGS 0 - #define DEBUG_GNU_VA_ARGS 1 - #endif -#elif( defined( __MWERKS__ ) ) - #define DEBUG_C99_VA_ARGS 1 - #define DEBUG_GNU_VA_ARGS 0 -#else - #define DEBUG_C99_VA_ARGS 0 - #define DEBUG_GNU_VA_ARGS 0 -#endif - -#if 0 -#pragma mark == Output == -#endif - -//--------------------------------------------------------------------------------------------------------------------------- -/*! @defined DEBUG_FPRINTF_ENABLED - - @abstract Enables ANSI C fprintf output. -*/ - -#if( !defined( DEBUG_FPRINTF_ENABLED ) ) - #if( !TARGET_API_MAC_OSX_KERNEL && !TARGET_OS_WINDOWS_CE ) - #define DEBUG_FPRINTF_ENABLED 1 - #else - #define DEBUG_FPRINTF_ENABLED 0 - #endif -#else - #if( TARGET_API_MAC_OSX_KERNEL || TARGET_OS_WINDOWS_CE ) - #error fprintf enabled, but not supported on Mac OS X kernel or Windows CE - #endif -#endif - -//--------------------------------------------------------------------------------------------------------------------------- -/*! @defined DEBUG_MAC_OS_X_IOLOG_ENABLED - - @abstract Enables IOLog (Mac OS X Kernel) output. -*/ - -#if( !defined( DEBUG_MAC_OS_X_IOLOG_ENABLED ) ) - #define DEBUG_MAC_OS_X_IOLOG_ENABLED TARGET_API_MAC_OSX_KERNEL -#endif - -//--------------------------------------------------------------------------------------------------------------------------- -/*! @defined DEBUG_KPRINTF_ENABLED - - @abstract Enables kprintf (Mac OS X Kernel) output. -*/ - -#if( !defined( DEBUG_KPRINTF_ENABLED ) ) - #define DEBUG_KPRINTF_ENABLED TARGET_API_MAC_OSX_KERNEL -#endif - -//--------------------------------------------------------------------------------------------------------------------------- -/*! @defined DEBUG_IDEBUG_ENABLED - - @abstract Enables iDebug (Mac OS X user and Kernel) output. - - @discussion - - For Mac OS X kernel development, iDebug is enabled by default because we can dynamically check for the presence - of iDebug via some exported IOKit symbols. Mac OS X app usage doesn't allow dynamic detection because it relies - on statically linking to the iDebugServices.cp file so for Mac OS X app usage, you have to manually enable iDebug. -*/ - -#if( !defined( DEBUG_IDEBUG_ENABLED ) ) - #define DEBUG_IDEBUG_ENABLED TARGET_API_MAC_OSX_KERNEL -#endif - -//--------------------------------------------------------------------------------------------------------------------------- -/*! @defined DEBUG_CORE_SERVICE_ASSERTS_ENABLED - - @abstract Controls whether Core Services assert handling is enabled. Enabling requires CoreServices framework. -*/ - -#if( !defined( DEBUG_CORE_SERVICE_ASSERTS_ENABLED ) ) - #if( defined( __DEBUGGING__ ) ) - #define DEBUG_CORE_SERVICE_ASSERTS_ENABLED 1 - #else - #define DEBUG_CORE_SERVICE_ASSERTS_ENABLED 0 - #endif -#endif - -//--------------------------------------------------------------------------------------------------------------------------- -/*! @typedef DebugOutputType - - @abstract Type of debug output (i.e. where the output goes). -*/ - -typedef uint32_t DebugOutputType; - -#define kDebugOutputTypeNone 0x6E6F6E65U // 'none' - no params -#define kDebugOutputTypeCustom 0x63757374U // 'cust' - 1st param = function ptr, 2nd param = context -#define kDebugOutputTypeFPrintF 0x66707269U // 'fpri' - 1st param = DebugOutputTypeFlags [, 2nd param = filename] -#define kDebugOutputTypeiDebug 0x69646267U // 'idbg' - no params -#define kDebugOutputTypeKPrintF 0x6B707266U // 'kprf' - no params -#define kDebugOutputTypeMacOSXIOLog 0x696C6F67U // 'ilog' - no params -#define kDebugOutputTypeMacOSXLog 0x786C6F67U // 'xlog' - no params -#define kDebugOutputTypeWindowsDebugger 0x77696E64U // 'wind' - no params -#define kDebugOutputTypeWindowsEventLog 0x7765766CU // 'wevl' - 1st param = C-string name, 2nd param = HMODULE or NULL. - -// Console meta output kind - Any kind of Console output (in horizontal order of preference): -// -// Mac OS X = ANSI printf (viewable in Console.app) -// Mac OS X Kernel = IOLog (/var/log/system.log) or kprintf (serial). -// Windows = ANSI printf (Console window) or OutputDebugString (debugger). -// Other = ANSI printf (viewer varies). - -#define kDebugOutputTypeMetaConsole 0x434F4E53U // 'CONS' - no params - -//--------------------------------------------------------------------------------------------------------------------------- -/*! @typedef DebugOutputTypeFlags - - @abstract Flags controlling how the output type is configured. - - @constant kDebugOutputTypeFlagsTypeMask Bit mask for the output type (e.g. stdout, stderr, file, etc.). - @constant kDebugOutputTypeFlagsStdOut fprintf should go to stdout. - @constant kDebugOutputTypeFlagsStdErr fprintf should go to stderr. - @constant kDebugOutputTypeFlagsFile fprintf should go to a specific file (filename passed as va_arg). -*/ - -typedef unsigned int DebugOutputTypeFlags; - -#define kDebugOutputTypeFlagsTypeMask 0xF -#define kDebugOutputTypeFlagsStdOut 1 -#define kDebugOutputTypeFlagsStdErr 2 -#define kDebugOutputTypeFlagsFile 10 - -//--------------------------------------------------------------------------------------------------------------------------- -/*! @typedef DebugOutputFunctionPtr - - @abstract Function ptr for a custom callback to print debug output. -*/ - -typedef void ( *DebugOutputFunctionPtr )( char *inData, size_t inSize, void *inContext ); - -//=========================================================================================================================== -// Constants -//=========================================================================================================================== - -#if 0 -#pragma mark == Flags == -#endif - -//--------------------------------------------------------------------------------------------------------------------------- -/*! @typedef DebugFlags - - @abstract Flags controlling how output is printed. -*/ - -typedef uint32_t DebugFlags; - -#define kDebugFlagsNone 0 -#define kDebugFlagsNoAddress ( 1 << 0 ) -#define kDebugFlagsNoOffset ( 1 << 1 ) -#define kDebugFlags32BitOffset ( 1 << 2 ) -#define kDebugFlagsNoASCII ( 1 << 3 ) -#define kDebugFlagsNoNewLine ( 1 << 4 ) -#define kDebugFlags8BitSeparator ( 1 << 5 ) -#define kDebugFlags16BitSeparator ( 1 << 6 ) -#define kDebugFlagsNo32BitSeparator ( 1 << 7 ) -#define kDebugFlagsNo16ByteHexPad ( 1 << 8 ) -#define kDebugFlagsNoByteCount ( 1 << 9 ) - -//--------------------------------------------------------------------------------------------------------------------------- -/*! @enum DebugTaskLevelFlags - - @abstract Flags indicating the task level. -*/ - -enum -{ - kDebugInterruptLevelShift = 0, - kDebugInterruptLevelMask = 0x00000007, - kDebugInVBLTaskMask = 0x00000010, - kDebugInDeferredTaskMask = 0x00000020, - kDebugInSecondaryInterruptHandlerMask = 0x00000040, - kDebugPageFaultFatalMask = 0x00000100, // There should be a "kPageFaultFatalMask" in Debugging.h. - kDebugMPTaskLevelMask = 0x00000200, // There should be a "kMPTaskLevelMask" in Debugging.h. - kDebugInterruptDepthShift = 16, - kDebugInterruptDepthMask = 0x00FF0000 -}; - -#define DebugExtractTaskLevelInterruptLevel( LEVEL ) \ - ( ( ( LEVEL ) & kDebugInterruptLevelMask ) >> kDebugInterruptLevelShift ) - -#define DebugExtractTaskLevelInterruptDepth( LEVEL ) \ - ( ( ( LEVEL ) & kDebugInterruptDepthMask ) >> kDebugInterruptDepthShift ) - -#if 0 -#pragma mark == Levels == -#endif - -//=========================================================================================================================== -// Constants & Types - Levels -//=========================================================================================================================== - -//--------------------------------------------------------------------------------------------------------------------------- -/*! @typedef DebugLevel - - @abstract Level used to control debug logging. -*/ - -typedef int32_t DebugLevel; - -// Levels - -#define kDebugLevelMask 0x0000FFFF -#define kDebugLevelChatty 100 -#define kDebugLevelVerbose 500 -#define kDebugLevelTrace 800 -#define kDebugLevelInfo 1000 -#define kDebugLevelNotice 3000 -#define kDebugLevelWarning 5000 -#define kDebugLevelAssert 6000 -#define kDebugLevelRequire 7000 -#define kDebugLevelError 8000 -#define kDebugLevelCritical 9000 -#define kDebugLevelAlert 10000 -#define kDebugLevelEmergency 11000 -#define kDebugLevelTragic 12000 -#define kDebugLevelMax 0x0000FFFF - -// Level Flags - -#define kDebugLevelFlagMask 0xFFFF0000 -#define kDebugLevelFlagStackTrace 0x00010000 -#define kDebugLevelFlagDebugBreak 0x00020000 - -//--------------------------------------------------------------------------------------------------------------------------- -/*! @typedef LogLevel - - @abstract Level used to control which events are logged. -*/ - -typedef int32_t LogLevel; - -#define kLogLevelUninitialized -1L -#define kLogLevelAll 0L -#define kLogLevelChatty 100L -#define kLogLevelVerbose 500L -#define kLogLevelTrace 800L -#define kLogLevelInfo 1000L -#define kLogLevelNotice 3000L -#define kLogLevelWarning 4000L -#define kLogLevelAssert 6000L -#define kLogLevelRequire 7000L -#define kLogLevelError 8000L -#define kLogLevelCritical 9000L -#define kLogLevelAlert 10000L -#define kLogLevelEmergency 11000L -#define kLogLevelTragic 12000L -#define kLogLevelOff 0x0000FFFEL - -#if 0 -#pragma mark == Properties == -#endif - -//--------------------------------------------------------------------------------------------------------------------------- -/*! @typedef DebugPropertyTag - - @abstract Tag for properties. -*/ - -typedef uint32_t DebugPropertyTag; - -#define kDebugPropertyTagPrintLevelMin 0x6D696E70U // 'minp' Get: 1st param = DebugLevel * - // Set: 1st param = DebugLevel - -#define kDebugPropertyTagPrintLevel kDebugPropertyTagPrintLevelMin - -#define kDebugPropertyTagPrintLevelMax 0x706D786CU // 'maxp' Get: 1st param = DebugLevel * - // Set: 1st param = DebugLevel - -#define kDebugPropertyTagBreakLevel 0x62726B6CU // 'brkl' Get: 1st param = DebugLevel * - // Set: 1st param = DebugLevel -#if 0 -#pragma mark == General macros == -#endif - -//--------------------------------------------------------------------------------------------------------------------------- -/*! @defined DEBUG_UNUSED - - @abstract Macro to mark a paramter as unused to avoid unused parameter warnings. - - @discussion - - There is no universally supported pragma/attribute for indicating a variable is unused. DEBUG_UNUSED lets us - indicate a variable is unused in a manner that is supported by most compilers. -*/ - -#define DEBUG_UNUSED( X ) (void)( X ) - -//--------------------------------------------------------------------------------------------------------------------------- -/*! @defined DEBUG_USE_ONLY - - @abstract Macro to mark a variable as used only when debugging is enabled. - - @discussion - - Variables are sometimes needed only for debugging. When debugging is turned off, these debug-only variables generate - compiler warnings about unused variables. To eliminate these warnings, use these macros to indicate variables that - are only used for debugging. -*/ - -#if( DEBUG ) - #define DEBUG_USE_ONLY( X ) -#else - #define DEBUG_USE_ONLY( X ) (void)( X ) -#endif - -//--------------------------------------------------------------------------------------------------------------------------- -/*! @defined DEBUG_LOCAL - - @abstract Macros to make variables and functions static when debugging is off, but extern when debugging is on. - - @discussion - - Rather than using "static" directly, using this macros allows you to access these variables external while - debugging without being penalized for production builds. -*/ - -#if( DEBUG ) - #define DEBUG_LOCAL -#else - #define DEBUG_LOCAL static -#endif - -//--------------------------------------------------------------------------------------------------------------------------- -/*! @defined DEBUG_STATIC - - @abstract Macros to make variables and functions static when debugging is off, but extern when debugging is on. - - @discussion - - Rather than using "static" directly, using this macros allows you to access these variables external while - debugging without being penalized for production builds. -*/ - -#if( DEBUG ) - #define DEBUG_STATIC -#else - #define DEBUG_STATIC static -#endif - -//--------------------------------------------------------------------------------------------------------------------------- -/*! @defined DEBUG_EXPORT - - @abstract Macros to export variables. - - @discussion - - "__private_extern__" is a hack for IOKit to allow symbols to be exported from compilation units, but - // not exported outside a driver (IOKit uses a lame global namespace for symbols). This still does not - // solve the problem of multiple drivers in the same dependency chain since they share symbols. -*/ - -#if( TARGET_API_MAC_OSX_KERNEL ) - #define DEBUG_EXPORT __private_extern__ -#else - #define DEBUG_EXPORT extern -#endif - -//--------------------------------------------------------------------------------------------------------------------------- -/*! @defined debug_add - - @abstract Macro to add (or subtract if negative) a value when debugging is on. Does nothing if debugging is off. -*/ - -#if( DEBUG ) - #define debug_add( A, B ) ( A ) += ( B ) -#else - #define debug_add( A, B ) -#endif - -//--------------------------------------------------------------------------------------------------------------------------- -/*! @defined debug_perform - - @abstract Macro to perform something in debug-only builds. -*/ - -#if( DEBUG ) - #define debug_perform( X ) do { X; } while( 0 ) -#else - #define debug_perform( X ) -#endif - -//--------------------------------------------------------------------------------------------------------------------------- -/*! @function translate_errno - - @abstract Returns 0 if the test success. If the test fails, returns errno if non-zero and othewise the alternate error. -*/ - -#define translate_errno( TEST, ERRNO, ALTERNATE_ERROR ) ( ( TEST ) ? 0 : ( ERRNO ) ? ( ERRNO ) : ( ALTERNATE_ERROR ) ) - -#if 0 -#pragma mark == Compile Time macros == -#endif - -//--------------------------------------------------------------------------------------------------------------------------- -/*! @defined check_compile_time - - @abstract Performs a compile-time check of something such as the size of an int. - - @discussion - - This declares an array with a size that is determined by a compile-time expression. If the expression evaluates - to 0, the array has a size of -1, which is illegal and generates a compile-time error. - - For example: - - check_compile_time( sizeof( int ) == 4 ); - - Note: This only works with compile-time expressions. - Note: This only works in places where extern declarations are allowed (e.g. global scope). - - References: - - - - - Note: The following macros differ from the macros on the www.jaggersoft.com web site because those versions do not - work with GCC due to GCC allow a zero-length array. Using a -1 condition turned out to be more portable. -*/ - -#define check_compile_time( X ) extern int debug_compile_time_name[ ( X ) ? 1 : -1 ] - -//--------------------------------------------------------------------------------------------------------------------------- -/*! @defined check_compile_time_code - - @abstract Perform a compile-time check, suitable for placement in code, of something such as the size of an int. - - @discussion - - This creates a switch statement with an existing case for 0 and an additional case using the result of a - compile-time expression. A switch statement cannot have two case labels with the same constant so if the - compile-time expression evaluates to 0, it is illegal and generates a compile-time error. If the compile-time - expression does not evaluate to 0, the resulting value is used as the case label and it compiles without error. - - For example: - - check_compile_time_code( sizeof( int ) == 4 ); - - Note: This only works with compile-time expressions. - Note: This does not work in a global scope so it must be inside a function. - - References: - - - -*/ - -#define check_compile_time_code( X ) switch( 0 ) { case 0: case X:; } - -#if 0 -#pragma mark == check macros == -#endif - -//--------------------------------------------------------------------------------------------------------------------------- -/*! @defined check - - @abstract Check that an expression is true (non-zero). - - @discussion - - If expression evalulates to false, this prints debugging information (actual expression string, file, line number, - function name, etc.) using the default debugging output method. - - Code inside check() statements is not compiled into production builds. -*/ - -#if( DEBUG_OVERRIDE_APPLE_MACROS ) - #undef check -#endif -#if( !defined( check ) ) - #if( DEBUG ) - #define check( X ) \ - do \ - { \ - if( !( X ) ) \ - { \ - debug_print_assert( 0, #X, NULL, __FILE__, __LINE__, __ROUTINE__ ); \ - } \ - } while( 0 ) - #else - #define check( X ) - #endif -#endif - -//--------------------------------------------------------------------------------------------------------------------------- -/*! @defined check_string - - @abstract Check that an expression is true (non-zero) with an explanation. - - @discussion - - If expression evalulates to false, this prints debugging information (actual expression string, file, line number, - function name, etc.) and a custom explanation string using the default debugging output method. - - Code inside check_string() statements is not compiled into production builds. -*/ - -#if( DEBUG_OVERRIDE_APPLE_MACROS ) - #undef check_string -#endif -#if( !defined( check_string ) ) - #if( DEBUG ) - #define check_string( X, STR ) \ - do \ - { \ - if( !( X ) ) \ - { \ - debug_print_assert( 0, #X, STR, __FILE__, __LINE__, __ROUTINE__ ); \ - } \ - \ - } while( 0 ) - #else - #define check_string( X, STR ) - #endif -#endif - -//--------------------------------------------------------------------------------------------------------------------------- -/*! @defined check_noerr - - @abstract Check that an error code is noErr (0). - - @discussion - - If the error code is non-0, this prints debugging information (actual expression string, file, line number, - function name, etc.) using the default debugging output method. - - Code inside check_noerr() statements is not compiled into production builds. -*/ - -#if( DEBUG_OVERRIDE_APPLE_MACROS ) - #undef check_noerr -#endif -#if( !defined( check_noerr ) ) - #if( DEBUG ) - #define check_noerr( ERR ) \ - do \ - { \ - int_least32_t localErr; \ - \ - localErr = (int_least32_t)( ERR ); \ - if( localErr != 0 ) \ - { \ - debug_print_assert( localErr, NULL, NULL, __FILE__, __LINE__, __ROUTINE__ ); \ - } \ - \ - } while( 0 ) - #else - #define check_noerr( ERR ) - #endif -#endif - -//--------------------------------------------------------------------------------------------------------------------------- -/*! @defined check_noerr_string - - @abstract Check that an error code is noErr (0) with an explanation. - - @discussion - - If the error code is non-0, this prints debugging information (actual expression string, file, line number, - function name, etc.) and a custom explanation string using the default debugging output method. - - Code inside check_noerr_string() statements is not compiled into production builds. -*/ - -#if( DEBUG_OVERRIDE_APPLE_MACROS ) - #undef check_noerr_string -#endif -#if( !defined( check_noerr_string ) ) - #if( DEBUG ) - #define check_noerr_string( ERR, STR ) \ - do \ - { \ - int_least32_t localErr; \ - \ - localErr = (int_least32_t)( ERR ); \ - if( localErr != 0 ) \ - { \ - debug_print_assert( localErr, NULL, STR, __FILE__, __LINE__, __ROUTINE__ ); \ - } \ - \ - } while( 0 ) - #else - #define check_noerr_string( ERR, STR ) - #endif -#endif - -//--------------------------------------------------------------------------------------------------------------------------- -/*! @defined check_translated_errno - - @abstract Check a condition and prints errno (if non-zero) to the log. - - @discussion - - Code inside check_translated_errno() statements is not compiled into production builds. -*/ - -#if( !defined( check_translated_errno ) ) - #if( DEBUG ) - #define check_translated_errno( TEST, ERRNO, ALTERNATE_ERROR ) \ - do \ - { \ - if( !( TEST ) ) \ - { \ - int_least32_t localErr; \ - \ - localErr = (int_least32_t)( ERRNO ); \ - localErr = ( localErr != 0 ) ? localErr : (int_least32_t)( ALTERNATE_ERROR ); \ - debug_print_assert( localErr, #TEST, NULL, __FILE__, __LINE__, __ROUTINE__ ); \ - } \ - \ - } while( 0 ) - #else - #define check_translated_errno( TEST, ERRNO, ALTERNATE_ERROR ) - #endif -#endif - -//--------------------------------------------------------------------------------------------------------------------------- -/*! @defined check_ptr_overlap - - @abstract Checks that two ptrs do not overlap. -*/ - -#define check_ptr_overlap( P1, P1_SIZE, P2, P2_SIZE ) \ - do \ - { \ - check( !( ( (uintptr_t)( P1 ) >= (uintptr_t)( P2 ) ) && \ - ( (uintptr_t)( P1 ) < ( ( (uintptr_t)( P2 ) ) + ( P2_SIZE ) ) ) ) ); \ - check( !( ( (uintptr_t)( P2 ) >= (uintptr_t)( P1 ) ) && \ - ( (uintptr_t)( P2 ) < ( ( (uintptr_t)( P1 ) ) + ( P1_SIZE ) ) ) ) ); \ - \ - } while( 0 ) - -#if 0 -#pragma mark == require macros == -#endif - -//--------------------------------------------------------------------------------------------------------------------------- -/*! @defined require - - @abstract Requires that an expression evaluate to true. - - @discussion - - If expression evalulates to false, this prints debugging information (actual expression string, file, line number, - function name, etc.) using the default debugging output method then jumps to a label. -*/ - -#if( DEBUG_OVERRIDE_APPLE_MACROS ) - #undef require -#endif -#if( !defined( require ) ) - #define require( X, LABEL ) \ - do \ - { \ - if( !( X ) ) \ - { \ - debug_print_assert( 0, #X, NULL, __FILE__, __LINE__, __ROUTINE__ ); \ - goto LABEL; \ - } \ - \ - } while( 0 ) -#endif - -//--------------------------------------------------------------------------------------------------------------------------- -/*! @defined require_string - - @abstract Requires that an expression evaluate to true with an explanation. - - @discussion - - If expression evalulates to false, this prints debugging information (actual expression string, file, line number, - function name, etc.) and a custom explanation string using the default debugging output method then jumps to a label. -*/ - -#if( DEBUG_OVERRIDE_APPLE_MACROS ) - #undef require_string -#endif -#if( !defined( require_string ) ) - #define require_string( X, LABEL, STR ) \ - do \ - { \ - if( !( X ) ) \ - { \ - debug_print_assert( 0, #X, STR, __FILE__, __LINE__, __ROUTINE__ ); \ - goto LABEL; \ - } \ - \ - } while( 0 ) -#endif - -//--------------------------------------------------------------------------------------------------------------------------- -/*! @defined require_quiet - - @abstract Requires that an expression evaluate to true. - - @discussion - - If expression evalulates to false, this jumps to a label. No debugging information is printed. -*/ - -#if( DEBUG_OVERRIDE_APPLE_MACROS ) - #undef require_quiet -#endif -#if( !defined( require_quiet ) ) - #define require_quiet( X, LABEL ) \ - do \ - { \ - if( !( X ) ) \ - { \ - goto LABEL; \ - } \ - \ - } while( 0 ) -#endif - -//--------------------------------------------------------------------------------------------------------------------------- -/*! @defined require_noerr - - @abstract Require that an error code is noErr (0). - - @discussion - - If the error code is non-0, this prints debugging information (actual expression string, file, line number, - function name, etc.) using the default debugging output method then jumps to a label. -*/ - -#if( DEBUG_OVERRIDE_APPLE_MACROS ) - #undef require_noerr -#endif -#if( !defined( require_noerr ) ) - #define require_noerr( ERR, LABEL ) \ - do \ - { \ - int_least32_t localErr; \ - \ - localErr = (int_least32_t)( ERR ); \ - if( localErr != 0 ) \ - { \ - debug_print_assert( localErr, NULL, NULL, __FILE__, __LINE__, __ROUTINE__ ); \ - goto LABEL; \ - } \ - \ - } while( 0 ) -#endif - -//--------------------------------------------------------------------------------------------------------------------------- -/*! @defined require_noerr_string - - @abstract Require that an error code is noErr (0). - - @discussion - - If the error code is non-0, this prints debugging information (actual expression string, file, line number, - function name, etc.), and a custom explanation string using the default debugging output method using the - default debugging output method then jumps to a label. -*/ - -#if( DEBUG_OVERRIDE_APPLE_MACROS ) - #undef require_noerr_string -#endif -#if( !defined( require_noerr_string ) ) - #define require_noerr_string( ERR, LABEL, STR ) \ - do \ - { \ - int_least32_t localErr; \ - \ - localErr = (int_least32_t)( ERR ); \ - if( localErr != 0 ) \ - { \ - debug_print_assert( localErr, NULL, STR, __FILE__, __LINE__, __ROUTINE__ ); \ - goto LABEL; \ - } \ - \ - } while( 0 ) -#endif - -//--------------------------------------------------------------------------------------------------------------------------- -/*! @defined require_noerr_action_string - - @abstract Require that an error code is noErr (0). - - @discussion - - If the error code is non-0, this prints debugging information (actual expression string, file, line number, - function name, etc.), and a custom explanation string using the default debugging output method using the - default debugging output method then executes an action and jumps to a label. -*/ - -#if( DEBUG_OVERRIDE_APPLE_MACROS ) - #undef require_noerr_action_string -#endif -#if( !defined( require_noerr_action_string ) ) - #define require_noerr_action_string( ERR, LABEL, ACTION, STR ) \ - do \ - { \ - int_least32_t localErr; \ - \ - localErr = (int_least32_t)( ERR ); \ - if( localErr != 0 ) \ - { \ - debug_print_assert( localErr, NULL, STR, __FILE__, __LINE__, __ROUTINE__ ); \ - { ACTION; } \ - goto LABEL; \ - } \ - \ - } while( 0 ) -#endif - -//--------------------------------------------------------------------------------------------------------------------------- -/*! @defined require_noerr_quiet - - @abstract Require that an error code is noErr (0). - - @discussion - - If the error code is non-0, this jumps to a label. No debugging information is printed. -*/ - -#if( DEBUG_OVERRIDE_APPLE_MACROS ) - #undef require_noerr_quiet -#endif -#if( !defined( require_noerr_quiet ) ) - #define require_noerr_quiet( ERR, LABEL ) \ - do \ - { \ - if( ( ERR ) != 0 ) \ - { \ - goto LABEL; \ - } \ - \ - } while( 0 ) -#endif - -//--------------------------------------------------------------------------------------------------------------------------- -/*! @defined require_noerr_action - - @abstract Require that an error code is noErr (0) with an action to execute otherwise. - - @discussion - - If the error code is non-0, this prints debugging information (actual expression string, file, line number, - function name, etc.) using the default debugging output method then executes an action and jumps to a label. -*/ - -#if( DEBUG_OVERRIDE_APPLE_MACROS ) - #undef require_noerr_action -#endif -#if( !defined( require_noerr_action ) ) - #define require_noerr_action( ERR, LABEL, ACTION ) \ - do \ - { \ - int_least32_t localErr; \ - \ - localErr = (int_least32_t)( ERR ); \ - if( localErr != 0 ) \ - { \ - debug_print_assert( localErr, NULL, NULL, __FILE__, __LINE__, __ROUTINE__ ); \ - { ACTION; } \ - goto LABEL; \ - } \ - \ - } while( 0 ) -#endif - -//--------------------------------------------------------------------------------------------------------------------------- -/*! @defined require_noerr_action_quiet - - @abstract Require that an error code is noErr (0) with an action to execute otherwise. - - @discussion - - If the error code is non-0, this executes an action and jumps to a label. No debugging information is printed. -*/ - -#if( DEBUG_OVERRIDE_APPLE_MACROS ) - #undef require_noerr_action_quiet -#endif -#if( !defined( require_noerr_action_quiet ) ) - #define require_noerr_action_quiet( ERR, LABEL, ACTION ) \ - do \ - { \ - if( ( ERR ) != 0 ) \ - { \ - { ACTION; } \ - goto LABEL; \ - } \ - \ - } while( 0 ) -#endif - -//--------------------------------------------------------------------------------------------------------------------------- -/*! @defined require_action - - @abstract Requires that an expression evaluate to true with an action to execute otherwise. - - @discussion - - If expression evalulates to false, this prints debugging information (actual expression string, file, line number, - function name, etc.) using the default debugging output method then executes an action and jumps to a label. -*/ - -#if( DEBUG_OVERRIDE_APPLE_MACROS ) - #undef require_action -#endif -#if( !defined( require_action ) ) - #define require_action( X, LABEL, ACTION ) \ - do \ - { \ - if( !( X ) ) \ - { \ - debug_print_assert( 0, #X, NULL, __FILE__, __LINE__, __ROUTINE__ ); \ - { ACTION; } \ - goto LABEL; \ - } \ - \ - } while( 0 ) -#endif - -//--------------------------------------------------------------------------------------------------------------------------- -/*! @defined require_action_quiet - - @abstract Requires that an expression evaluate to true with an action to execute otherwise. - - @discussion - - If expression evalulates to false, this executes an action and jumps to a label. No debugging information is printed. -*/ - -#if( DEBUG_OVERRIDE_APPLE_MACROS ) - #undef require_action_quiet -#endif -#if( !defined( require_action_quiet ) ) - #define require_action_quiet( X, LABEL, ACTION ) \ - do \ - { \ - if( !( X ) ) \ - { \ - { ACTION; } \ - goto LABEL; \ - } \ - \ - } while( 0 ) -#endif - -//--------------------------------------------------------------------------------------------------------------------------- -/*! @defined require_action_string - - @abstract Requires that an expression evaluate to true with an explanation and action to execute otherwise. - - @discussion - - If expression evalulates to false, this prints debugging information (actual expression string, file, line number, - function name, etc.) and a custom explanation string using the default debugging output method then executes an - action and jumps to a label. -*/ - -#if( DEBUG_OVERRIDE_APPLE_MACROS ) - #undef require_action_string -#endif -#if( !defined( require_action_string ) ) - #define require_action_string( X, LABEL, ACTION, STR ) \ - do \ - { \ - if( !( X ) ) \ - { \ - debug_print_assert( 0, #X, STR, __FILE__, __LINE__, __ROUTINE__ ); \ - { ACTION; } \ - goto LABEL; \ - } \ - \ - } while( 0 ) - -#endif - -//--------------------------------------------------------------------------------------------------------------------------- -/*! @defined require_throw - - @abstract Requires that an expression evaluates to true or an exception is thrown. - - @discussion - - If the expression evaluates to false, this prints debugging information (actual expression string, file, - line number, function name, etc.) using the default debugging output method then throws an exception. -*/ - -#if( defined( __cplusplus ) ) - #define require_throw( X ) \ - do \ - { \ - if( !( X ) ) \ - { \ - debug_print_assert( 0, #X, NULL, __FILE__, __LINE__, __ROUTINE__ ); \ - throw kUnknownErr; \ - } \ - \ - } while( 0 ) -#endif - -#if 0 -#pragma mark == Design-By-Contract macros == -#endif - -//=========================================================================================================================== -// Design-By-Contract macros -//=========================================================================================================================== - -#define ensure( X ) check( X ) -#define ensure_string( X, STR ) check_string( X, STR ) -#define ensure_noerr( ERR ) check_noerr( ERR ) -#define ensure_noerr_string( ERR, STR ) check_noerr_string( ERR, STR ) -#define ensure_translated_errno( TEST, ERRNO, ALTERNATE_ERROR ) check_translated_errno( TEST, ERRNO, ALTERNATE_ERROR ) - -// Note: Design-By-Contract "require" macros are already defined elsewhere. - -#if 0 -#pragma mark == Expect macros == -#endif - -//=========================================================================================================================== -// Expect macros -//=========================================================================================================================== - -// Expect macros allow code to include runtime checking of things that should not happen in shipping code (e.g. internal -// programmer errors, such as a NULL parameter where it is not allowed). Once the code has been verified to work correctly -// without asserting, the DEBUG_EXPECT_VERIFIED conditional can be set to eliminate the error checking entirely. It can -// also be useful to measure the cost of error checking code by profiling with it enable and with it disabled. - -#if( DEBUG_EXPECT_VERIFIED ) - #define require_expect - #define require_string_expect - #define require_quiet_expect - #define require_noerr_expect - #define require_noerr_string_expect - #define require_noerr_action_string_expect - #define require_noerr_quiet_expect - #define require_noerr_action_expect - #define require_noerr_action_quiet_expect - #define require_action_expect - #define require_action_quiet_expect - #define require_action_string_expect -#else - #define require_expect require - #define require_string_expect require_string - #define require_quiet_expect require_quiet - #define require_noerr_expect require_noerr - #define require_noerr_string_expect require_noerr_string - #define require_noerr_action_string_expect require_noerr_action_string - #define require_noerr_quiet_expect require_noerr_quiet - #define require_noerr_action_expect require_noerr_action - #define require_noerr_action_quiet_expect require_noerr_action_quiet - #define require_action_expect require_action - #define require_action_quiet_expect require_action_quiet - #define require_action_string_expect require_action_string -#endif - -#if 0 -#pragma mark == Output macros == -#endif - -//--------------------------------------------------------------------------------------------------------------------------- -/*! @defined debug_string - - @abstract Prints a debugging C string. -*/ - -#if( DEBUG_OVERRIDE_APPLE_MACROS ) - #undef debug_string -#endif -#if( !defined( debug_string ) ) - #if( DEBUG ) - #define debug_string( STR ) \ - do \ - { \ - debug_print_assert( 0, NULL, STR, __FILE__, __LINE__, __ROUTINE__ ); \ - \ - } while( 0 ) - #else - #define debug_string( STR ) - #endif -#endif - -//--------------------------------------------------------------------------------------------------------------------------- -/*! @defined debug_print_assert - - @abstract Prints an assertion. -*/ - -#if( DEBUG ) - #define debug_print_assert( ERROR_CODE, ASSERT_STRING, MESSAGE, FILENAME, LINE_NUMBER, FUNCTION ) \ - DebugPrintAssert( ERROR_CODE, ASSERT_STRING, MESSAGE, FILENAME, LINE_NUMBER, FUNCTION ) -#else - #define debug_print_assert( ERROR_CODE, ASSERT_STRING, MESSAGE, FILENAME, LINE_NUMBER, FUNCTION ) -#endif - -//--------------------------------------------------------------------------------------------------------------------------- -/*! @defined dlog - - @abstract Prints a debug-only message. -*/ - -#if( DEBUG ) - #if( DEBUG_C99_VA_ARGS ) - #define dlog( ... ) DebugPrintF( __VA_ARGS__ ) - #elif( DEBUG_GNU_VA_ARGS ) - #define dlog( ARGS... ) DebugPrintF( ## ARGS ) - #else - #define dlog DebugPrintF - #endif -#else - #if( DEBUG_C99_VA_ARGS ) - #define dlog( ... ) - #elif( DEBUG_GNU_VA_ARGS ) - #define dlog( ARGS... ) - #else - #define dlog while( 0 ) - #endif -#endif - -//--------------------------------------------------------------------------------------------------------------------------- -/*! @defined dlogv - - @abstract Prints a debug-only message. -*/ - -#if( DEBUG ) - #define dlogv( LEVEL, FORMAT, LIST ) DebugPrintFVAList( ( LEVEL ), ( FORMAT ), ( LIST ) ) -#else - #define dlogv( LEVEL, FORMAT, LIST ) -#endif - -//--------------------------------------------------------------------------------------------------------------------------- -/*! @defined dlogmem - - @abstract Prints a debug-only dump of memory. -*/ - -#if( DEBUG ) - #define dlogmem( LEVEL, PTR, SIZE ) \ - DebugHexDump( ( LEVEL ), 0, NULL, 0, 0, NULL, 0, ( PTR ), ( PTR ), ( SIZE ), kDebugFlagsNone, NULL, 0 ) -#else - #define dlogmem( LEVEL, PTR, SIZE ) -#endif - -//--------------------------------------------------------------------------------------------------------------------------- -/*! @defined DebugNSLog - - @abstract Debug-only macro for the Cocoa NSLog function. -*/ - -#if( DEBUG ) - #if( DEBUG_C99_VA_ARGS ) - #define DebugNSLog( ... ) NSLog( __VA_ARGS__ ) - #elif( DEBUG_GNU_VA_ARGS ) - #define DebugNSLog( ARGS... ) NSLog( ## ARGS ) - #else - #define DebugNSLog NSLog - #endif -#else - #if( DEBUG_C99_VA_ARGS ) - #define DebugNSLog( ... ) - #elif( DEBUG_GNU_VA_ARGS ) - #define DebugNSLog( ARGS... ) - #else - #define DebugNSLog while( 0 ) - #endif -#endif - -//--------------------------------------------------------------------------------------------------------------------------- -/*! @defined DebugLogMsg - - @abstract Debug-only macro for the VxWorks logMsg function. -*/ - -#if( TARGET_OS_VXWORKS ) - #if( DEBUG ) - #define DebugLogMsg( LEVEL, FORMAT, P1, P2, P3, P4, P5, P6 ) \ - do \ - { \ - if( ( inLevel >= gDebugPrintLevelMin ) || ( inLevel <= gDebugPrintLevelMax ) ) \ - { \ - logMsg( ( FORMAT ), ( P1 ), ( P2 ), ( P3 ), ( P4 ), ( P5 ), ( P6 ) ); \ - } \ - \ - } while( 0 ) - #else - #define DebugLogMsg( LEVEL, FORMAT, P1, P2, P3, P4, P5, P6 ) - #endif -#else - #define DebugLogMsg dlog -#endif - -#if 0 -#pragma mark == Routines - General == -#endif - -#ifdef __cplusplus - extern "C" { -#endif - -//--------------------------------------------------------------------------------------------------------------------------- -/*! @function DebugInitialize - - @abstract Initializes the debugging library for a specific kind of output. - - @param inType - @param varArg Variable number parameters, controlled by the "inType" parameter. -*/ - -#if( DEBUG ) - DEBUG_EXPORT OSStatus DebugInitialize( DebugOutputType inType, ... ); -#endif - -#if( DEBUG ) - #if( DEBUG_C99_VA_ARGS ) - #define debug_initialize( ... ) DebugInitialize( __VA_ARGS__ ) - #elif( DEBUG_GNU_VA_ARGS ) - #define debug_initialize( ARGS... ) DebugInitialize( ## ARGS ) - #else - #define debug_initialize DebugInitialize - #endif -#else - #if( DEBUG_C99_VA_ARGS ) - #define debug_initialize( ... ) - #elif( DEBUG_GNU_VA_ARGS ) - #define debug_initialize( ARGS... ) - #else - #define debug_initialize while( 0 ) - #endif -#endif - -//--------------------------------------------------------------------------------------------------------------------------- -/*! @function DebugFinalize - - @abstract Releases any resources used by the debugging library -*/ - -#if( DEBUG ) - DEBUG_EXPORT void DebugFinalize( void ); -#endif - -#if( DEBUG ) - #define debug_terminate() DebugFinalize() -#else - #define debug_terminate() -#endif - -//--------------------------------------------------------------------------------------------------------------------------- -/*! @function DebugGetProperty - - @abstract Gets the specified property from the debugging library. -*/ - -#if( DEBUG ) - DEBUG_EXPORT OSStatus DebugGetProperty( DebugPropertyTag inTag, ... ); -#endif - -#if( DEBUG ) - #if( DEBUG_C99_VA_ARGS ) - #define debug_get_property( ... ) DebugGetProperty( __VA_ARGS__ ) - #elif( DEBUG_GNU_VA_ARGS ) - #define debug_get_property( ARGS... ) DebugGetProperty( ## ARGS ) - #else - #define debug_get_property DebugGetProperty - #endif -#else - #if( DEBUG_C99_VA_ARGS ) - #define debug_get_property( ... ) - #elif( DEBUG_GNU_VA_ARGS ) - #define debug_get_property( ARGS... ) - #else - #define debug_get_property while( 0 ) - #endif -#endif - -//--------------------------------------------------------------------------------------------------------------------------- -/*! @function DebugSetProperty - - @abstract Sets the specified property from the debugging library. -*/ - -#if( DEBUG ) - DEBUG_EXPORT OSStatus DebugSetProperty( DebugPropertyTag inTag, ... ); -#endif - -#if( DEBUG ) - #if( DEBUG_C99_VA_ARGS ) - #define debug_set_property( ... ) DebugSetProperty( __VA_ARGS__ ) - #elif( DEBUG_GNU_VA_ARGS ) - #define debug_set_property( ARGS... ) DebugSetProperty( ## ARGS ) - #else - #define debug_set_property DebugSetProperty - #endif -#else - #if( DEBUG_C99_VA_ARGS ) - #define debug_set_property( ... ) - #elif( DEBUG_GNU_VA_ARGS ) - #define debug_set_property( ARGS... ) - #else - #define debug_set_property while( 0 ) - #endif -#endif - -#if 0 -#pragma mark == Routines - Debugging Output == -#endif - -//--------------------------------------------------------------------------------------------------------------------------- -/*! @function DebugPrintF - - @abstract Prints a debug message with printf-style formatting. - - @param inLevel Error that generated this assert or noErr. - - @param inFormatString - C string containing assertion text. - - @param VAR_ARG - Variable number of arguments depending on the format string. - - @result Number of bytes printed or -1 on error. -*/ - -#if( DEBUG ) - DEBUG_EXPORT size_t DebugPrintF( DebugLevel inLevel, const char *inFormat, ... ); -#endif - -//--------------------------------------------------------------------------------------------------------------------------- -/*! @function DebugPrintFVAList - - @abstract va_list version of DebugPrintF. See DebugPrintF for more info. -*/ - -#if( DEBUG ) - DEBUG_EXPORT size_t DebugPrintFVAList( DebugLevel inLevel, const char *inFormat, va_list inArgs ); -#endif - -//--------------------------------------------------------------------------------------------------------------------------- -/*! @function DebugPrintAssert - - @abstract Prints a message describing the reason the (e.g. an assert failed), an optional error message, - an optional source filename, an optional source line number. - - @param inErrorCode Error that generated this assert or noErr. - @param inAssertString C string containing assertion text. - @param inMessage C string containing a message about the assert. - @param inFileName C string containing path of file where the error occurred. - @param inLineNumber Line number in source file where the error occurred. - @param inFunction C string containing name of function where assert occurred. - - @discussion - - Example output: - - [ASSERT] assert: "dataPtr != NULL" allocate memory for object failed - [ASSERT] where: "MyFile.c", line 123, ("MyFunction") - - OR - - [ASSERT] error: -6728 (kNoMemoryErr) - [ASSERT] where: "MyFile.c", line 123, ("MyFunction") -*/ - -#if( DEBUG ) - DEBUG_EXPORT void - DebugPrintAssert( - int_least32_t inErrorCode, - const char * inAssertString, - const char * inMessage, - const char * inFilename, - int_least32_t inLineNumber, - const char * inFunction ); -#endif - -#if 0 -#pragma mark == Routines - Utilities == -#endif - -//--------------------------------------------------------------------------------------------------------------------------- -/*! @function DebugSNPrintF - - @abstract Debugging versions of standard C snprintf with extra features. - - @param sbuffer Buffer to receive result. Null terminated unless the buffer size is 0. - @param buflen Size of the buffer including space for the null terminator. - @param fmt printf-style format string. - @param VAR_ARG Variable number of arguments depending on the format string. - - @result Number of characters written (minus the null terminator). - - @discussion - - Extra features over the standard C snprintf: -
-		64-bit support for %d (%lld), %i (%lli), %u (%llu), %o (%llo), %x (%llx), and %b (%llb).
-		%@   - Cocoa/CoreFoundation object. Param is the object. Strings are used directly. Others use CFCopyDescription.
-		%a   - Network Address: %.4a=IPv4, %.6a=Ethernet, %.8a Fibre Channel, %.16a=IPv6. Arg=ptr to network address.
-		%#a  - IPv4 or IPv6 mDNSAddr. Arg=ptr to mDNSAddr.
-		%##a - IPv4 (if AF_INET defined) or IPv6 (if AF_INET6 defined) sockaddr. Arg=ptr to sockaddr.
-		%b   - Binary representation of integer (e.g. 01101011). Modifiers and arg=the same as %d, %x, etc.
-		%C   - Mac-style FourCharCode (e.g. 'APPL'). Arg=32-bit value to print as a Mac-style FourCharCode.
-		%H   - Hex Dump (e.g. "\x6b\xa7" -> "6B A7"). 1st arg=ptr, 2nd arg=size, 3rd arg=max size.
-		%#H  - Hex Dump & ASCII (e.g. "\x41\x62" -> "6B A7 'Ab'"). 1st arg=ptr, 2nd arg=size, 3rd arg=max size.
-		%m   - Error Message (e.g. 0 -> "kNoErr"). Modifiers and error code arg=the same as %d, %x, etc.
-		%#s  - Pascal-style length-prefixed string. Arg=ptr to string.
-		%##s - DNS label-sequence name. Arg=ptr to name.
-		%S   - UTF-16 string, 0x0000 terminated. Host order if no BOM. Precision is UTF-16 count. Precision includes BOM.
-		%#S  - Big Endian UTF-16 string (unless BOM overrides). Otherwise, the same as %S.
-		%##S - Little Endian UTF-16 string (unless BOM overrides). Otherwise, the same as %S.
-		%U   - Universally Unique Identifier (UUID) (e.g. 6ba7b810-9dad-11d1-80b4-00c04fd430c8). Arg=ptr to 16-byte UUID.
-	
-*/ - -#if( DEBUG ) - DEBUG_EXPORT size_t DebugSNPrintF(char *sbuffer, size_t buflen, const char *fmt, ...); -#endif - -//--------------------------------------------------------------------------------------------------------------------------- -/*! @function DebugSNPrintFVAList - - @abstract va_list version of DebugSNPrintF. See DebugSNPrintF for more info. -*/ - -#if( DEBUG ) - DEBUG_EXPORT size_t DebugSNPrintFVAList(char *sbuffer, size_t buflen, const char *fmt, va_list arg); -#endif - -//--------------------------------------------------------------------------------------------------------------------------- -/*! @function DebugGetErrorString - - @abstract Gets an error string from an error code. - - @param inStatus Error code to get the string for. - @param inBuffer Optional buffer to copy the string to for non-static strings. May be null. - @param inBufferSize Size of optional buffer. May be 0. - - @result C string containing error string for the error code. Guaranteed to be a valid, static string. If a - buffer is supplied, the return value will always be a pointer to the supplied buffer, which will - contain the best available description of the error code. If a buffer is not supplied, the return - value will be the best available description of the error code that can be represented as a static - string. This allows code that cannot use a temporary buffer to hold the result to still get a useful - error string in most cases, but also allows code that can use a temporary buffer to get the best - available description. -*/ - -#if( DEBUG ) - DEBUG_EXPORT const char * DebugGetErrorString( int_least32_t inErrorCode, char *inBuffer, size_t inBufferSize ); -#endif - -//--------------------------------------------------------------------------------------------------------------------------- -/*! @function DebugHexDump - - @abstract Hex dumps data to a string or to the output device. -*/ - -#if( DEBUG ) - DEBUG_EXPORT size_t - DebugHexDump( - DebugLevel inLevel, - int inIndent, - const char * inLabel, - size_t inLabelSize, - int inLabelMinWidth, - const char * inType, - size_t inTypeSize, - const void * inDataStart, - const void * inData, - size_t inDataSize, - DebugFlags inFlags, - char * outBuffer, - size_t inBufferSize ); -#endif - -#if( DEBUG ) - #define dloghex( LEVEL, INDENT, LABEL, LABEL_SIZE, LABEL_MIN_SIZE, TYPE, TYPE_SIZE, DATA_START, DATA, DATA_SIZE, FLAGS, BUFFER, BUFFER_SIZE ) \ - DebugHexDump( ( LEVEL ), (INDENT), ( LABEL ), ( LABEL_SIZE ), ( LABEL_MIN_SIZE ), ( TYPE ), ( TYPE_SIZE ), \ - ( DATA_START ), ( DATA ), ( DATA_SIZE ), ( FLAGS ), ( BUFFER ), ( BUFFER_SIZE ) ) -#else - #define dloghex( LEVEL, INDENT, LABEL, LABEL_SIZE, LABEL_MIN_SIZE, TYPE, TYPE_SIZE, DATA_START, DATA, DATA_SIZE, FLAGS, BUFFER, BUFFER_SIZE ) -#endif - -//--------------------------------------------------------------------------------------------------------------------------- -/*! @function DebugTaskLevel - - @abstract Returns the current task level. - - @result Current task level - - @discussion - - Bit masks to isolate portions of the result (note that some masks may also need bit shifts to right justify): -
-		kDebugInterruptLevelMask				- Indicates the current interrupt level (> 0 means interrupt time).
-		kDebugInVBLTaskMask						- Indicates if a VBL task is currently being executed.
-		kDebugInDeferredTaskMask				- Indicates if a Deferred Task is currently being executed.
-		kDebugInSecondaryInterruptHandlerMask	- Indicates if a Secondary Interrupt Handler is currently being executed.
-		kDebugPageFaultFatalMask				- Indicates if it is unsafe to cause a page fault (worse than interrupt time).
-		kDebugMPTaskLevelMask					- Indicates if being called from an MP task.
-		kDebugInterruptDepthMask				- 0 means task level, 1 means in interrupt, > 1 means in nested interrupt.
-	
- - Helpers: -
-		DebugExtractTaskLevelInterruptDepth()   - Macro to extract interrupt depth from task level value.
-	
-*/ - -#if( DEBUG ) - DEBUG_EXPORT uint32_t DebugTaskLevel( void ); -#endif - -//--------------------------------------------------------------------------------------------------------------------------- -/*! @function DebugServicesTest - - @abstract Unit test. -*/ - -#if( DEBUG ) - DEBUG_EXPORT OSStatus DebugServicesTest( void ); -#endif - -#ifdef __cplusplus - } -#endif - -#endif // __DEBUG_SERVICES__ diff -Nru qtcreator-2.5.0/src/tools/mdnssd/GenLinkedList.c qtcreator-2.5.2/src/tools/mdnssd/GenLinkedList.c --- qtcreator-2.5.0/src/tools/mdnssd/GenLinkedList.c 2012-05-09 12:13:19.000000000 +0000 +++ qtcreator-2.5.2/src/tools/mdnssd/GenLinkedList.c 1970-01-01 00:00:00.000000000 +0000 @@ -1,319 +0,0 @@ -/* -*- Mode: C; tab-width: 4 -*- - * - * Copyright (c) 2003 Apple Computer, Inc. All rights reserved. - * - * Licensed under the Apache License, Version 2.0 (the "License"); - * you may not use this file except in compliance with the License. - * You may obtain a copy of the License at - * - * http://www.apache.org/licenses/LICENSE-2.0 - * - * Unless required by applicable law or agreed to in writing, software - * distributed under the License is distributed on an "AS IS" BASIS, - * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. - * See the License for the specific language governing permissions and - * limitations under the License. - - File: GenLinkedList.c - - Contains: implementation of generic linked lists. - - Version: 1.0 - Tabs: 4 spaces - */ - -#include "GenLinkedList.h" - - -// Return the link pointer contained within element e at offset o. -#define GETLINK( e, o) ( *(void**)((char*) (e) + (o)) ) - -// Assign the link pointer l to element e at offset o. -#define ASSIGNLINK( e, l, o) ( *((void**)((char*) (e) + (o))) = (l)) - - -// GenLinkedList ///////////////////////////////////////////////////////////// - -void InitLinkedList( GenLinkedList *pList, size_t linkOffset) -/* Initialize the block of memory pointed to by pList as a linked list. */ -{ - pList->Head = NULL; - pList->Tail = NULL; - pList->LinkOffset = linkOffset; -} - - -void AddToTail( GenLinkedList *pList, void *elem) -/* Add a linked list element to the tail of the list. */ -{ - if ( pList->Tail) { - ASSIGNLINK( pList->Tail, elem, pList->LinkOffset); - } else - pList->Head = elem; - ASSIGNLINK( elem, NULL, pList->LinkOffset); - - pList->Tail = elem; -} - - -void AddToHead( GenLinkedList *pList, void *elem) -/* Add a linked list element to the head of the list. */ -{ - ASSIGNLINK( elem, pList->Head, pList->LinkOffset); - if ( pList->Tail == NULL) - pList->Tail = elem; - - pList->Head = elem; -} - - -int RemoveFromList( GenLinkedList *pList, void *elem) -/* Remove a linked list element from the list. Return 0 if it was not found. */ -/* If the element is removed, its link will be set to NULL. */ -{ -void *iElem, *lastElem; - - for ( iElem = pList->Head, lastElem = NULL; iElem; iElem = GETLINK( iElem, pList->LinkOffset)) { - if ( iElem == elem) { - if ( lastElem) { // somewhere past the head - ASSIGNLINK( lastElem, GETLINK( elem, pList->LinkOffset), pList->LinkOffset); - } else { // at the head - pList->Head = GETLINK( elem, pList->LinkOffset); - } - if ( pList->Tail == elem) - pList->Tail = lastElem ? lastElem : NULL; - ASSIGNLINK( elem, NULL, pList->LinkOffset); // maybe catch a stale reference bug. - return 1; - } - lastElem = iElem; - } - - return 0; -} - - -int ReplaceElem( GenLinkedList *pList, void *elemInList, void *newElem) -/* Replace an element in the list with a new element, in the same position. */ -{ -void *iElem, *lastElem; - - if ( elemInList == NULL || newElem == NULL) - return 0; - - for ( iElem = pList->Head, lastElem = NULL; iElem; iElem = GETLINK( iElem, pList->LinkOffset)) - { - if ( iElem == elemInList) - { - ASSIGNLINK( newElem, GETLINK( elemInList, pList->LinkOffset), pList->LinkOffset); - if ( lastElem) // somewhere past the head - { - ASSIGNLINK( lastElem, newElem, pList->LinkOffset); - } - else // at the head - { - pList->Head = newElem; - } - if ( pList->Tail == elemInList) - pList->Tail = newElem; - return 1; - } - lastElem = iElem; - } - - return 0; -} - - -// GenDoubleLinkedList ///////////////////////////////////////////////////////// - -void InitDoubleLinkedList( GenDoubleLinkedList *pList, size_t fwdLinkOffset, - size_t backLinkOffset) -/* Initialize the block of memory pointed to by pList as a double linked list. */ -{ - pList->Head = NULL; - pList->Tail = NULL; - pList->FwdLinkOffset = fwdLinkOffset; - pList->BackLinkOffset = backLinkOffset; -} - - -void DLLAddToHead( GenDoubleLinkedList *pList, void *elem) -/* Add a linked list element to the head of the list. */ -{ -void *pNext; - - pNext = pList->Head; - - // fix up the forward links - ASSIGNLINK( elem, pList->Head, pList->FwdLinkOffset); - pList->Head = elem; - - // fix up the backward links - if ( pNext) { - ASSIGNLINK( pNext, elem, pList->BackLinkOffset); - } else - pList->Tail = elem; - ASSIGNLINK( elem, NULL, pList->BackLinkOffset); -} - - -void DLLRemoveFromList( GenDoubleLinkedList *pList, void *elem) -/* Remove a linked list element from the list. */ -/* When the element is removed, its link will be set to NULL. */ -{ -void *pNext, *pPrev; - - pNext = GETLINK( elem, pList->FwdLinkOffset); - pPrev = GETLINK( elem, pList->BackLinkOffset); - - // fix up the forward links - if ( pPrev) - ASSIGNLINK( pPrev, pNext, pList->FwdLinkOffset); - else - pList->Head = pNext; - - // fix up the backward links - if ( pNext) - ASSIGNLINK( pNext, pPrev, pList->BackLinkOffset); - else - pList->Tail = pPrev; - - ASSIGNLINK( elem, NULL, pList->FwdLinkOffset); - ASSIGNLINK( elem, NULL, pList->BackLinkOffset); -} - - -// GenLinkedOffsetList ///////////////////////////////////////////////////// - -// Extract the Next offset from element -#define GETOFFSET( e, o) ( *(size_t*)((char*) (e) + (o)) ) - -static void AssignOffsetLink( void *elem, void *link, size_t linkOffset); - - -static void AssignOffsetLink( void *elem, void *link, size_t linkOffset) -// Assign link to elem as an offset from elem. Assign 0 to elem if link is NULL. -{ - GETOFFSET( elem, linkOffset) = link ? (size_t) link - (size_t) elem : 0; -} - - -void *GetHeadPtr( GenLinkedOffsetList *pList) -/* Return a pointer to the head element of a list, or NULL if none. */ -{ - return pList->Head ? ( (char*) (pList) + pList->Head) : NULL; -} - - -void *GetTailPtr( GenLinkedOffsetList *pList) -/* Return a pointer to the tail element of a list, or NULL if none. */ -{ - return pList->Tail ? ( (char*) (pList) + pList->Tail) : NULL; -} - - -void *GetOffsetLink( GenLinkedOffsetList *pList, void *elem) -/* Return the link pointer contained within element e for pList, or NULL if it is 0. */ -{ -size_t nextOffset; - - nextOffset = GETOFFSET( elem, pList->LinkOffset); - - return nextOffset ? (char*) elem + nextOffset : NULL; -} - - -void InitLinkedOffsetList( GenLinkedOffsetList *pList, size_t linkOffset) -/* Initialize the block of memory pointed to by pList as a linked list. */ -{ - pList->Head = 0; - pList->Tail = 0; - pList->LinkOffset = linkOffset; -} - - -void OffsetAddToTail( GenLinkedOffsetList *pList, void *elem) -/* Add a linked list element to the tail of the list. */ -{ - if ( pList->Tail) { - AssignOffsetLink( GetTailPtr( pList), elem, pList->LinkOffset); - } else - pList->Head = (size_t) elem - (size_t) pList; - AssignOffsetLink( elem, NULL, pList->LinkOffset); - - pList->Tail = (size_t) elem - (size_t) pList; -} - - -void OffsetAddToHead( GenLinkedOffsetList *pList, void *elem) -/* Add a linked list element to the head of the list. */ -{ - AssignOffsetLink( elem, GetHeadPtr( pList), pList->LinkOffset); - if ( pList->Tail == 0) - pList->Tail = (size_t) elem - (size_t) pList; - - pList->Head = (size_t) elem - (size_t) pList; -} - - -int OffsetRemoveFromList( GenLinkedOffsetList *pList, void *elem) -/* Remove a linked list element from the list. Return 0 if it was not found. */ -/* If the element is removed, its link will be set to NULL. */ -{ -void *iElem, *lastElem; - - for ( iElem = GetHeadPtr( pList), lastElem = NULL; iElem; - iElem = GetOffsetLink( pList, iElem)) - { - if ( iElem == elem) { - if ( lastElem) { // somewhere past the head - AssignOffsetLink( lastElem, GetOffsetLink( pList, elem), pList->LinkOffset); - } else { // at the head - iElem = GetOffsetLink( pList, elem); - pList->Head = iElem ? (size_t) iElem - (size_t) pList : 0; - } - if ( GetTailPtr( pList) == elem) - pList->Tail = lastElem ? (size_t) lastElem - (size_t) pList : 0; - AssignOffsetLink( elem, NULL, pList->LinkOffset); // maybe catch a stale reference bug. - return 1; - } - lastElem = iElem; - } - - return 0; -} - - -int OffsetReplaceElem( GenLinkedOffsetList *pList, void *elemInList, void *newElem) -/* Replace an element in the list with a new element, in the same position. */ -{ -void *iElem, *lastElem; - - if ( elemInList == NULL || newElem == NULL) - return 0; - - for ( iElem = GetHeadPtr( pList), lastElem = NULL; iElem; - iElem = GetOffsetLink( pList, iElem)) - { - if ( iElem == elemInList) - { - AssignOffsetLink( newElem, GetOffsetLink( pList, elemInList), pList->LinkOffset); - if ( lastElem) // somewhere past the head - { - AssignOffsetLink( lastElem, newElem, pList->LinkOffset); - } - else // at the head - { - pList->Head = (size_t) newElem - (size_t) pList; - } - if ( GetTailPtr( pList) == elemInList) - pList->Tail = (size_t) newElem - (size_t) pList; - return 1; - } - lastElem = iElem; - } - - return 0; -} - - diff -Nru qtcreator-2.5.0/src/tools/mdnssd/GenLinkedList.h qtcreator-2.5.2/src/tools/mdnssd/GenLinkedList.h --- qtcreator-2.5.0/src/tools/mdnssd/GenLinkedList.h 2012-05-09 12:13:19.000000000 +0000 +++ qtcreator-2.5.2/src/tools/mdnssd/GenLinkedList.h 1970-01-01 00:00:00.000000000 +0000 @@ -1,90 +0,0 @@ -/* -*- Mode: C; tab-width: 4 -*- - * - * Copyright (c) 2003 Apple Computer, Inc. All rights reserved. - * - * Licensed under the Apache License, Version 2.0 (the "License"); - * you may not use this file except in compliance with the License. - * You may obtain a copy of the License at - * - * http://www.apache.org/licenses/LICENSE-2.0 - * - * Unless required by applicable law or agreed to in writing, software - * distributed under the License is distributed on an "AS IS" BASIS, - * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. - * See the License for the specific language governing permissions and - * limitations under the License. - */ - -#ifndef __GenLinkedList__ -#define __GenLinkedList__ - - -#include - - -struct GenLinkedList -{ - void *Head, - *Tail; - size_t LinkOffset; -}; -typedef struct GenLinkedList GenLinkedList; - - -void InitLinkedList( GenLinkedList *pList, size_t linkOffset); - -void AddToHead( GenLinkedList *pList, void *elem); -void AddToTail( GenLinkedList *pList, void *elem); - -int RemoveFromList( GenLinkedList *pList, void *elem); - -int ReplaceElem( GenLinkedList *pList, void *elemInList, void *newElem); - - - -struct GenDoubleLinkedList -{ - void *Head, - *Tail; - size_t FwdLinkOffset, - BackLinkOffset; -}; -typedef struct GenDoubleLinkedList GenDoubleLinkedList; - - -void InitDoubleLinkedList( GenDoubleLinkedList *pList, size_t fwdLinkOffset, - size_t backLinkOffset); - -void DLLAddToHead( GenDoubleLinkedList *pList, void *elem); - -void DLLRemoveFromList( GenDoubleLinkedList *pList, void *elem); - - - -/* A GenLinkedOffsetList is like a GenLinkedList that stores the *Next field as a signed */ -/* offset from the address of the beginning of the element, rather than as a pointer. */ - -struct GenLinkedOffsetList -{ - size_t Head, - Tail; - size_t LinkOffset; -}; -typedef struct GenLinkedOffsetList GenLinkedOffsetList; - - -void InitLinkedOffsetList( GenLinkedOffsetList *pList, size_t linkOffset); - -void *GetHeadPtr( GenLinkedOffsetList *pList); -void *GetTailPtr( GenLinkedOffsetList *pList); -void *GetOffsetLink( GenLinkedOffsetList *pList, void *elem); - -void OffsetAddToHead( GenLinkedOffsetList *pList, void *elem); -void OffsetAddToTail( GenLinkedOffsetList *pList, void *elem); - -int OffsetRemoveFromList( GenLinkedOffsetList *pList, void *elem); - -int OffsetReplaceElem( GenLinkedOffsetList *pList, void *elemInList, void *newElem); - - -#endif // __GenLinkedList__ diff -Nru qtcreator-2.5.0/src/tools/mdnssd/PlatformCommon.c qtcreator-2.5.2/src/tools/mdnssd/PlatformCommon.c --- qtcreator-2.5.0/src/tools/mdnssd/PlatformCommon.c 2012-05-09 12:13:19.000000000 +0000 +++ qtcreator-2.5.2/src/tools/mdnssd/PlatformCommon.c 1970-01-01 00:00:00.000000000 +0000 @@ -1,196 +0,0 @@ -/* -*- Mode: C; tab-width: 4 -*- - * - * Copyright (c) 2004 Apple Computer, Inc. All rights reserved. - * - * Licensed under the Apache License, Version 2.0 (the "License"); - * you may not use this file except in compliance with the License. - * You may obtain a copy of the License at - * - * http://www.apache.org/licenses/LICENSE-2.0 - * - * Unless required by applicable law or agreed to in writing, software - * distributed under the License is distributed on an "AS IS" BASIS, - * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. - * See the License for the specific language governing permissions and - * limitations under the License. - */ - -#include // Needed for fopen() etc. -#include // Needed for close() -#include // Needed for strlen() etc. -#include // Needed for errno etc. -#include // Needed for socket() etc. -#include // Needed for sockaddr_in -#include - -#include "mDNSEmbeddedAPI.h" // Defines the interface provided to the client layer above -#include "DNSCommon.h" -#include "PlatformCommon.h" - -#ifdef NOT_HAVE_SOCKLEN_T - typedef unsigned int socklen_t; -#endif - -// Bind a UDP socket to find the source address to a destination -mDNSexport void mDNSPlatformSourceAddrForDest(mDNSAddr *const src, const mDNSAddr *const dst) - { - union { struct sockaddr s; struct sockaddr_in a4; struct sockaddr_in6 a6; } addr; - socklen_t len = sizeof(addr); - socklen_t inner_len = 0; - int sock = socket(AF_INET, SOCK_DGRAM, 0); - src->type = mDNSAddrType_None; - if (sock == -1) return; - if (dst->type == mDNSAddrType_IPv4) - { - inner_len = sizeof(addr.a4); - #ifndef NOT_HAVE_SA_LEN - addr.a4.sin_len = inner_len; - #endif - addr.a4.sin_family = AF_INET; - addr.a4.sin_port = 1; // Not important, any port will do - addr.a4.sin_addr.s_addr = dst->ip.v4.NotAnInteger; - } - else if (dst->type == mDNSAddrType_IPv6) - { - inner_len = sizeof(addr.a6); - #ifndef NOT_HAVE_SA_LEN - addr.a6.sin6_len = inner_len; - #endif - addr.a6.sin6_family = AF_INET6; - addr.a6.sin6_flowinfo = 0; - addr.a6.sin6_port = 1; // Not important, any port will do - addr.a6.sin6_addr = *(struct in6_addr*)&dst->ip.v6; - addr.a6.sin6_scope_id = 0; - } - else return; - - if ((connect(sock, &addr.s, inner_len)) < 0) - { LogMsg("mDNSPlatformSourceAddrForDest: connect %#a failed errno %d (%s)", dst, errno, strerror(errno)); goto exit; } - - if ((getsockname(sock, &addr.s, &len)) < 0) - { LogMsg("mDNSPlatformSourceAddrForDest: getsockname failed errno %d (%s)", errno, strerror(errno)); goto exit; } - - src->type = dst->type; - if (dst->type == mDNSAddrType_IPv4) src->ip.v4.NotAnInteger = addr.a4.sin_addr.s_addr; - else src->ip.v6 = *(mDNSv6Addr*)&addr.a6.sin6_addr; -exit: - close(sock); - } - -// dst must be at least MAX_ESCAPED_DOMAIN_NAME bytes, and option must be less than 32 bytes in length -mDNSlocal mDNSBool GetConfigOption(char *dst, const char *option, FILE *f) - { - char buf[32+1+MAX_ESCAPED_DOMAIN_NAME]; // Option name, one space, option value - unsigned int len = strlen(option); - if (len + 1 + MAX_ESCAPED_DOMAIN_NAME > sizeof(buf)-1) { LogMsg("GetConfigOption: option %s too long", option); return mDNSfalse; } - fseek(f, 0, SEEK_SET); // set position to beginning of stream - while (fgets(buf, sizeof(buf), f)) // Read at most sizeof(buf)-1 bytes from file, and append '\0' C-string terminator - { - if (!strncmp(buf, option, len)) - { - strncpy(dst, buf + len + 1, MAX_ESCAPED_DOMAIN_NAME-1); - if (dst[MAX_ESCAPED_DOMAIN_NAME-1]) dst[MAX_ESCAPED_DOMAIN_NAME-1] = '\0'; - len = strlen(dst); - if (len && dst[len-1] == '\n') dst[len-1] = '\0'; // chop newline - return mDNStrue; - } - } - debugf("Option %s not set", option); - return mDNSfalse; - } - -mDNSexport void ReadDDNSSettingsFromConfFile(mDNS *const m, const char *const filename, domainname *const hostname, domainname *const domain, mDNSBool *DomainDiscoveryDisabled) - { - char buf[MAX_ESCAPED_DOMAIN_NAME] = ""; - mStatus err; - FILE *f = fopen(filename, "r"); - - if (hostname) hostname->c[0] = 0; - if (domain) domain->c[0] = 0; - if (DomainDiscoveryDisabled) *DomainDiscoveryDisabled = mDNSfalse; - - if (f) - { - if (DomainDiscoveryDisabled && GetConfigOption(buf, "DomainDiscoveryDisabled", f) && !strcasecmp(buf, "true")) *DomainDiscoveryDisabled = mDNStrue; - if (hostname && GetConfigOption(buf, "hostname", f) && !MakeDomainNameFromDNSNameString(hostname, buf)) goto badf; - if (domain && GetConfigOption(buf, "zone", f) && !MakeDomainNameFromDNSNameString(domain, buf)) goto badf; - buf[0] = 0; - GetConfigOption(buf, "secret-64", f); // failure means no authentication - fclose(f); - f = NULL; - } - else - { - if (errno != ENOENT) LogMsg("ERROR: Config file exists, but cannot be opened."); - return; - } - - if (domain && domain->c[0] && buf[0]) - { - DomainAuthInfo *info = (DomainAuthInfo*)mDNSPlatformMemAllocate(sizeof(*info)); - // for now we assume keyname = service reg domain and we use same key for service and hostname registration - err = mDNS_SetSecretForDomain(m, info, domain, domain, buf, NULL, 0, NULL); - if (err) LogMsg("ERROR: mDNS_SetSecretForDomain returned %d for domain %##s", err, domain->c); - } - - return; - - badf: - LogMsg("ERROR: malformatted config file"); - if (f) fclose(f); - } - -#if MDNS_DEBUGMSGS -mDNSexport void mDNSPlatformWriteDebugMsg(const char *msg) - { - fprintf(stderr,"%s\n", msg); - fflush(stderr); - } -#endif - -mDNSexport void mDNSPlatformWriteLogMsg(const char *ident, const char *buffer, mDNSLogLevel_t loglevel) - { -#if APPLE_OSX_mDNSResponder && LogTimeStamps - extern mDNS mDNSStorage; - extern mDNSu32 mDNSPlatformClockDivisor; - mDNSs32 t = mDNSStorage.timenow ? mDNSStorage.timenow : mDNSPlatformClockDivisor ? mDNS_TimeNow_NoLock(&mDNSStorage) : 0; - int ms = ((t < 0) ? -t : t) % 1000; -#endif - - if (mDNS_DebugMode) // In debug mode we write to stderr - { -#if APPLE_OSX_mDNSResponder && LogTimeStamps - if (ident && ident[0] && mDNSPlatformClockDivisor) - fprintf(stderr,"%8d.%03d: %s\n", (int)(t/1000), ms, buffer); - else -#endif - fprintf(stderr,"%s\n", buffer); - fflush(stderr); - } - else // else, in production mode, we write to syslog - { - static int log_inited = 0; - - int syslog_level = LOG_ERR; - switch (loglevel) - { - case MDNS_LOG_MSG: syslog_level = LOG_ERR; break; - case MDNS_LOG_OPERATION: syslog_level = LOG_WARNING; break; - case MDNS_LOG_SPS: syslog_level = LOG_NOTICE; break; - case MDNS_LOG_INFO: syslog_level = LOG_INFO; break; - case MDNS_LOG_DEBUG: syslog_level = LOG_DEBUG; break; - default: - fprintf(stderr, "Unknown loglevel %d, assuming LOG_ERR\n", loglevel); - fflush(stderr); - } - - if (!log_inited) { openlog(ident, LOG_CONS, LOG_DAEMON); log_inited++; } - -#if APPLE_OSX_mDNSResponder && LogTimeStamps - if (ident && ident[0] && mDNSPlatformClockDivisor) - syslog(syslog_level, "%8d.%03d: %s", (int)(t/1000), ms, buffer); - else -#endif - syslog(syslog_level, "%s", buffer); - } - } diff -Nru qtcreator-2.5.0/src/tools/mdnssd/PlatformCommon.h qtcreator-2.5.2/src/tools/mdnssd/PlatformCommon.h --- qtcreator-2.5.0/src/tools/mdnssd/PlatformCommon.h 2012-05-09 12:13:19.000000000 +0000 +++ qtcreator-2.5.2/src/tools/mdnssd/PlatformCommon.h 1970-01-01 00:00:00.000000000 +0000 @@ -1,18 +0,0 @@ -/* -*- Mode: C; tab-width: 4 -*- - * - * Copyright (c) 2004 Apple Computer, Inc. All rights reserved. - * - * Licensed under the Apache License, Version 2.0 (the "License"); - * you may not use this file except in compliance with the License. - * You may obtain a copy of the License at - * - * http://www.apache.org/licenses/LICENSE-2.0 - * - * Unless required by applicable law or agreed to in writing, software - * distributed under the License is distributed on an "AS IS" BASIS, - * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. - * See the License for the specific language governing permissions and - * limitations under the License. - */ - -extern void ReadDDNSSettingsFromConfFile(mDNS *const m, const char *const filename, domainname *const hostname, domainname *const domain, mDNSBool *DomainDiscoveryDisabled); diff -Nru qtcreator-2.5.0/src/tools/mdnssd/PosixDaemon.c qtcreator-2.5.2/src/tools/mdnssd/PosixDaemon.c --- qtcreator-2.5.0/src/tools/mdnssd/PosixDaemon.c 2012-05-09 12:13:19.000000000 +0000 +++ qtcreator-2.5.2/src/tools/mdnssd/PosixDaemon.c 1970-01-01 00:00:00.000000000 +0000 @@ -1,258 +0,0 @@ -/* -*- Mode: C; tab-width: 4 -*- - * - * Copyright (c) 2003-2004 Apple Computer, Inc. All rights reserved. - * - * Licensed under the Apache License, Version 2.0 (the "License"); - * you may not use this file except in compliance with the License. - * You may obtain a copy of the License at - * - * http://www.apache.org/licenses/LICENSE-2.0 - * - * Unless required by applicable law or agreed to in writing, software - * distributed under the License is distributed on an "AS IS" BASIS, - * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. - * See the License for the specific language governing permissions and - * limitations under the License. - - File: daemon.c - - Contains: main & associated Application layer for mDNSResponder on Linux. - - */ - -#if __APPLE__ -// In Mac OS X 10.5 and later trying to use the daemon function gives a "'daemon' is deprecated" -// error, which prevents compilation because we build with "-Werror". -// Since this is supposed to be portable cross-platform code, we don't care that daemon is -// deprecated on Mac OS X 10.5, so we use this preprocessor trick to eliminate the error message. -#define daemon yes_we_know_that_daemon_is_deprecated_in_os_x_10_5_thankyou -#endif - -#include -#include -#include -#include -#include -#include -#include -#include -#include - -#if __APPLE__ -#undef daemon -extern int daemon(int, int); -#endif - -#include "mDNSEmbeddedAPI.h" -#include "mDNSPosix.h" -#include "mDNSUNP.h" // For daemon() -#include "uds_daemon.h" -#include "PlatformCommon.h" - -#define CONFIG_FILE "/etc/mdnsd.conf" -static domainname DynDNSZone; // Default wide-area zone for service registration -static domainname DynDNSHostname; - -#define RR_CACHE_SIZE 500 -static CacheEntity gRRCache[RR_CACHE_SIZE]; -static mDNS_PlatformSupport PlatformStorage; - -mDNSlocal void mDNS_StatusCallback(mDNS *const m, mStatus result) - { - (void)m; // Unused - if (result == mStatus_NoError) - { - // On successful registration of dot-local mDNS host name, daemon may want to check if - // any name conflict and automatic renaming took place, and if so, record the newly negotiated - // name in persistent storage for next time. It should also inform the user of the name change. - // On Mac OS X we store the current dot-local mDNS host name in the SCPreferences store, - // and notify the user with a CFUserNotification. - } - else if (result == mStatus_ConfigChanged) - { - udsserver_handle_configchange(m); - } - else if (result == mStatus_GrowCache) - { - // Allocate another chunk of cache storage - CacheEntity *storage = malloc(sizeof(CacheEntity) * RR_CACHE_SIZE); - if (storage) mDNS_GrowCache(m, storage, RR_CACHE_SIZE); - } - } - -// %%% Reconfigure() probably belongs in the platform support layer (mDNSPosix.c), not the daemon cde -// -- all client layers running on top of mDNSPosix.c need to handle network configuration changes, -// not only the Unix Domain Socket Daemon - -static void Reconfigure(mDNS *m) - { - mDNSAddr DynDNSIP; - const mDNSAddr dummy = { mDNSAddrType_IPv4, { { { 1, 1, 1, 1 } } } };; - mDNS_SetPrimaryInterfaceInfo(m, NULL, NULL, NULL); - if (ParseDNSServers(m, uDNS_SERVERS_FILE) < 0) - LogMsg("Unable to parse DNS server list. Unicast DNS-SD unavailable"); - ReadDDNSSettingsFromConfFile(m, CONFIG_FILE, &DynDNSHostname, &DynDNSZone, NULL); - mDNSPlatformSourceAddrForDest(&DynDNSIP, &dummy); - if (DynDNSHostname.c[0]) mDNS_AddDynDNSHostName(m, &DynDNSHostname, NULL, NULL); - if (DynDNSIP.type) mDNS_SetPrimaryInterfaceInfo(m, &DynDNSIP, NULL, NULL); - mDNS_ConfigChanged(m); - } - -// Do appropriate things at startup with command line arguments. Calls exit() if unhappy. -mDNSlocal void ParseCmdLinArgs(int argc, char **argv) - { - if (argc > 1) - { - if (0 == strcmp(argv[1], "-debug")) mDNS_DebugMode = mDNStrue; - else printf("Usage: %s [-debug]\n", argv[0]); - } - - if (!mDNS_DebugMode) - { - int result = daemon(0, 0); - if (result != 0) { LogMsg("Could not run as daemon - exiting"); exit(result); } -#if __APPLE__ - LogMsg("The POSIX mdnsd should only be used on OS X for testing - exiting"); - exit(-1); -#endif - } - } - -mDNSlocal void DumpStateLog(mDNS *const m) -// Dump a little log of what we've been up to. - { - LogMsg("---- BEGIN STATE LOG ----"); - udsserver_info(m); - LogMsg("---- END STATE LOG ----"); - } - -mDNSlocal mStatus MainLoop(mDNS *m) // Loop until we quit. - { - sigset_t signals; - mDNSBool gotData = mDNSfalse; - - mDNSPosixListenForSignalInEventLoop(SIGINT); - mDNSPosixListenForSignalInEventLoop(SIGTERM); - mDNSPosixListenForSignalInEventLoop(SIGUSR1); - mDNSPosixListenForSignalInEventLoop(SIGPIPE); - mDNSPosixListenForSignalInEventLoop(SIGHUP) ; - - for (; ;) - { - // Work out how long we expect to sleep before the next scheduled task - struct timeval timeout; - mDNSs32 ticks; - - // Only idle if we didn't find any data the last time around - if (!gotData) - { - mDNSs32 nextTimerEvent = mDNS_Execute(m); - nextTimerEvent = udsserver_idle(nextTimerEvent); - ticks = nextTimerEvent - mDNS_TimeNow(m); - if (ticks < 1) ticks = 1; - } - else // otherwise call EventLoop again with 0 timemout - ticks = 0; - - timeout.tv_sec = ticks / mDNSPlatformOneSecond; - timeout.tv_usec = (ticks % mDNSPlatformOneSecond) * 1000000 / mDNSPlatformOneSecond; - - (void) mDNSPosixRunEventLoopOnce(m, &timeout, &signals, &gotData); - - if (sigismember(&signals, SIGHUP )) Reconfigure(m); - if (sigismember(&signals, SIGUSR1)) DumpStateLog(m); - // SIGPIPE happens when we try to write to a dead client; death should be detected soon in request_callback() and cleaned up. - if (sigismember(&signals, SIGPIPE)) LogMsg("Received SIGPIPE - ignoring"); - if (sigismember(&signals, SIGINT) || sigismember(&signals, SIGTERM)) break; - } - return EINTR; - } - -int main(int argc, char **argv) - { - mStatus err; - - ParseCmdLinArgs(argc, argv); - - LogMsg("%s starting", mDNSResponderVersionString); - - err = mDNS_Init(&mDNSStorage, &PlatformStorage, gRRCache, RR_CACHE_SIZE, mDNS_Init_AdvertiseLocalAddresses, - mDNS_StatusCallback, mDNS_Init_NoInitCallbackContext); - - if (mStatus_NoError == err) - err = udsserver_init(mDNSNULL, 0); - - Reconfigure(&mDNSStorage); - - // Now that we're finished with anything privileged, switch over to running as "nobody" - if (mStatus_NoError == err) - { - const struct passwd *pw = getpwnam("nobody"); - if (pw != NULL) - setuid(pw->pw_uid); - else - LogMsg("WARNING: mdnsd continuing as root because user \"nobody\" does not exist"); - } - - if (mStatus_NoError == err) - err = MainLoop(&mDNSStorage); - - LogMsg("%s stopping", mDNSResponderVersionString); - - mDNS_Close(&mDNSStorage); - - if (udsserver_exit() < 0) - LogMsg("ExitCallback: udsserver_exit failed"); - - #if MDNS_DEBUGMSGS > 0 - printf("mDNSResponder exiting normally with %ld\n", err); - #endif - - return err; - } - -// uds_daemon support //////////////////////////////////////////////////////////// - -mStatus udsSupportAddFDToEventLoop(int fd, udsEventCallback callback, void *context, void **platform_data) -/* Support routine for uds_daemon.c */ - { - // Depends on the fact that udsEventCallback == mDNSPosixEventCallback - (void) platform_data; - return mDNSPosixAddFDToEventLoop(fd, callback, context); - } - -int udsSupportReadFD(dnssd_sock_t fd, char *buf, int len, int flags, void *platform_data) - { - (void) platform_data; - return recv(fd, buf, len, flags); - } - -mStatus udsSupportRemoveFDFromEventLoop(int fd, void *platform_data) // Note: This also CLOSES the file descriptor - { - mStatus err = mDNSPosixRemoveFDFromEventLoop(fd); - (void) platform_data; - close(fd); - return err; - } - -mDNSexport void RecordUpdatedNiceLabel(mDNS *const m, mDNSs32 delay) - { - (void)m; - (void)delay; - // No-op, for now - } - -#if _BUILDING_XCODE_PROJECT_ -// If the process crashes, then this string will be magically included in the automatically-generated crash log -const char *__crashreporter_info__ = mDNSResponderVersionString_SCCS + 5; -asm(".desc ___crashreporter_info__, 0x10"); -#endif - -// For convenience when using the "strings" command, this is the last thing in the file -#if mDNSResponderVersion > 1 -mDNSexport const char mDNSResponderVersionString_SCCS[] = "@(#) mDNSResponder-" STRINGIFY(mDNSResponderVersion) " (" __DATE__ " " __TIME__ ")"; -#elif MDNS_VERSIONSTR_NODTS -mDNSexport const char mDNSResponderVersionString_SCCS[] = "@(#) mDNSResponder (Engineering Build)"; -#else -mDNSexport const char mDNSResponderVersionString_SCCS[] = "@(#) mDNSResponder (Engineering Build) (" __DATE__ " " __TIME__ ")"; -#endif diff -Nru qtcreator-2.5.0/src/tools/mdnssd/dns_sd.h qtcreator-2.5.2/src/tools/mdnssd/dns_sd.h --- qtcreator-2.5.0/src/tools/mdnssd/dns_sd.h 2012-05-09 12:13:19.000000000 +0000 +++ qtcreator-2.5.2/src/tools/mdnssd/dns_sd.h 1970-01-01 00:00:00.000000000 +0000 @@ -1,2434 +0,0 @@ -/* -*- Mode: C; tab-width: 4 -*- - * - * Copyright (c) 2003-2004, Apple Computer, Inc. All rights reserved. - * - * Redistribution and use in source and binary forms, with or without - * modification, are permitted provided that the following conditions are met: - * - * 1. Redistributions of source code must retain the above copyright notice, - * this list of conditions and the following disclaimer. - * 2. Redistributions in binary form must reproduce the above copyright notice, - * this list of conditions and the following disclaimer in the documentation - * and/or other materials provided with the distribution. - * 3. Neither the name of Apple Computer, Inc. ("Apple") nor the names of its - * contributors may be used to endorse or promote products derived from this - * software without specific prior written permission. - * - * THIS SOFTWARE IS PROVIDED BY APPLE AND ITS CONTRIBUTORS "AS IS" AND ANY - * EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED - * WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE - * DISCLAIMED. IN NO EVENT SHALL APPLE OR ITS CONTRIBUTORS BE LIABLE FOR ANY - * DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES - * (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; - * LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND - * ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT - * (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS - * SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE. - */ - - -/*! @header DNS Service Discovery - * - * @discussion This section describes the functions, callbacks, and data structures - * that make up the DNS Service Discovery API. - * - * The DNS Service Discovery API is part of Bonjour, Apple's implementation - * of zero-configuration networking (ZEROCONF). - * - * Bonjour allows you to register a network service, such as a - * printer or file server, so that it can be found by name or browsed - * for by service type and domain. Using Bonjour, applications can - * discover what services are available on the network, along with - * all the information -- such as name, IP address, and port -- - * necessary to access a particular service. - * - * In effect, Bonjour combines the functions of a local DNS server and - * AppleTalk. Bonjour allows applications to provide user-friendly printer - * and server browsing, among other things, over standard IP networks. - * This behavior is a result of combining protocols such as multicast and - * DNS to add new functionality to the network (such as multicast DNS). - * - * Bonjour gives applications easy access to services over local IP - * networks without requiring the service or the application to support - * an AppleTalk or a Netbeui stack, and without requiring a DNS server - * for the local network. - */ - - -/* _DNS_SD_H contains the mDNSResponder version number for this header file, formatted as follows: - * Major part of the build number * 10000 + - * minor part of the build number * 100 - * For example, Mac OS X 10.4.9 has mDNSResponder-108.4, which would be represented as - * version 1080400. This allows C code to do simple greater-than and less-than comparisons: - * e.g. an application that requires the DNSServiceGetProperty() call (new in mDNSResponder-126) can check: - * - * #if _DNS_SD_H+0 >= 1260000 - * ... some C code that calls DNSServiceGetProperty() ... - * #endif - * - * The version defined in this header file symbol allows for compile-time - * checking, so that C code building with earlier versions of the header file - * can avoid compile errors trying to use functions that aren't even defined - * in those earlier versions. Similar checks may also be performed at run-time: - * => weak linking -- to avoid link failures if run with an earlier - * version of the library that's missing some desired symbol, or - * => DNSServiceGetProperty(DaemonVersion) -- to verify whether the running daemon - * ("system service" on Windows) meets some required minimum functionality level. - */ - -#ifndef _DNS_SD_H -#define _DNS_SD_H 3331000 - -#ifdef __cplusplus - extern "C" { -#endif - -/* Set to 1 if libdispatch is supported - * Note: May also be set by project and/or Makefile - */ -#ifndef _DNS_SD_LIBDISPATCH -#define _DNS_SD_LIBDISPATCH 0 -#endif /* ndef _DNS_SD_LIBDISPATCH */ - -/* standard calling convention under Win32 is __stdcall */ -/* Note: When compiling Intel EFI (Extensible Firmware Interface) under MS Visual Studio, the */ -/* _WIN32 symbol is defined by the compiler even though it's NOT compiling code for Windows32 */ -#if defined(_WIN32) && !defined(EFI32) && !defined(EFI64) -#define DNSSD_API __stdcall -#else -#define DNSSD_API -#endif - -/* stdint.h does not exist on FreeBSD 4.x; its types are defined in sys/types.h instead */ -#if defined(__FreeBSD__) && (__FreeBSD__ < 5) -#include - -/* Likewise, on Sun, standard integer types are in sys/types.h */ -#elif defined(__sun__) -#include - -/* EFI does not have stdint.h, or anything else equivalent */ -#elif defined(EFI32) || defined(EFI64) || defined(EFIX64) -#include "Tiano.h" -#if !defined(_STDINT_H_) -typedef UINT8 uint8_t; -typedef INT8 int8_t; -typedef UINT16 uint16_t; -typedef INT16 int16_t; -typedef UINT32 uint32_t; -typedef INT32 int32_t; -#endif -/* Windows has its own differences */ -#elif defined(_WIN32) -#include -#define _UNUSED -#ifndef _MSL_STDINT_H -typedef UINT8 uint8_t; -typedef INT8 int8_t; -typedef UINT16 uint16_t; -typedef INT16 int16_t; -typedef UINT32 uint32_t; -typedef INT32 int32_t; -#endif - -/* All other Posix platforms use stdint.h */ -#else -#include -#endif - -#if _DNS_SD_LIBDISPATCH -#include -#endif - -/* DNSServiceRef, DNSRecordRef - * - * Opaque internal data types. - * Note: client is responsible for serializing access to these structures if - * they are shared between concurrent threads. - */ - -typedef struct _DNSServiceRef_t *DNSServiceRef; -typedef struct _DNSRecordRef_t *DNSRecordRef; - -struct sockaddr; - -/*! @enum General flags - * Most DNS-SD API functions and callbacks include a DNSServiceFlags parameter. - * As a general rule, any given bit in the 32-bit flags field has a specific fixed meaning, - * regardless of the function or callback being used. For any given function or callback, - * typically only a subset of the possible flags are meaningful, and all others should be zero. - * The discussion section for each API call describes which flags are valid for that call - * and callback. In some cases, for a particular call, it may be that no flags are currently - * defined, in which case the DNSServiceFlags parameter exists purely to allow future expansion. - * In all cases, developers should expect that in future releases, it is possible that new flag - * values will be defined, and write code with this in mind. For example, code that tests - * if (flags == kDNSServiceFlagsAdd) ... - * will fail if, in a future release, another bit in the 32-bit flags field is also set. - * The reliable way to test whether a particular bit is set is not with an equality test, - * but with a bitwise mask: - * if (flags & kDNSServiceFlagsAdd) ... - */ -enum - { - kDNSServiceFlagsMoreComing = 0x1, - /* MoreComing indicates to a callback that at least one more result is - * queued and will be delivered following immediately after this one. - * When the MoreComing flag is set, applications should not immediately - * update their UI, because this can result in a great deal of ugly flickering - * on the screen, and can waste a great deal of CPU time repeatedly updating - * the screen with content that is then immediately erased, over and over. - * Applications should wait until until MoreComing is not set, and then - * update their UI when no more changes are imminent. - * When MoreComing is not set, that doesn't mean there will be no more - * answers EVER, just that there are no more answers immediately - * available right now at this instant. If more answers become available - * in the future they will be delivered as usual. - */ - - kDNSServiceFlagsAdd = 0x2, - kDNSServiceFlagsDefault = 0x4, - /* Flags for domain enumeration and browse/query reply callbacks. - * "Default" applies only to enumeration and is only valid in - * conjunction with "Add". An enumeration callback with the "Add" - * flag NOT set indicates a "Remove", i.e. the domain is no longer - * valid. - */ - - kDNSServiceFlagsNoAutoRename = 0x8, - /* Flag for specifying renaming behavior on name conflict when registering - * non-shared records. By default, name conflicts are automatically handled - * by renaming the service. NoAutoRename overrides this behavior - with this - * flag set, name conflicts will result in a callback. The NoAutorename flag - * is only valid if a name is explicitly specified when registering a service - * (i.e. the default name is not used.) - */ - - kDNSServiceFlagsShared = 0x10, - kDNSServiceFlagsUnique = 0x20, - /* Flag for registering individual records on a connected - * DNSServiceRef. Shared indicates that there may be multiple records - * with this name on the network (e.g. PTR records). Unique indicates that the - * record's name is to be unique on the network (e.g. SRV records). - */ - - kDNSServiceFlagsBrowseDomains = 0x40, - kDNSServiceFlagsRegistrationDomains = 0x80, - /* Flags for specifying domain enumeration type in DNSServiceEnumerateDomains. - * BrowseDomains enumerates domains recommended for browsing, RegistrationDomains - * enumerates domains recommended for registration. - */ - - kDNSServiceFlagsLongLivedQuery = 0x100, - /* Flag for creating a long-lived unicast query for the DNSServiceQueryRecord call. */ - - kDNSServiceFlagsAllowRemoteQuery = 0x200, - /* Flag for creating a record for which we will answer remote queries - * (queries from hosts more than one hop away; hosts not directly connected to the local link). - */ - - kDNSServiceFlagsForceMulticast = 0x400, - /* Flag for signifying that a query or registration should be performed exclusively via multicast - * DNS, even for a name in a domain (e.g. foo.apple.com.) that would normally imply unicast DNS. - */ - - kDNSServiceFlagsForce = 0x800, - /* Flag for signifying a "stronger" variant of an operation. - * Currently defined only for DNSServiceReconfirmRecord(), where it forces a record to - * be removed from the cache immediately, instead of querying for a few seconds before - * concluding that the record is no longer valid and then removing it. This flag should - * be used with caution because if a service browsing PTR record is indeed still valid - * on the network, forcing its removal will result in a user-interface flap -- the - * discovered service instance will disappear, and then re-appear moments later. - */ - - kDNSServiceFlagsReturnIntermediates = 0x1000, - /* Flag for returning intermediate results. - * For example, if a query results in an authoritative NXDomain (name does not exist) - * then that result is returned to the client. However the query is not implicitly - * cancelled -- it remains active and if the answer subsequently changes - * (e.g. because a VPN tunnel is subsequently established) then that positive - * result will still be returned to the client. - * Similarly, if a query results in a CNAME record, then in addition to following - * the CNAME referral, the intermediate CNAME result is also returned to the client. - * When this flag is not set, NXDomain errors are not returned, and CNAME records - * are followed silently without informing the client of the intermediate steps. - * (In earlier builds this flag was briefly calledkDNSServiceFlagsReturnCNAME) - */ - - kDNSServiceFlagsNonBrowsable = 0x2000, - /* A service registered with the NonBrowsable flag set can be resolved using - * DNSServiceResolve(), but will not be discoverable using DNSServiceBrowse(). - * This is for cases where the name is actually a GUID; it is found by other means; - * there is no end-user benefit to browsing to find a long list of opaque GUIDs. - * Using the NonBrowsable flag creates SRV+TXT without the cost of also advertising - * an associated PTR record. - */ - - kDNSServiceFlagsShareConnection = 0x4000, - /* For efficiency, clients that perform many concurrent operations may want to use a - * single Unix Domain Socket connection with the background daemon, instead of having a - * separate connection for each independent operation. To use this mode, clients first - * call DNSServiceCreateConnection(&MainRef) to initialize the main DNSServiceRef. - * For each subsequent operation that is to share that same connection, the client copies - * the MainRef, and then passes the address of that copy, setting the ShareConnection flag - * to tell the library that this DNSServiceRef is not a typical uninitialized DNSServiceRef; - * it's a copy of an existing DNSServiceRef whose connection information should be reused. - * - * For example: - * - * DNSServiceErrorType error; - * DNSServiceRef MainRef; - * error = DNSServiceCreateConnection(&MainRef); - * if (error) ... - * DNSServiceRef BrowseRef = MainRef; // Important: COPY the primary DNSServiceRef first... - * error = DNSServiceBrowse(&BrowseRef, kDNSServiceFlagsShareConnection, ...); // then use the copy - * if (error) ... - * ... - * DNSServiceRefDeallocate(BrowseRef); // Terminate the browse operation - * DNSServiceRefDeallocate(MainRef); // Terminate the shared connection - * - * Notes: - * - * 1. Collective kDNSServiceFlagsMoreComing flag - * When callbacks are invoked using a shared DNSServiceRef, the - * kDNSServiceFlagsMoreComing flag applies collectively to *all* active - * operations sharing the same parent DNSServiceRef. If the MoreComing flag is - * set it means that there are more results queued on this parent DNSServiceRef, - * but not necessarily more results for this particular callback function. - * The implication of this for client programmers is that when a callback - * is invoked with the MoreComing flag set, the code should update its - * internal data structures with the new result, and set a variable indicating - * that its UI needs to be updated. Then, later when a callback is eventually - * invoked with the MoreComing flag not set, the code should update *all* - * stale UI elements related to that shared parent DNSServiceRef that need - * updating, not just the UI elements related to the particular callback - * that happened to be the last one to be invoked. - * - * 2. Canceling operations and kDNSServiceFlagsMoreComing - * Whenever you cancel any operation for which you had deferred UI updates - * waiting because of a kDNSServiceFlagsMoreComing flag, you should perform - * those deferred UI updates. This is because, after cancelling the operation, - * you can no longer wait for a callback *without* MoreComing set, to tell - * you do perform your deferred UI updates (the operation has been canceled, - * so there will be no more callbacks). An implication of the collective - * kDNSServiceFlagsMoreComing flag for shared connections is that this - * guideline applies more broadly -- any time you cancel an operation on - * a shared connection, you should perform all deferred UI updates for all - * operations sharing that connection. This is because the MoreComing flag - * might have been referring to events coming for the operation you canceled, - * which will now not be coming because the operation has been canceled. - * - * 3. Only share DNSServiceRef's created with DNSServiceCreateConnection - * Calling DNSServiceCreateConnection(&ref) creates a special shareable DNSServiceRef. - * DNSServiceRef's created by other calls like DNSServiceBrowse() or DNSServiceResolve() - * cannot be shared by copying them and using kDNSServiceFlagsShareConnection. - * - * 4. Don't Double-Deallocate - * Calling DNSServiceRefDeallocate(ref) for a particular operation's DNSServiceRef terminates - * just that operation. Calling DNSServiceRefDeallocate(ref) for the main shared DNSServiceRef - * (the parent DNSServiceRef, originally created by DNSServiceCreateConnection(&ref)) - * automatically terminates the shared connection and all operations that were still using it. - * After doing this, DO NOT then attempt to deallocate any remaining subordinate DNSServiceRef's. - * The memory used by those subordinate DNSServiceRef's has already been freed, so any attempt - * to do a DNSServiceRefDeallocate (or any other operation) on them will result in accesses - * to freed memory, leading to crashes or other equally undesirable results. - * - * 5. Thread Safety - * The dns_sd.h API does not presuppose any particular threading model, and consequently - * does no locking of its own (which would require linking some specific threading library). - * If client code calls API routines on the same DNSServiceRef concurrently - * from multiple threads, it is the client's responsibility to use a mutext - * lock or take similar appropriate precautions to serialize those calls. - */ - - kDNSServiceFlagsSuppressUnusable = 0x8000, - /* - * This flag is meaningful only in DNSServiceQueryRecord which suppresses unusable queries on the - * wire. If "hostname" is a wide-area unicast DNS hostname (i.e. not a ".local." name) - * but this host has no routable IPv6 address, then the call will not try to look up IPv6 addresses - * for "hostname", since any addresses it found would be unlikely to be of any use anyway. Similarly, - * if this host has no routable IPv4 address, the call will not try to look up IPv4 addresses for - * "hostname". - */ - - kDNSServiceFlagsTimeout = 0x10000, - /* - * When kDNServiceFlagsTimeout is passed to DNSServiceQueryRecord or DNSServiceGetAddrInfo, the query is - * stopped after a certain number of seconds have elapsed. The time at which the query will be stopped - * is determined by the system and cannot be configured by the user. The query will be stopped irrespective - * of whether a response was given earlier or not. When the query is stopped, the callback will be called - * with an error code of kDNSServiceErr_Timeout and a NULL sockaddr will be returned for DNSServiceGetAddrInfo - * and zero length rdata will be returned for DNSServiceQueryRecord. - */ - - kDNSServiceFlagsIncludeP2P = 0x20000, - /* - * Include P2P interfaces when kDNSServiceInterfaceIndexAny is specified. - * By default, specifying kDNSServiceInterfaceIndexAny does not include P2P interfaces. - */ - kDNSServiceFlagsWakeOnResolve = 0x40000 - /* - * This flag is meaningful only in DNSServiceResolve. When set, it tries to send a magic packet - * to wake up the client. - */ - }; - -/* Possible protocols for DNSServiceNATPortMappingCreate(). */ -enum - { - kDNSServiceProtocol_IPv4 = 0x01, - kDNSServiceProtocol_IPv6 = 0x02, - /* 0x04 and 0x08 reserved for future internetwork protocols */ - - kDNSServiceProtocol_UDP = 0x10, - kDNSServiceProtocol_TCP = 0x20 - /* 0x40 and 0x80 reserved for future transport protocols, e.g. SCTP [RFC 2960] - * or DCCP [RFC 4340]. If future NAT gateways are created that support port - * mappings for these protocols, new constants will be defined here. - */ - }; - -/* - * The values for DNS Classes and Types are listed in RFC 1035, and are available - * on every OS in its DNS header file. Unfortunately every OS does not have the - * same header file containing DNS Class and Type constants, and the names of - * the constants are not consistent. For example, BIND 8 uses "T_A", - * BIND 9 uses "ns_t_a", Windows uses "DNS_TYPE_A", etc. - * For this reason, these constants are also listed here, so that code using - * the DNS-SD programming APIs can use these constants, so that the same code - * can compile on all our supported platforms. - */ - -enum - { - kDNSServiceClass_IN = 1 /* Internet */ - }; - -enum - { - kDNSServiceType_A = 1, /* Host address. */ - kDNSServiceType_NS = 2, /* Authoritative server. */ - kDNSServiceType_MD = 3, /* Mail destination. */ - kDNSServiceType_MF = 4, /* Mail forwarder. */ - kDNSServiceType_CNAME = 5, /* Canonical name. */ - kDNSServiceType_SOA = 6, /* Start of authority zone. */ - kDNSServiceType_MB = 7, /* Mailbox domain name. */ - kDNSServiceType_MG = 8, /* Mail group member. */ - kDNSServiceType_MR = 9, /* Mail rename name. */ - kDNSServiceType_NULL = 10, /* Null resource record. */ - kDNSServiceType_WKS = 11, /* Well known service. */ - kDNSServiceType_PTR = 12, /* Domain name pointer. */ - kDNSServiceType_HINFO = 13, /* Host information. */ - kDNSServiceType_MINFO = 14, /* Mailbox information. */ - kDNSServiceType_MX = 15, /* Mail routing information. */ - kDNSServiceType_TXT = 16, /* One or more text strings (NOT "zero or more..."). */ - kDNSServiceType_RP = 17, /* Responsible person. */ - kDNSServiceType_AFSDB = 18, /* AFS cell database. */ - kDNSServiceType_X25 = 19, /* X_25 calling address. */ - kDNSServiceType_ISDN = 20, /* ISDN calling address. */ - kDNSServiceType_RT = 21, /* Router. */ - kDNSServiceType_NSAP = 22, /* NSAP address. */ - kDNSServiceType_NSAP_PTR = 23, /* Reverse NSAP lookup (deprecated). */ - kDNSServiceType_SIG = 24, /* Security signature. */ - kDNSServiceType_KEY = 25, /* Security key. */ - kDNSServiceType_PX = 26, /* X.400 mail mapping. */ - kDNSServiceType_GPOS = 27, /* Geographical position (withdrawn). */ - kDNSServiceType_AAAA = 28, /* IPv6 Address. */ - kDNSServiceType_LOC = 29, /* Location Information. */ - kDNSServiceType_NXT = 30, /* Next domain (security). */ - kDNSServiceType_EID = 31, /* Endpoint identifier. */ - kDNSServiceType_NIMLOC = 32, /* Nimrod Locator. */ - kDNSServiceType_SRV = 33, /* Server Selection. */ - kDNSServiceType_ATMA = 34, /* ATM Address */ - kDNSServiceType_NAPTR = 35, /* Naming Authority PoinTeR */ - kDNSServiceType_KX = 36, /* Key Exchange */ - kDNSServiceType_CERT = 37, /* Certification record */ - kDNSServiceType_A6 = 38, /* IPv6 Address (deprecated) */ - kDNSServiceType_DNAME = 39, /* Non-terminal DNAME (for IPv6) */ - kDNSServiceType_SINK = 40, /* Kitchen sink (experimental) */ - kDNSServiceType_OPT = 41, /* EDNS0 option (meta-RR) */ - kDNSServiceType_APL = 42, /* Address Prefix List */ - kDNSServiceType_DS = 43, /* Delegation Signer */ - kDNSServiceType_SSHFP = 44, /* SSH Key Fingerprint */ - kDNSServiceType_IPSECKEY = 45, /* IPSECKEY */ - kDNSServiceType_RRSIG = 46, /* RRSIG */ - kDNSServiceType_NSEC = 47, /* Denial of Existence */ - kDNSServiceType_DNSKEY = 48, /* DNSKEY */ - kDNSServiceType_DHCID = 49, /* DHCP Client Identifier */ - kDNSServiceType_NSEC3 = 50, /* Hashed Authenticated Denial of Existence */ - kDNSServiceType_NSEC3PARAM = 51, /* Hashed Authenticated Denial of Existence */ - - kDNSServiceType_HIP = 55, /* Host Identity Protocol */ - - kDNSServiceType_SPF = 99, /* Sender Policy Framework for E-Mail */ - kDNSServiceType_UINFO = 100, /* IANA-Reserved */ - kDNSServiceType_UID = 101, /* IANA-Reserved */ - kDNSServiceType_GID = 102, /* IANA-Reserved */ - kDNSServiceType_UNSPEC = 103, /* IANA-Reserved */ - - kDNSServiceType_TKEY = 249, /* Transaction key */ - kDNSServiceType_TSIG = 250, /* Transaction signature. */ - kDNSServiceType_IXFR = 251, /* Incremental zone transfer. */ - kDNSServiceType_AXFR = 252, /* Transfer zone of authority. */ - kDNSServiceType_MAILB = 253, /* Transfer mailbox records. */ - kDNSServiceType_MAILA = 254, /* Transfer mail agent records. */ - kDNSServiceType_ANY = 255 /* Wildcard match. */ - }; - -/* possible error code values */ -enum - { - kDNSServiceErr_NoError = 0, - kDNSServiceErr_Unknown = -65537, /* 0xFFFE FFFF */ - kDNSServiceErr_NoSuchName = -65538, - kDNSServiceErr_NoMemory = -65539, - kDNSServiceErr_BadParam = -65540, - kDNSServiceErr_BadReference = -65541, - kDNSServiceErr_BadState = -65542, - kDNSServiceErr_BadFlags = -65543, - kDNSServiceErr_Unsupported = -65544, - kDNSServiceErr_NotInitialized = -65545, - kDNSServiceErr_AlreadyRegistered = -65547, - kDNSServiceErr_NameConflict = -65548, - kDNSServiceErr_Invalid = -65549, - kDNSServiceErr_Firewall = -65550, - kDNSServiceErr_Incompatible = -65551, /* client library incompatible with daemon */ - kDNSServiceErr_BadInterfaceIndex = -65552, - kDNSServiceErr_Refused = -65553, - kDNSServiceErr_NoSuchRecord = -65554, - kDNSServiceErr_NoAuth = -65555, - kDNSServiceErr_NoSuchKey = -65556, - kDNSServiceErr_NATTraversal = -65557, - kDNSServiceErr_DoubleNAT = -65558, - kDNSServiceErr_BadTime = -65559, /* Codes up to here existed in Tiger */ - kDNSServiceErr_BadSig = -65560, - kDNSServiceErr_BadKey = -65561, - kDNSServiceErr_Transient = -65562, - kDNSServiceErr_ServiceNotRunning = -65563, /* Background daemon not running */ - kDNSServiceErr_NATPortMappingUnsupported = -65564, /* NAT doesn't support NAT-PMP or UPnP */ - kDNSServiceErr_NATPortMappingDisabled = -65565, /* NAT supports NAT-PMP or UPnP but it's disabled by the administrator */ - kDNSServiceErr_NoRouter = -65566, /* No router currently configured (probably no network connectivity) */ - kDNSServiceErr_PollingMode = -65567, - kDNSServiceErr_Timeout = -65568 - - /* mDNS Error codes are in the range - * FFFE FF00 (-65792) to FFFE FFFF (-65537) */ - }; - -/* Maximum length, in bytes, of a service name represented as a */ -/* literal C-String, including the terminating NULL at the end. */ - -#define kDNSServiceMaxServiceName 64 - -/* Maximum length, in bytes, of a domain name represented as an *escaped* C-String */ -/* including the final trailing dot, and the C-String terminating NULL at the end. */ - -#define kDNSServiceMaxDomainName 1009 - -/* - * Notes on DNS Name Escaping - * -- or -- - * "Why is kDNSServiceMaxDomainName 1009, when the maximum legal domain name is 256 bytes?" - * - * All strings used in the DNS-SD APIs are UTF-8 strings. Apart from the exceptions noted below, - * the APIs expect the strings to be properly escaped, using the conventional DNS escaping rules: - * - * '\\' represents a single literal '\' in the name - * '\.' represents a single literal '.' in the name - * '\ddd', where ddd is a three-digit decimal value from 000 to 255, - * represents a single literal byte with that value. - * A bare unescaped '.' is a label separator, marking a boundary between domain and subdomain. - * - * The exceptions, that do not use escaping, are the routines where the full - * DNS name of a resource is broken, for convenience, into servicename/regtype/domain. - * In these routines, the "servicename" is NOT escaped. It does not need to be, since - * it is, by definition, just a single literal string. Any characters in that string - * represent exactly what they are. The "regtype" portion is, technically speaking, - * escaped, but since legal regtypes are only allowed to contain letters, digits, - * and hyphens, there is nothing to escape, so the issue is moot. The "domain" - * portion is also escaped, though most domains in use on the public Internet - * today, like regtypes, don't contain any characters that need to be escaped. - * As DNS-SD becomes more popular, rich-text domains for service discovery will - * become common, so software should be written to cope with domains with escaping. - * - * The servicename may be up to 63 bytes of UTF-8 text (not counting the C-String - * terminating NULL at the end). The regtype is of the form _service._tcp or - * _service._udp, where the "service" part is 1-15 characters, which may be - * letters, digits, or hyphens. The domain part of the three-part name may be - * any legal domain, providing that the resulting servicename+regtype+domain - * name does not exceed 256 bytes. - * - * For most software, these issues are transparent. When browsing, the discovered - * servicenames should simply be displayed as-is. When resolving, the discovered - * servicename/regtype/domain are simply passed unchanged to DNSServiceResolve(). - * When a DNSServiceResolve() succeeds, the returned fullname is already in - * the correct format to pass to standard system DNS APIs such as res_query(). - * For converting from servicename/regtype/domain to a single properly-escaped - * full DNS name, the helper function DNSServiceConstructFullName() is provided. - * - * The following (highly contrived) example illustrates the escaping process. - * Suppose you have an service called "Dr. Smith\Dr. Johnson", of type "_ftp._tcp" - * in subdomain "4th. Floor" of subdomain "Building 2" of domain "apple.com." - * The full (escaped) DNS name of this service's SRV record would be: - * Dr\.\032Smith\\Dr\.\032Johnson._ftp._tcp.4th\.\032Floor.Building\0322.apple.com. - */ - - -/* - * Constants for specifying an interface index - * - * Specific interface indexes are identified via a 32-bit unsigned integer returned - * by the if_nametoindex() family of calls. - * - * If the client passes 0 for interface index, that means "do the right thing", - * which (at present) means, "if the name is in an mDNS local multicast domain - * (e.g. 'local.', '254.169.in-addr.arpa.', '{8,9,A,B}.E.F.ip6.arpa.') then multicast - * on all applicable interfaces, otherwise send via unicast to the appropriate - * DNS server." Normally, most clients will use 0 for interface index to - * automatically get the default sensible behaviour. - * - * If the client passes a positive interface index, then for multicast names that - * indicates to do the operation only on that one interface. For unicast names the - * interface index is ignored unless kDNSServiceFlagsForceMulticast is also set. - * - * If the client passes kDNSServiceInterfaceIndexLocalOnly when registering - * a service, then that service will be found *only* by other local clients - * on the same machine that are browsing using kDNSServiceInterfaceIndexLocalOnly - * or kDNSServiceInterfaceIndexAny. - * If a client has a 'private' service, accessible only to other processes - * running on the same machine, this allows the client to advertise that service - * in a way such that it does not inadvertently appear in service lists on - * all the other machines on the network. - * - * If the client passes kDNSServiceInterfaceIndexLocalOnly when browsing - * then it will find *all* records registered on that same local machine. - * Clients explicitly wishing to discover *only* LocalOnly services can - * accomplish this by inspecting the interfaceIndex of each service reported - * to their DNSServiceBrowseReply() callback function, and discarding those - * where the interface index is not kDNSServiceInterfaceIndexLocalOnly. - * - * kDNSServiceInterfaceIndexP2P is meaningful only in Browse, QueryRecord, - * and Resolve operations. It should not be used in other DNSService APIs. - * - * - If kDNSServiceInterfaceIndexP2P is passed to DNSServiceBrowse or - * DNSServiceQueryRecord, it restricts the operation to P2P. - * - * - If kDNSServiceInterfaceIndexP2P is passed to DNSServiceResolve, it is - * mapped internally to kDNSServiceInterfaceIndexAny, because resolving - * a P2P service may create and/or enable an interface whose index is not - * known a priori. The resolve callback will indicate the index of the - * interface via which the service can be accessed. - * - * If applications pass kDNSServiceInterfaceIndexAny to DNSServiceBrowse - * or DNSServiceQueryRecord, they must set the kDNSServiceFlagsIncludeP2P flag - * to include P2P. In this case, if a service instance or the record being queried - * is found over P2P, the resulting ADD event will indicate kDNSServiceInterfaceIndexP2P - * as the interface index. - */ - -#define kDNSServiceInterfaceIndexAny 0 -#define kDNSServiceInterfaceIndexLocalOnly ((uint32_t)-1) -#define kDNSServiceInterfaceIndexUnicast ((uint32_t)-2) -#define kDNSServiceInterfaceIndexP2P ((uint32_t)-3) - -typedef uint32_t DNSServiceFlags; -typedef uint32_t DNSServiceProtocol; -typedef int32_t DNSServiceErrorType; - - -/********************************************************************************************* - * - * Version checking - * - *********************************************************************************************/ - -/* DNSServiceGetProperty() Parameters: - * - * property: The requested property. - * Currently the only property defined is kDNSServiceProperty_DaemonVersion. - * - * result: Place to store result. - * For retrieving DaemonVersion, this should be the address of a uint32_t. - * - * size: Pointer to uint32_t containing size of the result location. - * For retrieving DaemonVersion, this should be sizeof(uint32_t). - * On return the uint32_t is updated to the size of the data returned. - * For DaemonVersion, the returned size is always sizeof(uint32_t), but - * future properties could be defined which return variable-sized results. - * - * return value: Returns kDNSServiceErr_NoError on success, or kDNSServiceErr_ServiceNotRunning - * if the daemon (or "system service" on Windows) is not running. - */ - -DNSServiceErrorType DNSSD_API DNSServiceGetProperty - ( - const char *property, /* Requested property (i.e. kDNSServiceProperty_DaemonVersion) */ - void *result, /* Pointer to place to store result */ - uint32_t *size /* size of result location */ - ); - -/* - * When requesting kDNSServiceProperty_DaemonVersion, the result pointer must point - * to a 32-bit unsigned integer, and the size parameter must be set to sizeof(uint32_t). - * - * On return, the 32-bit unsigned integer contains the version number, formatted as follows: - * Major part of the build number * 10000 + - * minor part of the build number * 100 - * - * For example, Mac OS X 10.4.9 has mDNSResponder-108.4, which would be represented as - * version 1080400. This allows applications to do simple greater-than and less-than comparisons: - * e.g. an application that requires at least mDNSResponder-108.4 can check: - * - * if (version >= 1080400) ... - * - * Example usage: - * - * uint32_t version; - * uint32_t size = sizeof(version); - * DNSServiceErrorType err = DNSServiceGetProperty(kDNSServiceProperty_DaemonVersion, &version, &size); - * if (!err) printf("Bonjour version is %d.%d\n", version / 10000, version / 100 % 100); - */ - -#define kDNSServiceProperty_DaemonVersion "DaemonVersion" - - -/********************************************************************************************* - * - * Unix Domain Socket access, DNSServiceRef deallocation, and data processing functions - * - *********************************************************************************************/ - -/* DNSServiceRefSockFD() - * - * Access underlying Unix domain socket for an initialized DNSServiceRef. - * The DNS Service Discovery implementation uses this socket to communicate between the client and - * the mDNSResponder daemon. The application MUST NOT directly read from or write to this socket. - * Access to the socket is provided so that it can be used as a kqueue event source, a CFRunLoop - * event source, in a select() loop, etc. When the underlying event management subsystem (kqueue/ - * select/CFRunLoop etc.) indicates to the client that data is available for reading on the - * socket, the client should call DNSServiceProcessResult(), which will extract the daemon's - * reply from the socket, and pass it to the appropriate application callback. By using a run - * loop or select(), results from the daemon can be processed asynchronously. Alternatively, - * a client can choose to fork a thread and have it loop calling "DNSServiceProcessResult(ref);" - * If DNSServiceProcessResult() is called when no data is available for reading on the socket, it - * will block until data does become available, and then process the data and return to the caller. - * When data arrives on the socket, the client is responsible for calling DNSServiceProcessResult(ref) - * in a timely fashion -- if the client allows a large backlog of data to build up the daemon - * may terminate the connection. - * - * sdRef: A DNSServiceRef initialized by any of the DNSService calls. - * - * return value: The DNSServiceRef's underlying socket descriptor, or -1 on - * error. - */ - -int DNSSD_API DNSServiceRefSockFD(DNSServiceRef sdRef); - - -/* DNSServiceProcessResult() - * - * Read a reply from the daemon, calling the appropriate application callback. This call will - * block until the daemon's response is received. Use DNSServiceRefSockFD() in - * conjunction with a run loop or select() to determine the presence of a response from the - * server before calling this function to process the reply without blocking. Call this function - * at any point if it is acceptable to block until the daemon's response arrives. Note that the - * client is responsible for ensuring that DNSServiceProcessResult() is called whenever there is - * a reply from the daemon - the daemon may terminate its connection with a client that does not - * process the daemon's responses. - * - * sdRef: A DNSServiceRef initialized by any of the DNSService calls - * that take a callback parameter. - * - * return value: Returns kDNSServiceErr_NoError on success, otherwise returns - * an error code indicating the specific failure that occurred. - */ - -DNSServiceErrorType DNSSD_API DNSServiceProcessResult(DNSServiceRef sdRef); - - -/* DNSServiceRefDeallocate() - * - * Terminate a connection with the daemon and free memory associated with the DNSServiceRef. - * Any services or records registered with this DNSServiceRef will be deregistered. Any - * Browse, Resolve, or Query operations called with this reference will be terminated. - * - * Note: If the reference's underlying socket is used in a run loop or select() call, it should - * be removed BEFORE DNSServiceRefDeallocate() is called, as this function closes the reference's - * socket. - * - * Note: If the reference was initialized with DNSServiceCreateConnection(), any DNSRecordRefs - * created via this reference will be invalidated by this call - the resource records are - * deregistered, and their DNSRecordRefs may not be used in subsequent functions. Similarly, - * if the reference was initialized with DNSServiceRegister, and an extra resource record was - * added to the service via DNSServiceAddRecord(), the DNSRecordRef created by the Add() call - * is invalidated when this function is called - the DNSRecordRef may not be used in subsequent - * functions. - * - * Note: This call is to be used only with the DNSServiceRef defined by this API. It is - * not compatible with dns_service_discovery_ref objects defined in the legacy Mach-based - * DNSServiceDiscovery.h API. - * - * sdRef: A DNSServiceRef initialized by any of the DNSService calls. - * - */ - -void DNSSD_API DNSServiceRefDeallocate(DNSServiceRef sdRef); - - -/********************************************************************************************* - * - * Domain Enumeration - * - *********************************************************************************************/ - -/* DNSServiceEnumerateDomains() - * - * Asynchronously enumerate domains available for browsing and registration. - * - * The enumeration MUST be cancelled via DNSServiceRefDeallocate() when no more domains - * are to be found. - * - * Note that the names returned are (like all of DNS-SD) UTF-8 strings, - * and are escaped using standard DNS escaping rules. - * (See "Notes on DNS Name Escaping" earlier in this file for more details.) - * A graphical browser displaying a hierarchical tree-structured view should cut - * the names at the bare dots to yield individual labels, then de-escape each - * label according to the escaping rules, and then display the resulting UTF-8 text. - * - * DNSServiceDomainEnumReply Callback Parameters: - * - * sdRef: The DNSServiceRef initialized by DNSServiceEnumerateDomains(). - * - * flags: Possible values are: - * kDNSServiceFlagsMoreComing - * kDNSServiceFlagsAdd - * kDNSServiceFlagsDefault - * - * interfaceIndex: Specifies the interface on which the domain exists. (The index for a given - * interface is determined via the if_nametoindex() family of calls.) - * - * errorCode: Will be kDNSServiceErr_NoError (0) on success, otherwise indicates - * the failure that occurred (other parameters are undefined if errorCode is nonzero). - * - * replyDomain: The name of the domain. - * - * context: The context pointer passed to DNSServiceEnumerateDomains. - * - */ - -typedef void (DNSSD_API *DNSServiceDomainEnumReply) - ( - DNSServiceRef sdRef, - DNSServiceFlags flags, - uint32_t interfaceIndex, - DNSServiceErrorType errorCode, - const char *replyDomain, - void *context - ); - - -/* DNSServiceEnumerateDomains() Parameters: - * - * sdRef: A pointer to an uninitialized DNSServiceRef. If the call succeeds - * then it initializes the DNSServiceRef, returns kDNSServiceErr_NoError, - * and the enumeration operation will run indefinitely until the client - * terminates it by passing this DNSServiceRef to DNSServiceRefDeallocate(). - * - * flags: Possible values are: - * kDNSServiceFlagsBrowseDomains to enumerate domains recommended for browsing. - * kDNSServiceFlagsRegistrationDomains to enumerate domains recommended - * for registration. - * - * interfaceIndex: If non-zero, specifies the interface on which to look for domains. - * (the index for a given interface is determined via the if_nametoindex() - * family of calls.) Most applications will pass 0 to enumerate domains on - * all interfaces. See "Constants for specifying an interface index" for more details. - * - * callBack: The function to be called when a domain is found or the call asynchronously - * fails. - * - * context: An application context pointer which is passed to the callback function - * (may be NULL). - * - * return value: Returns kDNSServiceErr_NoError on success (any subsequent, asynchronous - * errors are delivered to the callback), otherwise returns an error code indicating - * the error that occurred (the callback is not invoked and the DNSServiceRef - * is not initialized). - */ - -DNSServiceErrorType DNSSD_API DNSServiceEnumerateDomains - ( - DNSServiceRef *sdRef, - DNSServiceFlags flags, - uint32_t interfaceIndex, - DNSServiceDomainEnumReply callBack, - void *context /* may be NULL */ - ); - - -/********************************************************************************************* - * - * Service Registration - * - *********************************************************************************************/ - -/* Register a service that is discovered via Browse() and Resolve() calls. - * - * DNSServiceRegisterReply() Callback Parameters: - * - * sdRef: The DNSServiceRef initialized by DNSServiceRegister(). - * - * flags: When a name is successfully registered, the callback will be - * invoked with the kDNSServiceFlagsAdd flag set. When Wide-Area - * DNS-SD is in use, it is possible for a single service to get - * more than one success callback (e.g. one in the "local" multicast - * DNS domain, and another in a wide-area unicast DNS domain). - * If a successfully-registered name later suffers a name conflict - * or similar problem and has to be deregistered, the callback will - * be invoked with the kDNSServiceFlagsAdd flag not set. The callback - * is *not* invoked in the case where the caller explicitly terminates - * the service registration by calling DNSServiceRefDeallocate(ref); - * - * errorCode: Will be kDNSServiceErr_NoError on success, otherwise will - * indicate the failure that occurred (including name conflicts, - * if the kDNSServiceFlagsNoAutoRename flag was used when registering.) - * Other parameters are undefined if errorCode is nonzero. - * - * name: The service name registered (if the application did not specify a name in - * DNSServiceRegister(), this indicates what name was automatically chosen). - * - * regtype: The type of service registered, as it was passed to the callout. - * - * domain: The domain on which the service was registered (if the application did not - * specify a domain in DNSServiceRegister(), this indicates the default domain - * on which the service was registered). - * - * context: The context pointer that was passed to the callout. - * - */ - -typedef void (DNSSD_API *DNSServiceRegisterReply) - ( - DNSServiceRef sdRef, - DNSServiceFlags flags, - DNSServiceErrorType errorCode, - const char *name, - const char *regtype, - const char *domain, - void *context - ); - - -/* DNSServiceRegister() Parameters: - * - * sdRef: A pointer to an uninitialized DNSServiceRef. If the call succeeds - * then it initializes the DNSServiceRef, returns kDNSServiceErr_NoError, - * and the registration will remain active indefinitely until the client - * terminates it by passing this DNSServiceRef to DNSServiceRefDeallocate(). - * - * interfaceIndex: If non-zero, specifies the interface on which to register the service - * (the index for a given interface is determined via the if_nametoindex() - * family of calls.) Most applications will pass 0 to register on all - * available interfaces. See "Constants for specifying an interface index" for more details. - * - * flags: Indicates the renaming behavior on name conflict (most applications - * will pass 0). See flag definitions above for details. - * - * name: If non-NULL, specifies the service name to be registered. - * Most applications will not specify a name, in which case the computer - * name is used (this name is communicated to the client via the callback). - * If a name is specified, it must be 1-63 bytes of UTF-8 text. - * If the name is longer than 63 bytes it will be automatically truncated - * to a legal length, unless the NoAutoRename flag is set, - * in which case kDNSServiceErr_BadParam will be returned. - * - * regtype: The service type followed by the protocol, separated by a dot - * (e.g. "_ftp._tcp"). The service type must be an underscore, followed - * by 1-15 characters, which may be letters, digits, or hyphens. - * The transport protocol must be "_tcp" or "_udp". New service types - * should be registered at . - * - * Additional subtypes of the primary service type (where a service - * type has defined subtypes) follow the primary service type in a - * comma-separated list, with no additional spaces, e.g. - * "_primarytype._tcp,_subtype1,_subtype2,_subtype3" - * Subtypes provide a mechanism for filtered browsing: A client browsing - * for "_primarytype._tcp" will discover all instances of this type; - * a client browsing for "_primarytype._tcp,_subtype2" will discover only - * those instances that were registered with "_subtype2" in their list of - * registered subtypes. - * - * The subtype mechanism can be illustrated with some examples using the - * dns-sd command-line tool: - * - * % dns-sd -R Simple _test._tcp "" 1001 & - * % dns-sd -R Better _test._tcp,HasFeatureA "" 1002 & - * % dns-sd -R Best _test._tcp,HasFeatureA,HasFeatureB "" 1003 & - * - * Now: - * % dns-sd -B _test._tcp # will find all three services - * % dns-sd -B _test._tcp,HasFeatureA # finds "Better" and "Best" - * % dns-sd -B _test._tcp,HasFeatureB # finds only "Best" - * - * Subtype labels may be up to 63 bytes long, and may contain any eight- - * bit byte values, including zero bytes. However, due to the nature of - * using a C-string-based API, conventional DNS escaping must be used for - * dots ('.'), commas (','), backslashes ('\') and zero bytes, as shown below: - * - * % dns-sd -R Test '_test._tcp,s\.one,s\,two,s\\three,s\000four' local 123 - * - * domain: If non-NULL, specifies the domain on which to advertise the service. - * Most applications will not specify a domain, instead automatically - * registering in the default domain(s). - * - * host: If non-NULL, specifies the SRV target host name. Most applications - * will not specify a host, instead automatically using the machine's - * default host name(s). Note that specifying a non-NULL host does NOT - * create an address record for that host - the application is responsible - * for ensuring that the appropriate address record exists, or creating it - * via DNSServiceRegisterRecord(). - * - * port: The port, in network byte order, on which the service accepts connections. - * Pass 0 for a "placeholder" service (i.e. a service that will not be discovered - * by browsing, but will cause a name conflict if another client tries to - * register that same name). Most clients will not use placeholder services. - * - * txtLen: The length of the txtRecord, in bytes. Must be zero if the txtRecord is NULL. - * - * txtRecord: The TXT record rdata. A non-NULL txtRecord MUST be a properly formatted DNS - * TXT record, i.e. ... - * Passing NULL for the txtRecord is allowed as a synonym for txtLen=1, txtRecord="", - * i.e. it creates a TXT record of length one containing a single empty string. - * RFC 1035 doesn't allow a TXT record to contain *zero* strings, so a single empty - * string is the smallest legal DNS TXT record. - * As with the other parameters, the DNSServiceRegister call copies the txtRecord - * data; e.g. if you allocated the storage for the txtRecord parameter with malloc() - * then you can safely free that memory right after the DNSServiceRegister call returns. - * - * callBack: The function to be called when the registration completes or asynchronously - * fails. The client MAY pass NULL for the callback - The client will NOT be notified - * of the default values picked on its behalf, and the client will NOT be notified of any - * asynchronous errors (e.g. out of memory errors, etc.) that may prevent the registration - * of the service. The client may NOT pass the NoAutoRename flag if the callback is NULL. - * The client may still deregister the service at any time via DNSServiceRefDeallocate(). - * - * context: An application context pointer which is passed to the callback function - * (may be NULL). - * - * return value: Returns kDNSServiceErr_NoError on success (any subsequent, asynchronous - * errors are delivered to the callback), otherwise returns an error code indicating - * the error that occurred (the callback is never invoked and the DNSServiceRef - * is not initialized). - */ - -DNSServiceErrorType DNSSD_API DNSServiceRegister - ( - DNSServiceRef *sdRef, - DNSServiceFlags flags, - uint32_t interfaceIndex, - const char *name, /* may be NULL */ - const char *regtype, - const char *domain, /* may be NULL */ - const char *host, /* may be NULL */ - uint16_t port, /* In network byte order */ - uint16_t txtLen, - const void *txtRecord, /* may be NULL */ - DNSServiceRegisterReply callBack, /* may be NULL */ - void *context /* may be NULL */ - ); - - -/* DNSServiceAddRecord() - * - * Add a record to a registered service. The name of the record will be the same as the - * registered service's name. - * The record can later be updated or deregistered by passing the RecordRef initialized - * by this function to DNSServiceUpdateRecord() or DNSServiceRemoveRecord(). - * - * Note that the DNSServiceAddRecord/UpdateRecord/RemoveRecord are *NOT* thread-safe - * with respect to a single DNSServiceRef. If you plan to have multiple threads - * in your program simultaneously add, update, or remove records from the same - * DNSServiceRef, then it's the caller's responsibility to use a mutext lock - * or take similar appropriate precautions to serialize those calls. - * - * Parameters; - * - * sdRef: A DNSServiceRef initialized by DNSServiceRegister(). - * - * RecordRef: A pointer to an uninitialized DNSRecordRef. Upon succesfull completion of this - * call, this ref may be passed to DNSServiceUpdateRecord() or DNSServiceRemoveRecord(). - * If the above DNSServiceRef is passed to DNSServiceRefDeallocate(), RecordRef is also - * invalidated and may not be used further. - * - * flags: Currently ignored, reserved for future use. - * - * rrtype: The type of the record (e.g. kDNSServiceType_TXT, kDNSServiceType_SRV, etc) - * - * rdlen: The length, in bytes, of the rdata. - * - * rdata: The raw rdata to be contained in the added resource record. - * - * ttl: The time to live of the resource record, in seconds. - * Most clients should pass 0 to indicate that the system should - * select a sensible default value. - * - * return value: Returns kDNSServiceErr_NoError on success, otherwise returns an - * error code indicating the error that occurred (the RecordRef is not initialized). - */ - -DNSServiceErrorType DNSSD_API DNSServiceAddRecord - ( - DNSServiceRef sdRef, - DNSRecordRef *RecordRef, - DNSServiceFlags flags, - uint16_t rrtype, - uint16_t rdlen, - const void *rdata, - uint32_t ttl - ); - - -/* DNSServiceUpdateRecord - * - * Update a registered resource record. The record must either be: - * - The primary txt record of a service registered via DNSServiceRegister() - * - A record added to a registered service via DNSServiceAddRecord() - * - An individual record registered by DNSServiceRegisterRecord() - * - * Parameters: - * - * sdRef: A DNSServiceRef that was initialized by DNSServiceRegister() - * or DNSServiceCreateConnection(). - * - * RecordRef: A DNSRecordRef initialized by DNSServiceAddRecord, or NULL to update the - * service's primary txt record. - * - * flags: Currently ignored, reserved for future use. - * - * rdlen: The length, in bytes, of the new rdata. - * - * rdata: The new rdata to be contained in the updated resource record. - * - * ttl: The time to live of the updated resource record, in seconds. - * Most clients should pass 0 to indicate that the system should - * select a sensible default value. - * - * return value: Returns kDNSServiceErr_NoError on success, otherwise returns an - * error code indicating the error that occurred. - */ - -DNSServiceErrorType DNSSD_API DNSServiceUpdateRecord - ( - DNSServiceRef sdRef, - DNSRecordRef RecordRef, /* may be NULL */ - DNSServiceFlags flags, - uint16_t rdlen, - const void *rdata, - uint32_t ttl - ); - - -/* DNSServiceRemoveRecord - * - * Remove a record previously added to a service record set via DNSServiceAddRecord(), or deregister - * an record registered individually via DNSServiceRegisterRecord(). - * - * Parameters: - * - * sdRef: A DNSServiceRef initialized by DNSServiceRegister() (if the - * record being removed was registered via DNSServiceAddRecord()) or by - * DNSServiceCreateConnection() (if the record being removed was registered via - * DNSServiceRegisterRecord()). - * - * recordRef: A DNSRecordRef initialized by a successful call to DNSServiceAddRecord() - * or DNSServiceRegisterRecord(). - * - * flags: Currently ignored, reserved for future use. - * - * return value: Returns kDNSServiceErr_NoError on success, otherwise returns an - * error code indicating the error that occurred. - */ - -DNSServiceErrorType DNSSD_API DNSServiceRemoveRecord - ( - DNSServiceRef sdRef, - DNSRecordRef RecordRef, - DNSServiceFlags flags - ); - - -/********************************************************************************************* - * - * Service Discovery - * - *********************************************************************************************/ - -/* Browse for instances of a service. - * - * DNSServiceBrowseReply() Parameters: - * - * sdRef: The DNSServiceRef initialized by DNSServiceBrowse(). - * - * flags: Possible values are kDNSServiceFlagsMoreComing and kDNSServiceFlagsAdd. - * See flag definitions for details. - * - * interfaceIndex: The interface on which the service is advertised. This index should - * be passed to DNSServiceResolve() when resolving the service. - * - * errorCode: Will be kDNSServiceErr_NoError (0) on success, otherwise will - * indicate the failure that occurred. Other parameters are undefined if - * the errorCode is nonzero. - * - * serviceName: The discovered service name. This name should be displayed to the user, - * and stored for subsequent use in the DNSServiceResolve() call. - * - * regtype: The service type, which is usually (but not always) the same as was passed - * to DNSServiceBrowse(). One case where the discovered service type may - * not be the same as the requested service type is when using subtypes: - * The client may want to browse for only those ftp servers that allow - * anonymous connections. The client will pass the string "_ftp._tcp,_anon" - * to DNSServiceBrowse(), but the type of the service that's discovered - * is simply "_ftp._tcp". The regtype for each discovered service instance - * should be stored along with the name, so that it can be passed to - * DNSServiceResolve() when the service is later resolved. - * - * domain: The domain of the discovered service instance. This may or may not be the - * same as the domain that was passed to DNSServiceBrowse(). The domain for each - * discovered service instance should be stored along with the name, so that - * it can be passed to DNSServiceResolve() when the service is later resolved. - * - * context: The context pointer that was passed to the callout. - * - */ - -typedef void (DNSSD_API *DNSServiceBrowseReply) - ( - DNSServiceRef sdRef, - DNSServiceFlags flags, - uint32_t interfaceIndex, - DNSServiceErrorType errorCode, - const char *serviceName, - const char *regtype, - const char *replyDomain, - void *context - ); - - -/* DNSServiceBrowse() Parameters: - * - * sdRef: A pointer to an uninitialized DNSServiceRef. If the call succeeds - * then it initializes the DNSServiceRef, returns kDNSServiceErr_NoError, - * and the browse operation will run indefinitely until the client - * terminates it by passing this DNSServiceRef to DNSServiceRefDeallocate(). - * - * flags: Currently ignored, reserved for future use. - * - * interfaceIndex: If non-zero, specifies the interface on which to browse for services - * (the index for a given interface is determined via the if_nametoindex() - * family of calls.) Most applications will pass 0 to browse on all available - * interfaces. See "Constants for specifying an interface index" for more details. - * - * regtype: The service type being browsed for followed by the protocol, separated by a - * dot (e.g. "_ftp._tcp"). The transport protocol must be "_tcp" or "_udp". - * A client may optionally specify a single subtype to perform filtered browsing: - * e.g. browsing for "_primarytype._tcp,_subtype" will discover only those - * instances of "_primarytype._tcp" that were registered specifying "_subtype" - * in their list of registered subtypes. - * - * domain: If non-NULL, specifies the domain on which to browse for services. - * Most applications will not specify a domain, instead browsing on the - * default domain(s). - * - * callBack: The function to be called when an instance of the service being browsed for - * is found, or if the call asynchronously fails. - * - * context: An application context pointer which is passed to the callback function - * (may be NULL). - * - * return value: Returns kDNSServiceErr_NoError on success (any subsequent, asynchronous - * errors are delivered to the callback), otherwise returns an error code indicating - * the error that occurred (the callback is not invoked and the DNSServiceRef - * is not initialized). - */ - -DNSServiceErrorType DNSSD_API DNSServiceBrowse - ( - DNSServiceRef *sdRef, - DNSServiceFlags flags, - uint32_t interfaceIndex, - const char *regtype, - const char *domain, /* may be NULL */ - DNSServiceBrowseReply callBack, - void *context /* may be NULL */ - ); - - -/* DNSServiceResolve() - * - * Resolve a service name discovered via DNSServiceBrowse() to a target host name, port number, and - * txt record. - * - * Note: Applications should NOT use DNSServiceResolve() solely for txt record monitoring - use - * DNSServiceQueryRecord() instead, as it is more efficient for this task. - * - * Note: When the desired results have been returned, the client MUST terminate the resolve by calling - * DNSServiceRefDeallocate(). - * - * Note: DNSServiceResolve() behaves correctly for typical services that have a single SRV record - * and a single TXT record. To resolve non-standard services with multiple SRV or TXT records, - * DNSServiceQueryRecord() should be used. - * - * DNSServiceResolveReply Callback Parameters: - * - * sdRef: The DNSServiceRef initialized by DNSServiceResolve(). - * - * flags: Possible values: kDNSServiceFlagsMoreComing - * - * interfaceIndex: The interface on which the service was resolved. - * - * errorCode: Will be kDNSServiceErr_NoError (0) on success, otherwise will - * indicate the failure that occurred. Other parameters are undefined if - * the errorCode is nonzero. - * - * fullname: The full service domain name, in the form ... - * (This name is escaped following standard DNS rules, making it suitable for - * passing to standard system DNS APIs such as res_query(), or to the - * special-purpose functions included in this API that take fullname parameters. - * See "Notes on DNS Name Escaping" earlier in this file for more details.) - * - * hosttarget: The target hostname of the machine providing the service. This name can - * be passed to functions like gethostbyname() to identify the host's IP address. - * - * port: The port, in network byte order, on which connections are accepted for this service. - * - * txtLen: The length of the txt record, in bytes. - * - * txtRecord: The service's primary txt record, in standard txt record format. - * - * context: The context pointer that was passed to the callout. - * - * NOTE: In earlier versions of this header file, the txtRecord parameter was declared "const char *" - * This is incorrect, since it contains length bytes which are values in the range 0 to 255, not -128 to +127. - * Depending on your compiler settings, this change may cause signed/unsigned mismatch warnings. - * These should be fixed by updating your own callback function definition to match the corrected - * function signature using "const unsigned char *txtRecord". Making this change may also fix inadvertent - * bugs in your callback function, where it could have incorrectly interpreted a length byte with value 250 - * as being -6 instead, with various bad consequences ranging from incorrect operation to software crashes. - * If you need to maintain portable code that will compile cleanly with both the old and new versions of - * this header file, you should update your callback function definition to use the correct unsigned value, - * and then in the place where you pass your callback function to DNSServiceResolve(), use a cast to eliminate - * the compiler warning, e.g.: - * DNSServiceResolve(sd, flags, index, name, regtype, domain, (DNSServiceResolveReply)MyCallback, context); - * This will ensure that your code compiles cleanly without warnings (and more importantly, works correctly) - * with both the old header and with the new corrected version. - * - */ - -typedef void (DNSSD_API *DNSServiceResolveReply) - ( - DNSServiceRef sdRef, - DNSServiceFlags flags, - uint32_t interfaceIndex, - DNSServiceErrorType errorCode, - const char *fullname, - const char *hosttarget, - uint16_t port, /* In network byte order */ - uint16_t txtLen, - const unsigned char *txtRecord, - void *context - ); - - -/* DNSServiceResolve() Parameters - * - * sdRef: A pointer to an uninitialized DNSServiceRef. If the call succeeds - * then it initializes the DNSServiceRef, returns kDNSServiceErr_NoError, - * and the resolve operation will run indefinitely until the client - * terminates it by passing this DNSServiceRef to DNSServiceRefDeallocate(). - * - * flags: Specifying kDNSServiceFlagsForceMulticast will cause query to be - * performed with a link-local mDNS query, even if the name is an - * apparently non-local name (i.e. a name not ending in ".local.") - * - * interfaceIndex: The interface on which to resolve the service. If this resolve call is - * as a result of a currently active DNSServiceBrowse() operation, then the - * interfaceIndex should be the index reported in the DNSServiceBrowseReply - * callback. If this resolve call is using information previously saved - * (e.g. in a preference file) for later use, then use interfaceIndex 0, because - * the desired service may now be reachable via a different physical interface. - * See "Constants for specifying an interface index" for more details. - * - * name: The name of the service instance to be resolved, as reported to the - * DNSServiceBrowseReply() callback. - * - * regtype: The type of the service instance to be resolved, as reported to the - * DNSServiceBrowseReply() callback. - * - * domain: The domain of the service instance to be resolved, as reported to the - * DNSServiceBrowseReply() callback. - * - * callBack: The function to be called when a result is found, or if the call - * asynchronously fails. - * - * context: An application context pointer which is passed to the callback function - * (may be NULL). - * - * return value: Returns kDNSServiceErr_NoError on success (any subsequent, asynchronous - * errors are delivered to the callback), otherwise returns an error code indicating - * the error that occurred (the callback is never invoked and the DNSServiceRef - * is not initialized). - */ - -DNSServiceErrorType DNSSD_API DNSServiceResolve - ( - DNSServiceRef *sdRef, - DNSServiceFlags flags, - uint32_t interfaceIndex, - const char *name, - const char *regtype, - const char *domain, - DNSServiceResolveReply callBack, - void *context /* may be NULL */ - ); - - -/********************************************************************************************* - * - * Querying Individual Specific Records - * - *********************************************************************************************/ - -/* DNSServiceQueryRecord - * - * Query for an arbitrary DNS record. - * - * DNSServiceQueryRecordReply() Callback Parameters: - * - * sdRef: The DNSServiceRef initialized by DNSServiceQueryRecord(). - * - * flags: Possible values are kDNSServiceFlagsMoreComing and - * kDNSServiceFlagsAdd. The Add flag is NOT set for PTR records - * with a ttl of 0, i.e. "Remove" events. - * - * interfaceIndex: The interface on which the query was resolved (the index for a given - * interface is determined via the if_nametoindex() family of calls). - * See "Constants for specifying an interface index" for more details. - * - * errorCode: Will be kDNSServiceErr_NoError on success, otherwise will - * indicate the failure that occurred. Other parameters are undefined if - * errorCode is nonzero. - * - * fullname: The resource record's full domain name. - * - * rrtype: The resource record's type (e.g. kDNSServiceType_PTR, kDNSServiceType_SRV, etc) - * - * rrclass: The class of the resource record (usually kDNSServiceClass_IN). - * - * rdlen: The length, in bytes, of the resource record rdata. - * - * rdata: The raw rdata of the resource record. - * - * ttl: If the client wishes to cache the result for performance reasons, - * the TTL indicates how long the client may legitimately hold onto - * this result, in seconds. After the TTL expires, the client should - * consider the result no longer valid, and if it requires this data - * again, it should be re-fetched with a new query. Of course, this - * only applies to clients that cancel the asynchronous operation when - * they get a result. Clients that leave the asynchronous operation - * running can safely assume that the data remains valid until they - * get another callback telling them otherwise. - * - * context: The context pointer that was passed to the callout. - * - */ - -typedef void (DNSSD_API *DNSServiceQueryRecordReply) - ( - DNSServiceRef sdRef, - DNSServiceFlags flags, - uint32_t interfaceIndex, - DNSServiceErrorType errorCode, - const char *fullname, - uint16_t rrtype, - uint16_t rrclass, - uint16_t rdlen, - const void *rdata, - uint32_t ttl, - void *context - ); - - -/* DNSServiceQueryRecord() Parameters: - * - * sdRef: A pointer to an uninitialized DNSServiceRef. If the call succeeds - * then it initializes the DNSServiceRef, returns kDNSServiceErr_NoError, - * and the query operation will run indefinitely until the client - * terminates it by passing this DNSServiceRef to DNSServiceRefDeallocate(). - * - * flags: kDNSServiceFlagsForceMulticast or kDNSServiceFlagsLongLivedQuery. - * Pass kDNSServiceFlagsLongLivedQuery to create a "long-lived" unicast - * query in a non-local domain. Without setting this flag, unicast queries - * will be one-shot - that is, only answers available at the time of the call - * will be returned. By setting this flag, answers (including Add and Remove - * events) that become available after the initial call is made will generate - * callbacks. This flag has no effect on link-local multicast queries. - * - * interfaceIndex: If non-zero, specifies the interface on which to issue the query - * (the index for a given interface is determined via the if_nametoindex() - * family of calls.) Passing 0 causes the name to be queried for on all - * interfaces. See "Constants for specifying an interface index" for more details. - * - * fullname: The full domain name of the resource record to be queried for. - * - * rrtype: The numerical type of the resource record to be queried for - * (e.g. kDNSServiceType_PTR, kDNSServiceType_SRV, etc) - * - * rrclass: The class of the resource record (usually kDNSServiceClass_IN). - * - * callBack: The function to be called when a result is found, or if the call - * asynchronously fails. - * - * context: An application context pointer which is passed to the callback function - * (may be NULL). - * - * return value: Returns kDNSServiceErr_NoError on success (any subsequent, asynchronous - * errors are delivered to the callback), otherwise returns an error code indicating - * the error that occurred (the callback is never invoked and the DNSServiceRef - * is not initialized). - */ - -DNSServiceErrorType DNSSD_API DNSServiceQueryRecord - ( - DNSServiceRef *sdRef, - DNSServiceFlags flags, - uint32_t interfaceIndex, - const char *fullname, - uint16_t rrtype, - uint16_t rrclass, - DNSServiceQueryRecordReply callBack, - void *context /* may be NULL */ - ); - - -/********************************************************************************************* - * - * Unified lookup of both IPv4 and IPv6 addresses for a fully qualified hostname - * - *********************************************************************************************/ - -/* DNSServiceGetAddrInfo - * - * Queries for the IP address of a hostname by using either Multicast or Unicast DNS. - * - * DNSServiceGetAddrInfoReply() parameters: - * - * sdRef: The DNSServiceRef initialized by DNSServiceGetAddrInfo(). - * - * flags: Possible values are kDNSServiceFlagsMoreComing and - * kDNSServiceFlagsAdd. - * - * interfaceIndex: The interface to which the answers pertain. - * - * errorCode: Will be kDNSServiceErr_NoError on success, otherwise will - * indicate the failure that occurred. Other parameters are - * undefined if errorCode is nonzero. - * - * hostname: The fully qualified domain name of the host to be queried for. - * - * address: IPv4 or IPv6 address. - * - * ttl: If the client wishes to cache the result for performance reasons, - * the TTL indicates how long the client may legitimately hold onto - * this result, in seconds. After the TTL expires, the client should - * consider the result no longer valid, and if it requires this data - * again, it should be re-fetched with a new query. Of course, this - * only applies to clients that cancel the asynchronous operation when - * they get a result. Clients that leave the asynchronous operation - * running can safely assume that the data remains valid until they - * get another callback telling them otherwise. - * - * context: The context pointer that was passed to the callout. - * - */ - -typedef void (DNSSD_API *DNSServiceGetAddrInfoReply) - ( - DNSServiceRef sdRef, - DNSServiceFlags flags, - uint32_t interfaceIndex, - DNSServiceErrorType errorCode, - const char *hostname, - const struct sockaddr *address, - uint32_t ttl, - void *context - ); - - -/* DNSServiceGetAddrInfo() Parameters: - * - * sdRef: A pointer to an uninitialized DNSServiceRef. If the call succeeds then it - * initializes the DNSServiceRef, returns kDNSServiceErr_NoError, and the query - * begins and will last indefinitely until the client terminates the query - * by passing this DNSServiceRef to DNSServiceRefDeallocate(). - * - * flags: kDNSServiceFlagsForceMulticast or kDNSServiceFlagsLongLivedQuery. - * Pass kDNSServiceFlagsLongLivedQuery to create a "long-lived" unicast - * query in a non-local domain. Without setting this flag, unicast queries - * will be one-shot - that is, only answers available at the time of the call - * will be returned. By setting this flag, answers (including Add and Remove - * events) that become available after the initial call is made will generate - * callbacks. This flag has no effect on link-local multicast queries. - * - * interfaceIndex: The interface on which to issue the query. Passing 0 causes the query to be - * sent on all active interfaces via Multicast or the primary interface via Unicast. - * - * protocol: Pass in kDNSServiceProtocol_IPv4 to look up IPv4 addresses, or kDNSServiceProtocol_IPv6 - * to look up IPv6 addresses, or both to look up both kinds. If neither flag is - * set, the system will apply an intelligent heuristic, which is (currently) - * that it will attempt to look up both, except: - * - * * If "hostname" is a wide-area unicast DNS hostname (i.e. not a ".local." name) - * but this host has no routable IPv6 address, then the call will not try to - * look up IPv6 addresses for "hostname", since any addresses it found would be - * unlikely to be of any use anyway. Similarly, if this host has no routable - * IPv4 address, the call will not try to look up IPv4 addresses for "hostname". - * - * hostname: The fully qualified domain name of the host to be queried for. - * - * callBack: The function to be called when the query succeeds or fails asynchronously. - * - * context: An application context pointer which is passed to the callback function - * (may be NULL). - * - * return value: Returns kDNSServiceErr_NoError on success (any subsequent, asynchronous - * errors are delivered to the callback), otherwise returns an error code indicating - * the error that occurred. - */ - -DNSServiceErrorType DNSSD_API DNSServiceGetAddrInfo - ( - DNSServiceRef *sdRef, - DNSServiceFlags flags, - uint32_t interfaceIndex, - DNSServiceProtocol protocol, - const char *hostname, - DNSServiceGetAddrInfoReply callBack, - void *context /* may be NULL */ - ); - - -/********************************************************************************************* - * - * Special Purpose Calls: - * DNSServiceCreateConnection(), DNSServiceRegisterRecord(), DNSServiceReconfirmRecord() - * (most applications will not use these) - * - *********************************************************************************************/ - -/* DNSServiceCreateConnection() - * - * Create a connection to the daemon allowing efficient registration of - * multiple individual records. - * - * Parameters: - * - * sdRef: A pointer to an uninitialized DNSServiceRef. Deallocating - * the reference (via DNSServiceRefDeallocate()) severs the - * connection and deregisters all records registered on this connection. - * - * return value: Returns kDNSServiceErr_NoError on success, otherwise returns - * an error code indicating the specific failure that occurred (in which - * case the DNSServiceRef is not initialized). - */ - -DNSServiceErrorType DNSSD_API DNSServiceCreateConnection(DNSServiceRef *sdRef); - - -/* DNSServiceRegisterRecord - * - * Register an individual resource record on a connected DNSServiceRef. - * - * Note that name conflicts occurring for records registered via this call must be handled - * by the client in the callback. - * - * DNSServiceRegisterRecordReply() parameters: - * - * sdRef: The connected DNSServiceRef initialized by - * DNSServiceCreateConnection(). - * - * RecordRef: The DNSRecordRef initialized by DNSServiceRegisterRecord(). If the above - * DNSServiceRef is passed to DNSServiceRefDeallocate(), this DNSRecordRef is - * invalidated, and may not be used further. - * - * flags: Currently unused, reserved for future use. - * - * errorCode: Will be kDNSServiceErr_NoError on success, otherwise will - * indicate the failure that occurred (including name conflicts.) - * Other parameters are undefined if errorCode is nonzero. - * - * context: The context pointer that was passed to the callout. - * - */ - - typedef void (DNSSD_API *DNSServiceRegisterRecordReply) - ( - DNSServiceRef sdRef, - DNSRecordRef RecordRef, - DNSServiceFlags flags, - DNSServiceErrorType errorCode, - void *context - ); - - -/* DNSServiceRegisterRecord() Parameters: - * - * sdRef: A DNSServiceRef initialized by DNSServiceCreateConnection(). - * - * RecordRef: A pointer to an uninitialized DNSRecordRef. Upon succesfull completion of this - * call, this ref may be passed to DNSServiceUpdateRecord() or DNSServiceRemoveRecord(). - * (To deregister ALL records registered on a single connected DNSServiceRef - * and deallocate each of their corresponding DNSServiceRecordRefs, call - * DNSServiceRefDeallocate()). - * - * flags: Possible values are kDNSServiceFlagsShared or kDNSServiceFlagsUnique - * (see flag type definitions for details). - * - * interfaceIndex: If non-zero, specifies the interface on which to register the record - * (the index for a given interface is determined via the if_nametoindex() - * family of calls.) Passing 0 causes the record to be registered on all interfaces. - * See "Constants for specifying an interface index" for more details. - * - * fullname: The full domain name of the resource record. - * - * rrtype: The numerical type of the resource record (e.g. kDNSServiceType_PTR, kDNSServiceType_SRV, etc) - * - * rrclass: The class of the resource record (usually kDNSServiceClass_IN) - * - * rdlen: Length, in bytes, of the rdata. - * - * rdata: A pointer to the raw rdata, as it is to appear in the DNS record. - * - * ttl: The time to live of the resource record, in seconds. - * Most clients should pass 0 to indicate that the system should - * select a sensible default value. - * - * callBack: The function to be called when a result is found, or if the call - * asynchronously fails (e.g. because of a name conflict.) - * - * context: An application context pointer which is passed to the callback function - * (may be NULL). - * - * return value: Returns kDNSServiceErr_NoError on success (any subsequent, asynchronous - * errors are delivered to the callback), otherwise returns an error code indicating - * the error that occurred (the callback is never invoked and the DNSRecordRef is - * not initialized). - */ - -DNSServiceErrorType DNSSD_API DNSServiceRegisterRecord - ( - DNSServiceRef sdRef, - DNSRecordRef *RecordRef, - DNSServiceFlags flags, - uint32_t interfaceIndex, - const char *fullname, - uint16_t rrtype, - uint16_t rrclass, - uint16_t rdlen, - const void *rdata, - uint32_t ttl, - DNSServiceRegisterRecordReply callBack, - void *context /* may be NULL */ - ); - - -/* DNSServiceReconfirmRecord - * - * Instruct the daemon to verify the validity of a resource record that appears - * to be out of date (e.g. because TCP connection to a service's target failed.) - * Causes the record to be flushed from the daemon's cache (as well as all other - * daemons' caches on the network) if the record is determined to be invalid. - * Use this routine conservatively. Reconfirming a record necessarily consumes - * network bandwidth, so this should not be done indiscriminately. - * - * Parameters: - * - * flags: Pass kDNSServiceFlagsForce to force immediate deletion of record, - * instead of after some number of reconfirmation queries have gone unanswered. - * - * interfaceIndex: Specifies the interface of the record in question. - * The caller must specify the interface. - * This API (by design) causes increased network traffic, so it requires - * the caller to be precise about which record should be reconfirmed. - * It is not possible to pass zero for the interface index to perform - * a "wildcard" reconfirmation, where *all* matching records are reconfirmed. - * - * fullname: The resource record's full domain name. - * - * rrtype: The resource record's type (e.g. kDNSServiceType_PTR, kDNSServiceType_SRV, etc) - * - * rrclass: The class of the resource record (usually kDNSServiceClass_IN). - * - * rdlen: The length, in bytes, of the resource record rdata. - * - * rdata: The raw rdata of the resource record. - * - */ - -DNSServiceErrorType DNSSD_API DNSServiceReconfirmRecord - ( - DNSServiceFlags flags, - uint32_t interfaceIndex, - const char *fullname, - uint16_t rrtype, - uint16_t rrclass, - uint16_t rdlen, - const void *rdata - ); - - -/********************************************************************************************* - * - * NAT Port Mapping - * - *********************************************************************************************/ - -/* DNSServiceNATPortMappingCreate - * - * Request a port mapping in the NAT gateway, which maps a port on the local machine - * to an external port on the NAT. The NAT should support either the NAT-PMP or the UPnP IGD - * protocol for this API to create a successful mapping. - * - * The port mapping will be renewed indefinitely until the client process exits, or - * explicitly terminates the port mapping request by calling DNSServiceRefDeallocate(). - * The client callback will be invoked, informing the client of the NAT gateway's - * external IP address and the external port that has been allocated for this client. - * The client should then record this external IP address and port using whatever - * directory service mechanism it is using to enable peers to connect to it. - * (Clients advertising services using Wide-Area DNS-SD DO NOT need to use this API - * -- when a client calls DNSServiceRegister() NAT mappings are automatically created - * and the external IP address and port for the service are recorded in the global DNS. - * Only clients using some directory mechanism other than Wide-Area DNS-SD need to use - * this API to explicitly map their own ports.) - * - * It's possible that the client callback could be called multiple times, for example - * if the NAT gateway's IP address changes, or if a configuration change results in a - * different external port being mapped for this client. Over the lifetime of any long-lived - * port mapping, the client should be prepared to handle these notifications of changes - * in the environment, and should update its recorded address and/or port as appropriate. - * - * NOTE: There are two unusual aspects of how the DNSServiceNATPortMappingCreate API works, - * which were intentionally designed to help simplify client code: - * - * 1. It's not an error to request a NAT mapping when the machine is not behind a NAT gateway. - * In other NAT mapping APIs, if you request a NAT mapping and the machine is not behind a NAT - * gateway, then the API returns an error code -- it can't get you a NAT mapping if there's no - * NAT gateway. The DNSServiceNATPortMappingCreate API takes a different view. Working out - * whether or not you need a NAT mapping can be tricky and non-obvious, particularly on - * a machine with multiple active network interfaces. Rather than make every client recreate - * this logic for deciding whether a NAT mapping is required, the PortMapping API does that - * work for you. If the client calls the PortMapping API when the machine already has a - * routable public IP address, then instead of complaining about it and giving an error, - * the PortMapping API just invokes your callback, giving the machine's public address - * and your own port number. This means you don't need to write code to work out whether - * your client needs to call the PortMapping API -- just call it anyway, and if it wasn't - * necessary, no harm is done: - * - * - If the machine already has a routable public IP address, then your callback - * will just be invoked giving your own address and port. - * - If a NAT mapping is required and obtained, then your callback will be invoked - * giving you the external address and port. - * - If a NAT mapping is required but not obtained from the local NAT gateway, - * or the machine has no network connectivity, then your callback will be - * invoked giving zero address and port. - * - * 2. In other NAT mapping APIs, if a laptop computer is put to sleep and woken up on a new - * network, it's the client's job to notice this, and work out whether a NAT mapping - * is required on the new network, and make a new NAT mapping request if necessary. - * The DNSServiceNATPortMappingCreate API does this for you, automatically. - * The client just needs to make one call to the PortMapping API, and its callback will - * be invoked any time the mapping state changes. This property complements point (1) above. - * If the client didn't make a NAT mapping request just because it determined that one was - * not required at that particular moment in time, the client would then have to monitor - * for network state changes to determine if a NAT port mapping later became necessary. - * By unconditionally making a NAT mapping request, even when a NAT mapping not to be - * necessary, the PortMapping API will then begin monitoring network state changes on behalf of - * the client, and if a NAT mapping later becomes necessary, it will automatically create a NAT - * mapping and inform the client with a new callback giving the new address and port information. - * - * DNSServiceNATPortMappingReply() parameters: - * - * sdRef: The DNSServiceRef initialized by DNSServiceNATPortMappingCreate(). - * - * flags: Currently unused, reserved for future use. - * - * interfaceIndex: The interface through which the NAT gateway is reached. - * - * errorCode: Will be kDNSServiceErr_NoError on success. - * Will be kDNSServiceErr_DoubleNAT when the NAT gateway is itself behind one or - * more layers of NAT, in which case the other parameters have the defined values. - * For other failures, will indicate the failure that occurred, and the other - * parameters are undefined. - * - * externalAddress: Four byte IPv4 address in network byte order. - * - * protocol: Will be kDNSServiceProtocol_UDP or kDNSServiceProtocol_TCP or both. - * - * internalPort: The port on the local machine that was mapped. - * - * externalPort: The actual external port in the NAT gateway that was mapped. - * This is likely to be different than the requested external port. - * - * ttl: The lifetime of the NAT port mapping created on the gateway. - * This controls how quickly stale mappings will be garbage-collected - * if the client machine crashes, suffers a power failure, is disconnected - * from the network, or suffers some other unfortunate demise which - * causes it to vanish without explicitly removing its NAT port mapping. - * It's possible that the ttl value will differ from the requested ttl value. - * - * context: The context pointer that was passed to the callout. - * - */ - -typedef void (DNSSD_API *DNSServiceNATPortMappingReply) - ( - DNSServiceRef sdRef, - DNSServiceFlags flags, - uint32_t interfaceIndex, - DNSServiceErrorType errorCode, - uint32_t externalAddress, /* four byte IPv4 address in network byte order */ - DNSServiceProtocol protocol, - uint16_t internalPort, /* In network byte order */ - uint16_t externalPort, /* In network byte order and may be different than the requested port */ - uint32_t ttl, /* may be different than the requested ttl */ - void *context - ); - - -/* DNSServiceNATPortMappingCreate() Parameters: - * - * sdRef: A pointer to an uninitialized DNSServiceRef. If the call succeeds then it - * initializes the DNSServiceRef, returns kDNSServiceErr_NoError, and the nat - * port mapping will last indefinitely until the client terminates the port - * mapping request by passing this DNSServiceRef to DNSServiceRefDeallocate(). - * - * flags: Currently ignored, reserved for future use. - * - * interfaceIndex: The interface on which to create port mappings in a NAT gateway. Passing 0 causes - * the port mapping request to be sent on the primary interface. - * - * protocol: To request a port mapping, pass in kDNSServiceProtocol_UDP, or kDNSServiceProtocol_TCP, - * or (kDNSServiceProtocol_UDP | kDNSServiceProtocol_TCP) to map both. - * The local listening port number must also be specified in the internalPort parameter. - * To just discover the NAT gateway's external IP address, pass zero for protocol, - * internalPort, externalPort and ttl. - * - * internalPort: The port number in network byte order on the local machine which is listening for packets. - * - * externalPort: The requested external port in network byte order in the NAT gateway that you would - * like to map to the internal port. Pass 0 if you don't care which external port is chosen for you. - * - * ttl: The requested renewal period of the NAT port mapping, in seconds. - * If the client machine crashes, suffers a power failure, is disconnected from - * the network, or suffers some other unfortunate demise which causes it to vanish - * unexpectedly without explicitly removing its NAT port mappings, then the NAT gateway - * will garbage-collect old stale NAT port mappings when their lifetime expires. - * Requesting a short TTL causes such orphaned mappings to be garbage-collected - * more promptly, but consumes system resources and network bandwidth with - * frequent renewal packets to keep the mapping from expiring. - * Requesting a long TTL is more efficient on the network, but in the event of the - * client vanishing, stale NAT port mappings will not be garbage-collected as quickly. - * Most clients should pass 0 to use a system-wide default value. - * - * callBack: The function to be called when the port mapping request succeeds or fails asynchronously. - * - * context: An application context pointer which is passed to the callback function - * (may be NULL). - * - * return value: Returns kDNSServiceErr_NoError on success (any subsequent, asynchronous - * errors are delivered to the callback), otherwise returns an error code indicating - * the error that occurred. - * - * If you don't actually want a port mapped, and are just calling the API - * because you want to find out the NAT's external IP address (e.g. for UI - * display) then pass zero for protocol, internalPort, externalPort and ttl. - */ - -DNSServiceErrorType DNSSD_API DNSServiceNATPortMappingCreate - ( - DNSServiceRef *sdRef, - DNSServiceFlags flags, - uint32_t interfaceIndex, - DNSServiceProtocol protocol, /* TCP and/or UDP */ - uint16_t internalPort, /* network byte order */ - uint16_t externalPort, /* network byte order */ - uint32_t ttl, /* time to live in seconds */ - DNSServiceNATPortMappingReply callBack, - void *context /* may be NULL */ - ); - - -/********************************************************************************************* - * - * General Utility Functions - * - *********************************************************************************************/ - -/* DNSServiceConstructFullName() - * - * Concatenate a three-part domain name (as returned by the above callbacks) into a - * properly-escaped full domain name. Note that callbacks in the above functions ALREADY ESCAPE - * strings where necessary. - * - * Parameters: - * - * fullName: A pointer to a buffer that where the resulting full domain name is to be written. - * The buffer must be kDNSServiceMaxDomainName (1009) bytes in length to - * accommodate the longest legal domain name without buffer overrun. - * - * service: The service name - any dots or backslashes must NOT be escaped. - * May be NULL (to construct a PTR record name, e.g. - * "_ftp._tcp.apple.com."). - * - * regtype: The service type followed by the protocol, separated by a dot - * (e.g. "_ftp._tcp"). - * - * domain: The domain name, e.g. "apple.com.". Literal dots or backslashes, - * if any, must be escaped, e.g. "1st\. Floor.apple.com." - * - * return value: Returns kDNSServiceErr_NoError (0) on success, kDNSServiceErr_BadParam on error. - * - */ - -DNSServiceErrorType DNSSD_API DNSServiceConstructFullName - ( - char * const fullName, - const char * const service, /* may be NULL */ - const char * const regtype, - const char * const domain - ); - - -/********************************************************************************************* - * - * TXT Record Construction Functions - * - *********************************************************************************************/ - -/* - * A typical calling sequence for TXT record construction is something like: - * - * Client allocates storage for TXTRecord data (e.g. declare buffer on the stack) - * TXTRecordCreate(); - * TXTRecordSetValue(); - * TXTRecordSetValue(); - * TXTRecordSetValue(); - * ... - * DNSServiceRegister( ... TXTRecordGetLength(), TXTRecordGetBytesPtr() ... ); - * TXTRecordDeallocate(); - * Explicitly deallocate storage for TXTRecord data (if not allocated on the stack) - */ - - -/* TXTRecordRef - * - * Opaque internal data type. - * Note: Represents a DNS-SD TXT record. - */ - -typedef union _TXTRecordRef_t { char PrivateData[16]; char *ForceNaturalAlignment; } TXTRecordRef; - - -/* TXTRecordCreate() - * - * Creates a new empty TXTRecordRef referencing the specified storage. - * - * If the buffer parameter is NULL, or the specified storage size is not - * large enough to hold a key subsequently added using TXTRecordSetValue(), - * then additional memory will be added as needed using malloc(). - * - * On some platforms, when memory is low, malloc() may fail. In this - * case, TXTRecordSetValue() will return kDNSServiceErr_NoMemory, and this - * error condition will need to be handled as appropriate by the caller. - * - * You can avoid the need to handle this error condition if you ensure - * that the storage you initially provide is large enough to hold all - * the key/value pairs that are to be added to the record. - * The caller can precompute the exact length required for all of the - * key/value pairs to be added, or simply provide a fixed-sized buffer - * known in advance to be large enough. - * A no-value (key-only) key requires (1 + key length) bytes. - * A key with empty value requires (1 + key length + 1) bytes. - * A key with non-empty value requires (1 + key length + 1 + value length). - * For most applications, DNS-SD TXT records are generally - * less than 100 bytes, so in most cases a simple fixed-sized - * 256-byte buffer will be more than sufficient. - * Recommended size limits for DNS-SD TXT Records are discussed in - * - * - * Note: When passing parameters to and from these TXT record APIs, - * the key name does not include the '=' character. The '=' character - * is the separator between the key and value in the on-the-wire - * packet format; it is not part of either the key or the value. - * - * txtRecord: A pointer to an uninitialized TXTRecordRef. - * - * bufferLen: The size of the storage provided in the "buffer" parameter. - * - * buffer: Optional caller-supplied storage used to hold the TXTRecord data. - * This storage must remain valid for as long as - * the TXTRecordRef. - */ - -void DNSSD_API TXTRecordCreate - ( - TXTRecordRef *txtRecord, - uint16_t bufferLen, - void *buffer - ); - - -/* TXTRecordDeallocate() - * - * Releases any resources allocated in the course of preparing a TXT Record - * using TXTRecordCreate()/TXTRecordSetValue()/TXTRecordRemoveValue(). - * Ownership of the buffer provided in TXTRecordCreate() returns to the client. - * - * txtRecord: A TXTRecordRef initialized by calling TXTRecordCreate(). - * - */ - -void DNSSD_API TXTRecordDeallocate - ( - TXTRecordRef *txtRecord - ); - - -/* TXTRecordSetValue() - * - * Adds a key (optionally with value) to a TXTRecordRef. If the "key" already - * exists in the TXTRecordRef, then the current value will be replaced with - * the new value. - * Keys may exist in four states with respect to a given TXT record: - * - Absent (key does not appear at all) - * - Present with no value ("key" appears alone) - * - Present with empty value ("key=" appears in TXT record) - * - Present with non-empty value ("key=value" appears in TXT record) - * For more details refer to "Data Syntax for DNS-SD TXT Records" in - * - * - * txtRecord: A TXTRecordRef initialized by calling TXTRecordCreate(). - * - * key: A null-terminated string which only contains printable ASCII - * values (0x20-0x7E), excluding '=' (0x3D). Keys should be - * 9 characters or fewer (not counting the terminating null). - * - * valueSize: The size of the value. - * - * value: Any binary value. For values that represent - * textual data, UTF-8 is STRONGLY recommended. - * For values that represent textual data, valueSize - * should NOT include the terminating null (if any) - * at the end of the string. - * If NULL, then "key" will be added with no value. - * If non-NULL but valueSize is zero, then "key=" will be - * added with empty value. - * - * return value: Returns kDNSServiceErr_NoError on success. - * Returns kDNSServiceErr_Invalid if the "key" string contains - * illegal characters. - * Returns kDNSServiceErr_NoMemory if adding this key would - * exceed the available storage. - */ - -DNSServiceErrorType DNSSD_API TXTRecordSetValue - ( - TXTRecordRef *txtRecord, - const char *key, - uint8_t valueSize, /* may be zero */ - const void *value /* may be NULL */ - ); - - -/* TXTRecordRemoveValue() - * - * Removes a key from a TXTRecordRef. The "key" must be an - * ASCII string which exists in the TXTRecordRef. - * - * txtRecord: A TXTRecordRef initialized by calling TXTRecordCreate(). - * - * key: A key name which exists in the TXTRecordRef. - * - * return value: Returns kDNSServiceErr_NoError on success. - * Returns kDNSServiceErr_NoSuchKey if the "key" does not - * exist in the TXTRecordRef. - */ - -DNSServiceErrorType DNSSD_API TXTRecordRemoveValue - ( - TXTRecordRef *txtRecord, - const char *key - ); - - -/* TXTRecordGetLength() - * - * Allows you to determine the length of the raw bytes within a TXTRecordRef. - * - * txtRecord: A TXTRecordRef initialized by calling TXTRecordCreate(). - * - * return value: Returns the size of the raw bytes inside a TXTRecordRef - * which you can pass directly to DNSServiceRegister() or - * to DNSServiceUpdateRecord(). - * Returns 0 if the TXTRecordRef is empty. - */ - -uint16_t DNSSD_API TXTRecordGetLength - ( - const TXTRecordRef *txtRecord - ); - - -/* TXTRecordGetBytesPtr() - * - * Allows you to retrieve a pointer to the raw bytes within a TXTRecordRef. - * - * txtRecord: A TXTRecordRef initialized by calling TXTRecordCreate(). - * - * return value: Returns a pointer to the raw bytes inside the TXTRecordRef - * which you can pass directly to DNSServiceRegister() or - * to DNSServiceUpdateRecord(). - */ - -const void * DNSSD_API TXTRecordGetBytesPtr - ( - const TXTRecordRef *txtRecord - ); - - -/********************************************************************************************* - * - * TXT Record Parsing Functions - * - *********************************************************************************************/ - -/* - * A typical calling sequence for TXT record parsing is something like: - * - * Receive TXT record data in DNSServiceResolve() callback - * if (TXTRecordContainsKey(txtLen, txtRecord, "key")) then do something - * val1ptr = TXTRecordGetValuePtr(txtLen, txtRecord, "key1", &len1); - * val2ptr = TXTRecordGetValuePtr(txtLen, txtRecord, "key2", &len2); - * ... - * memcpy(myval1, val1ptr, len1); - * memcpy(myval2, val2ptr, len2); - * ... - * return; - * - * If you wish to retain the values after return from the DNSServiceResolve() - * callback, then you need to copy the data to your own storage using memcpy() - * or similar, as shown in the example above. - * - * If for some reason you need to parse a TXT record you built yourself - * using the TXT record construction functions above, then you can do - * that using TXTRecordGetLength and TXTRecordGetBytesPtr calls: - * TXTRecordGetValue(TXTRecordGetLength(x), TXTRecordGetBytesPtr(x), key, &len); - * - * Most applications only fetch keys they know about from a TXT record and - * ignore the rest. - * However, some debugging tools wish to fetch and display all keys. - * To do that, use the TXTRecordGetCount() and TXTRecordGetItemAtIndex() calls. - */ - -/* TXTRecordContainsKey() - * - * Allows you to determine if a given TXT Record contains a specified key. - * - * txtLen: The size of the received TXT Record. - * - * txtRecord: Pointer to the received TXT Record bytes. - * - * key: A null-terminated ASCII string containing the key name. - * - * return value: Returns 1 if the TXT Record contains the specified key. - * Otherwise, it returns 0. - */ - -int DNSSD_API TXTRecordContainsKey - ( - uint16_t txtLen, - const void *txtRecord, - const char *key - ); - - -/* TXTRecordGetValuePtr() - * - * Allows you to retrieve the value for a given key from a TXT Record. - * - * txtLen: The size of the received TXT Record - * - * txtRecord: Pointer to the received TXT Record bytes. - * - * key: A null-terminated ASCII string containing the key name. - * - * valueLen: On output, will be set to the size of the "value" data. - * - * return value: Returns NULL if the key does not exist in this TXT record, - * or exists with no value (to differentiate between - * these two cases use TXTRecordContainsKey()). - * Returns pointer to location within TXT Record bytes - * if the key exists with empty or non-empty value. - * For empty value, valueLen will be zero. - * For non-empty value, valueLen will be length of value data. - */ - -const void * DNSSD_API TXTRecordGetValuePtr - ( - uint16_t txtLen, - const void *txtRecord, - const char *key, - uint8_t *valueLen - ); - - -/* TXTRecordGetCount() - * - * Returns the number of keys stored in the TXT Record. The count - * can be used with TXTRecordGetItemAtIndex() to iterate through the keys. - * - * txtLen: The size of the received TXT Record. - * - * txtRecord: Pointer to the received TXT Record bytes. - * - * return value: Returns the total number of keys in the TXT Record. - * - */ - -uint16_t DNSSD_API TXTRecordGetCount - ( - uint16_t txtLen, - const void *txtRecord - ); - - -/* TXTRecordGetItemAtIndex() - * - * Allows you to retrieve a key name and value pointer, given an index into - * a TXT Record. Legal index values range from zero to TXTRecordGetCount()-1. - * It's also possible to iterate through keys in a TXT record by simply - * calling TXTRecordGetItemAtIndex() repeatedly, beginning with index zero - * and increasing until TXTRecordGetItemAtIndex() returns kDNSServiceErr_Invalid. - * - * On return: - * For keys with no value, *value is set to NULL and *valueLen is zero. - * For keys with empty value, *value is non-NULL and *valueLen is zero. - * For keys with non-empty value, *value is non-NULL and *valueLen is non-zero. - * - * txtLen: The size of the received TXT Record. - * - * txtRecord: Pointer to the received TXT Record bytes. - * - * itemIndex: An index into the TXT Record. - * - * keyBufLen: The size of the string buffer being supplied. - * - * key: A string buffer used to store the key name. - * On return, the buffer contains a null-terminated C string - * giving the key name. DNS-SD TXT keys are usually - * 9 characters or fewer. To hold the maximum possible - * key name, the buffer should be 256 bytes long. - * - * valueLen: On output, will be set to the size of the "value" data. - * - * value: On output, *value is set to point to location within TXT - * Record bytes that holds the value data. - * - * return value: Returns kDNSServiceErr_NoError on success. - * Returns kDNSServiceErr_NoMemory if keyBufLen is too short. - * Returns kDNSServiceErr_Invalid if index is greater than - * TXTRecordGetCount()-1. - */ - -DNSServiceErrorType DNSSD_API TXTRecordGetItemAtIndex - ( - uint16_t txtLen, - const void *txtRecord, - uint16_t itemIndex, - uint16_t keyBufLen, - char *key, - uint8_t *valueLen, - const void **value - ); - -#if _DNS_SD_LIBDISPATCH -/* -* DNSServiceSetDispatchQueue -* -* Allows you to schedule a DNSServiceRef on a serial dispatch queue for receiving asynchronous -* callbacks. It's the clients responsibility to ensure that the provided dispatch queue is running. -* -* A typical application that uses CFRunLoopRun or dispatch_main on its main thread will -* usually schedule DNSServiceRefs on its main queue (which is always a serial queue) -* using "DNSServiceSetDispatchQueue(sdref, dispatch_get_main_queue());" -* -* If there is any error during the processing of events, the application callback will -* be called with an error code. For shared connections, each subordinate DNSServiceRef -* will get its own error callback. Currently these error callbacks only happen -* if the mDNSResponder daemon is manually terminated or crashes, and the error -* code in this case is kDNSServiceErr_ServiceNotRunning. The application must call -* DNSServiceRefDeallocate to free the DNSServiceRef when it gets such an error code. -* These error callbacks are rare and should not normally happen on customer machines, -* but application code should be written defensively to handle such error callbacks -* gracefully if they occur. -* -* After using DNSServiceSetDispatchQueue on a DNSServiceRef, calling DNSServiceProcessResult -* on the same DNSServiceRef will result in undefined behavior and should be avoided. -* -* Once the application successfully schedules a DNSServiceRef on a serial dispatch queue using -* DNSServiceSetDispatchQueue, it cannot remove the DNSServiceRef from the dispatch queue, or use -* DNSServiceSetDispatchQueue a second time to schedule the DNSServiceRef onto a different serial dispatch -* queue. Once scheduled onto a dispatch queue a DNSServiceRef will deliver events to that queue until -* the application no longer requires that operation and terminates it using DNSServiceRefDeallocate. -* -* service: DNSServiceRef that was allocated and returned to the application, when the -* application calls one of the DNSService API. -* -* queue: dispatch queue where the application callback will be scheduled -* -* return value: Returns kDNSServiceErr_NoError on success. -* Returns kDNSServiceErr_NoMemory if it cannot create a dispatch source -* Returns kDNSServiceErr_BadParam if the service param is invalid or the -* queue param is invalid -*/ - -DNSServiceErrorType DNSSD_API DNSServiceSetDispatchQueue - ( - DNSServiceRef service, - dispatch_queue_t queue - ); -#endif //_DNS_SD_LIBDISPATCH - -#ifdef __APPLE_API_PRIVATE - -#define kDNSServiceCompPrivateDNS "PrivateDNS" -#define kDNSServiceCompMulticastDNS "MulticastDNS" - -#endif //__APPLE_API_PRIVATE - -/* Some C compiler cleverness. We can make the compiler check certain things for us, - * and report errors at compile-time if anything is wrong. The usual way to do this would - * be to use a run-time "if" statement or the conventional run-time "assert" mechanism, but - * then you don't find out what's wrong until you run the software. This way, if the assertion - * condition is false, the array size is negative, and the complier complains immediately. - */ - -struct CompileTimeAssertionChecks_DNS_SD - { - char assert0[(sizeof(union _TXTRecordRef_t) == 16) ? 1 : -1]; - }; - -#ifdef __cplusplus - } -#endif - -#endif /* _DNS_SD_H */ diff -Nru qtcreator-2.5.0/src/tools/mdnssd/dnssd_ipc.c qtcreator-2.5.2/src/tools/mdnssd/dnssd_ipc.c --- qtcreator-2.5.0/src/tools/mdnssd/dnssd_ipc.c 2012-05-09 12:13:19.000000000 +0000 +++ qtcreator-2.5.2/src/tools/mdnssd/dnssd_ipc.c 1970-01-01 00:00:00.000000000 +0000 @@ -1,161 +0,0 @@ -/* -*- Mode: C; tab-width: 4 -*- - * - * Copyright (c) 2003-2004, Apple Computer, Inc. All rights reserved. - * - * Redistribution and use in source and binary forms, with or without - * modification, are permitted provided that the following conditions are met: - * - * 1. Redistributions of source code must retain the above copyright notice, - * this list of conditions and the following disclaimer. - * 2. Redistributions in binary form must reproduce the above copyright notice, - * this list of conditions and the following disclaimer in the documentation - * and/or other materials provided with the distribution. - * 3. Neither the name of Apple Computer, Inc. ("Apple") nor the names of its - * contributors may be used to endorse or promote products derived from this - * software without specific prior written permission. - * - * THIS SOFTWARE IS PROVIDED BY APPLE AND ITS CONTRIBUTORS "AS IS" AND ANY - * EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED - * WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE - * DISCLAIMED. IN NO EVENT SHALL APPLE OR ITS CONTRIBUTORS BE LIABLE FOR ANY - * DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES - * (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; - * LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND - * ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT - * (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS - * SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE. - */ - -#include "dnssd_ipc.h" - -#if defined(_WIN32) - -char *win32_strerror(int inErrorCode) - { - static char buffer[1024]; - DWORD n; - memset(buffer, 0, sizeof(buffer)); - n = FormatMessageA( - FORMAT_MESSAGE_FROM_SYSTEM | FORMAT_MESSAGE_IGNORE_INSERTS, - NULL, - (DWORD) inErrorCode, - MAKELANGID(LANG_NEUTRAL, SUBLANG_DEFAULT), - buffer, - sizeof(buffer), - NULL); - if (n > 0) - { - // Remove any trailing CR's or LF's since some messages have them. - while ((n > 0) && isspace(((unsigned char *) buffer)[n - 1])) - buffer[--n] = '\0'; - } - return buffer; - } - -#endif - -void put_uint32(const uint32_t l, char **ptr) - { - (*ptr)[0] = (char)((l >> 24) & 0xFF); - (*ptr)[1] = (char)((l >> 16) & 0xFF); - (*ptr)[2] = (char)((l >> 8) & 0xFF); - (*ptr)[3] = (char)((l ) & 0xFF); - *ptr += sizeof(uint32_t); - } - -uint32_t get_uint32(const char **ptr, const char *end) - { - if (!*ptr || *ptr + sizeof(uint32_t) > end) - { - *ptr = NULL; - return(0); - } - else - { - uint8_t *p = (uint8_t*) *ptr; - *ptr += sizeof(uint32_t); - return((uint32_t) ((uint32_t)p[0] << 24 | (uint32_t)p[1] << 16 | (uint32_t)p[2] << 8 | p[3])); - } - } - -void put_uint16(uint16_t s, char **ptr) - { - (*ptr)[0] = (char)((s >> 8) & 0xFF); - (*ptr)[1] = (char)((s ) & 0xFF); - *ptr += sizeof(uint16_t); - } - -uint16_t get_uint16(const char **ptr, const char *end) - { - if (!*ptr || *ptr + sizeof(uint16_t) > end) - { - *ptr = NULL; - return(0); - } - else - { - uint8_t *p = (uint8_t*) *ptr; - *ptr += sizeof(uint16_t); - return((uint16_t) ((uint16_t)p[0] << 8 | p[1])); - } - } - -int put_string(const char *str, char **ptr) - { - if (!str) str = ""; - strcpy(*ptr, str); - *ptr += strlen(str) + 1; - return 0; - } - -int get_string(const char **ptr, const char *const end, char *buffer, int buflen) - { - if (!*ptr) - { - *buffer = 0; - return(-1); - } - else - { - char *lim = buffer + buflen; // Calculate limit - while (*ptr < end && buffer < lim) - { - char c = *buffer++ = *(*ptr)++; - if (c == 0) return(0); // Success - } - if (buffer == lim) buffer--; - *buffer = 0; // Failed, so terminate string, - *ptr = NULL; // clear pointer, - return(-1); // and return failure indication - } - } - -void put_rdata(const int rdlen, const unsigned char *rdata, char **ptr) - { - memcpy(*ptr, rdata, rdlen); - *ptr += rdlen; - } - -const char *get_rdata(const char **ptr, const char *end, int rdlen) - { - if (!*ptr || *ptr + rdlen > end) - { - *ptr = NULL; - return(0); - } - else - { - const char *rd = *ptr; - *ptr += rdlen; - return rd; - } - } - -void ConvertHeaderBytes(ipc_msg_hdr *hdr) - { - hdr->version = htonl(hdr->version); - hdr->datalen = htonl(hdr->datalen); - hdr->ipc_flags = htonl(hdr->ipc_flags); - hdr->op = htonl(hdr->op ); - hdr->reg_index = htonl(hdr->reg_index); - } diff -Nru qtcreator-2.5.0/src/tools/mdnssd/dnssd_ipc.h qtcreator-2.5.2/src/tools/mdnssd/dnssd_ipc.h --- qtcreator-2.5.0/src/tools/mdnssd/dnssd_ipc.h 2012-05-09 12:13:19.000000000 +0000 +++ qtcreator-2.5.2/src/tools/mdnssd/dnssd_ipc.h 1970-01-01 00:00:00.000000000 +0000 @@ -1,217 +0,0 @@ -/* -*- Mode: C; tab-width: 4 -*- - * - * Copyright (c) 2003-2004, Apple Computer, Inc. All rights reserved. - * - * Redistribution and use in source and binary forms, with or without - * modification, are permitted provided that the following conditions are met: - * - * 1. Redistributions of source code must retain the above copyright notice, - * this list of conditions and the following disclaimer. - * 2. Redistributions in binary form must reproduce the above copyright notice, - * this list of conditions and the following disclaimer in the documentation - * and/or other materials provided with the distribution. - * 3. Neither the name of Apple Computer, Inc. ("Apple") nor the names of its - * contributors may be used to endorse or promote products derived from this - * software without specific prior written permission. - * - * THIS SOFTWARE IS PROVIDED BY APPLE AND ITS CONTRIBUTORS "AS IS" AND ANY - * EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED - * WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE - * DISCLAIMED. IN NO EVENT SHALL APPLE OR ITS CONTRIBUTORS BE LIABLE FOR ANY - * DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES - * (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; - * LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND - * ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT - * (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS - * SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE. - */ - -#ifndef DNSSD_IPC_H -#define DNSSD_IPC_H - -#include "dns_sd.h" - -// -// Common cross platform services -// -#if defined(WIN32) -# include -# define dnssd_InvalidSocket INVALID_SOCKET -# define dnssd_SocketValid(s) ((s) != INVALID_SOCKET) -# define dnssd_EWOULDBLOCK WSAEWOULDBLOCK -# define dnssd_EINTR WSAEINTR -# define dnssd_ECONNRESET WSAECONNRESET -# define dnssd_sock_t SOCKET -# define dnssd_socklen_t int -# define dnssd_close(sock) closesocket(sock) -# define dnssd_errno WSAGetLastError() -# define dnssd_strerror(X) win32_strerror(X) -# define ssize_t int -# define getpid _getpid -# define unlink _unlink -extern char *win32_strerror(int inErrorCode); -#else -# include -# include -# include -# include -# include -# include -# include -# include -# include -# define dnssd_InvalidSocket -1 -# define dnssd_SocketValid(s) ((s) >= 0) -# define dnssd_EWOULDBLOCK EWOULDBLOCK -# define dnssd_EINTR EINTR -# define dnssd_ECONNRESET ECONNRESET -# define dnssd_EPIPE EPIPE -# define dnssd_sock_t int -# define dnssd_socklen_t unsigned int -# define dnssd_close(sock) close(sock) -# define dnssd_errno errno -# define dnssd_strerror(X) strerror(X) -#endif - -#if defined(USE_TCP_LOOPBACK) -# define AF_DNSSD AF_INET -# define MDNS_TCP_SERVERADDR "127.0.0.1" -# define MDNS_TCP_SERVERPORT 5354 -# define LISTENQ 5 -# define dnssd_sockaddr_t struct sockaddr_in -#else -# define AF_DNSSD AF_LOCAL -# ifndef MDNS_UDS_SERVERPATH -# define MDNS_UDS_SERVERPATH "/var/run/mDNSResponder" -# endif -# define LISTENQ 100 - // longest legal control path length -# define MAX_CTLPATH 256 -# define dnssd_sockaddr_t struct sockaddr_un -#endif - -// Compatibility workaround -#ifndef AF_LOCAL -#define AF_LOCAL AF_UNIX -#endif - -// General UDS constants -#define TXT_RECORD_INDEX ((uint32_t)(-1)) // record index for default text record - -// IPC data encoding constants and types -#define VERSION 1 -#define IPC_FLAGS_NOREPLY 1 // set flag if no asynchronous replies are to be sent to client - -// Structure packing macro. If we're not using GNUC, it's not fatal. Most compilers naturally pack the on-the-wire -// structures correctly anyway, so a plain "struct" is usually fine. In the event that structures are not packed -// correctly, our compile-time assertion checks will catch it and prevent inadvertent generation of non-working code. -#ifndef packedstruct - #if ((__GNUC__ > 2) || ((__GNUC__ == 2) && (__GNUC_MINOR__ >= 9))) - #define packedstruct struct __attribute__((__packed__)) - #define packedunion union __attribute__((__packed__)) - #else - #define packedstruct struct - #define packedunion union - #endif -#endif - -typedef enum - { - request_op_none = 0, // No request yet received on this connection - connection_request = 1, // connected socket via DNSServiceConnect() - reg_record_request, // reg/remove record only valid for connected sockets - remove_record_request, - enumeration_request, - reg_service_request, - browse_request, - resolve_request, - query_request, - reconfirm_record_request, - add_record_request, - update_record_request, - setdomain_request, // Up to here is in Tiger and B4W 1.0.3 - getproperty_request, // New in B4W 1.0.4 - port_mapping_request, // New in Leopard and B4W 2.0 - addrinfo_request, - send_bpf, // New in SL - - cancel_request = 63 - } request_op_t; - -typedef enum - { - enumeration_reply_op = 64, - reg_service_reply_op, - browse_reply_op, - resolve_reply_op, - query_reply_op, - reg_record_reply_op, // Up to here is in Tiger and B4W 1.0.3 - getproperty_reply_op, // New in B4W 1.0.4 - port_mapping_reply_op, // New in Leopard and B4W 2.0 - addrinfo_reply_op - } reply_op_t; - -#if defined(_WIN64) -# pragma pack(push,4) -#endif - -// Define context object big enough to hold a 64-bit pointer, -// to accomodate 64-bit clients communicating with 32-bit daemon. -// There's no reason for the daemon to ever be a 64-bit process, but its clients might be -typedef packedunion - { - void *context; - uint32_t u32[2]; - } client_context_t; - -typedef packedstruct - { - uint32_t version; - uint32_t datalen; - uint32_t ipc_flags; - uint32_t op; // request_op_t or reply_op_t - client_context_t client_context; // context passed from client, returned by server in corresponding reply - uint32_t reg_index; // identifier for a record registered via DNSServiceRegisterRecord() on a - // socket connected by DNSServiceCreateConnection(). Must be unique in the scope of the connection, such that and - // index/socket pair uniquely identifies a record. (Used to select records for removal by DNSServiceRemoveRecord()) - } ipc_msg_hdr; - -#if defined(_WIN64) -# pragma pack(pop) -#endif - -// routines to write to and extract data from message buffers. -// caller responsible for bounds checking. -// ptr is the address of the pointer to the start of the field. -// it is advanced to point to the next field, or the end of the message - -void put_uint32(const uint32_t l, char **ptr); -uint32_t get_uint32(const char **ptr, const char *end); - -void put_uint16(uint16_t s, char **ptr); -uint16_t get_uint16(const char **ptr, const char *end); - -#define put_flags put_uint32 -#define get_flags get_uint32 - -#define put_error_code put_uint32 -#define get_error_code get_uint32 - -int put_string(const char *str, char **ptr); -int get_string(const char **ptr, const char *const end, char *buffer, int buflen); - -void put_rdata(const int rdlen, const unsigned char *rdata, char **ptr); -const char *get_rdata(const char **ptr, const char *end, int rdlen); // return value is rdata pointed to by *ptr - - // rdata is not copied from buffer. - -void ConvertHeaderBytes(ipc_msg_hdr *hdr); - -struct CompileTimeAssertionChecks_dnssd_ipc - { - // Check that the compiler generated our on-the-wire packet format structure definitions - // properly packed, without adding padding bytes to align fields on 32-bit or 64-bit boundaries. - char assert0[(sizeof(client_context_t) == 8) ? 1 : -1]; - char assert1[(sizeof(ipc_msg_hdr) == 28) ? 1 : -1]; - }; - -#endif // DNSSD_IPC_H diff -Nru qtcreator-2.5.0/src/tools/mdnssd/mDNS.c qtcreator-2.5.2/src/tools/mdnssd/mDNS.c --- qtcreator-2.5.0/src/tools/mdnssd/mDNS.c 2012-05-09 12:13:19.000000000 +0000 +++ qtcreator-2.5.2/src/tools/mdnssd/mDNS.c 1970-01-01 00:00:00.000000000 +0000 @@ -1,11521 +0,0 @@ -/* -*- Mode: C; tab-width: 4 -*- - * - * Copyright (c) 2002-2006 Apple Computer, Inc. All rights reserved. - * - * Licensed under the Apache License, Version 2.0 (the "License"); - * you may not use this file except in compliance with the License. - * You may obtain a copy of the License at - * - * http://www.apache.org/licenses/LICENSE-2.0 - * - * Unless required by applicable law or agreed to in writing, software - * distributed under the License is distributed on an "AS IS" BASIS, - * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. - * See the License for the specific language governing permissions and - * limitations under the License. - * - * This code is completely 100% portable C. It does not depend on any external header files - * from outside the mDNS project -- all the types it expects to find are defined right here. - * - * The previous point is very important: This file does not depend on any external - * header files. It should compile on *any* platform that has a C compiler, without - * making *any* assumptions about availability of so-called "standard" C functions, - * routines, or types (which may or may not be present on any given platform). - - * Formatting notes: - * This code follows the "Whitesmiths style" C indentation rules. Plenty of discussion - * on C indentation can be found on the web, such as , - * but for the sake of brevity here I will say just this: Curly braces are not syntactially - * part of an "if" statement; they are the beginning and ending markers of a compound statement; - * therefore common sense dictates that if they are part of a compound statement then they - * should be indented to the same level as everything else in that compound statement. - * Indenting curly braces at the same level as the "if" implies that curly braces are - * part of the "if", which is false. (This is as misleading as people who write "char* x,y;" - * thinking that variables x and y are both of type "char*" -- and anyone who doesn't - * understand why variable y is not of type "char*" just proves the point that poor code - * layout leads people to unfortunate misunderstandings about how the C language really works.) - */ - -#include "DNSCommon.h" // Defines general DNS untility routines -#include "uDNS.h" // Defines entry points into unicast-specific routines - -// Disable certain benign warnings with Microsoft compilers -#if(defined(_MSC_VER)) - // Disable "conditional expression is constant" warning for debug macros. - // Otherwise, this generates warnings for the perfectly natural construct "while(1)" - // If someone knows a variant way of writing "while(1)" that doesn't generate warning messages, please let us know - #pragma warning(disable:4127) - - // Disable "assignment within conditional expression". - // Other compilers understand the convention that if you place the assignment expression within an extra pair - // of parentheses, this signals to the compiler that you really intended an assignment and no warning is necessary. - // The Microsoft compiler doesn't understand this convention, so in the absense of any other way to signal - // to the compiler that the assignment is intentional, we have to just turn this warning off completely. - #pragma warning(disable:4706) -#endif - -#if APPLE_OSX_mDNSResponder - -#include - -#if ! NO_WCF -WCFConnection *WCFConnectionNew(void) __attribute__((weak_import)); -void WCFConnectionDealloc(WCFConnection* c) __attribute__((weak_import)); - -// Do we really need to define a macro for "if"? -#define CHECK_WCF_FUNCTION(X) if (X) -#endif // ! NO_WCF - -#else - -#define NO_WCF 1 -#endif // APPLE_OSX_mDNSResponder - -// Forward declarations -mDNSlocal void BeginSleepProcessing(mDNS *const m); -mDNSlocal void RetrySPSRegistrations(mDNS *const m); -mDNSlocal void SendWakeup(mDNS *const m, mDNSInterfaceID InterfaceID, mDNSEthAddr *EthAddr, mDNSOpaque48 *password); -mDNSlocal mDNSBool CacheRecordRmvEventsForQuestion(mDNS *const m, DNSQuestion *q); -mDNSlocal mDNSBool LocalRecordRmvEventsForQuestion(mDNS *const m, DNSQuestion *q); -mDNSlocal void mDNS_PurgeBeforeResolve(mDNS *const m, DNSQuestion *q); - -// *************************************************************************** -#if COMPILER_LIKES_PRAGMA_MARK -#pragma mark - Program Constants -#endif - -#define NO_HINFO 1 - - -// Any records bigger than this are considered 'large' records -#define SmallRecordLimit 1024 - -#define kMaxUpdateCredits 10 -#define kUpdateCreditRefreshInterval (mDNSPlatformOneSecond * 6) - -mDNSexport const char *const mDNS_DomainTypeNames[] = - { - "b._dns-sd._udp.", // Browse - "db._dns-sd._udp.", // Default Browse - "lb._dns-sd._udp.", // Automatic Browse - "r._dns-sd._udp.", // Registration - "dr._dns-sd._udp." // Default Registration - }; - -#ifdef UNICAST_DISABLED -#define uDNS_IsActiveQuery(q, u) mDNSfalse -#endif - -// *************************************************************************** -#if COMPILER_LIKES_PRAGMA_MARK -#pragma mark - -#pragma mark - General Utility Functions -#endif - -// If there is a authoritative LocalOnly record that answers questions of type A, AAAA and CNAME -// this returns true. Main use is to handle /etc/hosts records. -#define LORecordAnswersAddressType(rr) ((rr)->ARType == AuthRecordLocalOnly && \ - (rr)->resrec.RecordType & kDNSRecordTypeUniqueMask && \ - ((rr)->resrec.rrtype == kDNSType_A || (rr)->resrec.rrtype == kDNSType_AAAA || \ - (rr)->resrec.rrtype == kDNSType_CNAME)) - -#define FollowCNAME(q, rr, AddRecord) (AddRecord && (q)->qtype != kDNSType_CNAME && \ - (rr)->RecordType != kDNSRecordTypePacketNegative && \ - (rr)->rrtype == kDNSType_CNAME) - -mDNSlocal void SetNextQueryStopTime(mDNS *const m, const DNSQuestion *const q) - { - if (m->mDNS_busy != m->mDNS_reentrancy+1) - LogMsg("SetNextQueryTime: Lock not held! mDNS_busy (%ld) mDNS_reentrancy (%ld)", m->mDNS_busy, m->mDNS_reentrancy); - -#if ForceAlerts - if (m->mDNS_busy != m->mDNS_reentrancy+1) *(long*)0 = 0; -#endif - - if (m->NextScheduledStopTime - q->StopTime > 0) - m->NextScheduledStopTime = q->StopTime; - } - -mDNSexport void SetNextQueryTime(mDNS *const m, const DNSQuestion *const q) - { - if (m->mDNS_busy != m->mDNS_reentrancy+1) - LogMsg("SetNextQueryTime: Lock not held! mDNS_busy (%ld) mDNS_reentrancy (%ld)", m->mDNS_busy, m->mDNS_reentrancy); - -#if ForceAlerts - if (m->mDNS_busy != m->mDNS_reentrancy+1) *(long*)0 = 0; -#endif - - if (ActiveQuestion(q)) - { - // Depending on whether this is a multicast or unicast question we want to set either: - // m->NextScheduledQuery = NextQSendTime(q) or - // m->NextuDNSEvent = NextQSendTime(q) - mDNSs32 *const timer = mDNSOpaque16IsZero(q->TargetQID) ? &m->NextScheduledQuery : &m->NextuDNSEvent; - if (*timer - NextQSendTime(q) > 0) - *timer = NextQSendTime(q); - } - } - -mDNSlocal void ReleaseAuthEntity(AuthHash *r, AuthEntity *e) - { -#if APPLE_OSX_mDNSResponder && MACOSX_MDNS_MALLOC_DEBUGGING >= 1 - unsigned int i; - for (i=0; inext = r->rrauth_free; - r->rrauth_free = e; - r->rrauth_totalused--; - } - -mDNSlocal void ReleaseAuthGroup(AuthHash *r, AuthGroup **cp) - { - AuthEntity *e = (AuthEntity *)(*cp); - LogMsg("ReleaseAuthGroup: Releasing AuthGroup %##s", (*cp)->name->c); - if ((*cp)->rrauth_tail != &(*cp)->members) - LogMsg("ERROR: (*cp)->members == mDNSNULL but (*cp)->rrauth_tail != &(*cp)->members)"); - if ((*cp)->name != (domainname*)((*cp)->namestorage)) mDNSPlatformMemFree((*cp)->name); - (*cp)->name = mDNSNULL; - *cp = (*cp)->next; // Cut record from list - ReleaseAuthEntity(r, e); - } - -mDNSlocal AuthEntity *GetAuthEntity(AuthHash *r, const AuthGroup *const PreserveAG) - { - AuthEntity *e = mDNSNULL; - - if (r->rrauth_lock) { LogMsg("GetFreeCacheRR ERROR! Cache already locked!"); return(mDNSNULL); } - r->rrauth_lock = 1; - - if (!r->rrauth_free) - { - // We allocate just one AuthEntity at a time because we need to be able - // free them all individually which normally happens when we parse /etc/hosts into - // AuthHash where we add the "new" entries and discard (free) the already added - // entries. If we allocate as chunks, we can't free them individually. - AuthEntity *storage = mDNSPlatformMemAllocate(sizeof(AuthEntity)); - storage->next = mDNSNULL; - r->rrauth_free = storage; - } - - // If we still have no free records, recycle all the records we can. - // Enumerating the entire auth is moderately expensive, so when we do it, we reclaim all the records we can in one pass. - if (!r->rrauth_free) - { - mDNSu32 oldtotalused = r->rrauth_totalused; - mDNSu32 slot; - for (slot = 0; slot < AUTH_HASH_SLOTS; slot++) - { - AuthGroup **cp = &r->rrauth_hash[slot]; - while (*cp) - { - if ((*cp)->members || (*cp)==PreserveAG) cp=&(*cp)->next; - else ReleaseAuthGroup(r, cp); - } - } - LogInfo("GetAuthEntity: Recycled %d records to reduce auth cache from %d to %d", - oldtotalused - r->rrauth_totalused, oldtotalused, r->rrauth_totalused); - } - - if (r->rrauth_free) // If there are records in the free list, take one - { - e = r->rrauth_free; - r->rrauth_free = e->next; - if (++r->rrauth_totalused >= r->rrauth_report) - { - LogInfo("RR Auth now using %ld objects", r->rrauth_totalused); - if (r->rrauth_report < 100) r->rrauth_report += 10; - else if (r->rrauth_report < 1000) r->rrauth_report += 100; - else r->rrauth_report += 1000; - } - mDNSPlatformMemZero(e, sizeof(*e)); - } - - r->rrauth_lock = 0; - - return(e); - } - -mDNSexport AuthGroup *AuthGroupForName(AuthHash *r, const mDNSu32 slot, const mDNSu32 namehash, const domainname *const name) - { - AuthGroup *ag; - for (ag = r->rrauth_hash[slot]; ag; ag=ag->next) - if (ag->namehash == namehash && SameDomainName(ag->name, name)) - break; - return(ag); - } - -mDNSexport AuthGroup *AuthGroupForRecord(AuthHash *r, const mDNSu32 slot, const ResourceRecord *const rr) - { - return(AuthGroupForName(r, slot, rr->namehash, rr->name)); - } - -mDNSlocal AuthGroup *GetAuthGroup(AuthHash *r, const mDNSu32 slot, const ResourceRecord *const rr) - { - mDNSu16 namelen = DomainNameLength(rr->name); - AuthGroup *ag = (AuthGroup*)GetAuthEntity(r, mDNSNULL); - if (!ag) { LogMsg("GetAuthGroup: Failed to allocate memory for %##s", rr->name->c); return(mDNSNULL); } - ag->next = r->rrauth_hash[slot]; - ag->namehash = rr->namehash; - ag->members = mDNSNULL; - ag->rrauth_tail = &ag->members; - ag->name = (domainname*)ag->namestorage; - ag->NewLocalOnlyRecords = mDNSNULL; - if (namelen > InlineCacheGroupNameSize) ag->name = mDNSPlatformMemAllocate(namelen); - if (!ag->name) - { - LogMsg("GetAuthGroup: Failed to allocate name storage for %##s", rr->name->c); - ReleaseAuthEntity(r, (AuthEntity*)ag); - return(mDNSNULL); - } - AssignDomainName(ag->name, rr->name); - - if (AuthGroupForRecord(r, slot, rr)) LogMsg("GetAuthGroup: Already have AuthGroup for %##s", rr->name->c); - r->rrauth_hash[slot] = ag; - if (AuthGroupForRecord(r, slot, rr) != ag) LogMsg("GetAuthGroup: Not finding AuthGroup for %##s", rr->name->c); - - return(ag); - } - -// Returns the AuthGroup in which the AuthRecord was inserted -mDNSexport AuthGroup *InsertAuthRecord(mDNS *const m, AuthHash *r, AuthRecord *rr) - { - AuthGroup *ag; - const mDNSu32 slot = AuthHashSlot(rr->resrec.name); - ag = AuthGroupForRecord(r, slot, &rr->resrec); - if (!ag) ag = GetAuthGroup(r, slot, &rr->resrec); // If we don't have a AuthGroup for this name, make one now - if (ag) - { - LogInfo("InsertAuthRecord: inserting auth record %s from table", ARDisplayString(m, rr)); - *(ag->rrauth_tail) = rr; // Append this record to tail of cache slot list - ag->rrauth_tail = &(rr->next); // Advance tail pointer - } - return ag; - } - -mDNSexport AuthGroup *RemoveAuthRecord(mDNS *const m, AuthHash *r, AuthRecord *rr) - { - AuthGroup *a; - AuthGroup **ag = &a; - AuthRecord **rp; - const mDNSu32 slot = AuthHashSlot(rr->resrec.name); - - a = AuthGroupForRecord(r, slot, &rr->resrec); - if (!a) { LogMsg("RemoveAuthRecord: ERROR!! AuthGroup not found for %s", ARDisplayString(m, rr)); return mDNSNULL; } - rp = &(*ag)->members; - while (*rp) - { - if (*rp != rr) - rp=&(*rp)->next; - else - { - // We don't break here, so that we can set the tail below without tracking "prev" pointers - - LogInfo("RemoveAuthRecord: removing auth record %s from table", ARDisplayString(m, rr)); - *rp = (*rp)->next; // Cut record from list - } - } - // TBD: If there are no more members, release authgroup ? - (*ag)->rrauth_tail = rp; - return a; - } - -mDNSexport CacheGroup *CacheGroupForName(const mDNS *const m, const mDNSu32 slot, const mDNSu32 namehash, const domainname *const name) - { - CacheGroup *cg; - for (cg = m->rrcache_hash[slot]; cg; cg=cg->next) - if (cg->namehash == namehash && SameDomainName(cg->name, name)) - break; - return(cg); - } - -mDNSlocal CacheGroup *CacheGroupForRecord(const mDNS *const m, const mDNSu32 slot, const ResourceRecord *const rr) - { - return(CacheGroupForName(m, slot, rr->namehash, rr->name)); - } - -mDNSexport mDNSBool mDNS_AddressIsLocalSubnet(mDNS *const m, const mDNSInterfaceID InterfaceID, const mDNSAddr *addr) - { - NetworkInterfaceInfo *intf; - - if (addr->type == mDNSAddrType_IPv4) - { - // Normally we resist touching the NotAnInteger fields, but here we're doing tricky bitwise masking so we make an exception - if (mDNSv4AddressIsLinkLocal(&addr->ip.v4)) return(mDNStrue); - for (intf = m->HostInterfaces; intf; intf = intf->next) - if (intf->ip.type == addr->type && intf->InterfaceID == InterfaceID && intf->McastTxRx) - if (((intf->ip.ip.v4.NotAnInteger ^ addr->ip.v4.NotAnInteger) & intf->mask.ip.v4.NotAnInteger) == 0) - return(mDNStrue); - } - - if (addr->type == mDNSAddrType_IPv6) - { - if (mDNSv6AddressIsLinkLocal(&addr->ip.v6)) return(mDNStrue); - for (intf = m->HostInterfaces; intf; intf = intf->next) - if (intf->ip.type == addr->type && intf->InterfaceID == InterfaceID && intf->McastTxRx) - if ((((intf->ip.ip.v6.l[0] ^ addr->ip.v6.l[0]) & intf->mask.ip.v6.l[0]) == 0) && - (((intf->ip.ip.v6.l[1] ^ addr->ip.v6.l[1]) & intf->mask.ip.v6.l[1]) == 0) && - (((intf->ip.ip.v6.l[2] ^ addr->ip.v6.l[2]) & intf->mask.ip.v6.l[2]) == 0) && - (((intf->ip.ip.v6.l[3] ^ addr->ip.v6.l[3]) & intf->mask.ip.v6.l[3]) == 0)) - return(mDNStrue); - } - - return(mDNSfalse); - } - -mDNSlocal NetworkInterfaceInfo *FirstInterfaceForID(mDNS *const m, const mDNSInterfaceID InterfaceID) - { - NetworkInterfaceInfo *intf = m->HostInterfaces; - while (intf && intf->InterfaceID != InterfaceID) intf = intf->next; - return(intf); - } - -mDNSexport char *InterfaceNameForID(mDNS *const m, const mDNSInterfaceID InterfaceID) - { - NetworkInterfaceInfo *intf = FirstInterfaceForID(m, InterfaceID); - return(intf ? intf->ifname : mDNSNULL); - } - -// Caller should hold the lock -mDNSlocal void GenerateNegativeResponse(mDNS *const m) - { - DNSQuestion *q; - if (!m->CurrentQuestion) { LogMsg("GenerateNegativeResponse: ERROR!! CurrentQuestion not set"); return; } - q = m->CurrentQuestion; - LogInfo("GenerateNegativeResponse: Generating negative response for question %##s (%s)", q->qname.c, DNSTypeName(q->qtype)); - - MakeNegativeCacheRecord(m, &m->rec.r, &q->qname, q->qnamehash, q->qtype, q->qclass, 60, mDNSInterface_Any, mDNSNULL); - AnswerCurrentQuestionWithResourceRecord(m, &m->rec.r, QC_addnocache); - if (m->CurrentQuestion == q) { q->ThisQInterval = 0; } // Deactivate this question - // Don't touch the question after this - m->rec.r.resrec.RecordType = 0; // Clear RecordType to show we're not still using it - } - -mDNSlocal void AnswerQuestionByFollowingCNAME(mDNS *const m, DNSQuestion *q, ResourceRecord *rr) - { - const mDNSBool selfref = SameDomainName(&q->qname, &rr->rdata->u.name); - if (q->CNAMEReferrals >= 10 || selfref) - LogMsg("AnswerQuestionByFollowingCNAME: %p %##s (%s) NOT following CNAME referral %d%s for %s", - q, q->qname.c, DNSTypeName(q->qtype), q->CNAMEReferrals, selfref ? " (Self-Referential)" : "", RRDisplayString(m, rr)); - else - { - const mDNSu32 c = q->CNAMEReferrals + 1; // Stash a copy of the new q->CNAMEReferrals value - - // The SameDomainName check above is to ignore bogus CNAME records that point right back at - // themselves. Without that check we can get into a case where we have two duplicate questions, - // A and B, and when we stop question A, UpdateQuestionDuplicates copies the value of CNAMEReferrals - // from A to B, and then A is re-appended to the end of the list as a duplicate of B (because - // the target name is still the same), and then when we stop question B, UpdateQuestionDuplicates - // copies the B's value of CNAMEReferrals back to A, and we end up not incrementing CNAMEReferrals - // for either of them. This is not a problem for CNAME loops of two or more records because in - // those cases the newly re-appended question A has a different target name and therefore cannot be - // a duplicate of any other question ('B') which was itself a duplicate of the previous question A. - - // Right now we just stop and re-use the existing query. If we really wanted to be 100% perfect, - // and track CNAMEs coming and going, we should really create a subordinate query here, - // which we would subsequently cancel and retract if the CNAME referral record were removed. - // In reality this is such a corner case we'll ignore it until someone actually needs it. - - LogInfo("AnswerQuestionByFollowingCNAME: %p %##s (%s) following CNAME referral %d for %s", - q, q->qname.c, DNSTypeName(q->qtype), q->CNAMEReferrals, RRDisplayString(m, rr)); - - mDNS_StopQuery_internal(m, q); // Stop old query - AssignDomainName(&q->qname, &rr->rdata->u.name); // Update qname - q->qnamehash = DomainNameHashValue(&q->qname); // and namehash - // If a unicast query results in a CNAME that points to a .local, we need to re-try - // this as unicast. Setting the mDNSInterface_Unicast tells mDNS_StartQuery_internal - // to try this as unicast query even though it is a .local name - if (!mDNSOpaque16IsZero(q->TargetQID) && IsLocalDomain(&q->qname)) - { - LogInfo("AnswerQuestionByFollowingCNAME: Resolving a .local CNAME %p %##s (%s) Record %s", - q, q->qname.c, DNSTypeName(q->qtype), RRDisplayString(m, rr)); - q->InterfaceID = mDNSInterface_Unicast; - } - mDNS_StartQuery_internal(m, q); // start new query - // Record how many times we've done this. We need to do this *after* mDNS_StartQuery_internal, - // because mDNS_StartQuery_internal re-initializes CNAMEReferrals to zero - q->CNAMEReferrals = c; - } - } - -// For a single given DNSQuestion pointed to by CurrentQuestion, deliver an add/remove result for the single given AuthRecord -// Note: All the callers should use the m->CurrentQuestion to see if the question is still valid or not -mDNSlocal void AnswerLocalQuestionWithLocalAuthRecord(mDNS *const m, AuthRecord *rr, QC_result AddRecord) - { - DNSQuestion *q = m->CurrentQuestion; - mDNSBool followcname; - - if (!q) - { - LogMsg("AnswerLocalQuestionWithLocalAuthRecord: ERROR!! CurrentQuestion NULL while answering with %s", ARDisplayString(m, rr)); - return; - } - - followcname = FollowCNAME(q, &rr->resrec, AddRecord); - - // We should not be delivering results for record types Unregistered, Deregistering, and (unverified) Unique - if (!(rr->resrec.RecordType & kDNSRecordTypeActiveMask)) - { - LogMsg("AnswerLocalQuestionWithLocalAuthRecord: *NOT* delivering %s event for local record type %X %s", - AddRecord ? "Add" : "Rmv", rr->resrec.RecordType, ARDisplayString(m, rr)); - return; - } - - // Indicate that we've given at least one positive answer for this record, so we should be prepared to send a goodbye for it - if (AddRecord) rr->AnsweredLocalQ = mDNStrue; - mDNS_DropLockBeforeCallback(); // Allow client to legally make mDNS API calls from the callback - if (q->QuestionCallback && !q->NoAnswer) - { - q->CurrentAnswers += AddRecord ? 1 : -1; - if (LORecordAnswersAddressType(rr)) - { - if (!followcname || q->ReturnIntermed) - { - // Don't send this packet on the wire as we answered from /etc/hosts - q->ThisQInterval = 0; - q->LOAddressAnswers += AddRecord ? 1 : -1; - q->QuestionCallback(m, q, &rr->resrec, AddRecord); - } - mDNS_ReclaimLockAfterCallback(); // Decrement mDNS_reentrancy to block mDNS API calls again - // The callback above could have caused the question to stop. Detect that - // using m->CurrentQuestion - if (followcname && m->CurrentQuestion == q) - AnswerQuestionByFollowingCNAME(m, q, &rr->resrec); - return; - } - else - q->QuestionCallback(m, q, &rr->resrec, AddRecord); - } - mDNS_ReclaimLockAfterCallback(); // Decrement mDNS_reentrancy to block mDNS API calls again - } - -mDNSlocal void AnswerInterfaceAnyQuestionsWithLocalAuthRecord(mDNS *const m, AuthRecord *rr, QC_result AddRecord) - { - if (m->CurrentQuestion) - LogMsg("AnswerInterfaceAnyQuestionsWithLocalAuthRecord: ERROR m->CurrentQuestion already set: %##s (%s)", - m->CurrentQuestion->qname.c, DNSTypeName(m->CurrentQuestion->qtype)); - m->CurrentQuestion = m->Questions; - while (m->CurrentQuestion && m->CurrentQuestion != m->NewQuestions) - { - mDNSBool answered; - DNSQuestion *q = m->CurrentQuestion; - if (RRAny(rr)) - answered = ResourceRecordAnswersQuestion(&rr->resrec, q); - else - answered = LocalOnlyRecordAnswersQuestion(rr, q); - if (answered) - AnswerLocalQuestionWithLocalAuthRecord(m, rr, AddRecord); // MUST NOT dereference q again - if (m->CurrentQuestion == q) // If m->CurrentQuestion was not auto-advanced, do it ourselves now - m->CurrentQuestion = q->next; - } - m->CurrentQuestion = mDNSNULL; - } - -// When a new local AuthRecord is created or deleted, AnswerAllLocalQuestionsWithLocalAuthRecord() -// delivers the appropriate add/remove events to listening questions: -// 1. It runs though all our LocalOnlyQuestions delivering answers as appropriate, -// stopping if it reaches a NewLocalOnlyQuestion -- brand-new questions are handled by AnswerNewLocalOnlyQuestion(). -// 2. If the AuthRecord is marked mDNSInterface_LocalOnly or mDNSInterface_P2P, then it also runs though -// our main question list, delivering answers to mDNSInterface_Any questions as appropriate, -// stopping if it reaches a NewQuestion -- brand-new questions are handled by AnswerNewQuestion(). -// -// AnswerAllLocalQuestionsWithLocalAuthRecord is used by the m->NewLocalRecords loop in mDNS_Execute(), -// and by mDNS_Deregister_internal() - -mDNSlocal void AnswerAllLocalQuestionsWithLocalAuthRecord(mDNS *const m, AuthRecord *rr, QC_result AddRecord) - { - if (m->CurrentQuestion) - LogMsg("AnswerAllLocalQuestionsWithLocalAuthRecord ERROR m->CurrentQuestion already set: %##s (%s)", - m->CurrentQuestion->qname.c, DNSTypeName(m->CurrentQuestion->qtype)); - - m->CurrentQuestion = m->LocalOnlyQuestions; - while (m->CurrentQuestion && m->CurrentQuestion != m->NewLocalOnlyQuestions) - { - mDNSBool answered; - DNSQuestion *q = m->CurrentQuestion; - // We are called with both LocalOnly/P2P record or a regular AuthRecord - if (RRAny(rr)) - answered = ResourceRecordAnswersQuestion(&rr->resrec, q); - else - answered = LocalOnlyRecordAnswersQuestion(rr, q); - if (answered) - AnswerLocalQuestionWithLocalAuthRecord(m, rr, AddRecord); // MUST NOT dereference q again - if (m->CurrentQuestion == q) // If m->CurrentQuestion was not auto-advanced, do it ourselves now - m->CurrentQuestion = q->next; - } - - m->CurrentQuestion = mDNSNULL; - - // If this AuthRecord is marked LocalOnly or P2P, then we want to deliver it to all local 'mDNSInterface_Any' questions - if (rr->ARType == AuthRecordLocalOnly || rr->ARType == AuthRecordP2P) - AnswerInterfaceAnyQuestionsWithLocalAuthRecord(m, rr, AddRecord); - - } - -// *************************************************************************** -#if COMPILER_LIKES_PRAGMA_MARK -#pragma mark - -#pragma mark - Resource Record Utility Functions -#endif - -#define RRTypeIsAddressType(T) ((T) == kDNSType_A || (T) == kDNSType_AAAA) - -#define ResourceRecordIsValidAnswer(RR) ( ((RR)-> resrec.RecordType & kDNSRecordTypeActiveMask) && \ - ((RR)->Additional1 == mDNSNULL || ((RR)->Additional1->resrec.RecordType & kDNSRecordTypeActiveMask)) && \ - ((RR)->Additional2 == mDNSNULL || ((RR)->Additional2->resrec.RecordType & kDNSRecordTypeActiveMask)) && \ - ((RR)->DependentOn == mDNSNULL || ((RR)->DependentOn->resrec.RecordType & kDNSRecordTypeActiveMask)) ) - -#define ResourceRecordIsValidInterfaceAnswer(RR, INTID) \ - (ResourceRecordIsValidAnswer(RR) && \ - ((RR)->resrec.InterfaceID == mDNSInterface_Any || (RR)->resrec.InterfaceID == (INTID))) - -#define DefaultProbeCountForTypeUnique ((mDNSu8)3) -#define DefaultProbeCountForRecordType(X) ((X) == kDNSRecordTypeUnique ? DefaultProbeCountForTypeUnique : (mDNSu8)0) - -#define InitialAnnounceCount ((mDNSu8)8) - -// For goodbye packets we set the count to 3, and for wakeups we set it to 18 -// (which will be up to 15 wakeup attempts over the course of 30 seconds, -// and then if the machine fails to wake, 3 goodbye packets). -#define GoodbyeCount ((mDNSu8)3) -#define WakeupCount ((mDNSu8)18) - -// Number of wakeups we send if WakeOnResolve is set in the question -#define InitialWakeOnResolveCount ((mDNSu8)3) - -// Note that the announce intervals use exponential backoff, doubling each time. The probe intervals do not. -// This means that because the announce interval is doubled after sending the first packet, the first -// observed on-the-wire inter-packet interval between announcements is actually one second. -// The half-second value here may be thought of as a conceptual (non-existent) half-second delay *before* the first packet is sent. -#define DefaultProbeIntervalForTypeUnique (mDNSPlatformOneSecond/4) -#define DefaultAnnounceIntervalForTypeShared (mDNSPlatformOneSecond/2) -#define DefaultAnnounceIntervalForTypeUnique (mDNSPlatformOneSecond/2) - -#define DefaultAPIntervalForRecordType(X) ((X) & kDNSRecordTypeActiveSharedMask ? DefaultAnnounceIntervalForTypeShared : \ - (X) & kDNSRecordTypeUnique ? DefaultProbeIntervalForTypeUnique : \ - (X) & kDNSRecordTypeActiveUniqueMask ? DefaultAnnounceIntervalForTypeUnique : 0) - -#define TimeToAnnounceThisRecord(RR,time) ((RR)->AnnounceCount && (time) - ((RR)->LastAPTime + (RR)->ThisAPInterval) >= 0) -#define TimeToSendThisRecord(RR,time) ((TimeToAnnounceThisRecord(RR,time) || (RR)->ImmedAnswer) && ResourceRecordIsValidAnswer(RR)) -#define TicksTTL(RR) ((mDNSs32)(RR)->resrec.rroriginalttl * mDNSPlatformOneSecond) -#define RRExpireTime(RR) ((RR)->TimeRcvd + TicksTTL(RR)) - -#define MaxUnansweredQueries 4 - -// SameResourceRecordSignature returns true if two resources records have the same name, type, and class, and may be sent -// (or were received) on the same interface (i.e. if *both* records specify an interface, then it has to match). -// TTL and rdata may differ. -// This is used for cache flush management: -// When sending a unique record, all other records matching "SameResourceRecordSignature" must also be sent -// When receiving a unique record, all old cache records matching "SameResourceRecordSignature" are flushed - -// SameResourceRecordNameClassInterface is functionally the same as SameResourceRecordSignature, except rrtype does not have to match - -#define SameResourceRecordSignature(A,B) (A)->resrec.rrtype == (B)->resrec.rrtype && SameResourceRecordNameClassInterface((A),(B)) - -mDNSlocal mDNSBool SameResourceRecordNameClassInterface(const AuthRecord *const r1, const AuthRecord *const r2) - { - if (!r1) { LogMsg("SameResourceRecordSignature ERROR: r1 is NULL"); return(mDNSfalse); } - if (!r2) { LogMsg("SameResourceRecordSignature ERROR: r2 is NULL"); return(mDNSfalse); } - if (r1->resrec.InterfaceID && - r2->resrec.InterfaceID && - r1->resrec.InterfaceID != r2->resrec.InterfaceID) return(mDNSfalse); - return(mDNSBool)( - r1->resrec.rrclass == r2->resrec.rrclass && - r1->resrec.namehash == r2->resrec.namehash && - SameDomainName(r1->resrec.name, r2->resrec.name)); - } - -// PacketRRMatchesSignature behaves as SameResourceRecordSignature, except that types may differ if our -// authoratative record is unique (as opposed to shared). For unique records, we are supposed to have -// complete ownership of *all* types for this name, so *any* record type with the same name is a conflict. -// In addition, when probing we send our questions with the wildcard type kDNSQType_ANY, -// so a response of any type should match, even if it is not actually the type the client plans to use. - -// For now, to make it easier to avoid false conflicts, we treat SPS Proxy records like shared records, -// and require the rrtypes to match for the rdata to be considered potentially conflicting -mDNSlocal mDNSBool PacketRRMatchesSignature(const CacheRecord *const pktrr, const AuthRecord *const authrr) - { - if (!pktrr) { LogMsg("PacketRRMatchesSignature ERROR: pktrr is NULL"); return(mDNSfalse); } - if (!authrr) { LogMsg("PacketRRMatchesSignature ERROR: authrr is NULL"); return(mDNSfalse); } - if (pktrr->resrec.InterfaceID && - authrr->resrec.InterfaceID && - pktrr->resrec.InterfaceID != authrr->resrec.InterfaceID) return(mDNSfalse); - if (!(authrr->resrec.RecordType & kDNSRecordTypeUniqueMask) || authrr->WakeUp.HMAC.l[0]) - if (pktrr->resrec.rrtype != authrr->resrec.rrtype) return(mDNSfalse); - return(mDNSBool)( - pktrr->resrec.rrclass == authrr->resrec.rrclass && - pktrr->resrec.namehash == authrr->resrec.namehash && - SameDomainName(pktrr->resrec.name, authrr->resrec.name)); - } - -// CacheRecord *ka is the CacheRecord from the known answer list in the query. -// This is the information that the requester believes to be correct. -// AuthRecord *rr is the answer we are proposing to give, if not suppressed. -// This is the information that we believe to be correct. -// We've already determined that we plan to give this answer on this interface -// (either the record is non-specific, or it is specific to this interface) -// so now we just need to check the name, type, class, rdata and TTL. -mDNSlocal mDNSBool ShouldSuppressKnownAnswer(const CacheRecord *const ka, const AuthRecord *const rr) - { - // If RR signature is different, or data is different, then don't suppress our answer - if (!IdenticalResourceRecord(&ka->resrec, &rr->resrec)) return(mDNSfalse); - - // If the requester's indicated TTL is less than half the real TTL, - // we need to give our answer before the requester's copy expires. - // If the requester's indicated TTL is at least half the real TTL, - // then we can suppress our answer this time. - // If the requester's indicated TTL is greater than the TTL we believe, - // then that's okay, and we don't need to do anything about it. - // (If two responders on the network are offering the same information, - // that's okay, and if they are offering the information with different TTLs, - // the one offering the lower TTL should defer to the one offering the higher TTL.) - return(mDNSBool)(ka->resrec.rroriginalttl >= rr->resrec.rroriginalttl / 2); - } - -mDNSlocal void SetNextAnnounceProbeTime(mDNS *const m, const AuthRecord *const rr) - { - if (rr->resrec.RecordType == kDNSRecordTypeUnique) - { - if ((rr->LastAPTime + rr->ThisAPInterval) - m->timenow > mDNSPlatformOneSecond * 10) - { - LogMsg("SetNextAnnounceProbeTime: ProbeCount %d Next in %d %s", rr->ProbeCount, (rr->LastAPTime + rr->ThisAPInterval) - m->timenow, ARDisplayString(m, rr)); - LogMsg("SetNextAnnounceProbeTime: m->SuppressProbes %d m->timenow %d diff %d", m->SuppressProbes, m->timenow, m->SuppressProbes - m->timenow); - } - if (m->NextScheduledProbe - (rr->LastAPTime + rr->ThisAPInterval) >= 0) - m->NextScheduledProbe = (rr->LastAPTime + rr->ThisAPInterval); - // Some defensive code: - // If (rr->LastAPTime + rr->ThisAPInterval) happens to be far in the past, we don't want to allow - // NextScheduledProbe to be set excessively in the past, because that can cause bad things to happen. - // See: mDNS: Sometimes advertising stops working and record interval is set to zero - if (m->NextScheduledProbe - m->timenow < 0) - m->NextScheduledProbe = m->timenow; - } - else if (rr->AnnounceCount && (ResourceRecordIsValidAnswer(rr) || rr->resrec.RecordType == kDNSRecordTypeDeregistering)) - { - if (m->NextScheduledResponse - (rr->LastAPTime + rr->ThisAPInterval) >= 0) - m->NextScheduledResponse = (rr->LastAPTime + rr->ThisAPInterval); - } - } - -mDNSlocal void InitializeLastAPTime(mDNS *const m, AuthRecord *const rr) - { - // For reverse-mapping Sleep Proxy PTR records, probe interval is one second - rr->ThisAPInterval = rr->AddressProxy.type ? mDNSPlatformOneSecond : DefaultAPIntervalForRecordType(rr->resrec.RecordType); - - // * If this is a record type that's going to probe, then we use the m->SuppressProbes time. - // * Otherwise, if it's not going to probe, but m->SuppressProbes is set because we have other - // records that are going to probe, then we delay its first announcement so that it will - // go out synchronized with the first announcement for the other records that *are* probing. - // This is a minor performance tweak that helps keep groups of related records synchronized together. - // The addition of "interval / 2" is to make sure that, in the event that any of the probes are - // delayed by a few milliseconds, this announcement does not inadvertently go out *before* the probing is complete. - // When the probing is complete and those records begin to announce, these records will also be picked up and accelerated, - // because they will meet the criterion of being at least half-way to their scheduled announcement time. - // * If it's not going to probe and m->SuppressProbes is not already set then we should announce immediately. - - if (rr->ProbeCount) - { - // If we have no probe suppression time set, or it is in the past, set it now - if (m->SuppressProbes == 0 || m->SuppressProbes - m->timenow < 0) - { - // To allow us to aggregate probes when a group of services are registered together, - // the first probe is delayed 1/4 second. This means the common-case behaviour is: - // 1/4 second wait; probe - // 1/4 second wait; probe - // 1/4 second wait; probe - // 1/4 second wait; announce (i.e. service is normally announced exactly one second after being registered) - m->SuppressProbes = NonZeroTime(m->timenow + DefaultProbeIntervalForTypeUnique/2 + mDNSRandom(DefaultProbeIntervalForTypeUnique/2)); - - // If we already have a *probe* scheduled to go out sooner, then use that time to get better aggregation - if (m->SuppressProbes - m->NextScheduledProbe >= 0) - m->SuppressProbes = NonZeroTime(m->NextScheduledProbe); - if (m->SuppressProbes - m->timenow < 0) // Make sure we don't set m->SuppressProbes excessively in the past - m->SuppressProbes = m->timenow; - - // If we already have a *query* scheduled to go out sooner, then use that time to get better aggregation - if (m->SuppressProbes - m->NextScheduledQuery >= 0) - m->SuppressProbes = NonZeroTime(m->NextScheduledQuery); - if (m->SuppressProbes - m->timenow < 0) // Make sure we don't set m->SuppressProbes excessively in the past - m->SuppressProbes = m->timenow; - - // except... don't expect to be able to send before the m->SuppressSending timer fires - if (m->SuppressSending && m->SuppressProbes - m->SuppressSending < 0) - m->SuppressProbes = NonZeroTime(m->SuppressSending); - - if (m->SuppressProbes - m->timenow > mDNSPlatformOneSecond * 8) - { - LogMsg("InitializeLastAPTime ERROR m->SuppressProbes %d m->NextScheduledProbe %d m->NextScheduledQuery %d m->SuppressSending %d %d", - m->SuppressProbes - m->timenow, - m->NextScheduledProbe - m->timenow, - m->NextScheduledQuery - m->timenow, - m->SuppressSending, - m->SuppressSending - m->timenow); - m->SuppressProbes = NonZeroTime(m->timenow + DefaultProbeIntervalForTypeUnique/2 + mDNSRandom(DefaultProbeIntervalForTypeUnique/2)); - } - } - rr->LastAPTime = m->SuppressProbes - rr->ThisAPInterval; - } - else if (m->SuppressProbes && m->SuppressProbes - m->timenow >= 0) - rr->LastAPTime = m->SuppressProbes - rr->ThisAPInterval + DefaultProbeIntervalForTypeUnique * DefaultProbeCountForTypeUnique + rr->ThisAPInterval / 2; - else - rr->LastAPTime = m->timenow - rr->ThisAPInterval; - - // For reverse-mapping Sleep Proxy PTR records we don't want to start probing instantly -- we - // wait one second to give the client a chance to go to sleep, and then start our ARP/NDP probing. - // After three probes one second apart with no answer, we conclude the client is now sleeping - // and we can begin broadcasting our announcements to take over ownership of that IP address. - // If we don't wait for the client to go to sleep, then when the client sees our ARP Announcements there's a risk - // (depending on the OS and networking stack it's using) that it might interpret it as a conflict and change its IP address. - if (rr->AddressProxy.type) rr->LastAPTime = m->timenow; - - // Unsolicited Neighbor Advertisements (RFC 2461 Section 7.2.6) give us fast address cache updating, - // but some older IPv6 clients get confused by them, so for now we don't send them. Without Unsolicited - // Neighbor Advertisements we have to rely on Neighbor Unreachability Detection instead, which is slower. - // Given this, we'll do our best to wake for existing IPv6 connections, but we don't want to encourage - // new ones for sleeping clients, so we'll we send deletions for our SPS clients' AAAA records. - if (m->KnownBugs & mDNS_KnownBug_LimitedIPv6) - if (rr->WakeUp.HMAC.l[0] && rr->resrec.rrtype == kDNSType_AAAA) - rr->LastAPTime = m->timenow - rr->ThisAPInterval + mDNSPlatformOneSecond * 10; - - // Set LastMCTime to now, to inhibit multicast responses - // (no need to send additional multicast responses when we're announcing anyway) - rr->LastMCTime = m->timenow; - rr->LastMCInterface = mDNSInterfaceMark; - - SetNextAnnounceProbeTime(m, rr); - } - -mDNSlocal const domainname *SetUnicastTargetToHostName(mDNS *const m, AuthRecord *rr) - { - const domainname *target; - if (rr->AutoTarget) - { - // For autotunnel services pointing at our IPv6 ULA we don't need or want a NAT mapping, but for all other - // advertised services referencing our uDNS hostname, we want NAT mappings automatically created as appropriate, - // with the port number in our advertised SRV record automatically tracking the external mapped port. - DomainAuthInfo *AuthInfo = GetAuthInfoForName_internal(m, rr->resrec.name); - if (!AuthInfo || !AuthInfo->AutoTunnel) rr->AutoTarget = Target_AutoHostAndNATMAP; - } - - target = GetServiceTarget(m, rr); - if (!target || target->c[0] == 0) - { - // defer registration until we've got a target - LogInfo("SetUnicastTargetToHostName No target for %s", ARDisplayString(m, rr)); - rr->state = regState_NoTarget; - return mDNSNULL; - } - else - { - LogInfo("SetUnicastTargetToHostName target %##s for resource record %s", target->c, ARDisplayString(m,rr)); - return target; - } - } - -// Right now this only applies to mDNS (.local) services where the target host is always m->MulticastHostname -// Eventually we should unify this with GetServiceTarget() in uDNS.c -mDNSlocal void SetTargetToHostName(mDNS *const m, AuthRecord *const rr) - { - domainname *const target = GetRRDomainNameTarget(&rr->resrec); - const domainname *newname = &m->MulticastHostname; - - if (!target) LogInfo("SetTargetToHostName: Don't know how to set the target of rrtype %s", DNSTypeName(rr->resrec.rrtype)); - - if (!(rr->ForceMCast || rr->ARType == AuthRecordLocalOnly || rr->ARType == AuthRecordP2P || IsLocalDomain(&rr->namestorage))) - { - const domainname *const n = SetUnicastTargetToHostName(m, rr); - if (n) newname = n; - else { target->c[0] = 0; SetNewRData(&rr->resrec, mDNSNULL, 0); return; } - } - - if (target && SameDomainName(target, newname)) - debugf("SetTargetToHostName: Target of %##s is already %##s", rr->resrec.name->c, target->c); - - if (target && !SameDomainName(target, newname)) - { - AssignDomainName(target, newname); - SetNewRData(&rr->resrec, mDNSNULL, 0); // Update rdlength, rdestimate, rdatahash - - // If we're in the middle of probing this record, we need to start again, - // because changing its rdata may change the outcome of the tie-breaker. - // (If the record type is kDNSRecordTypeUnique (unconfirmed unique) then DefaultProbeCountForRecordType is non-zero.) - rr->ProbeCount = DefaultProbeCountForRecordType(rr->resrec.RecordType); - - // If we've announced this record, we really should send a goodbye packet for the old rdata before - // changing to the new rdata. However, in practice, we only do SetTargetToHostName for unique records, - // so when we announce them we'll set the kDNSClass_UniqueRRSet and clear any stale data that way. - if (rr->RequireGoodbye && rr->resrec.RecordType == kDNSRecordTypeShared) - debugf("Have announced shared record %##s (%s) at least once: should have sent a goodbye packet before updating", - rr->resrec.name->c, DNSTypeName(rr->resrec.rrtype)); - - rr->AnnounceCount = InitialAnnounceCount; - rr->RequireGoodbye = mDNSfalse; - InitializeLastAPTime(m, rr); - } - } - -mDNSlocal void AcknowledgeRecord(mDNS *const m, AuthRecord *const rr) - { - if (rr->RecordCallback) - { - // CAUTION: MUST NOT do anything more with rr after calling rr->Callback(), because the client's callback function - // is allowed to do anything, including starting/stopping queries, registering/deregistering records, etc. - rr->Acknowledged = mDNStrue; - mDNS_DropLockBeforeCallback(); // Allow client to legally make mDNS API calls from the callback - rr->RecordCallback(m, rr, mStatus_NoError); - mDNS_ReclaimLockAfterCallback(); // Decrement mDNS_reentrancy to block mDNS API calls again - } - } - -mDNSexport void ActivateUnicastRegistration(mDNS *const m, AuthRecord *const rr) - { - // Make sure that we don't activate the SRV record and associated service records, if it is in - // NoTarget state. First time when a service is being instantiated, SRV record may be in NoTarget state. - // We should not activate any of the other reords (PTR, TXT) that are part of the service. When - // the target becomes available, the records will be reregistered. - if (rr->resrec.rrtype != kDNSType_SRV) - { - AuthRecord *srvRR = mDNSNULL; - if (rr->resrec.rrtype == kDNSType_PTR) - srvRR = rr->Additional1; - else if (rr->resrec.rrtype == kDNSType_TXT) - srvRR = rr->DependentOn; - if (srvRR) - { - if (srvRR->resrec.rrtype != kDNSType_SRV) - { - LogMsg("ActivateUnicastRegistration: ERROR!! Resource record %s wrong, expecting SRV type", ARDisplayString(m, srvRR)); - } - else - { - LogInfo("ActivateUnicastRegistration: Found Service Record %s in state %d for %##s (%s)", - ARDisplayString(m, srvRR), srvRR->state, rr->resrec.name->c, DNSTypeName(rr->resrec.rrtype)); - rr->state = srvRR->state; - } - } - } - - if (rr->state == regState_NoTarget) - { - LogInfo("ActivateUnicastRegistration record %s in regState_NoTarget, not activating", ARDisplayString(m, rr)); - return; - } - // When we wake up from sleep, we call ActivateUnicastRegistration. It is possible that just before we went to sleep, - // the service/record was being deregistered. In that case, we should not try to register again. For the cases where - // the records are deregistered due to e.g., no target for the SRV record, we would have returned from above if it - // was already in NoTarget state. If it was in the process of deregistration but did not complete fully before we went - // to sleep, then it is okay to start in Pending state as we will go back to NoTarget state if we don't have a target. - if (rr->resrec.RecordType == kDNSRecordTypeDeregistering) - { - LogInfo("ActivateUnicastRegistration: Resource record %s, current state %d, moving to DeregPending", ARDisplayString(m, rr), rr->state); - rr->state = regState_DeregPending; - } - else - { - LogInfo("ActivateUnicastRegistration: Resource record %s, current state %d, moving to Pending", ARDisplayString(m, rr), rr->state); - rr->state = regState_Pending; - } - rr->ProbeCount = 0; - rr->AnnounceCount = 0; - rr->ThisAPInterval = INIT_RECORD_REG_INTERVAL; - rr->LastAPTime = m->timenow - rr->ThisAPInterval; - rr->expire = 0; // Forget about all the leases, start fresh - rr->uselease = mDNStrue; - rr->updateid = zeroID; - rr->SRVChanged = mDNSfalse; - rr->updateError = mStatus_NoError; - // RestartRecordGetZoneData calls this function whenever a new interface gets registered with core. - // The records might already be registered with the server and hence could have NAT state. - if (rr->NATinfo.clientContext) - { - mDNS_StopNATOperation_internal(m, &rr->NATinfo); - rr->NATinfo.clientContext = mDNSNULL; - } - if (rr->nta) { CancelGetZoneData(m, rr->nta); rr->nta = mDNSNULL; } - if (rr->tcp) { DisposeTCPConn(rr->tcp); rr->tcp = mDNSNULL; } - if (m->NextuDNSEvent - (rr->LastAPTime + rr->ThisAPInterval) >= 0) - m->NextuDNSEvent = (rr->LastAPTime + rr->ThisAPInterval); - } - -// Two records qualify to be local duplicates if: -// (a) the RecordTypes are the same, or -// (b) one is Unique and the other Verified -// (c) either is in the process of deregistering -#define RecordLDT(A,B) ((A)->resrec.RecordType == (B)->resrec.RecordType || \ - ((A)->resrec.RecordType | (B)->resrec.RecordType) == (kDNSRecordTypeUnique | kDNSRecordTypeVerified) || \ - ((A)->resrec.RecordType == kDNSRecordTypeDeregistering || (B)->resrec.RecordType == kDNSRecordTypeDeregistering)) - -#define RecordIsLocalDuplicate(A,B) \ - ((A)->resrec.InterfaceID == (B)->resrec.InterfaceID && RecordLDT((A),(B)) && IdenticalResourceRecord(&(A)->resrec, &(B)->resrec)) - -mDNSlocal AuthRecord *CheckAuthIdenticalRecord(AuthHash *r, AuthRecord *rr) - { - AuthGroup *a; - AuthGroup **ag = &a; - AuthRecord **rp; - const mDNSu32 slot = AuthHashSlot(rr->resrec.name); - - a = AuthGroupForRecord(r, slot, &rr->resrec); - if (!a) return mDNSNULL; - rp = &(*ag)->members; - while (*rp) - { - if (!RecordIsLocalDuplicate(*rp, rr)) - rp=&(*rp)->next; - else - { - if ((*rp)->resrec.RecordType == kDNSRecordTypeDeregistering) - { - (*rp)->AnnounceCount = 0; - rp=&(*rp)->next; - } - else return *rp; - } - } - return (mDNSNULL); - } - -mDNSlocal mDNSBool CheckAuthRecordConflict(AuthHash *r, AuthRecord *rr) - { - AuthGroup *a; - AuthGroup **ag = &a; - AuthRecord **rp; - const mDNSu32 slot = AuthHashSlot(rr->resrec.name); - - a = AuthGroupForRecord(r, slot, &rr->resrec); - if (!a) return mDNSfalse; - rp = &(*ag)->members; - while (*rp) - { - const AuthRecord *s1 = rr->RRSet ? rr->RRSet : rr; - const AuthRecord *s2 = (*rp)->RRSet ? (*rp)->RRSet : *rp; - if (s1 != s2 && SameResourceRecordSignature((*rp), rr) && !IdenticalSameNameRecord(&(*rp)->resrec, &rr->resrec)) - return mDNStrue; - else - rp=&(*rp)->next; - } - return (mDNSfalse); - } - -// checks to see if "rr" is already present -mDNSlocal AuthRecord *CheckAuthSameRecord(AuthHash *r, AuthRecord *rr) - { - AuthGroup *a; - AuthGroup **ag = &a; - AuthRecord **rp; - const mDNSu32 slot = AuthHashSlot(rr->resrec.name); - - a = AuthGroupForRecord(r, slot, &rr->resrec); - if (!a) return mDNSNULL; - rp = &(*ag)->members; - while (*rp) - { - if (*rp != rr) - rp=&(*rp)->next; - else - { - return *rp; - } - } - return (mDNSNULL); - } - -// Exported so uDNS.c can call this -mDNSexport mStatus mDNS_Register_internal(mDNS *const m, AuthRecord *const rr) - { - domainname *target = GetRRDomainNameTarget(&rr->resrec); - AuthRecord *r; - AuthRecord **p = &m->ResourceRecords; - AuthRecord **d = &m->DuplicateRecords; - - if ((mDNSs32)rr->resrec.rroriginalttl <= 0) - { LogMsg("mDNS_Register_internal: TTL %X should be 1 - 0x7FFFFFFF %s", rr->resrec.rroriginalttl, ARDisplayString(m, rr)); return(mStatus_BadParamErr); } - - if (!rr->resrec.RecordType) - { LogMsg("mDNS_Register_internal: RecordType must be non-zero %s", ARDisplayString(m, rr)); return(mStatus_BadParamErr); } - - if (m->ShutdownTime) - { LogMsg("mDNS_Register_internal: Shutting down, can't register %s", ARDisplayString(m, rr)); return(mStatus_ServiceNotRunning); } - - if (m->DivertMulticastAdvertisements && !AuthRecord_uDNS(rr)) - { - mDNSInterfaceID previousID = rr->resrec.InterfaceID; - if (rr->resrec.InterfaceID == mDNSInterface_Any || rr->resrec.InterfaceID == mDNSInterface_P2P) - { - rr->resrec.InterfaceID = mDNSInterface_LocalOnly; - rr->ARType = AuthRecordLocalOnly; - } - if (rr->resrec.InterfaceID != mDNSInterface_LocalOnly) - { - NetworkInterfaceInfo *intf = FirstInterfaceForID(m, rr->resrec.InterfaceID); - if (intf && !intf->Advertise){ rr->resrec.InterfaceID = mDNSInterface_LocalOnly; rr->ARType = AuthRecordLocalOnly; } - } - if (rr->resrec.InterfaceID != previousID) - LogInfo("mDNS_Register_internal: Diverting record to local-only %s", ARDisplayString(m, rr)); - } - - if (RRLocalOnly(rr)) - { - if (CheckAuthSameRecord(&m->rrauth, rr)) - { - LogMsg("mDNS_Register_internal: ERROR!! Tried to register LocalOnly AuthRecord %p %##s (%s) that's already in the list", - rr, rr->resrec.name->c, DNSTypeName(rr->resrec.rrtype)); - return(mStatus_AlreadyRegistered); - } - } - else - { - while (*p && *p != rr) p=&(*p)->next; - if (*p) - { - LogMsg("mDNS_Register_internal: ERROR!! Tried to register AuthRecord %p %##s (%s) that's already in the list", - rr, rr->resrec.name->c, DNSTypeName(rr->resrec.rrtype)); - return(mStatus_AlreadyRegistered); - } - } - - while (*d && *d != rr) d=&(*d)->next; - if (*d) - { - LogMsg("mDNS_Register_internal: ERROR!! Tried to register AuthRecord %p %##s (%s) that's already in the Duplicate list", - rr, rr->resrec.name->c, DNSTypeName(rr->resrec.rrtype)); - return(mStatus_AlreadyRegistered); - } - - if (rr->DependentOn) - { - if (rr->resrec.RecordType == kDNSRecordTypeUnique) - rr->resrec.RecordType = kDNSRecordTypeVerified; - else - { - LogMsg("mDNS_Register_internal: ERROR! %##s (%s): rr->DependentOn && RecordType != kDNSRecordTypeUnique", - rr->resrec.name->c, DNSTypeName(rr->resrec.rrtype)); - return(mStatus_Invalid); - } - if (!(rr->DependentOn->resrec.RecordType & (kDNSRecordTypeUnique | kDNSRecordTypeVerified | kDNSRecordTypeKnownUnique))) - { - LogMsg("mDNS_Register_internal: ERROR! %##s (%s): rr->DependentOn->RecordType bad type %X", - rr->resrec.name->c, DNSTypeName(rr->resrec.rrtype), rr->DependentOn->resrec.RecordType); - return(mStatus_Invalid); - } - } - - // If this resource record is referencing a specific interface, make sure it exists. - // Skip checks for LocalOnly and P2P as they are not valid InterfaceIDs. Also, for scoped - // entries in /etc/hosts skip that check as that interface may not be valid at this time. - if (rr->resrec.InterfaceID && rr->ARType != AuthRecordLocalOnly && rr->ARType != AuthRecordP2P) - { - NetworkInterfaceInfo *intf = FirstInterfaceForID(m, rr->resrec.InterfaceID); - if (!intf) - { - debugf("mDNS_Register_internal: Bogus InterfaceID %p in resource record", rr->resrec.InterfaceID); - return(mStatus_BadReferenceErr); - } - } - - rr->next = mDNSNULL; - - // Field Group 1: The actual information pertaining to this resource record - // Set up by client prior to call - - // Field Group 2: Persistent metadata for Authoritative Records -// rr->Additional1 = set to mDNSNULL in mDNS_SetupResourceRecord; may be overridden by client -// rr->Additional2 = set to mDNSNULL in mDNS_SetupResourceRecord; may be overridden by client -// rr->DependentOn = set to mDNSNULL in mDNS_SetupResourceRecord; may be overridden by client -// rr->RRSet = set to mDNSNULL in mDNS_SetupResourceRecord; may be overridden by client -// rr->Callback = already set in mDNS_SetupResourceRecord -// rr->Context = already set in mDNS_SetupResourceRecord -// rr->RecordType = already set in mDNS_SetupResourceRecord -// rr->HostTarget = set to mDNSfalse in mDNS_SetupResourceRecord; may be overridden by client -// rr->AllowRemoteQuery = set to mDNSfalse in mDNS_SetupResourceRecord; may be overridden by client - // Make sure target is not uninitialized data, or we may crash writing debugging log messages - if (rr->AutoTarget && target) target->c[0] = 0; - - // Field Group 3: Transient state for Authoritative Records - rr->Acknowledged = mDNSfalse; - rr->ProbeCount = DefaultProbeCountForRecordType(rr->resrec.RecordType); - rr->AnnounceCount = InitialAnnounceCount; - rr->RequireGoodbye = mDNSfalse; - rr->AnsweredLocalQ = mDNSfalse; - rr->IncludeInProbe = mDNSfalse; - rr->ImmedUnicast = mDNSfalse; - rr->SendNSECNow = mDNSNULL; - rr->ImmedAnswer = mDNSNULL; - rr->ImmedAdditional = mDNSNULL; - rr->SendRNow = mDNSNULL; - rr->v4Requester = zerov4Addr; - rr->v6Requester = zerov6Addr; - rr->NextResponse = mDNSNULL; - rr->NR_AnswerTo = mDNSNULL; - rr->NR_AdditionalTo = mDNSNULL; - if (!rr->AutoTarget) InitializeLastAPTime(m, rr); -// rr->LastAPTime = Set for us in InitializeLastAPTime() -// rr->LastMCTime = Set for us in InitializeLastAPTime() -// rr->LastMCInterface = Set for us in InitializeLastAPTime() - rr->NewRData = mDNSNULL; - rr->newrdlength = 0; - rr->UpdateCallback = mDNSNULL; - rr->UpdateCredits = kMaxUpdateCredits; - rr->NextUpdateCredit = 0; - rr->UpdateBlocked = 0; - - // For records we're holding as proxy (except reverse-mapping PTR records) two announcements is sufficient - if (rr->WakeUp.HMAC.l[0] && !rr->AddressProxy.type) rr->AnnounceCount = 2; - - // Field Group 4: Transient uDNS state for Authoritative Records - rr->state = regState_Zero; - rr->uselease = 0; - rr->expire = 0; - rr->Private = 0; - rr->updateid = zeroID; - rr->zone = rr->resrec.name; - rr->nta = mDNSNULL; - rr->tcp = mDNSNULL; - rr->OrigRData = 0; - rr->OrigRDLen = 0; - rr->InFlightRData = 0; - rr->InFlightRDLen = 0; - rr->QueuedRData = 0; - rr->QueuedRDLen = 0; - //mDNSPlatformMemZero(&rr->NATinfo, sizeof(rr->NATinfo)); - // We should be recording the actual internal port for this service record here. Once we initiate our NAT mapping - // request we'll subsequently overwrite srv.port with the allocated external NAT port -- potentially multiple - // times with different values if the external NAT port changes during the lifetime of the service registration. - //if (rr->resrec.rrtype == kDNSType_SRV) rr->NATinfo.IntPort = rr->resrec.rdata->u.srv.port; - -// rr->resrec.interface = already set in mDNS_SetupResourceRecord -// rr->resrec.name->c = MUST be set by client -// rr->resrec.rrtype = already set in mDNS_SetupResourceRecord -// rr->resrec.rrclass = already set in mDNS_SetupResourceRecord -// rr->resrec.rroriginalttl = already set in mDNS_SetupResourceRecord -// rr->resrec.rdata = MUST be set by client, unless record type is CNAME or PTR and rr->HostTarget is set - - // BIND named (name daemon) doesn't allow TXT records with zero-length rdata. This is strictly speaking correct, - // since RFC 1035 specifies a TXT record as "One or more s", not "Zero or more s". - // Since some legacy apps try to create zero-length TXT records, we'll silently correct it here. - if (rr->resrec.rrtype == kDNSType_TXT && rr->resrec.rdlength == 0) { rr->resrec.rdlength = 1; rr->resrec.rdata->u.txt.c[0] = 0; } - - if (rr->AutoTarget) - { - SetTargetToHostName(m, rr); // Also sets rdlength and rdestimate for us, and calls InitializeLastAPTime(); -#ifndef UNICAST_DISABLED - // If we have no target record yet, SetTargetToHostName will set rr->state == regState_NoTarget - // In this case we leave the record half-formed in the list, and later we'll remove it from the list and re-add it properly. - if (rr->state == regState_NoTarget) - { - // Initialize the target so that we don't crash while logging etc. - domainname *tar = GetRRDomainNameTarget(&rr->resrec); - if (tar) tar->c[0] = 0; - LogInfo("mDNS_Register_internal: record %s in NoTarget state", ARDisplayString(m, rr)); - } -#endif - } - else - { - rr->resrec.rdlength = GetRDLength(&rr->resrec, mDNSfalse); - rr->resrec.rdestimate = GetRDLength(&rr->resrec, mDNStrue); - } - - if (!ValidateDomainName(rr->resrec.name)) - { LogMsg("Attempt to register record with invalid name: %s", ARDisplayString(m, rr)); return(mStatus_Invalid); } - - // Don't do this until *after* we've set rr->resrec.rdlength - if (!ValidateRData(rr->resrec.rrtype, rr->resrec.rdlength, rr->resrec.rdata)) - { LogMsg("Attempt to register record with invalid rdata: %s", ARDisplayString(m, rr)); return(mStatus_Invalid); } - - rr->resrec.namehash = DomainNameHashValue(rr->resrec.name); - rr->resrec.rdatahash = target ? DomainNameHashValue(target) : RDataHashValue(&rr->resrec); - - if (RRLocalOnly(rr)) - { - // If this is supposed to be unique, make sure we don't have any name conflicts. - // If we found a conflict, we may still want to insert the record in the list but mark it appropriately - // (kDNSRecordTypeDeregistering) so that we deliver RMV events to the application. But this causes more - // complications and not clear whether there are any benefits. See rdar:9304275 for details. - // Hence, just bail out. - if (rr->resrec.RecordType & kDNSRecordTypeUniqueMask) - { - if (CheckAuthRecordConflict(&m->rrauth, rr)) - { - LogInfo("mDNS_Register_internal: Name conflict %s (%p), InterfaceID %p", ARDisplayString(m, rr), rr, rr->resrec.InterfaceID); - return mStatus_NameConflict; - } - } - } - - // For uDNS records, we don't support duplicate checks at this time. -#ifndef UNICAST_DISABLED - if (AuthRecord_uDNS(rr)) - { - if (!m->NewLocalRecords) m->NewLocalRecords = rr; - // When we called SetTargetToHostName, it may have caused mDNS_Register_internal to be re-entered, appending new - // records to the list, so we now need to update p to advance to the new end to the list before appending our new record. - // Note that for AutoTunnel this should never happen, but this check makes the code future-proof. - while (*p) p=&(*p)->next; - *p = rr; - if (rr->resrec.RecordType == kDNSRecordTypeUnique) rr->resrec.RecordType = kDNSRecordTypeVerified; - rr->ProbeCount = 0; - rr->AnnounceCount = 0; - if (rr->state != regState_NoTarget) ActivateUnicastRegistration(m, rr); - return(mStatus_NoError); // <--- Note: For unicast records, code currently bails out at this point - } -#endif - - // Now that we've finished building our new record, make sure it's not identical to one we already have - if (RRLocalOnly(rr)) - { - rr->ProbeCount = 0; - rr->AnnounceCount = 0; - r = CheckAuthIdenticalRecord(&m->rrauth, rr); - } - else - { - for (r = m->ResourceRecords; r; r=r->next) - if (RecordIsLocalDuplicate(r, rr)) - { - if (r->resrec.RecordType == kDNSRecordTypeDeregistering) r->AnnounceCount = 0; - else break; - } - } - - if (r) - { - debugf("mDNS_Register_internal:Adding to duplicate list %s", ARDisplayString(m,rr)); - *d = rr; - // If the previous copy of this record is already verified unique, - // then indicate that we should move this record promptly to kDNSRecordTypeUnique state. - // Setting ProbeCount to zero will cause SendQueries() to advance this record to - // kDNSRecordTypeVerified state and call the client callback at the next appropriate time. - if (rr->resrec.RecordType == kDNSRecordTypeUnique && r->resrec.RecordType == kDNSRecordTypeVerified) - rr->ProbeCount = 0; - } - else - { - debugf("mDNS_Register_internal: Adding to active record list %s", ARDisplayString(m,rr)); - if (RRLocalOnly(rr)) - { - AuthGroup *ag; - ag = InsertAuthRecord(m, &m->rrauth, rr); - if (ag && !ag->NewLocalOnlyRecords) { - m->NewLocalOnlyRecords = mDNStrue; - ag->NewLocalOnlyRecords = rr; - } - // No probing for LocalOnly records, Acknowledge them right away - if (rr->resrec.RecordType == kDNSRecordTypeUnique) rr->resrec.RecordType = kDNSRecordTypeVerified; - AcknowledgeRecord(m, rr); - return(mStatus_NoError); - } - else - { - if (!m->NewLocalRecords) m->NewLocalRecords = rr; - *p = rr; - } - } - - if (!AuthRecord_uDNS(rr)) // This check is superfluous, given that for unicast records we (currently) bail out above - { - // For records that are not going to probe, acknowledge them right away - if (rr->resrec.RecordType != kDNSRecordTypeUnique && rr->resrec.RecordType != kDNSRecordTypeDeregistering) - AcknowledgeRecord(m, rr); - - // Adding a record may affect whether or not we should sleep - mDNS_UpdateAllowSleep(m); - } - - return(mStatus_NoError); - } - -mDNSlocal void RecordProbeFailure(mDNS *const m, const AuthRecord *const rr) - { - m->ProbeFailTime = m->timenow; - m->NumFailedProbes++; - // If we've had fifteen or more probe failures, rate-limit to one every five seconds. - // If a bunch of hosts have all been configured with the same name, then they'll all - // conflict and run through the same series of names: name-2, name-3, name-4, etc., - // up to name-10. After that they'll start adding random increments in the range 1-100, - // so they're more likely to branch out in the available namespace and settle on a set of - // unique names quickly. If after five more tries the host is still conflicting, then we - // may have a serious problem, so we start rate-limiting so we don't melt down the network. - if (m->NumFailedProbes >= 15) - { - m->SuppressProbes = NonZeroTime(m->timenow + mDNSPlatformOneSecond * 5); - LogMsg("Excessive name conflicts (%lu) for %##s (%s); rate limiting in effect", - m->NumFailedProbes, rr->resrec.name->c, DNSTypeName(rr->resrec.rrtype)); - } - } - -mDNSlocal void CompleteRDataUpdate(mDNS *const m, AuthRecord *const rr) - { - RData *OldRData = rr->resrec.rdata; - mDNSu16 OldRDLen = rr->resrec.rdlength; - SetNewRData(&rr->resrec, rr->NewRData, rr->newrdlength); // Update our rdata - rr->NewRData = mDNSNULL; // Clear the NewRData pointer ... - if (rr->UpdateCallback) - rr->UpdateCallback(m, rr, OldRData, OldRDLen); // ... and let the client know - } - -// Note: mDNS_Deregister_internal can call a user callback, which may change the record list and/or question list. -// Any code walking either list must use the CurrentQuestion and/or CurrentRecord mechanism to protect against this. -// Exported so uDNS.c can call this -mDNSexport mStatus mDNS_Deregister_internal(mDNS *const m, AuthRecord *const rr, mDNS_Dereg_type drt) - { - AuthRecord *r2; - mDNSu8 RecordType = rr->resrec.RecordType; - AuthRecord **p = &m->ResourceRecords; // Find this record in our list of active records - mDNSBool dupList = mDNSfalse; - - if (RRLocalOnly(rr)) - { - AuthGroup *a; - AuthGroup **ag = &a; - AuthRecord **rp; - const mDNSu32 slot = AuthHashSlot(rr->resrec.name); - - a = AuthGroupForRecord(&m->rrauth, slot, &rr->resrec); - if (!a) return mDNSfalse; - rp = &(*ag)->members; - while (*rp && *rp != rr) rp=&(*rp)->next; - p = rp; - } - else - { - while (*p && *p != rr) p=&(*p)->next; - } - - if (*p) - { - // We found our record on the main list. See if there are any duplicates that need special handling. - if (drt == mDNS_Dereg_conflict) // If this was a conflict, see that all duplicates get the same treatment - { - // Scan for duplicates of rr, and mark them for deregistration at the end of this routine, after we've finished - // deregistering rr. We need to do this scan *before* we give the client the chance to free and reuse the rr memory. - for (r2 = m->DuplicateRecords; r2; r2=r2->next) if (RecordIsLocalDuplicate(r2, rr)) r2->ProbeCount = 0xFF; - } - else - { - // Before we delete the record (and potentially send a goodbye packet) - // first see if we have a record on the duplicate list ready to take over from it. - AuthRecord **d = &m->DuplicateRecords; - while (*d && !RecordIsLocalDuplicate(*d, rr)) d=&(*d)->next; - if (*d) - { - AuthRecord *dup = *d; - debugf("mDNS_Register_internal: Duplicate record %p taking over from %p %##s (%s)", - dup, rr, rr->resrec.name->c, DNSTypeName(rr->resrec.rrtype)); - *d = dup->next; // Cut replacement record from DuplicateRecords list - if (RRLocalOnly(rr)) - { - dup->next = mDNSNULL; - if (!InsertAuthRecord(m, &m->rrauth, dup)) LogMsg("mDNS_Deregister_internal: ERROR!! cannot insert %s", ARDisplayString(m, dup)); - } - else - { - dup->next = rr->next; // And then... - rr->next = dup; // ... splice it in right after the record we're about to delete - } - dup->resrec.RecordType = rr->resrec.RecordType; - dup->ProbeCount = rr->ProbeCount; - dup->AnnounceCount = rr->AnnounceCount; - dup->RequireGoodbye = rr->RequireGoodbye; - dup->AnsweredLocalQ = rr->AnsweredLocalQ; - dup->ImmedAnswer = rr->ImmedAnswer; - dup->ImmedUnicast = rr->ImmedUnicast; - dup->ImmedAdditional = rr->ImmedAdditional; - dup->v4Requester = rr->v4Requester; - dup->v6Requester = rr->v6Requester; - dup->ThisAPInterval = rr->ThisAPInterval; - dup->LastAPTime = rr->LastAPTime; - dup->LastMCTime = rr->LastMCTime; - dup->LastMCInterface = rr->LastMCInterface; - dup->Private = rr->Private; - dup->state = rr->state; - rr->RequireGoodbye = mDNSfalse; - rr->AnsweredLocalQ = mDNSfalse; - } - } - } - else - { - // We didn't find our record on the main list; try the DuplicateRecords list instead. - p = &m->DuplicateRecords; - while (*p && *p != rr) p=&(*p)->next; - // If we found our record on the duplicate list, then make sure we don't send a goodbye for it - if (*p) { rr->RequireGoodbye = mDNSfalse; dupList = mDNStrue; } - if (*p) debugf("mDNS_Deregister_internal: Deleting DuplicateRecord %p %##s (%s)", - rr, rr->resrec.name->c, DNSTypeName(rr->resrec.rrtype)); - } - - if (!*p) - { - // No need to log an error message if we already know this is a potentially repeated deregistration - if (drt != mDNS_Dereg_repeat) - LogMsg("mDNS_Deregister_internal: Record %p not found in list %s", rr, ARDisplayString(m,rr)); - return(mStatus_BadReferenceErr); - } - - // If this is a shared record and we've announced it at least once, - // we need to retract that announcement before we delete the record - - // If this is a record (including mDNSInterface_LocalOnly records) for which we've given local-only answers then - // it's tempting to just do "AnswerAllLocalQuestionsWithLocalAuthRecord(m, rr, mDNSfalse)" here, but that would not not be safe. - // The AnswerAllLocalQuestionsWithLocalAuthRecord routine walks the question list invoking client callbacks, using the "m->CurrentQuestion" - // mechanism to cope with the client callback modifying the question list while that's happening. - // However, mDNS_Deregister could have been called from a client callback (e.g. from the domain enumeration callback FoundDomain) - // which means that the "m->CurrentQuestion" mechanism is already in use to protect that list, so we can't use it twice. - // More generally, if we invoke callbacks from within a client callback, then those callbacks could deregister other - // records, thereby invoking yet more callbacks, without limit. - // The solution is to defer delivering the "Remove" events until mDNS_Execute time, just like we do for sending - // actual goodbye packets. - -#ifndef UNICAST_DISABLED - if (AuthRecord_uDNS(rr)) - { - if (rr->RequireGoodbye) - { - if (rr->tcp) { DisposeTCPConn(rr->tcp); rr->tcp = mDNSNULL; } - rr->resrec.RecordType = kDNSRecordTypeDeregistering; - m->LocalRemoveEvents = mDNStrue; - uDNS_DeregisterRecord(m, rr); - // At this point unconditionally we bail out - // Either uDNS_DeregisterRecord will have completed synchronously, and called CompleteDeregistration, - // which calls us back here with RequireGoodbye set to false, or it will have initiated the deregistration - // process and will complete asynchronously. Either way we don't need to do anything more here. - return(mStatus_NoError); - } - // Sometimes the records don't complete proper deregistration i.e., don't wait for a response - // from the server. In that case, if the records have been part of a group update, clear the - // state here. Some recors e.g., AutoTunnel gets reused without ever being completely initialized - rr->updateid = zeroID; - - // We defer cleaning up NAT state only after sending goodbyes. This is important because - // RecordRegistrationGotZoneData guards against creating NAT state if clientContext is non-NULL. - // This happens today when we turn on/off interface where we get multiple network transitions - // and RestartRecordGetZoneData triggers re-registration of the resource records even though - // they may be in Registered state which causes NAT information to be setup multiple times. Defering - // the cleanup here keeps clientContext non-NULL and hence prevents that. Note that cleaning up - // NAT state here takes care of the case where we did not send goodbyes at all. - if (rr->NATinfo.clientContext) - { - mDNS_StopNATOperation_internal(m, &rr->NATinfo); - rr->NATinfo.clientContext = mDNSNULL; - } - if (rr->nta) { CancelGetZoneData(m, rr->nta); rr->nta = mDNSNULL; } - if (rr->tcp) { DisposeTCPConn(rr->tcp); rr->tcp = mDNSNULL; } - } -#endif // UNICAST_DISABLED - - if (RecordType == kDNSRecordTypeUnregistered) - LogMsg("mDNS_Deregister_internal: %s already marked kDNSRecordTypeUnregistered", ARDisplayString(m, rr)); - else if (RecordType == kDNSRecordTypeDeregistering) - { - LogMsg("mDNS_Deregister_internal: %s already marked kDNSRecordTypeDeregistering", ARDisplayString(m, rr)); - return(mStatus_BadReferenceErr); - } - - // Local-only questions don't get remove events for unique records - // We may want to consider changing this code so that we generate local-only question "rmv" - // events (and maybe goodbye packets too) for unique records as well as for shared records - // Note: If we change the logic for this "if" statement, need to ensure that the code in - // CompleteDeregistration() sets the appropriate state variables to gaurantee that "else" - // clause will execute here and the record will be cut from the list. - if (rr->WakeUp.HMAC.l[0] || - (RecordType == kDNSRecordTypeShared && (rr->RequireGoodbye || rr->AnsweredLocalQ))) - { - verbosedebugf("mDNS_Deregister_internal: Starting deregistration for %s", ARDisplayString(m, rr)); - rr->resrec.RecordType = kDNSRecordTypeDeregistering; - rr->resrec.rroriginalttl = 0; - rr->AnnounceCount = rr->WakeUp.HMAC.l[0] ? WakeupCount : (drt == mDNS_Dereg_rapid) ? 1 : GoodbyeCount; - rr->ThisAPInterval = mDNSPlatformOneSecond * 2; - rr->LastAPTime = m->timenow - rr->ThisAPInterval; - m->LocalRemoveEvents = mDNStrue; - if (m->NextScheduledResponse - (m->timenow + mDNSPlatformOneSecond/10) >= 0) - m->NextScheduledResponse = (m->timenow + mDNSPlatformOneSecond/10); - } - else - { - if (!dupList && RRLocalOnly(rr)) - { - AuthGroup *ag = RemoveAuthRecord(m, &m->rrauth, rr); - if (ag->NewLocalOnlyRecords == rr) ag->NewLocalOnlyRecords = rr->next; - } - else - { - *p = rr->next; // Cut this record from the list - if (m->NewLocalRecords == rr) m->NewLocalRecords = rr->next; - } - // If someone is about to look at this, bump the pointer forward - if (m->CurrentRecord == rr) m->CurrentRecord = rr->next; - rr->next = mDNSNULL; - - // Should we generate local remove events here? - // i.e. something like: - // if (rr->AnsweredLocalQ) { AnswerAllLocalQuestionsWithLocalAuthRecord(m, rr, mDNSfalse); rr->AnsweredLocalQ = mDNSfalse; } - - verbosedebugf("mDNS_Deregister_internal: Deleting record for %s", ARDisplayString(m, rr)); - rr->resrec.RecordType = kDNSRecordTypeUnregistered; - - if ((drt == mDNS_Dereg_conflict || drt == mDNS_Dereg_repeat) && RecordType == kDNSRecordTypeShared) - debugf("mDNS_Deregister_internal: Cannot have a conflict on a shared record! %##s (%s)", - rr->resrec.name->c, DNSTypeName(rr->resrec.rrtype)); - - // If we have an update queued up which never executed, give the client a chance to free that memory - if (rr->NewRData) CompleteRDataUpdate(m, rr); // Update our rdata, clear the NewRData pointer, and return memory to the client - - - // CAUTION: MUST NOT do anything more with rr after calling rr->Callback(), because the client's callback function - // is allowed to do anything, including starting/stopping queries, registering/deregistering records, etc. - // In this case the likely client action to the mStatus_MemFree message is to free the memory, - // so any attempt to touch rr after this is likely to lead to a crash. - if (drt != mDNS_Dereg_conflict) - { - mDNS_DropLockBeforeCallback(); // Allow client to legally make mDNS API calls from the callback - LogInfo("mDNS_Deregister_internal: mStatus_MemFree for %s", ARDisplayString(m, rr)); - if (rr->RecordCallback) - rr->RecordCallback(m, rr, mStatus_MemFree); // MUST NOT touch rr after this - mDNS_ReclaimLockAfterCallback(); // Decrement mDNS_reentrancy to block mDNS API calls again - } - else - { - RecordProbeFailure(m, rr); - mDNS_DropLockBeforeCallback(); // Allow client to legally make mDNS API calls from the callback - if (rr->RecordCallback) - rr->RecordCallback(m, rr, mStatus_NameConflict); // MUST NOT touch rr after this - mDNS_ReclaimLockAfterCallback(); // Decrement mDNS_reentrancy to block mDNS API calls again - // Now that we've finished deregistering rr, check our DuplicateRecords list for any that we marked previously. - // Note that with all the client callbacks going on, by the time we get here all the - // records we marked may have been explicitly deregistered by the client anyway. - r2 = m->DuplicateRecords; - while (r2) - { - if (r2->ProbeCount != 0xFF) r2 = r2->next; - else { mDNS_Deregister_internal(m, r2, mDNS_Dereg_conflict); r2 = m->DuplicateRecords; } - } - } - } - mDNS_UpdateAllowSleep(m); - return(mStatus_NoError); - } - -// *************************************************************************** -#if COMPILER_LIKES_PRAGMA_MARK -#pragma mark - -#pragma mark - Packet Sending Functions -#endif - -mDNSlocal void AddRecordToResponseList(AuthRecord ***nrpp, AuthRecord *rr, AuthRecord *add) - { - if (rr->NextResponse == mDNSNULL && *nrpp != &rr->NextResponse) - { - **nrpp = rr; - // NR_AdditionalTo must point to a record with NR_AnswerTo set (and not NR_AdditionalTo) - // If 'add' does not meet this requirement, then follow its NR_AdditionalTo pointer to a record that does - // The referenced record will definitely be acceptable (by recursive application of this rule) - if (add && add->NR_AdditionalTo) add = add->NR_AdditionalTo; - rr->NR_AdditionalTo = add; - *nrpp = &rr->NextResponse; - } - debugf("AddRecordToResponseList: %##s (%s) already in list", rr->resrec.name->c, DNSTypeName(rr->resrec.rrtype)); - } - -mDNSlocal void AddAdditionalsToResponseList(mDNS *const m, AuthRecord *ResponseRecords, AuthRecord ***nrpp, const mDNSInterfaceID InterfaceID) - { - AuthRecord *rr, *rr2; - for (rr=ResponseRecords; rr; rr=rr->NextResponse) // For each record we plan to put - { - // (Note: This is an "if", not a "while". If we add a record, we'll find it again - // later in the "for" loop, and we will follow further "additional" links then.) - if (rr->Additional1 && ResourceRecordIsValidInterfaceAnswer(rr->Additional1, InterfaceID)) - AddRecordToResponseList(nrpp, rr->Additional1, rr); - - if (rr->Additional2 && ResourceRecordIsValidInterfaceAnswer(rr->Additional2, InterfaceID)) - AddRecordToResponseList(nrpp, rr->Additional2, rr); - - // For SRV records, automatically add the Address record(s) for the target host - if (rr->resrec.rrtype == kDNSType_SRV) - { - for (rr2=m->ResourceRecords; rr2; rr2=rr2->next) // Scan list of resource records - if (RRTypeIsAddressType(rr2->resrec.rrtype) && // For all address records (A/AAAA) ... - ResourceRecordIsValidInterfaceAnswer(rr2, InterfaceID) && // ... which are valid for answer ... - rr->resrec.rdatahash == rr2->resrec.namehash && // ... whose name is the name of the SRV target - SameDomainName(&rr->resrec.rdata->u.srv.target, rr2->resrec.name)) - AddRecordToResponseList(nrpp, rr2, rr); - } - else if (RRTypeIsAddressType(rr->resrec.rrtype)) // For A or AAAA, put counterpart as additional - { - for (rr2=m->ResourceRecords; rr2; rr2=rr2->next) // Scan list of resource records - if (RRTypeIsAddressType(rr2->resrec.rrtype) && // For all address records (A/AAAA) ... - ResourceRecordIsValidInterfaceAnswer(rr2, InterfaceID) && // ... which are valid for answer ... - rr->resrec.namehash == rr2->resrec.namehash && // ... and have the same name - SameDomainName(rr->resrec.name, rr2->resrec.name)) - AddRecordToResponseList(nrpp, rr2, rr); - } - else if (rr->resrec.rrtype == kDNSType_PTR) // For service PTR, see if we want to add DeviceInfo record - { - if (ResourceRecordIsValidInterfaceAnswer(&m->DeviceInfo, InterfaceID) && - SameDomainLabel(rr->resrec.rdata->u.name.c, m->DeviceInfo.resrec.name->c)) - AddRecordToResponseList(nrpp, &m->DeviceInfo, rr); - } - } - } - -mDNSlocal void SendDelayedUnicastResponse(mDNS *const m, const mDNSAddr *const dest, const mDNSInterfaceID InterfaceID) - { - AuthRecord *rr; - AuthRecord *ResponseRecords = mDNSNULL; - AuthRecord **nrp = &ResponseRecords; - - // Make a list of all our records that need to be unicast to this destination - for (rr = m->ResourceRecords; rr; rr=rr->next) - { - // If we find we can no longer unicast this answer, clear ImmedUnicast - if (rr->ImmedAnswer == mDNSInterfaceMark || - mDNSSameIPv4Address(rr->v4Requester, onesIPv4Addr) || - mDNSSameIPv6Address(rr->v6Requester, onesIPv6Addr) ) - rr->ImmedUnicast = mDNSfalse; - - if (rr->ImmedUnicast && rr->ImmedAnswer == InterfaceID) - if ((dest->type == mDNSAddrType_IPv4 && mDNSSameIPv4Address(rr->v4Requester, dest->ip.v4)) || - (dest->type == mDNSAddrType_IPv6 && mDNSSameIPv6Address(rr->v6Requester, dest->ip.v6))) - { - rr->ImmedAnswer = mDNSNULL; // Clear the state fields - rr->ImmedUnicast = mDNSfalse; - rr->v4Requester = zerov4Addr; - rr->v6Requester = zerov6Addr; - if (rr->NextResponse == mDNSNULL && nrp != &rr->NextResponse) // rr->NR_AnswerTo - { rr->NR_AnswerTo = (mDNSu8*)~0; *nrp = rr; nrp = &rr->NextResponse; } - } - } - - AddAdditionalsToResponseList(m, ResponseRecords, &nrp, InterfaceID); - - while (ResponseRecords) - { - mDNSu8 *responseptr = m->omsg.data; - mDNSu8 *newptr; - InitializeDNSMessage(&m->omsg.h, zeroID, ResponseFlags); - - // Put answers in the packet - while (ResponseRecords && ResponseRecords->NR_AnswerTo) - { - rr = ResponseRecords; - if (rr->resrec.RecordType & kDNSRecordTypeUniqueMask) - rr->resrec.rrclass |= kDNSClass_UniqueRRSet; // Temporarily set the cache flush bit so PutResourceRecord will set it - newptr = PutResourceRecord(&m->omsg, responseptr, &m->omsg.h.numAnswers, &rr->resrec); - rr->resrec.rrclass &= ~kDNSClass_UniqueRRSet; // Make sure to clear cache flush bit back to normal state - if (!newptr && m->omsg.h.numAnswers) break; // If packet full, send it now - if (newptr) responseptr = newptr; - ResponseRecords = rr->NextResponse; - rr->NextResponse = mDNSNULL; - rr->NR_AnswerTo = mDNSNULL; - rr->NR_AdditionalTo = mDNSNULL; - rr->RequireGoodbye = mDNStrue; - } - - // Add additionals, if there's space - while (ResponseRecords && !ResponseRecords->NR_AnswerTo) - { - rr = ResponseRecords; - if (rr->resrec.RecordType & kDNSRecordTypeUniqueMask) - rr->resrec.rrclass |= kDNSClass_UniqueRRSet; // Temporarily set the cache flush bit so PutResourceRecord will set it - newptr = PutResourceRecord(&m->omsg, responseptr, &m->omsg.h.numAdditionals, &rr->resrec); - rr->resrec.rrclass &= ~kDNSClass_UniqueRRSet; // Make sure to clear cache flush bit back to normal state - - if (newptr) responseptr = newptr; - if (newptr && m->omsg.h.numAnswers) rr->RequireGoodbye = mDNStrue; - else if (rr->resrec.RecordType & kDNSRecordTypeUniqueMask) rr->ImmedAnswer = mDNSInterfaceMark; - ResponseRecords = rr->NextResponse; - rr->NextResponse = mDNSNULL; - rr->NR_AnswerTo = mDNSNULL; - rr->NR_AdditionalTo = mDNSNULL; - } - - if (m->omsg.h.numAnswers) - mDNSSendDNSMessage(m, &m->omsg, responseptr, InterfaceID, mDNSNULL, dest, MulticastDNSPort, mDNSNULL, mDNSNULL); - } - } - -// CompleteDeregistration guarantees that on exit the record will have been cut from the m->ResourceRecords list -// and the client's mStatus_MemFree callback will have been invoked -mDNSexport void CompleteDeregistration(mDNS *const m, AuthRecord *rr) - { - LogInfo("CompleteDeregistration: called for Resource record %s", ARDisplayString(m, rr)); - // Clearing rr->RequireGoodbye signals mDNS_Deregister_internal() that - // it should go ahead and immediately dispose of this registration - rr->resrec.RecordType = kDNSRecordTypeShared; - rr->RequireGoodbye = mDNSfalse; - rr->WakeUp.HMAC = zeroEthAddr; - if (rr->AnsweredLocalQ) { AnswerAllLocalQuestionsWithLocalAuthRecord(m, rr, mDNSfalse); rr->AnsweredLocalQ = mDNSfalse; } - mDNS_Deregister_internal(m, rr, mDNS_Dereg_normal); // Don't touch rr after this - } - -// DiscardDeregistrations is used on shutdown and sleep to discard (forcibly and immediately) -// any deregistering records that remain in the m->ResourceRecords list. -// DiscardDeregistrations calls mDNS_Deregister_internal which can call a user callback, -// which may change the record list and/or question list. -// Any code walking either list must use the CurrentQuestion and/or CurrentRecord mechanism to protect against this. -mDNSlocal void DiscardDeregistrations(mDNS *const m) - { - if (m->CurrentRecord) - LogMsg("DiscardDeregistrations ERROR m->CurrentRecord already set %s", ARDisplayString(m, m->CurrentRecord)); - m->CurrentRecord = m->ResourceRecords; - - while (m->CurrentRecord) - { - AuthRecord *rr = m->CurrentRecord; - if (!AuthRecord_uDNS(rr) && rr->resrec.RecordType == kDNSRecordTypeDeregistering) - CompleteDeregistration(m, rr); // Don't touch rr after this - else - m->CurrentRecord = rr->next; - } - } - -mDNSlocal mStatus GetLabelDecimalValue(const mDNSu8 *const src, mDNSu8 *dst) - { - int i, val = 0; - if (src[0] < 1 || src[0] > 3) return(mStatus_Invalid); - for (i=1; i<=src[0]; i++) - { - if (src[i] < '0' || src[i] > '9') return(mStatus_Invalid); - val = val * 10 + src[i] - '0'; - } - if (val > 255) return(mStatus_Invalid); - *dst = (mDNSu8)val; - return(mStatus_NoError); - } - -mDNSlocal mStatus GetIPv4FromName(mDNSAddr *const a, const domainname *const name) - { - int skip = CountLabels(name) - 6; - if (skip < 0) { LogMsg("GetIPFromName: Need six labels in IPv4 reverse mapping name %##s", name); return mStatus_Invalid; } - if (GetLabelDecimalValue(SkipLeadingLabels(name, skip+3)->c, &a->ip.v4.b[0]) || - GetLabelDecimalValue(SkipLeadingLabels(name, skip+2)->c, &a->ip.v4.b[1]) || - GetLabelDecimalValue(SkipLeadingLabels(name, skip+1)->c, &a->ip.v4.b[2]) || - GetLabelDecimalValue(SkipLeadingLabels(name, skip+0)->c, &a->ip.v4.b[3])) return mStatus_Invalid; - a->type = mDNSAddrType_IPv4; - return(mStatus_NoError); - } - -#define HexVal(X) ( ((X) >= '0' && (X) <= '9') ? ((X) - '0' ) : \ - ((X) >= 'A' && (X) <= 'F') ? ((X) - 'A' + 10) : \ - ((X) >= 'a' && (X) <= 'f') ? ((X) - 'a' + 10) : -1) - -mDNSlocal mStatus GetIPv6FromName(mDNSAddr *const a, const domainname *const name) - { - int i, h, l; - const domainname *n; - - int skip = CountLabels(name) - 34; - if (skip < 0) { LogMsg("GetIPFromName: Need 34 labels in IPv6 reverse mapping name %##s", name); return mStatus_Invalid; } - - n = SkipLeadingLabels(name, skip); - for (i=0; i<16; i++) - { - if (n->c[0] != 1) return mStatus_Invalid; - l = HexVal(n->c[1]); - n = (const domainname *)(n->c + 2); - - if (n->c[0] != 1) return mStatus_Invalid; - h = HexVal(n->c[1]); - n = (const domainname *)(n->c + 2); - - if (l<0 || h<0) return mStatus_Invalid; - a->ip.v6.b[15-i] = (mDNSu8)((h << 4) | l); - } - - a->type = mDNSAddrType_IPv6; - return(mStatus_NoError); - } - -mDNSlocal mDNSs32 ReverseMapDomainType(const domainname *const name) - { - int skip = CountLabels(name) - 2; - if (skip >= 0) - { - const domainname *suffix = SkipLeadingLabels(name, skip); - if (SameDomainName(suffix, (const domainname*)"\x7" "in-addr" "\x4" "arpa")) return mDNSAddrType_IPv4; - if (SameDomainName(suffix, (const domainname*)"\x3" "ip6" "\x4" "arpa")) return mDNSAddrType_IPv6; - } - return(mDNSAddrType_None); - } - -mDNSlocal void SendARP(mDNS *const m, const mDNSu8 op, const AuthRecord *const rr, - const mDNSv4Addr *const spa, const mDNSEthAddr *const tha, const mDNSv4Addr *const tpa, const mDNSEthAddr *const dst) - { - int i; - mDNSu8 *ptr = m->omsg.data; - NetworkInterfaceInfo *intf = FirstInterfaceForID(m, rr->resrec.InterfaceID); - if (!intf) { LogMsg("SendARP: No interface with InterfaceID %p found %s", rr->resrec.InterfaceID, ARDisplayString(m,rr)); return; } - - // 0x00 Destination address - for (i=0; i<6; i++) *ptr++ = dst->b[i]; - - // 0x06 Source address (Note: Since we don't currently set the BIOCSHDRCMPLT option, BPF will fill in the real interface address for us) - for (i=0; i<6; i++) *ptr++ = intf->MAC.b[0]; - - // 0x0C ARP Ethertype (0x0806) - *ptr++ = 0x08; *ptr++ = 0x06; - - // 0x0E ARP header - *ptr++ = 0x00; *ptr++ = 0x01; // Hardware address space; Ethernet = 1 - *ptr++ = 0x08; *ptr++ = 0x00; // Protocol address space; IP = 0x0800 - *ptr++ = 6; // Hardware address length - *ptr++ = 4; // Protocol address length - *ptr++ = 0x00; *ptr++ = op; // opcode; Request = 1, Response = 2 - - // 0x16 Sender hardware address (our MAC address) - for (i=0; i<6; i++) *ptr++ = intf->MAC.b[i]; - - // 0x1C Sender protocol address - for (i=0; i<4; i++) *ptr++ = spa->b[i]; - - // 0x20 Target hardware address - for (i=0; i<6; i++) *ptr++ = tha->b[i]; - - // 0x26 Target protocol address - for (i=0; i<4; i++) *ptr++ = tpa->b[i]; - - // 0x2A Total ARP Packet length 42 bytes - mDNSPlatformSendRawPacket(m->omsg.data, ptr, rr->resrec.InterfaceID); - } - -mDNSlocal mDNSu16 CheckSum(const void *const data, mDNSs32 length, mDNSu32 sum) - { - const mDNSu16 *ptr = data; - while (length > 0) { length -= 2; sum += *ptr++; } - sum = (sum & 0xFFFF) + (sum >> 16); - sum = (sum & 0xFFFF) + (sum >> 16); - return(sum != 0xFFFF ? sum : 0); - } - -mDNSlocal mDNSu16 IPv6CheckSum(const mDNSv6Addr *const src, const mDNSv6Addr *const dst, const mDNSu8 protocol, const void *const data, const mDNSu32 length) - { - IPv6PseudoHeader ph; - ph.src = *src; - ph.dst = *dst; - ph.len.b[0] = length >> 24; - ph.len.b[1] = length >> 16; - ph.len.b[2] = length >> 8; - ph.len.b[3] = length; - ph.pro.b[0] = 0; - ph.pro.b[1] = 0; - ph.pro.b[2] = 0; - ph.pro.b[3] = protocol; - return CheckSum(&ph, sizeof(ph), CheckSum(data, length, 0)); - } - -mDNSlocal void SendNDP(mDNS *const m, const mDNSu8 op, const mDNSu8 flags, const AuthRecord *const rr, - const mDNSv6Addr *const spa, const mDNSEthAddr *const tha, const mDNSv6Addr *const tpa, const mDNSEthAddr *const dst) - { - int i; - mDNSOpaque16 checksum; - mDNSu8 *ptr = m->omsg.data; - // Some recipient hosts seem to ignore Neighbor Solicitations if the IPv6-layer destination address is not the - // appropriate IPv6 solicited node multicast address, so we use that IPv6-layer destination address, even though - // at the Ethernet-layer we unicast the packet to the intended target, to avoid wasting network bandwidth. - const mDNSv6Addr mc = { { 0xFF,0x02,0x00,0x00, 0,0,0,0, 0,0,0,1, 0xFF,tpa->b[0xD],tpa->b[0xE],tpa->b[0xF] } }; - const mDNSv6Addr *const v6dst = (op == NDP_Sol) ? &mc : tpa; - NetworkInterfaceInfo *intf = FirstInterfaceForID(m, rr->resrec.InterfaceID); - if (!intf) { LogMsg("SendNDP: No interface with InterfaceID %p found %s", rr->resrec.InterfaceID, ARDisplayString(m,rr)); return; } - - // 0x00 Destination address - for (i=0; i<6; i++) *ptr++ = dst->b[i]; - // Right now we only send Neighbor Solicitations to verify whether the host we're proxying for has gone to sleep yet. - // Since we know who we're looking for, we send it via Ethernet-layer unicast, rather than bothering every host on the - // link with a pointless link-layer multicast. - // Should we want to send traditional Neighbor Solicitations in the future, where we really don't know in advance what - // Ethernet-layer address we're looking for, we'll need to send to the appropriate Ethernet-layer multicast address: - // *ptr++ = 0x33; - // *ptr++ = 0x33; - // *ptr++ = 0xFF; - // *ptr++ = tpa->b[0xD]; - // *ptr++ = tpa->b[0xE]; - // *ptr++ = tpa->b[0xF]; - - // 0x06 Source address (Note: Since we don't currently set the BIOCSHDRCMPLT option, BPF will fill in the real interface address for us) - for (i=0; i<6; i++) *ptr++ = (tha ? *tha : intf->MAC).b[i]; - - // 0x0C IPv6 Ethertype (0x86DD) - *ptr++ = 0x86; *ptr++ = 0xDD; - - // 0x0E IPv6 header - *ptr++ = 0x60; *ptr++ = 0x00; *ptr++ = 0x00; *ptr++ = 0x00; // Version, Traffic Class, Flow Label - *ptr++ = 0x00; *ptr++ = 0x20; // Length - *ptr++ = 0x3A; // Protocol == ICMPv6 - *ptr++ = 0xFF; // Hop Limit - - // 0x16 Sender IPv6 address - for (i=0; i<16; i++) *ptr++ = spa->b[i]; - - // 0x26 Destination IPv6 address - for (i=0; i<16; i++) *ptr++ = v6dst->b[i]; - - // 0x36 NDP header - *ptr++ = op; // 0x87 == Neighbor Solicitation, 0x88 == Neighbor Advertisement - *ptr++ = 0x00; // Code - *ptr++ = 0x00; *ptr++ = 0x00; // Checksum placeholder (0x38, 0x39) - *ptr++ = flags; - *ptr++ = 0x00; *ptr++ = 0x00; *ptr++ = 0x00; - - if (op == NDP_Sol) // Neighbor Solicitation. The NDP "target" is the address we seek. - { - // 0x3E NDP target. - for (i=0; i<16; i++) *ptr++ = tpa->b[i]; - // 0x4E Source Link-layer Address - // - // MUST NOT be included when the source IP address is the unspecified address. - // Otherwise, on link layers that have addresses this option MUST be included - // in multicast solicitations and SHOULD be included in unicast solicitations. - if (!mDNSIPv6AddressIsZero(*spa)) - { - *ptr++ = NDP_SrcLL; // Option Type 1 == Source Link-layer Address - *ptr++ = 0x01; // Option length 1 (in units of 8 octets) - for (i=0; i<6; i++) *ptr++ = (tha ? *tha : intf->MAC).b[i]; - } - } - else // Neighbor Advertisement. The NDP "target" is the address we're giving information about. - { - // 0x3E NDP target. - for (i=0; i<16; i++) *ptr++ = spa->b[i]; - // 0x4E Target Link-layer Address - *ptr++ = NDP_TgtLL; // Option Type 2 == Target Link-layer Address - *ptr++ = 0x01; // Option length 1 (in units of 8 octets) - for (i=0; i<6; i++) *ptr++ = (tha ? *tha : intf->MAC).b[i]; - } - - // 0x4E or 0x56 Total NDP Packet length 78 or 86 bytes - m->omsg.data[0x13] = ptr - &m->omsg.data[0x36]; // Compute actual length - checksum.NotAnInteger = ~IPv6CheckSum(spa, v6dst, 0x3A, &m->omsg.data[0x36], m->omsg.data[0x13]); - m->omsg.data[0x38] = checksum.b[0]; - m->omsg.data[0x39] = checksum.b[1]; - - mDNSPlatformSendRawPacket(m->omsg.data, ptr, rr->resrec.InterfaceID); - } - -mDNSlocal void SetupOwnerOpt(const mDNS *const m, const NetworkInterfaceInfo *const intf, rdataOPT *const owner) - { - owner->u.owner.vers = 0; - owner->u.owner.seq = m->SleepSeqNum; - owner->u.owner.HMAC = m->PrimaryMAC; - owner->u.owner.IMAC = intf->MAC; - owner->u.owner.password = zeroEthAddr; - - // Don't try to compute the optlen until *after* we've set up the data fields - // Right now the DNSOpt_Owner_Space macro does not depend on the owner->u.owner being set up correctly, but in the future it might - owner->opt = kDNSOpt_Owner; - owner->optlen = DNSOpt_Owner_Space(&m->PrimaryMAC, &intf->MAC) - 4; - } - -mDNSlocal void GrantUpdateCredit(AuthRecord *rr) - { - if (++rr->UpdateCredits >= kMaxUpdateCredits) rr->NextUpdateCredit = 0; - else rr->NextUpdateCredit = NonZeroTime(rr->NextUpdateCredit + kUpdateCreditRefreshInterval); - } - -// Note about acceleration of announcements to facilitate automatic coalescing of -// multiple independent threads of announcements into a single synchronized thread: -// The announcements in the packet may be at different stages of maturity; -// One-second interval, two-second interval, four-second interval, and so on. -// After we've put in all the announcements that are due, we then consider -// whether there are other nearly-due announcements that are worth accelerating. -// To be eligible for acceleration, a record MUST NOT be older (further along -// its timeline) than the most mature record we've already put in the packet. -// In other words, younger records can have their timelines accelerated to catch up -// with their elder bretheren; this narrows the age gap and helps them eventually get in sync. -// Older records cannot have their timelines accelerated; this would just widen -// the gap between them and their younger bretheren and get them even more out of sync. - -// Note: SendResponses calls mDNS_Deregister_internal which can call a user callback, which may change -// the record list and/or question list. -// Any code walking either list must use the CurrentQuestion and/or CurrentRecord mechanism to protect against this. -mDNSlocal void SendResponses(mDNS *const m) - { - int pktcount = 0; - AuthRecord *rr, *r2; - mDNSs32 maxExistingAnnounceInterval = 0; - const NetworkInterfaceInfo *intf = GetFirstActiveInterface(m->HostInterfaces); - - m->NextScheduledResponse = m->timenow + 0x78000000; - - if (m->SleepState == SleepState_Transferring) RetrySPSRegistrations(m); - - for (rr = m->ResourceRecords; rr; rr=rr->next) - if (rr->ImmedUnicast) - { - mDNSAddr v4 = { mDNSAddrType_IPv4, {{{0}}} }; - mDNSAddr v6 = { mDNSAddrType_IPv6, {{{0}}} }; - v4.ip.v4 = rr->v4Requester; - v6.ip.v6 = rr->v6Requester; - if (!mDNSIPv4AddressIsZero(rr->v4Requester)) SendDelayedUnicastResponse(m, &v4, rr->ImmedAnswer); - if (!mDNSIPv6AddressIsZero(rr->v6Requester)) SendDelayedUnicastResponse(m, &v6, rr->ImmedAnswer); - if (rr->ImmedUnicast) - { - LogMsg("SendResponses: ERROR: rr->ImmedUnicast still set: %s", ARDisplayString(m, rr)); - rr->ImmedUnicast = mDNSfalse; - } - } - - // *** - // *** 1. Setup: Set the SendRNow and ImmedAnswer fields to indicate which interface(s) the records need to be sent on - // *** - - // Run through our list of records, and decide which ones we're going to announce on all interfaces - for (rr = m->ResourceRecords; rr; rr=rr->next) - { - while (rr->NextUpdateCredit && m->timenow - rr->NextUpdateCredit >= 0) GrantUpdateCredit(rr); - if (TimeToAnnounceThisRecord(rr, m->timenow)) - { - if (rr->resrec.RecordType == kDNSRecordTypeDeregistering) - { - if (!rr->WakeUp.HMAC.l[0]) - { - if (rr->AnnounceCount) rr->ImmedAnswer = mDNSInterfaceMark; // Send goodbye packet on all interfaces - } - else - { - LogSPS("SendResponses: Sending wakeup %2d for %.6a %s", rr->AnnounceCount-3, &rr->WakeUp.IMAC, ARDisplayString(m, rr)); - SendWakeup(m, rr->resrec.InterfaceID, &rr->WakeUp.IMAC, &rr->WakeUp.password); - for (r2 = rr; r2; r2=r2->next) - if (r2->AnnounceCount && r2->resrec.InterfaceID == rr->resrec.InterfaceID && mDNSSameEthAddress(&r2->WakeUp.IMAC, &rr->WakeUp.IMAC)) - { - // For now we only want to send a single Unsolicited Neighbor Advertisement restoring the address to the original - // owner, because these packets can cause some IPv6 stacks to falsely conclude that there's an address conflict. - if (r2->AddressProxy.type == mDNSAddrType_IPv6 && r2->AnnounceCount == WakeupCount) - { - LogSPS("NDP Announcement %2d Releasing traffic for H-MAC %.6a I-MAC %.6a %s", - r2->AnnounceCount-3, &r2->WakeUp.HMAC, &r2->WakeUp.IMAC, ARDisplayString(m,r2)); - SendNDP(m, NDP_Adv, NDP_Override, r2, &r2->AddressProxy.ip.v6, &r2->WakeUp.IMAC, &AllHosts_v6, &AllHosts_v6_Eth); - } - r2->LastAPTime = m->timenow; - // After 15 wakeups without success (maybe host has left the network) send three goodbyes instead - if (--r2->AnnounceCount <= GoodbyeCount) r2->WakeUp.HMAC = zeroEthAddr; - } - } - } - else if (ResourceRecordIsValidAnswer(rr)) - { - if (rr->AddressProxy.type) - { - rr->AnnounceCount--; - rr->ThisAPInterval *= 2; - rr->LastAPTime = m->timenow; - if (rr->AddressProxy.type == mDNSAddrType_IPv4) - { - LogSPS("ARP Announcement %2d Capturing traffic for H-MAC %.6a I-MAC %.6a %s", - rr->AnnounceCount, &rr->WakeUp.HMAC, &rr->WakeUp.IMAC, ARDisplayString(m,rr)); - SendARP(m, 1, rr, &rr->AddressProxy.ip.v4, &zeroEthAddr, &rr->AddressProxy.ip.v4, &onesEthAddr); - } - else if (rr->AddressProxy.type == mDNSAddrType_IPv6) - { - LogSPS("NDP Announcement %2d Capturing traffic for H-MAC %.6a I-MAC %.6a %s", - rr->AnnounceCount, &rr->WakeUp.HMAC, &rr->WakeUp.IMAC, ARDisplayString(m,rr)); - SendNDP(m, NDP_Adv, NDP_Override, rr, &rr->AddressProxy.ip.v6, mDNSNULL, &AllHosts_v6, &AllHosts_v6_Eth); - } - } - else - { - rr->ImmedAnswer = mDNSInterfaceMark; // Send on all interfaces - if (maxExistingAnnounceInterval < rr->ThisAPInterval) - maxExistingAnnounceInterval = rr->ThisAPInterval; - if (rr->UpdateBlocked) rr->UpdateBlocked = 0; - } - } - } - } - - // Any interface-specific records we're going to send are marked as being sent on all appropriate interfaces (which is just one) - // Eligible records that are more than half-way to their announcement time are accelerated - for (rr = m->ResourceRecords; rr; rr=rr->next) - if ((rr->resrec.InterfaceID && rr->ImmedAnswer) || - (rr->ThisAPInterval <= maxExistingAnnounceInterval && - TimeToAnnounceThisRecord(rr, m->timenow + rr->ThisAPInterval/2) && - !rr->AddressProxy.type && // Don't include ARP Annoucements when considering which records to accelerate - ResourceRecordIsValidAnswer(rr))) - rr->ImmedAnswer = mDNSInterfaceMark; // Send on all interfaces - - // When sending SRV records (particularly when announcing a new service) automatically add related Address record(s) as additionals - // Note: Currently all address records are interface-specific, so it's safe to set ImmedAdditional to their InterfaceID, - // which will be non-null. If by some chance there is an address record that's not interface-specific (should never happen) - // then all that means is that it won't get sent -- which would not be the end of the world. - for (rr = m->ResourceRecords; rr; rr=rr->next) - { - if (rr->ImmedAnswer && rr->resrec.rrtype == kDNSType_SRV) - for (r2=m->ResourceRecords; r2; r2=r2->next) // Scan list of resource records - if (RRTypeIsAddressType(r2->resrec.rrtype) && // For all address records (A/AAAA) ... - ResourceRecordIsValidAnswer(r2) && // ... which are valid for answer ... - rr->LastMCTime - r2->LastMCTime >= 0 && // ... which we have not sent recently ... - rr->resrec.rdatahash == r2->resrec.namehash && // ... whose name is the name of the SRV target - SameDomainName(&rr->resrec.rdata->u.srv.target, r2->resrec.name) && - (rr->ImmedAnswer == mDNSInterfaceMark || rr->ImmedAnswer == r2->resrec.InterfaceID)) - r2->ImmedAdditional = r2->resrec.InterfaceID; // ... then mark this address record for sending too - // We also make sure we send the DeviceInfo TXT record too, if necessary - // We check for RecordType == kDNSRecordTypeShared because we don't want to tag the - // DeviceInfo TXT record onto a goodbye packet (RecordType == kDNSRecordTypeDeregistering). - if (rr->ImmedAnswer && rr->resrec.RecordType == kDNSRecordTypeShared && rr->resrec.rrtype == kDNSType_PTR) - if (ResourceRecordIsValidAnswer(&m->DeviceInfo) && SameDomainLabel(rr->resrec.rdata->u.name.c, m->DeviceInfo.resrec.name->c)) - { - if (!m->DeviceInfo.ImmedAnswer) m->DeviceInfo.ImmedAnswer = rr->ImmedAnswer; - else m->DeviceInfo.ImmedAnswer = mDNSInterfaceMark; - } - } - - // If there's a record which is supposed to be unique that we're going to send, then make sure that we give - // the whole RRSet as an atomic unit. That means that if we have any other records with the same name/type/class - // then we need to mark them for sending too. Otherwise, if we set the kDNSClass_UniqueRRSet bit on a - // record, then other RRSet members that have not been sent recently will get flushed out of client caches. - // -- If a record is marked to be sent on a certain interface, make sure the whole set is marked to be sent on that interface - // -- If any record is marked to be sent on all interfaces, make sure the whole set is marked to be sent on all interfaces - for (rr = m->ResourceRecords; rr; rr=rr->next) - if (rr->resrec.RecordType & kDNSRecordTypeUniqueMask) - { - if (rr->ImmedAnswer) // If we're sending this as answer, see that its whole RRSet is similarly marked - { - for (r2 = m->ResourceRecords; r2; r2=r2->next) - if (ResourceRecordIsValidAnswer(r2)) - if (r2->ImmedAnswer != mDNSInterfaceMark && - r2->ImmedAnswer != rr->ImmedAnswer && SameResourceRecordSignature(r2, rr)) - r2->ImmedAnswer = !r2->ImmedAnswer ? rr->ImmedAnswer : mDNSInterfaceMark; - } - else if (rr->ImmedAdditional) // If we're sending this as additional, see that its whole RRSet is similarly marked - { - for (r2 = m->ResourceRecords; r2; r2=r2->next) - if (ResourceRecordIsValidAnswer(r2)) - if (r2->ImmedAdditional != rr->ImmedAdditional && SameResourceRecordSignature(r2, rr)) - r2->ImmedAdditional = rr->ImmedAdditional; - } - } - - // Now set SendRNow state appropriately - for (rr = m->ResourceRecords; rr; rr=rr->next) - { - if (rr->ImmedAnswer == mDNSInterfaceMark) // Sending this record on all appropriate interfaces - { - rr->SendRNow = !intf ? mDNSNULL : (rr->resrec.InterfaceID) ? rr->resrec.InterfaceID : intf->InterfaceID; - rr->ImmedAdditional = mDNSNULL; // No need to send as additional if sending as answer - rr->LastMCTime = m->timenow; - rr->LastMCInterface = rr->ImmedAnswer; - // If we're announcing this record, and it's at least half-way to its ordained time, then consider this announcement done - if (TimeToAnnounceThisRecord(rr, m->timenow + rr->ThisAPInterval/2)) - { - rr->AnnounceCount--; - if (rr->resrec.RecordType != kDNSRecordTypeDeregistering) - rr->ThisAPInterval *= 2; - rr->LastAPTime = m->timenow; - debugf("Announcing %##s (%s) %d", rr->resrec.name->c, DNSTypeName(rr->resrec.rrtype), rr->AnnounceCount); - } - } - else if (rr->ImmedAnswer) // Else, just respond to a single query on single interface: - { - rr->SendRNow = rr->ImmedAnswer; // Just respond on that interface - rr->ImmedAdditional = mDNSNULL; // No need to send as additional too - rr->LastMCTime = m->timenow; - rr->LastMCInterface = rr->ImmedAnswer; - } - SetNextAnnounceProbeTime(m, rr); - //if (rr->SendRNow) LogMsg("%-15.4a %s", &rr->v4Requester, ARDisplayString(m, rr)); - } - - // *** - // *** 2. Loop through interface list, sending records as appropriate - // *** - - while (intf) - { - const int OwnerRecordSpace = (m->AnnounceOwner && intf->MAC.l[0]) ? DNSOpt_Header_Space + DNSOpt_Owner_Space(&m->PrimaryMAC, &intf->MAC) : 0; - int numDereg = 0; - int numAnnounce = 0; - int numAnswer = 0; - mDNSu8 *responseptr = m->omsg.data; - mDNSu8 *newptr; - InitializeDNSMessage(&m->omsg.h, zeroID, ResponseFlags); - - // First Pass. Look for: - // 1. Deregistering records that need to send their goodbye packet - // 2. Updated records that need to retract their old data - // 3. Answers and announcements we need to send - for (rr = m->ResourceRecords; rr; rr=rr->next) - { - - // Skip this interface if the record InterfaceID is *Any and the record is not - // appropriate for the interface type. - if ((rr->SendRNow == intf->InterfaceID) && - ((rr->resrec.InterfaceID == mDNSInterface_Any) && !mDNSPlatformValidRecordForInterface(rr, intf))) - { - LogInfo("SendResponses: Not Sending %s, on %s", ARDisplayString(m, rr), InterfaceNameForID(m, rr->SendRNow)); - rr->SendRNow = GetNextActiveInterfaceID(intf); - } - else if (rr->SendRNow == intf->InterfaceID) - { - RData *OldRData = rr->resrec.rdata; - mDNSu16 oldrdlength = rr->resrec.rdlength; - mDNSu8 active = (mDNSu8) - (rr->resrec.RecordType != kDNSRecordTypeDeregistering && - (m->SleepState != SleepState_Sleeping || intf->SPSAddr[0].type || intf->SPSAddr[1].type || intf->SPSAddr[2].type)); - newptr = mDNSNULL; - if (rr->NewRData && active) - { - // See if we should send a courtesy "goodbye" for the old data before we replace it. - if (ResourceRecordIsValidAnswer(rr) && rr->resrec.RecordType == kDNSRecordTypeShared && rr->RequireGoodbye) - { - newptr = PutRR_OS_TTL(responseptr, &m->omsg.h.numAnswers, &rr->resrec, 0); - if (newptr) { responseptr = newptr; numDereg++; rr->RequireGoodbye = mDNSfalse; } - else continue; // If this packet is already too full to hold the goodbye for this record, skip it for now and we'll retry later - } - SetNewRData(&rr->resrec, rr->NewRData, rr->newrdlength); - } - - if (rr->resrec.RecordType & kDNSRecordTypeUniqueMask) - rr->resrec.rrclass |= kDNSClass_UniqueRRSet; // Temporarily set the cache flush bit so PutResourceRecord will set it - newptr = PutRR_OS_TTL(responseptr, &m->omsg.h.numAnswers, &rr->resrec, active ? rr->resrec.rroriginalttl : 0); - rr->resrec.rrclass &= ~kDNSClass_UniqueRRSet; // Make sure to clear cache flush bit back to normal state - if (newptr) - { - responseptr = newptr; - rr->RequireGoodbye = active; - if (rr->resrec.RecordType == kDNSRecordTypeDeregistering) numDereg++; - else if (rr->LastAPTime == m->timenow) numAnnounce++; else numAnswer++; - } - - if (rr->NewRData && active) - SetNewRData(&rr->resrec, OldRData, oldrdlength); - - // The first time through (pktcount==0), if this record is verified unique - // (i.e. typically A, AAAA, SRV, TXT and reverse-mapping PTR), set the flag to add an NSEC too. - if (!pktcount && active && (rr->resrec.RecordType & kDNSRecordTypeActiveUniqueMask) && !rr->SendNSECNow) - rr->SendNSECNow = mDNSInterfaceMark; - - if (newptr) // If succeeded in sending, advance to next interface - { - // If sending on all interfaces, go to next interface; else we're finished now - if (rr->ImmedAnswer == mDNSInterfaceMark && rr->resrec.InterfaceID == mDNSInterface_Any) - rr->SendRNow = GetNextActiveInterfaceID(intf); - else - rr->SendRNow = mDNSNULL; - } - } - } - - // Second Pass. Add additional records, if there's space. - newptr = responseptr; - for (rr = m->ResourceRecords; rr; rr=rr->next) - if (rr->ImmedAdditional == intf->InterfaceID) - if (ResourceRecordIsValidAnswer(rr)) - { - // If we have at least one answer already in the packet, then plan to add additionals too - mDNSBool SendAdditional = (m->omsg.h.numAnswers > 0); - - // If we're not planning to send any additionals, but this record is a unique one, then - // make sure we haven't already sent any other members of its RRSet -- if we have, then they - // will have had the cache flush bit set, so now we need to finish the job and send the rest. - if (!SendAdditional && (rr->resrec.RecordType & kDNSRecordTypeUniqueMask)) - { - const AuthRecord *a; - for (a = m->ResourceRecords; a; a=a->next) - if (a->LastMCTime == m->timenow && - a->LastMCInterface == intf->InterfaceID && - SameResourceRecordSignature(a, rr)) { SendAdditional = mDNStrue; break; } - } - if (!SendAdditional) // If we don't want to send this after all, - rr->ImmedAdditional = mDNSNULL; // then cancel its ImmedAdditional field - else if (newptr) // Else, try to add it if we can - { - // The first time through (pktcount==0), if this record is verified unique - // (i.e. typically A, AAAA, SRV, TXT and reverse-mapping PTR), set the flag to add an NSEC too. - if (!pktcount && (rr->resrec.RecordType & kDNSRecordTypeActiveUniqueMask) && !rr->SendNSECNow) - rr->SendNSECNow = mDNSInterfaceMark; - - if (rr->resrec.RecordType & kDNSRecordTypeUniqueMask) - rr->resrec.rrclass |= kDNSClass_UniqueRRSet; // Temporarily set the cache flush bit so PutResourceRecord will set it - newptr = PutRR_OS(newptr, &m->omsg.h.numAdditionals, &rr->resrec); - rr->resrec.rrclass &= ~kDNSClass_UniqueRRSet; // Make sure to clear cache flush bit back to normal state - if (newptr) - { - responseptr = newptr; - rr->ImmedAdditional = mDNSNULL; - rr->RequireGoodbye = mDNStrue; - // If we successfully put this additional record in the packet, we record LastMCTime & LastMCInterface. - // This matters particularly in the case where we have more than one IPv6 (or IPv4) address, because otherwise, - // when we see our own multicast with the cache flush bit set, if we haven't set LastMCTime, then we'll get - // all concerned and re-announce our record again to make sure it doesn't get flushed from peer caches. - rr->LastMCTime = m->timenow; - rr->LastMCInterface = intf->InterfaceID; - } - } - } - - // Third Pass. Add NSEC records, if there's space. - // When we're generating an NSEC record in response to a specify query for that type - // (recognized by rr->SendNSECNow == intf->InterfaceID) we should really put the NSEC in the Answer Section, - // not Additional Section, but for now it's easier to handle both cases in this Additional Section loop here. - for (rr = m->ResourceRecords; rr; rr=rr->next) - if (rr->SendNSECNow == mDNSInterfaceMark || rr->SendNSECNow == intf->InterfaceID) - { - AuthRecord nsec; - mDNS_SetupResourceRecord(&nsec, mDNSNULL, mDNSInterface_Any, kDNSType_NSEC, rr->resrec.rroriginalttl, kDNSRecordTypeUnique, AuthRecordAny, mDNSNULL, mDNSNULL); - nsec.resrec.rrclass |= kDNSClass_UniqueRRSet; - AssignDomainName(&nsec.namestorage, rr->resrec.name); - mDNSPlatformMemZero(nsec.rdatastorage.u.nsec.bitmap, sizeof(nsec.rdatastorage.u.nsec.bitmap)); - for (r2 = m->ResourceRecords; r2; r2=r2->next) - if (ResourceRecordIsValidAnswer(r2) && SameResourceRecordNameClassInterface(r2, rr)) - { - if (r2->resrec.rrtype >= kDNSQType_ANY) { LogMsg("Can't create NSEC for record %s", ARDisplayString(m, r2)); break; } - else nsec.rdatastorage.u.nsec.bitmap[r2->resrec.rrtype >> 3] |= 128 >> (r2->resrec.rrtype & 7); - } - newptr = responseptr; - if (!r2) // If we successfully built our NSEC record, add it to the packet now - { - newptr = PutRR_OS(responseptr, &m->omsg.h.numAdditionals, &nsec.resrec); - if (newptr) responseptr = newptr; - } - - // If we successfully put the NSEC record, clear the SendNSECNow flag - // If we consider this NSEC optional, then we unconditionally clear the SendNSECNow flag, even if we fail to put this additional record - if (newptr || rr->SendNSECNow == mDNSInterfaceMark) - { - rr->SendNSECNow = mDNSNULL; - // Run through remainder of list clearing SendNSECNow flag for all other records which would generate the same NSEC - for (r2 = rr->next; r2; r2=r2->next) - if (SameResourceRecordNameClassInterface(r2, rr)) - if (r2->SendNSECNow == mDNSInterfaceMark || r2->SendNSECNow == intf->InterfaceID) - r2->SendNSECNow = mDNSNULL; - } - } - - if (m->omsg.h.numAnswers || m->omsg.h.numAdditionals) - { - // If we have data to send, add OWNER option if necessary, then send packet - - if (OwnerRecordSpace) - { - AuthRecord opt; - mDNS_SetupResourceRecord(&opt, mDNSNULL, mDNSInterface_Any, kDNSType_OPT, kStandardTTL, kDNSRecordTypeKnownUnique, AuthRecordAny, mDNSNULL, mDNSNULL); - opt.resrec.rrclass = NormalMaxDNSMessageData; - opt.resrec.rdlength = sizeof(rdataOPT); // One option in this OPT record - opt.resrec.rdestimate = sizeof(rdataOPT); - SetupOwnerOpt(m, intf, &opt.resrec.rdata->u.opt[0]); - newptr = PutResourceRecord(&m->omsg, responseptr, &m->omsg.h.numAdditionals, &opt.resrec); - if (newptr) { responseptr = newptr; LogSPS("SendResponses put %s", ARDisplayString(m, &opt)); } - else if (m->omsg.h.numAnswers + m->omsg.h.numAuthorities + m->omsg.h.numAdditionals == 1) - LogSPS("SendResponses: No space in packet for Owner OPT record (%d/%d/%d/%d) %s", - m->omsg.h.numQuestions, m->omsg.h.numAnswers, m->omsg.h.numAuthorities, m->omsg.h.numAdditionals, ARDisplayString(m, &opt)); - else - LogMsg("SendResponses: How did we fail to have space for Owner OPT record (%d/%d/%d/%d) %s", - m->omsg.h.numQuestions, m->omsg.h.numAnswers, m->omsg.h.numAuthorities, m->omsg.h.numAdditionals, ARDisplayString(m, &opt)); - } - - debugf("SendResponses: Sending %d Deregistration%s, %d Announcement%s, %d Answer%s, %d Additional%s on %p", - numDereg, numDereg == 1 ? "" : "s", - numAnnounce, numAnnounce == 1 ? "" : "s", - numAnswer, numAnswer == 1 ? "" : "s", - m->omsg.h.numAdditionals, m->omsg.h.numAdditionals == 1 ? "" : "s", intf->InterfaceID); - - if (intf->IPv4Available) mDNSSendDNSMessage(m, &m->omsg, responseptr, intf->InterfaceID, mDNSNULL, &AllDNSLinkGroup_v4, MulticastDNSPort, mDNSNULL, mDNSNULL); - if (intf->IPv6Available) mDNSSendDNSMessage(m, &m->omsg, responseptr, intf->InterfaceID, mDNSNULL, &AllDNSLinkGroup_v6, MulticastDNSPort, mDNSNULL, mDNSNULL); - if (!m->SuppressSending) m->SuppressSending = NonZeroTime(m->timenow + (mDNSPlatformOneSecond+9)/10); - if (++pktcount >= 1000) { LogMsg("SendResponses exceeded loop limit %d: giving up", pktcount); break; } - // There might be more things to send on this interface, so go around one more time and try again. - } - else // Nothing more to send on this interface; go to next - { - const NetworkInterfaceInfo *next = GetFirstActiveInterface(intf->next); - #if MDNS_DEBUGMSGS && 0 - const char *const msg = next ? "SendResponses: Nothing more on %p; moving to %p" : "SendResponses: Nothing more on %p"; - debugf(msg, intf, next); - #endif - intf = next; - pktcount = 0; // When we move to a new interface, reset packet count back to zero -- NSEC generation logic uses it - } - } - - // *** - // *** 3. Cleanup: Now that everything is sent, call client callback functions, and reset state variables - // *** - - if (m->CurrentRecord) - LogMsg("SendResponses ERROR m->CurrentRecord already set %s", ARDisplayString(m, m->CurrentRecord)); - m->CurrentRecord = m->ResourceRecords; - while (m->CurrentRecord) - { - rr = m->CurrentRecord; - m->CurrentRecord = rr->next; - - if (rr->SendRNow) - { - if (rr->ARType != AuthRecordLocalOnly && rr->ARType != AuthRecordP2P) - LogMsg("SendResponses: No active interface %p to send: %p %02X %s", rr->SendRNow, rr->resrec.InterfaceID, rr->resrec.RecordType, ARDisplayString(m, rr)); - rr->SendRNow = mDNSNULL; - } - - if (rr->ImmedAnswer || rr->resrec.RecordType == kDNSRecordTypeDeregistering) - { - if (rr->NewRData) CompleteRDataUpdate(m, rr); // Update our rdata, clear the NewRData pointer, and return memory to the client - - if (rr->resrec.RecordType == kDNSRecordTypeDeregistering && rr->AnnounceCount == 0) - { - // For Unicast, when we get the response from the server, we will call CompleteDeregistration - if (!AuthRecord_uDNS(rr)) CompleteDeregistration(m, rr); // Don't touch rr after this - } - else - { - rr->ImmedAnswer = mDNSNULL; - rr->ImmedUnicast = mDNSfalse; - rr->v4Requester = zerov4Addr; - rr->v6Requester = zerov6Addr; - } - } - } - verbosedebugf("SendResponses: Next in %ld ticks", m->NextScheduledResponse - m->timenow); - } - -// Calling CheckCacheExpiration() is an expensive operation because it has to look at the entire cache, -// so we want to be lazy about how frequently we do it. -// 1. If a cache record is currently referenced by *no* active questions, -// then we don't mind expiring it up to a minute late (who will know?) -// 2. Else, if a cache record is due for some of its final expiration queries, -// we'll allow them to be late by up to 2% of the TTL -// 3. Else, if a cache record has completed all its final expiration queries without success, -// and is expiring, and had an original TTL more than ten seconds, we'll allow it to be one second late -// 4. Else, it is expiring and had an original TTL of ten seconds or less (includes explicit goodbye packets), -// so allow at most 1/10 second lateness -// 5. For records with rroriginalttl set to zero, that means we really want to delete them immediately -// (we have a new record with DelayDelivery set, waiting for the old record to go away before we can notify clients). -#define CacheCheckGracePeriod(RR) ( \ - ((RR)->CRActiveQuestion == mDNSNULL ) ? (60 * mDNSPlatformOneSecond) : \ - ((RR)->UnansweredQueries < MaxUnansweredQueries) ? (TicksTTL(rr)/50) : \ - ((RR)->resrec.rroriginalttl > 10 ) ? (mDNSPlatformOneSecond) : \ - ((RR)->resrec.rroriginalttl > 0 ) ? (mDNSPlatformOneSecond/10) : 0) - -#define NextCacheCheckEvent(RR) ((RR)->NextRequiredQuery + CacheCheckGracePeriod(RR)) - -mDNSexport void ScheduleNextCacheCheckTime(mDNS *const m, const mDNSu32 slot, const mDNSs32 event) - { - if (m->rrcache_nextcheck[slot] - event > 0) - m->rrcache_nextcheck[slot] = event; - if (m->NextCacheCheck - event > 0) - m->NextCacheCheck = event; - } - -// Note: MUST call SetNextCacheCheckTimeForRecord any time we change: -// rr->TimeRcvd -// rr->resrec.rroriginalttl -// rr->UnansweredQueries -// rr->CRActiveQuestion -mDNSlocal void SetNextCacheCheckTimeForRecord(mDNS *const m, CacheRecord *const rr) - { - rr->NextRequiredQuery = RRExpireTime(rr); - - // If we have an active question, then see if we want to schedule a refresher query for this record. - // Usually we expect to do four queries, at 80-82%, 85-87%, 90-92% and then 95-97% of the TTL. - if (rr->CRActiveQuestion && rr->UnansweredQueries < MaxUnansweredQueries) - { - rr->NextRequiredQuery -= TicksTTL(rr)/20 * (MaxUnansweredQueries - rr->UnansweredQueries); - rr->NextRequiredQuery += mDNSRandom((mDNSu32)TicksTTL(rr)/50); - verbosedebugf("SetNextCacheCheckTimeForRecord: NextRequiredQuery in %ld sec CacheCheckGracePeriod %d ticks for %s", - (rr->NextRequiredQuery - m->timenow) / mDNSPlatformOneSecond, CacheCheckGracePeriod(rr), CRDisplayString(m,rr)); - } - - ScheduleNextCacheCheckTime(m, HashSlot(rr->resrec.name), NextCacheCheckEvent(rr)); - } - -#define kMinimumReconfirmTime ((mDNSu32)mDNSPlatformOneSecond * 5) -#define kDefaultReconfirmTimeForWake ((mDNSu32)mDNSPlatformOneSecond * 5) -#define kDefaultReconfirmTimeForNoAnswer ((mDNSu32)mDNSPlatformOneSecond * 5) -#define kDefaultReconfirmTimeForFlappingInterface ((mDNSu32)mDNSPlatformOneSecond * 30) - -mDNSlocal mStatus mDNS_Reconfirm_internal(mDNS *const m, CacheRecord *const rr, mDNSu32 interval) - { - if (interval < kMinimumReconfirmTime) - interval = kMinimumReconfirmTime; - if (interval > 0x10000000) // Make sure interval doesn't overflow when we multiply by four below - interval = 0x10000000; - - // If the expected expiration time for this record is more than interval+33%, then accelerate its expiration - if (RRExpireTime(rr) - m->timenow > (mDNSs32)((interval * 4) / 3)) - { - // Add a 33% random amount to the interval, to avoid synchronization between multiple hosts - // For all the reconfirmations in a given batch, we want to use the same random value - // so that the reconfirmation questions can be grouped into a single query packet - if (!m->RandomReconfirmDelay) m->RandomReconfirmDelay = 1 + mDNSRandom(0x3FFFFFFF); - interval += m->RandomReconfirmDelay % ((interval/3) + 1); - rr->TimeRcvd = m->timenow - (mDNSs32)interval * 3; - rr->resrec.rroriginalttl = (interval * 4 + mDNSPlatformOneSecond - 1) / mDNSPlatformOneSecond; - SetNextCacheCheckTimeForRecord(m, rr); - } - debugf("mDNS_Reconfirm_internal:%6ld ticks to go for %s %p", - RRExpireTime(rr) - m->timenow, CRDisplayString(m, rr), rr->CRActiveQuestion); - return(mStatus_NoError); - } - -#define MaxQuestionInterval (3600 * mDNSPlatformOneSecond) - -// BuildQuestion puts a question into a DNS Query packet and if successful, updates the value of queryptr. -// It also appends to the list of known answer records that need to be included, -// and updates the forcast for the size of the known answer section. -mDNSlocal mDNSBool BuildQuestion(mDNS *const m, DNSMessage *query, mDNSu8 **queryptr, DNSQuestion *q, - CacheRecord ***kalistptrptr, mDNSu32 *answerforecast) - { - mDNSBool ucast = (q->LargeAnswers || q->RequestUnicast) && m->CanReceiveUnicastOn5353; - mDNSu16 ucbit = (mDNSu16)(ucast ? kDNSQClass_UnicastResponse : 0); - const mDNSu8 *const limit = query->data + NormalMaxDNSMessageData; - mDNSu8 *newptr = putQuestion(query, *queryptr, limit - *answerforecast, &q->qname, q->qtype, (mDNSu16)(q->qclass | ucbit)); - if (!newptr) - { - debugf("BuildQuestion: No more space in this packet for question %##s (%s)", q->qname.c, DNSTypeName(q->qtype)); - return(mDNSfalse); - } - else - { - mDNSu32 forecast = *answerforecast; - const mDNSu32 slot = HashSlot(&q->qname); - const CacheGroup *const cg = CacheGroupForName(m, slot, q->qnamehash, &q->qname); - CacheRecord *rr; - CacheRecord **ka = *kalistptrptr; // Make a working copy of the pointer we're going to update - - for (rr = cg ? cg->members : mDNSNULL; rr; rr=rr->next) // If we have a resource record in our cache, - if (rr->resrec.InterfaceID == q->SendQNow && // received on this interface - !(rr->resrec.RecordType & kDNSRecordTypeUniqueMask) && // which is a shared (i.e. not unique) record type - rr->NextInKAList == mDNSNULL && ka != &rr->NextInKAList && // which is not already in the known answer list - rr->resrec.rdlength <= SmallRecordLimit && // which is small enough to sensibly fit in the packet - SameNameRecordAnswersQuestion(&rr->resrec, q) && // which answers our question - rr->TimeRcvd + TicksTTL(rr)/2 - m->timenow > // and its half-way-to-expiry time is at least 1 second away - mDNSPlatformOneSecond) // (also ensures we never include goodbye records with TTL=1) - { - // We don't want to include unique records in the Known Answer section. The Known Answer section - // is intended to suppress floods of shared-record replies from many other devices on the network. - // That concept really does not apply to unique records, and indeed if we do send a query for - // which we have a unique record already in our cache, then including that unique record as a - // Known Answer, so as to suppress the only answer we were expecting to get, makes little sense. - - *ka = rr; // Link this record into our known answer chain - ka = &rr->NextInKAList; - // We forecast: compressed name (2) type (2) class (2) TTL (4) rdlength (2) rdata (n) - forecast += 12 + rr->resrec.rdestimate; - // If we're trying to put more than one question in this packet, and it doesn't fit - // then undo that last question and try again next time - if (query->h.numQuestions > 1 && newptr + forecast >= limit) - { - debugf("BuildQuestion: Retracting question %##s (%s) new forecast total %d", - q->qname.c, DNSTypeName(q->qtype), newptr + forecast - query->data); - query->h.numQuestions--; - ka = *kalistptrptr; // Go back to where we started and retract these answer records - while (*ka) { CacheRecord *c = *ka; *ka = mDNSNULL; ka = &c->NextInKAList; } - return(mDNSfalse); // Return false, so we'll try again in the next packet - } - } - - // Success! Update our state pointers, increment UnansweredQueries as appropriate, and return - *queryptr = newptr; // Update the packet pointer - *answerforecast = forecast; // Update the forecast - *kalistptrptr = ka; // Update the known answer list pointer - if (ucast) q->ExpectUnicastResp = NonZeroTime(m->timenow); - - for (rr = cg ? cg->members : mDNSNULL; rr; rr=rr->next) // For every resource record in our cache, - if (rr->resrec.InterfaceID == q->SendQNow && // received on this interface - rr->NextInKAList == mDNSNULL && ka != &rr->NextInKAList && // which is not in the known answer list - SameNameRecordAnswersQuestion(&rr->resrec, q)) // which answers our question - { - rr->UnansweredQueries++; // indicate that we're expecting a response - rr->LastUnansweredTime = m->timenow; - SetNextCacheCheckTimeForRecord(m, rr); - } - - return(mDNStrue); - } - } - -// When we have a query looking for a specified name, but there appear to be no answers with -// that name, ReconfirmAntecedents() is called with depth=0 to start the reconfirmation process -// for any records in our cache that reference the given name (e.g. PTR and SRV records). -// For any such cache record we find, we also recursively call ReconfirmAntecedents() for *its* name. -// We increment depth each time we recurse, to guard against possible infinite loops, with a limit of 5. -// A typical reconfirmation scenario might go like this: -// Depth 0: Name "myhost.local" has no address records -// Depth 1: SRV "My Service._example._tcp.local." refers to "myhost.local"; may be stale -// Depth 2: PTR "_example._tcp.local." refers to "My Service"; may be stale -// Depth 3: PTR "_services._dns-sd._udp.local." refers to "_example._tcp.local."; may be stale -// Currently depths 4 and 5 are not expected to occur; if we did get to depth 5 we'd reconfim any records we -// found referring to the given name, but not recursively descend any further reconfirm *their* antecedents. -mDNSlocal void ReconfirmAntecedents(mDNS *const m, const domainname *const name, const mDNSu32 namehash, const int depth) - { - mDNSu32 slot; - CacheGroup *cg; - CacheRecord *cr; - debugf("ReconfirmAntecedents (depth=%d) for %##s", depth, name->c); - FORALL_CACHERECORDS(slot, cg, cr) - { - domainname *crtarget = GetRRDomainNameTarget(&cr->resrec); - if (crtarget && cr->resrec.rdatahash == namehash && SameDomainName(crtarget, name)) - { - LogInfo("ReconfirmAntecedents: Reconfirming (depth=%d) %s", depth, CRDisplayString(m, cr)); - mDNS_Reconfirm_internal(m, cr, kDefaultReconfirmTimeForNoAnswer); - if (depth < 5) ReconfirmAntecedents(m, cr->resrec.name, cr->resrec.namehash, depth+1); - } - } - } - -// If we get no answer for a AAAA query, then before doing an automatic implicit ReconfirmAntecedents -// we check if we have an address record for the same name. If we do have an IPv4 address for a given -// name but not an IPv6 address, that's okay (it just means the device doesn't do IPv6) so the failure -// to get a AAAA response is not grounds to doubt the PTR/SRV chain that lead us to that name. -mDNSlocal const CacheRecord *CacheHasAddressTypeForName(mDNS *const m, const domainname *const name, const mDNSu32 namehash) - { - CacheGroup *const cg = CacheGroupForName(m, HashSlot(name), namehash, name); - const CacheRecord *cr = cg ? cg->members : mDNSNULL; - while (cr && !RRTypeIsAddressType(cr->resrec.rrtype)) cr=cr->next; - return(cr); - } - -mDNSlocal const CacheRecord *FindSPSInCache1(mDNS *const m, const DNSQuestion *const q, const CacheRecord *const c0, const CacheRecord *const c1) - { - CacheGroup *const cg = CacheGroupForName(m, HashSlot(&q->qname), q->qnamehash, &q->qname); - const CacheRecord *cr, *bestcr = mDNSNULL; - mDNSu32 bestmetric = 1000000; - for (cr = cg ? cg->members : mDNSNULL; cr; cr=cr->next) - if (cr->resrec.rrtype == kDNSType_PTR && cr->resrec.rdlength >= 6) // If record is PTR type, with long enough name, - if (cr != c0 && cr != c1) // that's not one we've seen before, - if (SameNameRecordAnswersQuestion(&cr->resrec, q)) // and answers our browse query, - if (!IdenticalSameNameRecord(&cr->resrec, &m->SPSRecords.RR_PTR.resrec)) // and is not our own advertised service... - { - mDNSu32 metric = SPSMetric(cr->resrec.rdata->u.name.c); - if (bestmetric > metric) { bestmetric = metric; bestcr = cr; } - } - return(bestcr); - } - -// Finds the three best Sleep Proxies we currently have in our cache -mDNSexport void FindSPSInCache(mDNS *const m, const DNSQuestion *const q, const CacheRecord *sps[3]) - { - sps[0] = FindSPSInCache1(m, q, mDNSNULL, mDNSNULL); - sps[1] = !sps[0] ? mDNSNULL : FindSPSInCache1(m, q, sps[0], mDNSNULL); - sps[2] = !sps[1] ? mDNSNULL : FindSPSInCache1(m, q, sps[0], sps[1]); - } - -// Only DupSuppressInfos newer than the specified 'time' are allowed to remain active -mDNSlocal void ExpireDupSuppressInfo(DupSuppressInfo ds[DupSuppressInfoSize], mDNSs32 time) - { - int i; - for (i=0; iIPv4Available; // If this interface doesn't do v4, we don't need to find a v4 duplicate of this query - mDNSBool v6 = !intf->IPv6Available; // If this interface doesn't do v6, we don't need to find a v6 duplicate of this query - for (i=0; iInterfaceID) - { - if (ds[i].Type == mDNSAddrType_IPv4) v4 = mDNStrue; - else if (ds[i].Type == mDNSAddrType_IPv6) v6 = mDNStrue; - if (v4 && v6) return(mDNStrue); - } - return(mDNSfalse); - } - -mDNSlocal int RecordDupSuppressInfo(DupSuppressInfo ds[DupSuppressInfoSize], mDNSs32 Time, mDNSInterfaceID InterfaceID, mDNSs32 Type) - { - int i, j; - - // See if we have this one in our list somewhere already - for (i=0; i= DupSuppressInfoSize) - { - i = 0; - for (j=1; jInterfaceID; - domainname *d = &q->qname; - - // We can't send magic packets without knowing which interface to send it on. - if (InterfaceID == mDNSInterface_Any || InterfaceID == mDNSInterface_LocalOnly || InterfaceID == mDNSInterface_P2P) - { - LogMsg("mDNSSendWakeOnResolve: ERROR!! Invalid InterfaceID %p for question %##s", InterfaceID, q->qname.c); - return; - } - - // Split MAC@IPAddress and pass them separately - len = d->c[0]; - i = 1; - cnt = 0; - for (i = 1; i < len; i++) - { - if (d->c[i] == '@') - { - char EthAddr[18]; // ethernet adddress : 12 bytes + 5 ":" + 1 NULL byte - char IPAddr[47]; // Max IP address len: 46 bytes (IPv6) + 1 NULL byte - if (cnt != 5) - { - LogMsg("mDNSSendWakeOnResolve: ERROR!! Malformed Ethernet address %##s, cnt %d", q->qname.c, cnt); - return; - } - if ((i - 1) > (int) (sizeof(EthAddr) - 1)) - { - LogMsg("mDNSSendWakeOnResolve: ERROR!! Malformed Ethernet address %##s, length %d", q->qname.c, i - 1); - return; - } - if ((len - i) > (int)(sizeof(IPAddr) - 1)) - { - LogMsg("mDNSSendWakeOnResolve: ERROR!! Malformed IP address %##s, length %d", q->qname.c, len - i); - return; - } - mDNSPlatformMemCopy(EthAddr, &d->c[1], i - 1); - EthAddr[i - 1] = 0; - mDNSPlatformMemCopy(IPAddr, &d->c[i + 1], len - i); - IPAddr[len - i] = 0; - mDNSPlatformSendWakeupPacket(m, InterfaceID, EthAddr, IPAddr, InitialWakeOnResolveCount - q->WakeOnResolveCount); - return; - } - else if (d->c[i] == ':') - cnt++; - } - LogMsg("mDNSSendWakeOnResolve: ERROR!! Malformed WakeOnResolve name %##s", q->qname.c); - } - - -mDNSlocal mDNSBool AccelerateThisQuery(mDNS *const m, DNSQuestion *q) - { - // If more than 90% of the way to the query time, we should unconditionally accelerate it - if (TimeToSendThisQuestion(q, m->timenow + q->ThisQInterval/10)) - return(mDNStrue); - - // If half-way to next scheduled query time, only accelerate if it will add less than 512 bytes to the packet - if (TimeToSendThisQuestion(q, m->timenow + q->ThisQInterval/2)) - { - // We forecast: qname (n) type (2) class (2) - mDNSu32 forecast = (mDNSu32)DomainNameLength(&q->qname) + 4; - const mDNSu32 slot = HashSlot(&q->qname); - const CacheGroup *const cg = CacheGroupForName(m, slot, q->qnamehash, &q->qname); - const CacheRecord *rr; - for (rr = cg ? cg->members : mDNSNULL; rr; rr=rr->next) // If we have a resource record in our cache, - if (rr->resrec.rdlength <= SmallRecordLimit && // which is small enough to sensibly fit in the packet - SameNameRecordAnswersQuestion(&rr->resrec, q) && // which answers our question - rr->TimeRcvd + TicksTTL(rr)/2 - m->timenow >= 0 && // and it is less than half-way to expiry - rr->NextRequiredQuery - (m->timenow + q->ThisQInterval) > 0)// and we'll ask at least once again before NextRequiredQuery - { - // We forecast: compressed name (2) type (2) class (2) TTL (4) rdlength (2) rdata (n) - forecast += 12 + rr->resrec.rdestimate; - if (forecast >= 512) return(mDNSfalse); // If this would add 512 bytes or more to the packet, don't accelerate - } - return(mDNStrue); - } - - return(mDNSfalse); - } - -// How Standard Queries are generated: -// 1. The Question Section contains the question -// 2. The Additional Section contains answers we already know, to suppress duplicate responses - -// How Probe Queries are generated: -// 1. The Question Section contains queries for the name we intend to use, with QType=ANY because -// if some other host is already using *any* records with this name, we want to know about it. -// 2. The Authority Section contains the proposed values we intend to use for one or more -// of our records with that name (analogous to the Update section of DNS Update packets) -// because if some other host is probing at the same time, we each want to know what the other is -// planning, in order to apply the tie-breaking rule to see who gets to use the name and who doesn't. - -mDNSlocal void SendQueries(mDNS *const m) - { - mDNSu32 slot; - CacheGroup *cg; - CacheRecord *cr; - AuthRecord *ar; - int pktcount = 0; - DNSQuestion *q; - // For explanation of maxExistingQuestionInterval logic, see comments for maxExistingAnnounceInterval - mDNSs32 maxExistingQuestionInterval = 0; - const NetworkInterfaceInfo *intf = GetFirstActiveInterface(m->HostInterfaces); - CacheRecord *KnownAnswerList = mDNSNULL; - - // 1. If time for a query, work out what we need to do - - // We're expecting to send a query anyway, so see if any expiring cache records are close enough - // to their NextRequiredQuery to be worth batching them together with this one - FORALL_CACHERECORDS(slot, cg, cr) - if (cr->CRActiveQuestion && cr->UnansweredQueries < MaxUnansweredQueries) - if (m->timenow + TicksTTL(cr)/50 - cr->NextRequiredQuery >= 0) - { - debugf("Sending %d%% cache expiration query for %s", 80 + 5 * cr->UnansweredQueries, CRDisplayString(m, cr)); - q = cr->CRActiveQuestion; - ExpireDupSuppressInfoOnInterface(q->DupSuppress, m->timenow - TicksTTL(cr)/20, cr->resrec.InterfaceID); - // For uDNS queries (TargetQID non-zero) we adjust LastQTime, - // and bump UnansweredQueries so that we don't spin trying to send the same cache expiration query repeatedly - if (q->Target.type) q->SendQNow = mDNSInterfaceMark; // If targeted query, mark it - else if (!mDNSOpaque16IsZero(q->TargetQID)) { q->LastQTime = m->timenow - q->ThisQInterval; cr->UnansweredQueries++; } - else if (q->SendQNow == mDNSNULL) q->SendQNow = cr->resrec.InterfaceID; - else if (q->SendQNow != cr->resrec.InterfaceID) q->SendQNow = mDNSInterfaceMark; - } - - // Scan our list of questions to see which: - // *WideArea* queries need to be sent - // *unicast* queries need to be sent - // *multicast* queries we're definitely going to send - if (m->CurrentQuestion) - LogMsg("SendQueries ERROR m->CurrentQuestion already set: %##s (%s)", m->CurrentQuestion->qname.c, DNSTypeName(m->CurrentQuestion->qtype)); - m->CurrentQuestion = m->Questions; - while (m->CurrentQuestion && m->CurrentQuestion != m->NewQuestions) - { - q = m->CurrentQuestion; - if (q->Target.type && (q->SendQNow || TimeToSendThisQuestion(q, m->timenow))) - { - mDNSu8 *qptr = m->omsg.data; - const mDNSu8 *const limit = m->omsg.data + sizeof(m->omsg.data); - - // If we fail to get a new on-demand socket (should only happen cases of the most extreme resource exhaustion), we'll try again next time - if (!q->LocalSocket) q->LocalSocket = mDNSPlatformUDPSocket(m, zeroIPPort); - if (q->LocalSocket) - { - InitializeDNSMessage(&m->omsg.h, q->TargetQID, QueryFlags); - qptr = putQuestion(&m->omsg, qptr, limit, &q->qname, q->qtype, q->qclass); - mDNSSendDNSMessage(m, &m->omsg, qptr, mDNSInterface_Any, q->LocalSocket, &q->Target, q->TargetPort, mDNSNULL, mDNSNULL); - q->ThisQInterval *= QuestionIntervalStep; - } - if (q->ThisQInterval > MaxQuestionInterval) - q->ThisQInterval = MaxQuestionInterval; - q->LastQTime = m->timenow; - q->LastQTxTime = m->timenow; - q->RecentAnswerPkts = 0; - q->SendQNow = mDNSNULL; - q->ExpectUnicastResp = NonZeroTime(m->timenow); - } - else if (mDNSOpaque16IsZero(q->TargetQID) && !q->Target.type && TimeToSendThisQuestion(q, m->timenow)) - { - //LogInfo("Time to send %##s (%s) %d", q->qname.c, DNSTypeName(q->qtype), m->timenow - NextQSendTime(q)); - q->SendQNow = mDNSInterfaceMark; // Mark this question for sending on all interfaces - if (maxExistingQuestionInterval < q->ThisQInterval) - maxExistingQuestionInterval = q->ThisQInterval; - } - // If m->CurrentQuestion wasn't modified out from under us, advance it now - // We can't do this at the start of the loop because uDNS_CheckCurrentQuestion() depends on having - // m->CurrentQuestion point to the right question - if (q == m->CurrentQuestion) m->CurrentQuestion = m->CurrentQuestion->next; - } - while (m->CurrentQuestion) - { - LogInfo("SendQueries question loop 1: Skipping NewQuestion %##s (%s)", m->CurrentQuestion->qname.c, DNSTypeName(m->CurrentQuestion->qtype)); - m->CurrentQuestion = m->CurrentQuestion->next; - } - m->CurrentQuestion = mDNSNULL; - - // Scan our list of questions - // (a) to see if there are any more that are worth accelerating, and - // (b) to update the state variables for *all* the questions we're going to send - // Note: Don't set NextScheduledQuery until here, because uDNS_CheckCurrentQuestion in the loop above can add new questions to the list, - // which causes NextScheduledQuery to get (incorrectly) set to m->timenow. Setting it here is the right place, because the very - // next thing we do is scan the list and call SetNextQueryTime() for every question we find, so we know we end up with the right value. - m->NextScheduledQuery = m->timenow + 0x78000000; - for (q = m->Questions; q && q != m->NewQuestions; q=q->next) - { - if (mDNSOpaque16IsZero(q->TargetQID) && (q->SendQNow || - (!q->Target.type && ActiveQuestion(q) && q->ThisQInterval <= maxExistingQuestionInterval && AccelerateThisQuery(m,q)))) - { - // If at least halfway to next query time, advance to next interval - // If less than halfway to next query time, then - // treat this as logically a repeat of the last transmission, without advancing the interval - if (m->timenow - (q->LastQTime + (q->ThisQInterval/2)) >= 0) - { - //LogInfo("Accelerating %##s (%s) %d", q->qname.c, DNSTypeName(q->qtype), m->timenow - NextQSendTime(q)); - q->SendQNow = mDNSInterfaceMark; // Mark this question for sending on all interfaces - debugf("SendQueries: %##s (%s) next interval %d seconds RequestUnicast = %d", - q->qname.c, DNSTypeName(q->qtype), q->ThisQInterval / InitialQuestionInterval, q->RequestUnicast); - q->ThisQInterval *= QuestionIntervalStep; - if (q->ThisQInterval > MaxQuestionInterval) - q->ThisQInterval = MaxQuestionInterval; - else if (q->CurrentAnswers == 0 && q->ThisQInterval == InitialQuestionInterval * QuestionIntervalStep3 && !q->RequestUnicast && - !(RRTypeIsAddressType(q->qtype) && CacheHasAddressTypeForName(m, &q->qname, q->qnamehash))) - { - // Generally don't need to log this. - // It's not especially noteworthy if a query finds no results -- this usually happens for domain - // enumeration queries in the LL subdomain (e.g. "db._dns-sd._udp.0.0.254.169.in-addr.arpa") - // and when there simply happen to be no instances of the service the client is looking - // for (e.g. iTunes is set to look for RAOP devices, and the current network has none). - debugf("SendQueries: Zero current answers for %##s (%s); will reconfirm antecedents", - q->qname.c, DNSTypeName(q->qtype)); - // Sending third query, and no answers yet; time to begin doubting the source - ReconfirmAntecedents(m, &q->qname, q->qnamehash, 0); - } - } - - // Mark for sending. (If no active interfaces, then don't even try.) - q->SendOnAll = (q->SendQNow == mDNSInterfaceMark); - if (q->SendOnAll) - { - q->SendQNow = !intf ? mDNSNULL : (q->InterfaceID) ? q->InterfaceID : intf->InterfaceID; - q->LastQTime = m->timenow; - } - - // If we recorded a duplicate suppression for this question less than half an interval ago, - // then we consider it recent enough that we don't need to do an identical query ourselves. - ExpireDupSuppressInfo(q->DupSuppress, m->timenow - q->ThisQInterval/2); - - q->LastQTxTime = m->timenow; - q->RecentAnswerPkts = 0; - if (q->RequestUnicast) q->RequestUnicast--; - } - // For all questions (not just the ones we're sending) check what the next scheduled event will be - // We don't need to consider NewQuestions here because for those we'll set m->NextScheduledQuery in AnswerNewQuestion - SetNextQueryTime(m,q); - } - - // 2. Scan our authoritative RR list to see what probes we might need to send - - m->NextScheduledProbe = m->timenow + 0x78000000; - - if (m->CurrentRecord) - LogMsg("SendQueries ERROR m->CurrentRecord already set %s", ARDisplayString(m, m->CurrentRecord)); - m->CurrentRecord = m->ResourceRecords; - while (m->CurrentRecord) - { - ar = m->CurrentRecord; - m->CurrentRecord = ar->next; - if (!AuthRecord_uDNS(ar) && ar->resrec.RecordType == kDNSRecordTypeUnique) // For all records that are still probing... - { - // 1. If it's not reached its probe time, just make sure we update m->NextScheduledProbe correctly - if (m->timenow - (ar->LastAPTime + ar->ThisAPInterval) < 0) - { - SetNextAnnounceProbeTime(m, ar); - } - // 2. else, if it has reached its probe time, mark it for sending and then update m->NextScheduledProbe correctly - else if (ar->ProbeCount) - { - if (ar->AddressProxy.type == mDNSAddrType_IPv4) - { - LogSPS("SendQueries ARP Probe %d %s %s", ar->ProbeCount, InterfaceNameForID(m, ar->resrec.InterfaceID), ARDisplayString(m,ar)); - SendARP(m, 1, ar, &zerov4Addr, &zeroEthAddr, &ar->AddressProxy.ip.v4, &ar->WakeUp.IMAC); - } - else if (ar->AddressProxy.type == mDNSAddrType_IPv6) - { - LogSPS("SendQueries NDP Probe %d %s %s", ar->ProbeCount, InterfaceNameForID(m, ar->resrec.InterfaceID), ARDisplayString(m,ar)); - // IPv6 source = zero - // No target hardware address - // IPv6 target address is address we're probing - // Ethernet destination address is Ethernet interface address of the Sleep Proxy client we're probing - SendNDP(m, NDP_Sol, 0, ar, &zerov6Addr, mDNSNULL, &ar->AddressProxy.ip.v6, &ar->WakeUp.IMAC); - } - // Mark for sending. (If no active interfaces, then don't even try.) - ar->SendRNow = (!intf || ar->WakeUp.HMAC.l[0]) ? mDNSNULL : ar->resrec.InterfaceID ? ar->resrec.InterfaceID : intf->InterfaceID; - ar->LastAPTime = m->timenow; - // When we have a late conflict that resets a record to probing state we use a special marker value greater - // than DefaultProbeCountForTypeUnique. Here we detect that state and reset ar->ProbeCount back to the right value. - if (ar->ProbeCount > DefaultProbeCountForTypeUnique) - ar->ProbeCount = DefaultProbeCountForTypeUnique; - ar->ProbeCount--; - SetNextAnnounceProbeTime(m, ar); - if (ar->ProbeCount == 0) - { - // If this is the last probe for this record, then see if we have any matching records - // on our duplicate list which should similarly have their ProbeCount cleared to zero... - AuthRecord *r2; - for (r2 = m->DuplicateRecords; r2; r2=r2->next) - if (r2->resrec.RecordType == kDNSRecordTypeUnique && RecordIsLocalDuplicate(r2, ar)) - r2->ProbeCount = 0; - // ... then acknowledge this record to the client. - // We do this optimistically, just as we're about to send the third probe. - // This helps clients that both advertise and browse, and want to filter themselves - // from the browse results list, because it helps ensure that the registration - // confirmation will be delivered 1/4 second *before* the browse "add" event. - // A potential downside is that we could deliver a registration confirmation and then find out - // moments later that there's a name conflict, but applications have to be prepared to handle - // late conflicts anyway (e.g. on connection of network cable, etc.), so this is nothing new. - if (!ar->Acknowledged) AcknowledgeRecord(m, ar); - } - } - // else, if it has now finished probing, move it to state Verified, - // and update m->NextScheduledResponse so it will be announced - else - { - if (!ar->Acknowledged) AcknowledgeRecord(m, ar); // Defensive, just in case it got missed somehow - ar->resrec.RecordType = kDNSRecordTypeVerified; - ar->ThisAPInterval = DefaultAnnounceIntervalForTypeUnique; - ar->LastAPTime = m->timenow - DefaultAnnounceIntervalForTypeUnique; - SetNextAnnounceProbeTime(m, ar); - } - } - } - m->CurrentRecord = m->DuplicateRecords; - while (m->CurrentRecord) - { - ar = m->CurrentRecord; - m->CurrentRecord = ar->next; - if (ar->resrec.RecordType == kDNSRecordTypeUnique && ar->ProbeCount == 0 && !ar->Acknowledged) - AcknowledgeRecord(m, ar); - } - - // 3. Now we know which queries and probes we're sending, - // go through our interface list sending the appropriate queries on each interface - while (intf) - { - const int OwnerRecordSpace = (m->AnnounceOwner && intf->MAC.l[0]) ? DNSOpt_Header_Space + DNSOpt_Owner_Space(&m->PrimaryMAC, &intf->MAC) : 0; - mDNSu8 *queryptr = m->omsg.data; - InitializeDNSMessage(&m->omsg.h, zeroID, QueryFlags); - if (KnownAnswerList) verbosedebugf("SendQueries: KnownAnswerList set... Will continue from previous packet"); - if (!KnownAnswerList) - { - // Start a new known-answer list - CacheRecord **kalistptr = &KnownAnswerList; - mDNSu32 answerforecast = OwnerRecordSpace; // We start by assuming we'll need at least enough space to put the Owner Option - - // Put query questions in this packet - for (q = m->Questions; q && q != m->NewQuestions; q=q->next) - { - if (mDNSOpaque16IsZero(q->TargetQID) && (q->SendQNow == intf->InterfaceID)) - { - debugf("SendQueries: %s question for %##s (%s) at %d forecast total %d", - SuppressOnThisInterface(q->DupSuppress, intf) ? "Suppressing" : "Putting ", - q->qname.c, DNSTypeName(q->qtype), queryptr - m->omsg.data, queryptr + answerforecast - m->omsg.data); - - // If we're suppressing this question, or we successfully put it, update its SendQNow state - if (SuppressOnThisInterface(q->DupSuppress, intf) || - BuildQuestion(m, &m->omsg, &queryptr, q, &kalistptr, &answerforecast)) - { - q->SendQNow = (q->InterfaceID || !q->SendOnAll) ? mDNSNULL : GetNextActiveInterfaceID(intf); - if (q->WakeOnResolveCount) - { - mDNSSendWakeOnResolve(m, q); - q->WakeOnResolveCount--; - } - } - } - } - - // Put probe questions in this packet - for (ar = m->ResourceRecords; ar; ar=ar->next) - if (ar->SendRNow == intf->InterfaceID) - { - mDNSBool ucast = (ar->ProbeCount >= DefaultProbeCountForTypeUnique-1) && m->CanReceiveUnicastOn5353; - mDNSu16 ucbit = (mDNSu16)(ucast ? kDNSQClass_UnicastResponse : 0); - const mDNSu8 *const limit = m->omsg.data + (m->omsg.h.numQuestions ? NormalMaxDNSMessageData : AbsoluteMaxDNSMessageData); - // We forecast: compressed name (2) type (2) class (2) TTL (4) rdlength (2) rdata (n) - mDNSu32 forecast = answerforecast + 12 + ar->resrec.rdestimate; - mDNSu8 *newptr = putQuestion(&m->omsg, queryptr, limit - forecast, ar->resrec.name, kDNSQType_ANY, (mDNSu16)(ar->resrec.rrclass | ucbit)); - if (newptr) - { - queryptr = newptr; - answerforecast = forecast; - ar->SendRNow = (ar->resrec.InterfaceID) ? mDNSNULL : GetNextActiveInterfaceID(intf); - ar->IncludeInProbe = mDNStrue; - verbosedebugf("SendQueries: Put Question %##s (%s) probecount %d", - ar->resrec.name->c, DNSTypeName(ar->resrec.rrtype), ar->ProbeCount); - } - } - } - - // Put our known answer list (either new one from this question or questions, or remainder of old one from last time) - while (KnownAnswerList) - { - CacheRecord *ka = KnownAnswerList; - mDNSu32 SecsSinceRcvd = ((mDNSu32)(m->timenow - ka->TimeRcvd)) / mDNSPlatformOneSecond; - mDNSu8 *newptr = PutResourceRecordTTLWithLimit(&m->omsg, queryptr, &m->omsg.h.numAnswers, - &ka->resrec, ka->resrec.rroriginalttl - SecsSinceRcvd, m->omsg.data + NormalMaxDNSMessageData - OwnerRecordSpace); - if (newptr) - { - verbosedebugf("SendQueries: Put %##s (%s) at %d - %d", - ka->resrec.name->c, DNSTypeName(ka->resrec.rrtype), queryptr - m->omsg.data, newptr - m->omsg.data); - queryptr = newptr; - KnownAnswerList = ka->NextInKAList; - ka->NextInKAList = mDNSNULL; - } - else - { - // If we ran out of space and we have more than one question in the packet, that's an error -- - // we shouldn't have put more than one question if there was a risk of us running out of space. - if (m->omsg.h.numQuestions > 1) - LogMsg("SendQueries: Put %d answers; No more space for known answers", m->omsg.h.numAnswers); - m->omsg.h.flags.b[0] |= kDNSFlag0_TC; - break; - } - } - - for (ar = m->ResourceRecords; ar; ar=ar->next) - if (ar->IncludeInProbe) - { - mDNSu8 *newptr = PutResourceRecord(&m->omsg, queryptr, &m->omsg.h.numAuthorities, &ar->resrec); - ar->IncludeInProbe = mDNSfalse; - if (newptr) queryptr = newptr; - else LogMsg("SendQueries: How did we fail to have space for the Update record %s", ARDisplayString(m,ar)); - } - - if (queryptr > m->omsg.data) - { - if (OwnerRecordSpace) - { - AuthRecord opt; - mDNS_SetupResourceRecord(&opt, mDNSNULL, mDNSInterface_Any, kDNSType_OPT, kStandardTTL, kDNSRecordTypeKnownUnique, AuthRecordAny, mDNSNULL, mDNSNULL); - opt.resrec.rrclass = NormalMaxDNSMessageData; - opt.resrec.rdlength = sizeof(rdataOPT); // One option in this OPT record - opt.resrec.rdestimate = sizeof(rdataOPT); - SetupOwnerOpt(m, intf, &opt.resrec.rdata->u.opt[0]); - LogSPS("SendQueries putting %s", ARDisplayString(m, &opt)); - queryptr = PutResourceRecordTTLWithLimit(&m->omsg, queryptr, &m->omsg.h.numAdditionals, - &opt.resrec, opt.resrec.rroriginalttl, m->omsg.data + AbsoluteMaxDNSMessageData); - if (!queryptr) - LogMsg("SendQueries: How did we fail to have space for the OPT record (%d/%d/%d/%d) %s", - m->omsg.h.numQuestions, m->omsg.h.numAnswers, m->omsg.h.numAuthorities, m->omsg.h.numAdditionals, ARDisplayString(m, &opt)); - if (queryptr > m->omsg.data + NormalMaxDNSMessageData) - if (m->omsg.h.numQuestions != 1 || m->omsg.h.numAnswers != 0 || m->omsg.h.numAuthorities != 1 || m->omsg.h.numAdditionals != 1) - LogMsg("SendQueries: Why did we generate oversized packet with OPT record %p %p %p (%d/%d/%d/%d) %s", - m->omsg.data, m->omsg.data + NormalMaxDNSMessageData, queryptr, - m->omsg.h.numQuestions, m->omsg.h.numAnswers, m->omsg.h.numAuthorities, m->omsg.h.numAdditionals, ARDisplayString(m, &opt)); - } - - if ((m->omsg.h.flags.b[0] & kDNSFlag0_TC) && m->omsg.h.numQuestions > 1) - LogMsg("SendQueries: Should not have more than one question (%d) in a truncated packet", m->omsg.h.numQuestions); - debugf("SendQueries: Sending %d Question%s %d Answer%s %d Update%s on %p", - m->omsg.h.numQuestions, m->omsg.h.numQuestions == 1 ? "" : "s", - m->omsg.h.numAnswers, m->omsg.h.numAnswers == 1 ? "" : "s", - m->omsg.h.numAuthorities, m->omsg.h.numAuthorities == 1 ? "" : "s", intf->InterfaceID); - if (intf->IPv4Available) mDNSSendDNSMessage(m, &m->omsg, queryptr, intf->InterfaceID, mDNSNULL, &AllDNSLinkGroup_v4, MulticastDNSPort, mDNSNULL, mDNSNULL); - if (intf->IPv6Available) mDNSSendDNSMessage(m, &m->omsg, queryptr, intf->InterfaceID, mDNSNULL, &AllDNSLinkGroup_v6, MulticastDNSPort, mDNSNULL, mDNSNULL); - if (!m->SuppressSending) m->SuppressSending = NonZeroTime(m->timenow + (mDNSPlatformOneSecond+9)/10); - if (++pktcount >= 1000) - { LogMsg("SendQueries exceeded loop limit %d: giving up", pktcount); break; } - // There might be more records left in the known answer list, or more questions to send - // on this interface, so go around one more time and try again. - } - else // Nothing more to send on this interface; go to next - { - const NetworkInterfaceInfo *next = GetFirstActiveInterface(intf->next); - #if MDNS_DEBUGMSGS && 0 - const char *const msg = next ? "SendQueries: Nothing more on %p; moving to %p" : "SendQueries: Nothing more on %p"; - debugf(msg, intf, next); - #endif - intf = next; - } - } - - // 4. Final housekeeping - - // 4a. Debugging check: Make sure we announced all our records - for (ar = m->ResourceRecords; ar; ar=ar->next) - if (ar->SendRNow) - { - if (ar->ARType != AuthRecordLocalOnly && ar->ARType != AuthRecordP2P) - LogMsg("SendQueries: No active interface %p to send probe: %p %s", ar->SendRNow, ar->resrec.InterfaceID, ARDisplayString(m, ar)); - ar->SendRNow = mDNSNULL; - } - - // 4b. When we have lingering cache records that we're keeping around for a few seconds in the hope - // that their interface which went away might come back again, the logic will want to send queries - // for those records, but we can't because their interface isn't here any more, so to keep the - // state machine ticking over we just pretend we did so. - // If the interface does not come back in time, the cache record will expire naturally - FORALL_CACHERECORDS(slot, cg, cr) - if (cr->CRActiveQuestion && cr->UnansweredQueries < MaxUnansweredQueries) - if (m->timenow + TicksTTL(cr)/50 - cr->NextRequiredQuery >= 0) - { - cr->UnansweredQueries++; - cr->CRActiveQuestion->SendQNow = mDNSNULL; - SetNextCacheCheckTimeForRecord(m, cr); - } - - // 4c. Debugging check: Make sure we sent all our planned questions - // Do this AFTER the lingering cache records check above, because that will prevent spurious warnings for questions - // we legitimately couldn't send because the interface is no longer available - for (q = m->Questions; q; q=q->next) - if (q->SendQNow) - { - DNSQuestion *x; - for (x = m->NewQuestions; x; x=x->next) if (x == q) break; // Check if this question is a NewQuestion - LogMsg("SendQueries: No active interface %p to send %s question: %p %##s (%s)", q->SendQNow, x ? "new" : "old", q->InterfaceID, q->qname.c, DNSTypeName(q->qtype)); - q->SendQNow = mDNSNULL; - } - } - -mDNSlocal void SendWakeup(mDNS *const m, mDNSInterfaceID InterfaceID, mDNSEthAddr *EthAddr, mDNSOpaque48 *password) - { - int i, j; - mDNSu8 *ptr = m->omsg.data; - NetworkInterfaceInfo *intf = FirstInterfaceForID(m, InterfaceID); - if (!intf) { LogMsg("SendARP: No interface with InterfaceID %p found", InterfaceID); return; } - - // 0x00 Destination address - for (i=0; i<6; i++) *ptr++ = EthAddr->b[i]; - - // 0x06 Source address (Note: Since we don't currently set the BIOCSHDRCMPLT option, BPF will fill in the real interface address for us) - for (i=0; i<6; i++) *ptr++ = intf->MAC.b[0]; - - // 0x0C Ethertype (0x0842) - *ptr++ = 0x08; - *ptr++ = 0x42; - - // 0x0E Wakeup sync sequence - for (i=0; i<6; i++) *ptr++ = 0xFF; - - // 0x14 Wakeup data - for (j=0; j<16; j++) for (i=0; i<6; i++) *ptr++ = EthAddr->b[i]; - - // 0x74 Password - for (i=0; i<6; i++) *ptr++ = password->b[i]; - - mDNSPlatformSendRawPacket(m->omsg.data, ptr, InterfaceID); - - // For Ethernet switches that don't flood-foward packets with unknown unicast destination MAC addresses, - // broadcast is the only reliable way to get a wakeup packet to the intended target machine. - // For 802.11 WPA networks, where a sleeping target machine may have missed a broadcast/multicast - // key rotation, unicast is the only way to get a wakeup packet to the intended target machine. - // So, we send one of each, unicast first, then broadcast second. - for (i=0; i<6; i++) m->omsg.data[i] = 0xFF; - mDNSPlatformSendRawPacket(m->omsg.data, ptr, InterfaceID); - } - -// *************************************************************************** -#if COMPILER_LIKES_PRAGMA_MARK -#pragma mark - -#pragma mark - RR List Management & Task Management -#endif - -// Whenever a question is answered, reset its state so that we don't query -// the network repeatedly. This happens first time when we answer the question and -// and later when we refresh the cache. -mDNSlocal void ResetQuestionState(mDNS *const m, DNSQuestion *q) - { - q->LastQTime = m->timenow; - q->LastQTxTime = m->timenow; - q->RecentAnswerPkts = 0; - q->ThisQInterval = MaxQuestionInterval; - q->RequestUnicast = mDNSfalse; - // Reset unansweredQueries so that we don't penalize this server later when we - // start sending queries when the cache expires. - q->unansweredQueries = 0; - debugf("ResetQuestionState: Set MaxQuestionInterval for %##s (%s)", q->qname.c, DNSTypeName(q->qtype)); - } - -// Note: AnswerCurrentQuestionWithResourceRecord can call a user callback, which may change the record list and/or question list. -// Any code walking either list must use the m->CurrentQuestion (and possibly m->CurrentRecord) mechanism to protect against this. -// In fact, to enforce this, the routine will *only* answer the question currently pointed to by m->CurrentQuestion, -// which will be auto-advanced (possibly to NULL) if the client callback cancels the question. -mDNSexport void AnswerCurrentQuestionWithResourceRecord(mDNS *const m, CacheRecord *const rr, const QC_result AddRecord) - { - DNSQuestion *const q = m->CurrentQuestion; - mDNSBool followcname = FollowCNAME(q, &rr->resrec, AddRecord); - - verbosedebugf("AnswerCurrentQuestionWithResourceRecord:%4lu %s TTL %d %s", - q->CurrentAnswers, AddRecord ? "Add" : "Rmv", rr->resrec.rroriginalttl, CRDisplayString(m, rr)); - - // Normally we don't send out the unicast query if we have answered using our local only auth records e.g., /etc/hosts. - // But if the query for "A" record has a local answer but query for "AAAA" record has no local answer, we might - // send the AAAA query out which will come back with CNAME and will also answer the "A" query. To prevent that, - // we check to see if that query already has a unique local answer. - if (q->LOAddressAnswers) - { - LogInfo("AnswerCurrentQuestionWithResourceRecord: Question %p %##s (%s) not answering with record %s due to " - "LOAddressAnswers %d", q, q->qname.c, DNSTypeName(q->qtype), ARDisplayString(m, rr), - q->LOAddressAnswers); - return; - } - - if (QuerySuppressed(q)) - { - // If the query is suppressed, then we don't want to answer from the cache. But if this query is - // supposed to time out, we still want to callback the clients. We do this only for TimeoutQuestions - // that are timing out, which we know are answered with Negative cache record when timing out. - if (!q->TimeoutQuestion || rr->resrec.RecordType != kDNSRecordTypePacketNegative || (m->timenow - q->StopTime < 0)) - return; - } - - // Note: Use caution here. In the case of records with rr->DelayDelivery set, AnswerCurrentQuestionWithResourceRecord(... mDNStrue) - // may be called twice, once when the record is received, and again when it's time to notify local clients. - // If any counters or similar are added here, care must be taken to ensure that they are not double-incremented by this. - - rr->LastUsed = m->timenow; - if (AddRecord == QC_add && !q->DuplicateOf && rr->CRActiveQuestion != q) - { - if (!rr->CRActiveQuestion) m->rrcache_active++; // If not previously active, increment rrcache_active count - debugf("AnswerCurrentQuestionWithResourceRecord: Updating CRActiveQuestion from %p to %p for cache record %s, CurrentAnswer %d", - rr->CRActiveQuestion, q, CRDisplayString(m,rr), q->CurrentAnswers); - rr->CRActiveQuestion = q; // We know q is non-null - SetNextCacheCheckTimeForRecord(m, rr); - } - - // If this is: - // (a) a no-cache add, where we've already done at least one 'QM' query, or - // (b) a normal add, where we have at least one unique-type answer, - // then there's no need to keep polling the network. - // (If we have an answer in the cache, then we'll automatically ask again in time to stop it expiring.) - // We do this for mDNS questions and uDNS one-shot questions, but not for - // uDNS LongLived questions, because that would mess up our LLQ lease renewal timing. - if ((AddRecord == QC_addnocache && !q->RequestUnicast) || - (AddRecord == QC_add && (q->ExpectUnique || (rr->resrec.RecordType & kDNSRecordTypePacketUniqueMask)))) - if (ActiveQuestion(q) && (mDNSOpaque16IsZero(q->TargetQID) || !q->LongLived)) - { - ResetQuestionState(m, q); - } - - if (rr->DelayDelivery) return; // We'll come back later when CacheRecordDeferredAdd() calls us - - // Only deliver negative answers if client has explicitly requested them - if (rr->resrec.RecordType == kDNSRecordTypePacketNegative || (q->qtype != kDNSType_NSEC && RRAssertsNonexistence(&rr->resrec, q->qtype))) - if (!AddRecord || !q->ReturnIntermed) return; - - // For CNAME results to non-CNAME questions, only inform the client if they explicitly requested that - if (q->QuestionCallback && !q->NoAnswer && (!followcname || q->ReturnIntermed)) - { - mDNS_DropLockBeforeCallback(); // Allow client (and us) to legally make mDNS API calls - if (q->qtype != kDNSType_NSEC && RRAssertsNonexistence(&rr->resrec, q->qtype)) - { - CacheRecord neg; - MakeNegativeCacheRecord(m, &neg, &q->qname, q->qnamehash, q->qtype, q->qclass, 1, rr->resrec.InterfaceID, q->qDNSServer); - q->QuestionCallback(m, q, &neg.resrec, AddRecord); - } - else - q->QuestionCallback(m, q, &rr->resrec, AddRecord); - mDNS_ReclaimLockAfterCallback(); // Decrement mDNS_reentrancy to block mDNS API calls again - } - // Note: Proceed with caution here because client callback function is allowed to do anything, - // including starting/stopping queries, registering/deregistering records, etc. - - if (followcname && m->CurrentQuestion == q) - AnswerQuestionByFollowingCNAME(m, q, &rr->resrec); - } - -// New Questions are answered through AnswerNewQuestion. But there may not have been any -// matching cache records for the questions when it is called. There are two possibilities. -// -// 1) There are no cache records -// 2) There are cache records but the DNSServers between question and cache record don't match. -// -// In the case of (1), where there are no cache records and later we add them when we get a response, -// CacheRecordAdd/CacheRecordDeferredAdd will take care of adding the cache and delivering the ADD -// events to the application. If we already have a cache entry, then no ADD events are delivered -// unless the RDATA has changed -// -// In the case of (2) where we had the cache records and did not answer because of the DNSServer mismatch, -// we need to answer them whenever we change the DNSServer. But we can't do it at the instant the DNSServer -// changes because when we do the callback, the question can get deleted and the calling function would not -// know how to handle it. So, we run this function from mDNS_Execute to handle DNSServer changes on the -// question - -mDNSlocal void AnswerQuestionsForDNSServerChanges(mDNS *const m) - { - DNSQuestion *q; - DNSQuestion *qnext; - CacheRecord *rr; - mDNSu32 slot; - CacheGroup *cg; - - if (m->CurrentQuestion) - LogMsg("AnswerQuestionsForDNSServerChanges: ERROR m->CurrentQuestion already set: %##s (%s)", - m->CurrentQuestion->qname.c, DNSTypeName(m->CurrentQuestion->qtype)); - - for (q = m->Questions; q && q != m->NewQuestions; q = qnext) - { - qnext = q->next; - - // multicast or DNSServers did not change. - if (mDNSOpaque16IsZero(q->TargetQID)) continue; - if (!q->deliverAddEvents) continue; - - // We are going to look through the cache for this question since it changed - // its DNSserver last time. Reset it so that we don't call them again. Calling - // them again will deliver duplicate events to the application - q->deliverAddEvents = mDNSfalse; - if (QuerySuppressed(q)) continue; - m->CurrentQuestion = q; - slot = HashSlot(&q->qname); - cg = CacheGroupForName(m, slot, q->qnamehash, &q->qname); - for (rr = cg ? cg->members : mDNSNULL; rr; rr=rr->next) - { - if (SameNameRecordAnswersQuestion(&rr->resrec, q)) - { - LogInfo("AnswerQuestionsForDNSServerChanges: Calling AnswerCurrentQuestionWithResourceRecord for question %p %##s using resource record %s", - q, q->qname.c, CRDisplayString(m, rr)); - // When this question penalizes a DNS server and has no more DNS servers to pick, we normally - // deliver a negative cache response and suspend the question for 60 seconds (see uDNS_CheckCurrentQuestion). - // But sometimes we may already find the negative cache entry and deliver that here as the process - // of changing DNS servers. When the cache entry is about to expire, we will resend the question and - // that time, we need to make sure that we have a valid DNS server. Otherwise, we will deliver - // a negative cache response without trying the server. - if (!q->qDNSServer && !q->DuplicateOf && rr->resrec.RecordType == kDNSRecordTypePacketNegative) - { - DNSQuestion *qptr; - SetValidDNSServers(m, q); - q->qDNSServer = GetServerForQuestion(m, q); - for (qptr = q->next ; qptr; qptr = qptr->next) - if (qptr->DuplicateOf == q) { qptr->validDNSServers = q->validDNSServers; qptr->qDNSServer = q->qDNSServer; } - } - q->CurrentAnswers++; - if (rr->resrec.rdlength > SmallRecordLimit) q->LargeAnswers++; - if (rr->resrec.RecordType & kDNSRecordTypePacketUniqueMask) q->UniqueAnswers++; - AnswerCurrentQuestionWithResourceRecord(m, rr, QC_add); - if (m->CurrentQuestion != q) break; // If callback deleted q, then we're finished here - } - } - } - m->CurrentQuestion = mDNSNULL; - } - -mDNSlocal void CacheRecordDeferredAdd(mDNS *const m, CacheRecord *rr) - { - rr->DelayDelivery = 0; - if (m->CurrentQuestion) - LogMsg("CacheRecordDeferredAdd ERROR m->CurrentQuestion already set: %##s (%s)", - m->CurrentQuestion->qname.c, DNSTypeName(m->CurrentQuestion->qtype)); - m->CurrentQuestion = m->Questions; - while (m->CurrentQuestion && m->CurrentQuestion != m->NewQuestions) - { - DNSQuestion *q = m->CurrentQuestion; - if (ResourceRecordAnswersQuestion(&rr->resrec, q)) - AnswerCurrentQuestionWithResourceRecord(m, rr, QC_add); - if (m->CurrentQuestion == q) // If m->CurrentQuestion was not auto-advanced, do it ourselves now - m->CurrentQuestion = q->next; - } - m->CurrentQuestion = mDNSNULL; - } - -mDNSlocal mDNSs32 CheckForSoonToExpireRecords(mDNS *const m, const domainname *const name, const mDNSu32 namehash, const mDNSu32 slot) - { - const mDNSs32 threshhold = m->timenow + mDNSPlatformOneSecond; // See if there are any records expiring within one second - const mDNSs32 start = m->timenow - 0x10000000; - mDNSs32 delay = start; - CacheGroup *cg = CacheGroupForName(m, slot, namehash, name); - const CacheRecord *rr; - for (rr = cg ? cg->members : mDNSNULL; rr; rr=rr->next) - if (threshhold - RRExpireTime(rr) >= 0) // If we have records about to expire within a second - if (delay - RRExpireTime(rr) < 0) // then delay until after they've been deleted - delay = RRExpireTime(rr); - if (delay - start > 0) return(NonZeroTime(delay)); - else return(0); - } - -// CacheRecordAdd is only called from CreateNewCacheEntry, *never* directly as a result of a client API call. -// If new questions are created as a result of invoking client callbacks, they will be added to -// the end of the question list, and m->NewQuestions will be set to indicate the first new question. -// rr is a new CacheRecord just received into our cache -// (kDNSRecordTypePacketAns/PacketAnsUnique/PacketAdd/PacketAddUnique). -// Note: CacheRecordAdd calls AnswerCurrentQuestionWithResourceRecord which can call a user callback, -// which may change the record list and/or question list. -// Any code walking either list must use the CurrentQuestion and/or CurrentRecord mechanism to protect against this. -mDNSlocal void CacheRecordAdd(mDNS *const m, CacheRecord *rr) - { - DNSQuestion *q; - - // We stop when we get to NewQuestions -- if we increment their CurrentAnswers/LargeAnswers/UniqueAnswers - // counters here we'll end up double-incrementing them when we do it again in AnswerNewQuestion(). - for (q = m->Questions; q && q != m->NewQuestions; q=q->next) - { - if (ResourceRecordAnswersQuestion(&rr->resrec, q)) - { - // If this question is one that's actively sending queries, and it's received ten answers within one - // second of sending the last query packet, then that indicates some radical network topology change, - // so reset its exponential backoff back to the start. We must be at least at the eight-second interval - // to do this. If we're at the four-second interval, or less, there's not much benefit accelerating - // because we will anyway send another query within a few seconds. The first reset query is sent out - // randomized over the next four seconds to reduce possible synchronization between machines. - if (q->LastAnswerPktNum != m->PktNum) - { - q->LastAnswerPktNum = m->PktNum; - if (mDNSOpaque16IsZero(q->TargetQID) && ActiveQuestion(q) && ++q->RecentAnswerPkts >= 10 && - q->ThisQInterval > InitialQuestionInterval * QuestionIntervalStep3 && m->timenow - q->LastQTxTime < mDNSPlatformOneSecond) - { - LogMsg("CacheRecordAdd: %##s (%s) got immediate answer burst (%d); restarting exponential backoff sequence (%d)", - q->qname.c, DNSTypeName(q->qtype), q->RecentAnswerPkts, q->ThisQInterval); - q->LastQTime = m->timenow - InitialQuestionInterval + (mDNSs32)mDNSRandom((mDNSu32)mDNSPlatformOneSecond*4); - q->ThisQInterval = InitialQuestionInterval; - SetNextQueryTime(m,q); - } - } - verbosedebugf("CacheRecordAdd %p %##s (%s) %lu %#a:%d question %p", rr, rr->resrec.name->c, - DNSTypeName(rr->resrec.rrtype), rr->resrec.rroriginalttl, rr->resrec.rDNSServer ? - &rr->resrec.rDNSServer->addr : mDNSNULL, mDNSVal16(rr->resrec.rDNSServer ? - rr->resrec.rDNSServer->port : zeroIPPort), q); - q->CurrentAnswers++; - q->unansweredQueries = 0; - if (rr->resrec.rdlength > SmallRecordLimit) q->LargeAnswers++; - if (rr->resrec.RecordType & kDNSRecordTypePacketUniqueMask) q->UniqueAnswers++; - if (q->CurrentAnswers > 4000) - { - static int msgcount = 0; - if (msgcount++ < 10) - LogMsg("CacheRecordAdd: %##s (%s) has %d answers; shedding records to resist DOS attack", - q->qname.c, DNSTypeName(q->qtype), q->CurrentAnswers); - rr->resrec.rroriginalttl = 0; - rr->UnansweredQueries = MaxUnansweredQueries; - } - } - } - - if (!rr->DelayDelivery) - { - if (m->CurrentQuestion) - LogMsg("CacheRecordAdd ERROR m->CurrentQuestion already set: %##s (%s)", m->CurrentQuestion->qname.c, DNSTypeName(m->CurrentQuestion->qtype)); - m->CurrentQuestion = m->Questions; - while (m->CurrentQuestion && m->CurrentQuestion != m->NewQuestions) - { - q = m->CurrentQuestion; - if (ResourceRecordAnswersQuestion(&rr->resrec, q)) - AnswerCurrentQuestionWithResourceRecord(m, rr, QC_add); - if (m->CurrentQuestion == q) // If m->CurrentQuestion was not auto-advanced, do it ourselves now - m->CurrentQuestion = q->next; - } - m->CurrentQuestion = mDNSNULL; - } - - SetNextCacheCheckTimeForRecord(m, rr); - } - -// NoCacheAnswer is only called from mDNSCoreReceiveResponse, *never* directly as a result of a client API call. -// If new questions are created as a result of invoking client callbacks, they will be added to -// the end of the question list, and m->NewQuestions will be set to indicate the first new question. -// rr is a new CacheRecord just received from the wire (kDNSRecordTypePacketAns/AnsUnique/Add/AddUnique) -// but we don't have any place to cache it. We'll deliver question 'add' events now, but we won't have any -// way to deliver 'remove' events in future, nor will we be able to include this in known-answer lists, -// so we immediately bump ThisQInterval up to MaxQuestionInterval to avoid pounding the network. -// Note: NoCacheAnswer calls AnswerCurrentQuestionWithResourceRecord which can call a user callback, -// which may change the record list and/or question list. -// Any code walking either list must use the CurrentQuestion and/or CurrentRecord mechanism to protect against this. -mDNSlocal void NoCacheAnswer(mDNS *const m, CacheRecord *rr) - { - LogMsg("No cache space: Delivering non-cached result for %##s", m->rec.r.resrec.name->c); - if (m->CurrentQuestion) - LogMsg("NoCacheAnswer ERROR m->CurrentQuestion already set: %##s (%s)", m->CurrentQuestion->qname.c, DNSTypeName(m->CurrentQuestion->qtype)); - m->CurrentQuestion = m->Questions; - // We do this for *all* questions, not stopping when we get to m->NewQuestions, - // since we're not caching the record and we'll get no opportunity to do this later - while (m->CurrentQuestion) - { - DNSQuestion *q = m->CurrentQuestion; - if (ResourceRecordAnswersQuestion(&rr->resrec, q)) - AnswerCurrentQuestionWithResourceRecord(m, rr, QC_addnocache); // QC_addnocache means "don't expect remove events for this" - if (m->CurrentQuestion == q) // If m->CurrentQuestion was not auto-advanced, do it ourselves now - m->CurrentQuestion = q->next; - } - m->CurrentQuestion = mDNSNULL; - } - -// CacheRecordRmv is only called from CheckCacheExpiration, which is called from mDNS_Execute. -// Note that CacheRecordRmv is *only* called for records that are referenced by at least one active question. -// If new questions are created as a result of invoking client callbacks, they will be added to -// the end of the question list, and m->NewQuestions will be set to indicate the first new question. -// rr is an existing cache CacheRecord that just expired and is being deleted -// (kDNSRecordTypePacketAns/PacketAnsUnique/PacketAdd/PacketAddUnique). -// Note: CacheRecordRmv calls AnswerCurrentQuestionWithResourceRecord which can call a user callback, -// which may change the record list and/or question list. -// Any code walking either list must use the CurrentQuestion and/or CurrentRecord mechanism to protect against this. -mDNSlocal void CacheRecordRmv(mDNS *const m, CacheRecord *rr) - { - if (m->CurrentQuestion) - LogMsg("CacheRecordRmv ERROR m->CurrentQuestion already set: %##s (%s)", - m->CurrentQuestion->qname.c, DNSTypeName(m->CurrentQuestion->qtype)); - m->CurrentQuestion = m->Questions; - - // We stop when we get to NewQuestions -- for new questions their CurrentAnswers/LargeAnswers/UniqueAnswers counters - // will all still be zero because we haven't yet gone through the cache counting how many answers we have for them. - while (m->CurrentQuestion && m->CurrentQuestion != m->NewQuestions) - { - DNSQuestion *q = m->CurrentQuestion; - // When a question enters suppressed state, we generate RMV events and generate a negative - // response. A cache may be present that answers this question e.g., cache entry generated - // before the question became suppressed. We need to skip the suppressed questions here as - // the RMV event has already been generated. - if (!QuerySuppressed(q) && ResourceRecordAnswersQuestion(&rr->resrec, q)) - { - verbosedebugf("CacheRecordRmv %p %s", rr, CRDisplayString(m, rr)); - q->FlappingInterface1 = mDNSNULL; - q->FlappingInterface2 = mDNSNULL; - - // When a question changes DNS server, it is marked with deliverAddEvents if we find any - // cache entry corresponding to the new DNS server. Before we deliver the ADD event, the - // cache entry may be removed in which case CurrentAnswers can be zero. - if (q->deliverAddEvents && !q->CurrentAnswers) - { - LogInfo("CacheRecordRmv: Question %p %##s (%s) deliverAddEvents set, DNSServer %#a:%d", - q, q->qname.c, DNSTypeName(q->qtype), q->qDNSServer ? &q->qDNSServer->addr : mDNSNULL, - mDNSVal16(q->qDNSServer ? q->qDNSServer->port : zeroIPPort)); - m->CurrentQuestion = q->next; - continue; - } - if (q->CurrentAnswers == 0) - LogMsg("CacheRecordRmv ERROR!!: How can CurrentAnswers already be zero for %p %##s (%s) DNSServer %#a:%d", - q, q->qname.c, DNSTypeName(q->qtype), q->qDNSServer ? &q->qDNSServer->addr : mDNSNULL, - mDNSVal16(q->qDNSServer ? q->qDNSServer->port : zeroIPPort)); - else - { - q->CurrentAnswers--; - if (rr->resrec.rdlength > SmallRecordLimit) q->LargeAnswers--; - if (rr->resrec.RecordType & kDNSRecordTypePacketUniqueMask) q->UniqueAnswers--; - } - if (rr->resrec.rdata->MaxRDLength) // Never generate "remove" events for negative results - { - if (q->CurrentAnswers == 0) - { - LogInfo("CacheRecordRmv: Last answer for %##s (%s) expired from cache; will reconfirm antecedents", - q->qname.c, DNSTypeName(q->qtype)); - ReconfirmAntecedents(m, &q->qname, q->qnamehash, 0); - } - AnswerCurrentQuestionWithResourceRecord(m, rr, QC_rmv); - } - } - if (m->CurrentQuestion == q) // If m->CurrentQuestion was not auto-advanced, do it ourselves now - m->CurrentQuestion = q->next; - } - m->CurrentQuestion = mDNSNULL; - } - -mDNSlocal void ReleaseCacheEntity(mDNS *const m, CacheEntity *e) - { -#if APPLE_OSX_mDNSResponder && MACOSX_MDNS_MALLOC_DEBUGGING >= 1 - unsigned int i; - for (i=0; inext = m->rrcache_free; - m->rrcache_free = e; - m->rrcache_totalused--; - } - -mDNSlocal void ReleaseCacheGroup(mDNS *const m, CacheGroup **cp) - { - CacheEntity *e = (CacheEntity *)(*cp); - //LogMsg("ReleaseCacheGroup: Releasing CacheGroup for %p, %##s", (*cp)->name->c, (*cp)->name->c); - if ((*cp)->rrcache_tail != &(*cp)->members) - LogMsg("ERROR: (*cp)->members == mDNSNULL but (*cp)->rrcache_tail != &(*cp)->members)"); - //if ((*cp)->name != (domainname*)((*cp)->namestorage)) - // LogMsg("ReleaseCacheGroup: %##s, %p %p", (*cp)->name->c, (*cp)->name, (domainname*)((*cp)->namestorage)); - if ((*cp)->name != (domainname*)((*cp)->namestorage)) mDNSPlatformMemFree((*cp)->name); - (*cp)->name = mDNSNULL; - *cp = (*cp)->next; // Cut record from list - ReleaseCacheEntity(m, e); - } - -mDNSlocal void ReleaseCacheRecord(mDNS *const m, CacheRecord *r) - { - //LogMsg("ReleaseCacheRecord: Releasing %s", CRDisplayString(m, r)); - if (r->resrec.rdata && r->resrec.rdata != (RData*)&r->smallrdatastorage) mDNSPlatformMemFree(r->resrec.rdata); - r->resrec.rdata = mDNSNULL; - ReleaseCacheEntity(m, (CacheEntity *)r); - } - -// Note: We want to be careful that we deliver all the CacheRecordRmv calls before delivering -// CacheRecordDeferredAdd calls. The in-order nature of the cache lists ensures that all -// callbacks for old records are delivered before callbacks for newer records. -mDNSlocal void CheckCacheExpiration(mDNS *const m, const mDNSu32 slot, CacheGroup *const cg) - { - CacheRecord **rp = &cg->members; - - if (m->lock_rrcache) { LogMsg("CheckCacheExpiration ERROR! Cache already locked!"); return; } - m->lock_rrcache = 1; - - while (*rp) - { - CacheRecord *const rr = *rp; - mDNSs32 event = RRExpireTime(rr); - if (m->timenow - event >= 0) // If expired, delete it - { - *rp = rr->next; // Cut it from the list - verbosedebugf("CheckCacheExpiration: Deleting%7d %7d %p %s", - m->timenow - rr->TimeRcvd, rr->resrec.rroriginalttl, rr->CRActiveQuestion, CRDisplayString(m, rr)); - if (rr->CRActiveQuestion) // If this record has one or more active questions, tell them it's going away - { - DNSQuestion *q = rr->CRActiveQuestion; - // When a cache record is about to expire, we expect to do four queries at 80-82%, 85-87%, 90-92% and - // then 95-97% of the TTL. If the DNS server does not respond, then we will remove the cache entry - // before we pick a new DNS server. As the question interval is set to MaxQuestionInterval, we may - // not send out a query anytime soon. Hence, we need to reset the question interval. If this is - // a normal deferred ADD case, then AnswerCurrentQuestionWithResourceRecord will reset it to - // MaxQuestionInterval. If we have inactive questions referring to negative cache entries, - // don't ressurect them as they will deliver duplicate "No such Record" ADD events - if (!mDNSOpaque16IsZero(q->TargetQID) && !q->LongLived && ActiveQuestion(q)) - { - q->ThisQInterval = InitialQuestionInterval; - q->LastQTime = m->timenow - q->ThisQInterval; - SetNextQueryTime(m, q); - } - CacheRecordRmv(m, rr); - m->rrcache_active--; - } - ReleaseCacheRecord(m, rr); - } - else // else, not expired; see if we need to query - { - // If waiting to delay delivery, do nothing until then - if (rr->DelayDelivery && rr->DelayDelivery - m->timenow > 0) - event = rr->DelayDelivery; - else - { - if (rr->DelayDelivery) CacheRecordDeferredAdd(m, rr); - if (rr->CRActiveQuestion && rr->UnansweredQueries < MaxUnansweredQueries) - { - if (m->timenow - rr->NextRequiredQuery < 0) // If not yet time for next query - event = NextCacheCheckEvent(rr); // then just record when we want the next query - else // else trigger our question to go out now - { - // Set NextScheduledQuery to timenow so that SendQueries() will run. - // SendQueries() will see that we have records close to expiration, and send FEQs for them. - m->NextScheduledQuery = m->timenow; - // After sending the query we'll increment UnansweredQueries and call SetNextCacheCheckTimeForRecord(), - // which will correctly update m->NextCacheCheck for us. - event = m->timenow + 0x3FFFFFFF; - } - } - } - verbosedebugf("CheckCacheExpiration:%6d %5d %s", - (event - m->timenow) / mDNSPlatformOneSecond, CacheCheckGracePeriod(rr), CRDisplayString(m, rr)); - if (m->rrcache_nextcheck[slot] - event > 0) - m->rrcache_nextcheck[slot] = event; - rp = &rr->next; - } - } - if (cg->rrcache_tail != rp) verbosedebugf("CheckCacheExpiration: Updating CacheGroup tail from %p to %p", cg->rrcache_tail, rp); - cg->rrcache_tail = rp; - m->lock_rrcache = 0; - } - -mDNSlocal void AnswerNewQuestion(mDNS *const m) - { - mDNSBool ShouldQueryImmediately = mDNStrue; - DNSQuestion *const q = m->NewQuestions; // Grab the question we're going to answer - mDNSu32 slot = HashSlot(&q->qname); - CacheGroup *const cg = CacheGroupForName(m, slot, q->qnamehash, &q->qname); - AuthRecord *lr; - AuthGroup *ag; - mDNSBool AnsweredFromCache = mDNSfalse; - - verbosedebugf("AnswerNewQuestion: Answering %##s (%s)", q->qname.c, DNSTypeName(q->qtype)); - - if (cg) CheckCacheExpiration(m, slot, cg); - if (m->NewQuestions != q) { LogInfo("AnswerNewQuestion: Question deleted while doing CheckCacheExpiration"); goto exit; } - m->NewQuestions = q->next; - // Advance NewQuestions to the next *after* calling CheckCacheExpiration, because if we advance it first - // then CheckCacheExpiration may give this question add/remove callbacks, and it's not yet ready for that. - // - // Also, CheckCacheExpiration() calls CacheRecordDeferredAdd() and CacheRecordRmv(), which invoke - // client callbacks, which may delete their own or any other question. Our mechanism for detecting - // whether our current m->NewQuestions question got deleted by one of these callbacks is to store the - // value of m->NewQuestions in 'q' before calling CheckCacheExpiration(), and then verify afterwards - // that they're still the same. If m->NewQuestions has changed (because mDNS_StopQuery_internal - // advanced it), that means the question was deleted, so we no longer need to worry about answering - // it (and indeed 'q' is now a dangling pointer, so dereferencing it at all would be bad, and the - // values we computed for slot and cg are now stale and relate to a question that no longer exists). - // - // We can't use the usual m->CurrentQuestion mechanism for this because CacheRecordDeferredAdd() and - // CacheRecordRmv() both use that themselves when walking the list of (non-new) questions generating callbacks. - // Fortunately mDNS_StopQuery_internal auto-advances both m->CurrentQuestion *AND* m->NewQuestions when - // deleting a question, so luckily we have an easy alternative way of detecting if our question got deleted. - - if (m->lock_rrcache) LogMsg("AnswerNewQuestion ERROR! Cache already locked!"); - // This should be safe, because calling the client's question callback may cause the - // question list to be modified, but should not ever cause the rrcache list to be modified. - // If the client's question callback deletes the question, then m->CurrentQuestion will - // be advanced, and we'll exit out of the loop - m->lock_rrcache = 1; - if (m->CurrentQuestion) - LogMsg("AnswerNewQuestion ERROR m->CurrentQuestion already set: %##s (%s)", - m->CurrentQuestion->qname.c, DNSTypeName(m->CurrentQuestion->qtype)); - m->CurrentQuestion = q; // Indicate which question we're answering, so we'll know if it gets deleted - - if (q->NoAnswer == NoAnswer_Fail) - { - LogMsg("AnswerNewQuestion: NoAnswer_Fail %##s (%s)", q->qname.c, DNSTypeName(q->qtype)); - MakeNegativeCacheRecord(m, &m->rec.r, &q->qname, q->qnamehash, q->qtype, q->qclass, 60, mDNSInterface_Any, q->qDNSServer); - q->NoAnswer = NoAnswer_Normal; // Temporarily turn off answer suppression - AnswerCurrentQuestionWithResourceRecord(m, &m->rec.r, QC_addnocache); - // Don't touch the question if it has been stopped already - if (m->CurrentQuestion == q) q->NoAnswer = NoAnswer_Fail; // Restore NoAnswer state - m->rec.r.resrec.RecordType = 0; // Clear RecordType to show we're not still using it - } - if (m->CurrentQuestion != q) { LogInfo("AnswerNewQuestion: Question deleted while generating NoAnswer_Fail response"); goto exit; } - - // See if we want to tell it about LocalOnly records - if (m->CurrentRecord) - LogMsg("AnswerNewQuestion ERROR m->CurrentRecord already set %s", ARDisplayString(m, m->CurrentRecord)); - slot = AuthHashSlot(&q->qname); - ag = AuthGroupForName(&m->rrauth, slot, q->qnamehash, &q->qname); - if (ag) - { - m->CurrentRecord = ag->members; - while (m->CurrentRecord && m->CurrentRecord != ag->NewLocalOnlyRecords) - { - AuthRecord *rr = m->CurrentRecord; - m->CurrentRecord = rr->next; - // - // If the question is mDNSInterface_LocalOnly, all records local to the machine should be used - // to answer the query. This is handled in AnswerNewLocalOnlyQuestion. - // - // We handle mDNSInterface_Any and scoped questions here. See LocalOnlyRecordAnswersQuestion for more - // details on how we handle this case. For P2P we just handle "Interface_Any" questions. For LocalOnly - // we handle both mDNSInterface_Any and scoped questions. - - if (rr->ARType == AuthRecordLocalOnly || (rr->ARType == AuthRecordP2P && q->InterfaceID == mDNSInterface_Any)) - if (LocalOnlyRecordAnswersQuestion(rr, q)) - { - AnswerLocalQuestionWithLocalAuthRecord(m, rr, mDNStrue); - if (m->CurrentQuestion != q) break; // If callback deleted q, then we're finished here - } - } - } - m->CurrentRecord = mDNSNULL; - - if (m->CurrentQuestion != q) { LogInfo("AnswerNewQuestion: Question deleted while while giving LocalOnly record answers"); goto exit; } - - if (q->LOAddressAnswers) - { - LogInfo("AnswerNewQuestion: Question %p %##s (%s) answered using local auth records LOAddressAnswers %d", - q, q->qname.c, DNSTypeName(q->qtype), q->LOAddressAnswers); - goto exit; - } - - // Before we go check the cache and ship this query on the wire, we have to be sure that there are - // no local records that could possibly answer this question. As we did not check the NewLocalRecords, we - // need to just peek at them to see whether it will answer this question. If it would answer, pretend - // that we answered. AnswerAllLocalQuestionsWithLocalAuthRecord will answer shortly. This happens normally - // when we add new /etc/hosts entries and restart the question. It is a new question and also a new record. - if (ag) - { - lr = ag->NewLocalOnlyRecords; - while (lr) - { - if (LORecordAnswersAddressType(lr) && LocalOnlyRecordAnswersQuestion(lr, q)) - { - LogInfo("AnswerNewQuestion: Question %p %##s (%s) will be answered using new local auth records " - " LOAddressAnswers %d", q, q->qname.c, DNSTypeName(q->qtype), q->LOAddressAnswers); - goto exit; - } - lr = lr->next; - } - } - - - // If we are not supposed to answer this question, generate a negative response. - // Temporarily suspend the SuppressQuery so that AnswerCurrentQuestionWithResourceRecord can answer the question - if (QuerySuppressed(q)) { q->SuppressQuery = mDNSfalse; GenerateNegativeResponse(m); q->SuppressQuery = mDNStrue; } - else - { - CacheRecord *rr; - for (rr = cg ? cg->members : mDNSNULL; rr; rr=rr->next) - if (SameNameRecordAnswersQuestion(&rr->resrec, q)) - { - // SecsSinceRcvd is whole number of elapsed seconds, rounded down - mDNSu32 SecsSinceRcvd = ((mDNSu32)(m->timenow - rr->TimeRcvd)) / mDNSPlatformOneSecond; - if (rr->resrec.rroriginalttl <= SecsSinceRcvd) - { - LogMsg("AnswerNewQuestion: How is rr->resrec.rroriginalttl %lu <= SecsSinceRcvd %lu for %s %d %d", - rr->resrec.rroriginalttl, SecsSinceRcvd, CRDisplayString(m, rr), m->timenow, rr->TimeRcvd); - continue; // Go to next one in loop - } - - // If this record set is marked unique, then that means we can reasonably assume we have the whole set - // -- we don't need to rush out on the network and query immediately to see if there are more answers out there - if ((rr->resrec.RecordType & kDNSRecordTypePacketUniqueMask) || (q->ExpectUnique)) - ShouldQueryImmediately = mDNSfalse; - q->CurrentAnswers++; - if (rr->resrec.rdlength > SmallRecordLimit) q->LargeAnswers++; - if (rr->resrec.RecordType & kDNSRecordTypePacketUniqueMask) q->UniqueAnswers++; - AnsweredFromCache = mDNStrue; - AnswerCurrentQuestionWithResourceRecord(m, rr, QC_add); - if (m->CurrentQuestion != q) break; // If callback deleted q, then we're finished here - } - else if (RRTypeIsAddressType(rr->resrec.rrtype) && RRTypeIsAddressType(q->qtype)) - ShouldQueryImmediately = mDNSfalse; - } - // We don't use LogInfo for this "Question deleted" message because it happens so routinely that - // it's not remotely remarkable, and therefore unlikely to be of much help tracking down bugs. - if (m->CurrentQuestion != q) { debugf("AnswerNewQuestion: Question deleted while giving cache answers"); goto exit; } - - // Neither a local record nor a cache entry could answer this question. If this question need to be retried - // with search domains, generate a negative response which will now retry after appending search domains. - // If the query was suppressed above, we already generated a negative response. When it gets unsuppressed, - // we will retry with search domains. - if (!QuerySuppressed(q) && !AnsweredFromCache && q->RetryWithSearchDomains) - { - LogInfo("AnswerNewQuestion: Generating response for retrying with search domains %##s (%s)", q->qname.c, DNSTypeName(q->qtype)); - GenerateNegativeResponse(m); - } - - if (m->CurrentQuestion != q) { debugf("AnswerNewQuestion: Question deleted while giving negative answer"); goto exit; } - - // Note: When a query gets suppressed or retried with search domains, we de-activate the question. - // Hence we don't execute the following block of code for those cases. - if (ShouldQueryImmediately && ActiveQuestion(q)) - { - debugf("AnswerNewQuestion: ShouldQueryImmediately %##s (%s)", q->qname.c, DNSTypeName(q->qtype)); - q->ThisQInterval = InitialQuestionInterval; - q->LastQTime = m->timenow - q->ThisQInterval; - if (mDNSOpaque16IsZero(q->TargetQID)) // For mDNS, spread packets to avoid a burst of simultaneous queries - { - // Compute random delay in the range 1-6 seconds, then divide by 50 to get 20-120ms - if (!m->RandomQueryDelay) - m->RandomQueryDelay = (mDNSPlatformOneSecond + mDNSRandom(mDNSPlatformOneSecond*5) - 1) / 50 + 1; - q->LastQTime += m->RandomQueryDelay; - } - } - - // IN ALL CASES make sure that m->NextScheduledQuery is set appropriately. - // In cases where m->NewQuestions->DelayAnswering is set, we may have delayed generating our - // answers for this question until *after* its scheduled transmission time, in which case - // m->NextScheduledQuery may now be set to 'never', and in that case -- even though we're *not* doing - // ShouldQueryImmediately -- we still need to make sure we set m->NextScheduledQuery correctly. - SetNextQueryTime(m,q); - -exit: - m->CurrentQuestion = mDNSNULL; - m->lock_rrcache = 0; - } - -// When a NewLocalOnlyQuestion is created, AnswerNewLocalOnlyQuestion runs though our ResourceRecords delivering any -// appropriate answers, stopping if it reaches a NewLocalOnlyRecord -- these will be handled by AnswerAllLocalQuestionsWithLocalAuthRecord -mDNSlocal void AnswerNewLocalOnlyQuestion(mDNS *const m) - { - mDNSu32 slot; - AuthGroup *ag; - DNSQuestion *q = m->NewLocalOnlyQuestions; // Grab the question we're going to answer - m->NewLocalOnlyQuestions = q->next; // Advance NewLocalOnlyQuestions to the next (if any) - - debugf("AnswerNewLocalOnlyQuestion: Answering %##s (%s)", q->qname.c, DNSTypeName(q->qtype)); - - if (m->CurrentQuestion) - LogMsg("AnswerNewLocalOnlyQuestion ERROR m->CurrentQuestion already set: %##s (%s)", - m->CurrentQuestion->qname.c, DNSTypeName(m->CurrentQuestion->qtype)); - m->CurrentQuestion = q; // Indicate which question we're answering, so we'll know if it gets deleted - - if (m->CurrentRecord) - LogMsg("AnswerNewLocalOnlyQuestion ERROR m->CurrentRecord already set %s", ARDisplayString(m, m->CurrentRecord)); - - // 1. First walk the LocalOnly records answering the LocalOnly question - // 2. As LocalOnly questions should also be answered by any other Auth records local to the machine, - // walk the ResourceRecords list delivering the answers - slot = AuthHashSlot(&q->qname); - ag = AuthGroupForName(&m->rrauth, slot, q->qnamehash, &q->qname); - if (ag) - { - m->CurrentRecord = ag->members; - while (m->CurrentRecord && m->CurrentRecord != ag->NewLocalOnlyRecords) - { - AuthRecord *rr = m->CurrentRecord; - m->CurrentRecord = rr->next; - if (LocalOnlyRecordAnswersQuestion(rr, q)) - { - AnswerLocalQuestionWithLocalAuthRecord(m, rr, mDNStrue); - if (m->CurrentQuestion != q) break; // If callback deleted q, then we're finished here - } - } - } - - if (m->CurrentQuestion == q) - { - m->CurrentRecord = m->ResourceRecords; - - while (m->CurrentRecord && m->CurrentRecord != m->NewLocalRecords) - { - AuthRecord *rr = m->CurrentRecord; - m->CurrentRecord = rr->next; - if (ResourceRecordAnswersQuestion(&rr->resrec, q)) - { - AnswerLocalQuestionWithLocalAuthRecord(m, rr, mDNStrue); - if (m->CurrentQuestion != q) break; // If callback deleted q, then we're finished here - } - } - } - - m->CurrentQuestion = mDNSNULL; - m->CurrentRecord = mDNSNULL; - } - -mDNSlocal CacheEntity *GetCacheEntity(mDNS *const m, const CacheGroup *const PreserveCG) - { - CacheEntity *e = mDNSNULL; - - if (m->lock_rrcache) { LogMsg("GetFreeCacheRR ERROR! Cache already locked!"); return(mDNSNULL); } - m->lock_rrcache = 1; - - // If we have no free records, ask the client layer to give us some more memory - if (!m->rrcache_free && m->MainCallback) - { - if (m->rrcache_totalused != m->rrcache_size) - LogMsg("GetFreeCacheRR: count mismatch: m->rrcache_totalused %lu != m->rrcache_size %lu", - m->rrcache_totalused, m->rrcache_size); - - // We don't want to be vulnerable to a malicious attacker flooding us with an infinite - // number of bogus records so that we keep growing our cache until the machine runs out of memory. - // To guard against this, if our cache grows above 512kB (approx 3168 records at 164 bytes each), - // and we're actively using less than 1/32 of that cache, then we purge all the unused records - // and recycle them, instead of allocating more memory. - if (m->rrcache_size > 5000 && m->rrcache_size / 32 > m->rrcache_active) - LogInfo("Possible denial-of-service attack in progress: m->rrcache_size %lu; m->rrcache_active %lu", - m->rrcache_size, m->rrcache_active); - else - { - mDNS_DropLockBeforeCallback(); // Allow client to legally make mDNS API calls from the callback - m->MainCallback(m, mStatus_GrowCache); - mDNS_ReclaimLockAfterCallback(); // Decrement mDNS_reentrancy to block mDNS API calls again - } - } - - // If we still have no free records, recycle all the records we can. - // Enumerating the entire cache is moderately expensive, so when we do it, we reclaim all the records we can in one pass. - if (!m->rrcache_free) - { - mDNSu32 oldtotalused = m->rrcache_totalused; - mDNSu32 slot; - for (slot = 0; slot < CACHE_HASH_SLOTS; slot++) - { - CacheGroup **cp = &m->rrcache_hash[slot]; - while (*cp) - { - CacheRecord **rp = &(*cp)->members; - while (*rp) - { - // Records that answer still-active questions are not candidates for recycling - // Records that are currently linked into the CacheFlushRecords list may not be recycled, or we'll crash - if ((*rp)->CRActiveQuestion || (*rp)->NextInCFList) - rp=&(*rp)->next; - else - { - CacheRecord *rr = *rp; - *rp = (*rp)->next; // Cut record from list - ReleaseCacheRecord(m, rr); - } - } - if ((*cp)->rrcache_tail != rp) - verbosedebugf("GetFreeCacheRR: Updating rrcache_tail[%lu] from %p to %p", slot, (*cp)->rrcache_tail, rp); - (*cp)->rrcache_tail = rp; - if ((*cp)->members || (*cp)==PreserveCG) cp=&(*cp)->next; - else ReleaseCacheGroup(m, cp); - } - } - LogInfo("GetCacheEntity recycled %d records to reduce cache from %d to %d", - oldtotalused - m->rrcache_totalused, oldtotalused, m->rrcache_totalused); - } - - if (m->rrcache_free) // If there are records in the free list, take one - { - e = m->rrcache_free; - m->rrcache_free = e->next; - if (++m->rrcache_totalused >= m->rrcache_report) - { - LogInfo("RR Cache now using %ld objects", m->rrcache_totalused); - if (m->rrcache_report < 100) m->rrcache_report += 10; - else if (m->rrcache_report < 1000) m->rrcache_report += 100; - else m->rrcache_report += 1000; - } - mDNSPlatformMemZero(e, sizeof(*e)); - } - - m->lock_rrcache = 0; - - return(e); - } - -mDNSlocal CacheRecord *GetCacheRecord(mDNS *const m, CacheGroup *cg, mDNSu16 RDLength) - { - CacheRecord *r = (CacheRecord *)GetCacheEntity(m, cg); - if (r) - { - r->resrec.rdata = (RData*)&r->smallrdatastorage; // By default, assume we're usually going to be using local storage - if (RDLength > InlineCacheRDSize) // If RDLength is too big, allocate extra storage - { - r->resrec.rdata = (RData*)mDNSPlatformMemAllocate(sizeofRDataHeader + RDLength); - if (r->resrec.rdata) r->resrec.rdata->MaxRDLength = r->resrec.rdlength = RDLength; - else { ReleaseCacheEntity(m, (CacheEntity*)r); r = mDNSNULL; } - } - } - return(r); - } - -mDNSlocal CacheGroup *GetCacheGroup(mDNS *const m, const mDNSu32 slot, const ResourceRecord *const rr) - { - mDNSu16 namelen = DomainNameLength(rr->name); - CacheGroup *cg = (CacheGroup*)GetCacheEntity(m, mDNSNULL); - if (!cg) { LogMsg("GetCacheGroup: Failed to allocate memory for %##s", rr->name->c); return(mDNSNULL); } - cg->next = m->rrcache_hash[slot]; - cg->namehash = rr->namehash; - cg->members = mDNSNULL; - cg->rrcache_tail = &cg->members; - cg->name = (domainname*)cg->namestorage; - //LogMsg("GetCacheGroup: %-10s %d-byte cache name %##s", - // (namelen > InlineCacheGroupNameSize) ? "Allocating" : "Inline", namelen, rr->name->c); - if (namelen > InlineCacheGroupNameSize) cg->name = mDNSPlatformMemAllocate(namelen); - if (!cg->name) - { - LogMsg("GetCacheGroup: Failed to allocate name storage for %##s", rr->name->c); - ReleaseCacheEntity(m, (CacheEntity*)cg); - return(mDNSNULL); - } - AssignDomainName(cg->name, rr->name); - - if (CacheGroupForRecord(m, slot, rr)) LogMsg("GetCacheGroup: Already have CacheGroup for %##s", rr->name->c); - m->rrcache_hash[slot] = cg; - if (CacheGroupForRecord(m, slot, rr) != cg) LogMsg("GetCacheGroup: Not finding CacheGroup for %##s", rr->name->c); - - return(cg); - } - -mDNSexport void mDNS_PurgeCacheResourceRecord(mDNS *const m, CacheRecord *rr) - { - if (m->mDNS_busy != m->mDNS_reentrancy+1) - LogMsg("mDNS_PurgeCacheResourceRecord: Lock not held! mDNS_busy (%ld) mDNS_reentrancy (%ld)", m->mDNS_busy, m->mDNS_reentrancy); - // Make sure we mark this record as thoroughly expired -- we don't ever want to give - // a positive answer using an expired record (e.g. from an interface that has gone away). - // We don't want to clear CRActiveQuestion here, because that would leave the record subject to - // summary deletion without giving the proper callback to any questions that are monitoring it. - // By setting UnansweredQueries to MaxUnansweredQueries we ensure it won't trigger any further expiration queries. - rr->TimeRcvd = m->timenow - mDNSPlatformOneSecond * 60; - rr->UnansweredQueries = MaxUnansweredQueries; - rr->resrec.rroriginalttl = 0; - SetNextCacheCheckTimeForRecord(m, rr); - } - -mDNSexport mDNSs32 mDNS_TimeNow(const mDNS *const m) - { - mDNSs32 time; - mDNSPlatformLock(m); - if (m->mDNS_busy) - { - LogMsg("mDNS_TimeNow called while holding mDNS lock. This is incorrect. Code protected by lock should just use m->timenow."); - if (!m->timenow) LogMsg("mDNS_TimeNow: m->mDNS_busy is %ld but m->timenow not set", m->mDNS_busy); - } - - if (m->timenow) time = m->timenow; - else time = mDNS_TimeNow_NoLock(m); - mDNSPlatformUnlock(m); - return(time); - } - -// To avoid pointless CPU thrash, we use SetSPSProxyListChanged(X) to record the last interface that -// had its Sleep Proxy client list change, and defer to actual BPF reconfiguration to mDNS_Execute(). -// (GetNextScheduledEvent() returns "now" when m->SPSProxyListChanged is set) -#define SetSPSProxyListChanged(X) do { \ - if (m->SPSProxyListChanged && m->SPSProxyListChanged != (X)) mDNSPlatformUpdateProxyList(m, m->SPSProxyListChanged); \ - m->SPSProxyListChanged = (X); } while(0) - -// Called from mDNS_Execute() to expire stale proxy records -mDNSlocal void CheckProxyRecords(mDNS *const m, AuthRecord *list) - { - m->CurrentRecord = list; - while (m->CurrentRecord) - { - AuthRecord *rr = m->CurrentRecord; - if (rr->resrec.RecordType != kDNSRecordTypeDeregistering && rr->WakeUp.HMAC.l[0]) - { - // If m->SPSSocket is NULL that means we're not acting as a sleep proxy any more, - // so we need to cease proxying for *all* records we may have, expired or not. - if (m->SPSSocket && m->timenow - rr->TimeExpire < 0) // If proxy record not expired yet, update m->NextScheduledSPS - { - if (m->NextScheduledSPS - rr->TimeExpire > 0) - m->NextScheduledSPS = rr->TimeExpire; - } - else // else proxy record expired, so remove it - { - LogSPS("CheckProxyRecords: Removing %d H-MAC %.6a I-MAC %.6a %d %s", - m->ProxyRecords, &rr->WakeUp.HMAC, &rr->WakeUp.IMAC, rr->WakeUp.seq, ARDisplayString(m, rr)); - SetSPSProxyListChanged(rr->resrec.InterfaceID); - mDNS_Deregister_internal(m, rr, mDNS_Dereg_normal); - // Don't touch rr after this -- memory may have been free'd - } - } - // Mustn't advance m->CurrentRecord until *after* mDNS_Deregister_internal, because - // new records could have been added to the end of the list as a result of that call. - if (m->CurrentRecord == rr) // If m->CurrentRecord was not advanced for us, do it now - m->CurrentRecord = rr->next; - } - } - -mDNSlocal void CheckRmvEventsForLocalRecords(mDNS *const m) - { - while (m->CurrentRecord) - { - AuthRecord *rr = m->CurrentRecord; - if (rr->AnsweredLocalQ && rr->resrec.RecordType == kDNSRecordTypeDeregistering) - { - debugf("CheckRmvEventsForLocalRecords: Generating local RMV events for %s", ARDisplayString(m, rr)); - rr->resrec.RecordType = kDNSRecordTypeShared; - AnswerAllLocalQuestionsWithLocalAuthRecord(m, rr, mDNSfalse); - if (m->CurrentRecord == rr) // If rr still exists in list, restore its state now - { - rr->resrec.RecordType = kDNSRecordTypeDeregistering; - rr->AnsweredLocalQ = mDNSfalse; - // SendResponses normally calls CompleteDeregistration after sending goodbyes. - // For LocalOnly records, we don't do that and hence we need to do that here. - if (RRLocalOnly(rr)) CompleteDeregistration(m, rr); - } - } - if (m->CurrentRecord == rr) // If m->CurrentRecord was not auto-advanced, do it ourselves now - m->CurrentRecord = rr->next; - } - } - -mDNSlocal void TimeoutQuestions(mDNS *const m) - { - m->NextScheduledStopTime = m->timenow + 0x3FFFFFFF; - if (m->CurrentQuestion) - LogMsg("TimeoutQuestions ERROR m->CurrentQuestion already set: %##s (%s)", m->CurrentQuestion->qname.c, - DNSTypeName(m->CurrentQuestion->qtype)); - m->CurrentQuestion = m->Questions; - while (m->CurrentQuestion) - { - DNSQuestion *const q = m->CurrentQuestion; - if (q->StopTime) - { - if (m->timenow - q->StopTime >= 0) - { - LogInfo("TimeoutQuestions: question %##s timed out, time %d", q->qname.c, m->timenow - q->StopTime); - GenerateNegativeResponse(m); - if (m->CurrentQuestion == q) q->StopTime = 0; - } - else - { - if (m->NextScheduledStopTime - q->StopTime > 0) - m->NextScheduledStopTime = q->StopTime; - } - } - // If m->CurrentQuestion wasn't modified out from under us, advance it now - // We can't do this at the start of the loop because GenerateNegativeResponse - // depends on having m->CurrentQuestion point to the right question - if (m->CurrentQuestion == q) - m->CurrentQuestion = q->next; - } - m->CurrentQuestion = mDNSNULL; - } - -mDNSexport mDNSs32 mDNS_Execute(mDNS *const m) - { - mDNS_Lock(m); // Must grab lock before trying to read m->timenow - - if (m->timenow - m->NextScheduledEvent >= 0) - { - int i; - AuthRecord *head, *tail; - mDNSu32 slot; - AuthGroup *ag; - - verbosedebugf("mDNS_Execute"); - - if (m->CurrentQuestion) - LogMsg("mDNS_Execute: ERROR m->CurrentQuestion already set: %##s (%s)", - m->CurrentQuestion->qname.c, DNSTypeName(m->CurrentQuestion->qtype)); - - if (m->CurrentRecord) - LogMsg("mDNS_Execute: ERROR m->CurrentRecord already set: %s", ARDisplayString(m, m->CurrentRecord)); - - // 1. If we're past the probe suppression time, we can clear it - if (m->SuppressProbes && m->timenow - m->SuppressProbes >= 0) m->SuppressProbes = 0; - - // 2. If it's been more than ten seconds since the last probe failure, we can clear the counter - if (m->NumFailedProbes && m->timenow - m->ProbeFailTime >= mDNSPlatformOneSecond * 10) m->NumFailedProbes = 0; - - // 3. Purge our cache of stale old records - if (m->rrcache_size && m->timenow - m->NextCacheCheck >= 0) - { - mDNSu32 numchecked = 0; - m->NextCacheCheck = m->timenow + 0x3FFFFFFF; - for (slot = 0; slot < CACHE_HASH_SLOTS; slot++) - { - if (m->timenow - m->rrcache_nextcheck[slot] >= 0) - { - CacheGroup **cp = &m->rrcache_hash[slot]; - m->rrcache_nextcheck[slot] = m->timenow + 0x3FFFFFFF; - while (*cp) - { - debugf("m->NextCacheCheck %4d Slot %3d %##s", numchecked, slot, *cp ? (*cp)->name : (domainname*)"\x04NULL"); - numchecked++; - CheckCacheExpiration(m, slot, *cp); - if ((*cp)->members) cp=&(*cp)->next; - else ReleaseCacheGroup(m, cp); - } - } - // Even if we didn't need to actually check this slot yet, still need to - // factor its nextcheck time into our overall NextCacheCheck value - if (m->NextCacheCheck - m->rrcache_nextcheck[slot] > 0) - m->NextCacheCheck = m->rrcache_nextcheck[slot]; - } - debugf("m->NextCacheCheck %4d checked, next in %d", numchecked, m->NextCacheCheck - m->timenow); - } - - if (m->timenow - m->NextScheduledSPS >= 0) - { - m->NextScheduledSPS = m->timenow + 0x3FFFFFFF; - CheckProxyRecords(m, m->DuplicateRecords); // Clear m->DuplicateRecords first, then m->ResourceRecords - CheckProxyRecords(m, m->ResourceRecords); - } - - SetSPSProxyListChanged(mDNSNULL); // Perform any deferred BPF reconfiguration now - - // Clear AnnounceOwner if necessary. (Do this *before* SendQueries() and SendResponses().) - if (m->AnnounceOwner && m->timenow - m->AnnounceOwner >= 0) m->AnnounceOwner = 0; - - if (m->DelaySleep && m->timenow - m->DelaySleep >= 0) - { - m->DelaySleep = 0; - if (m->SleepState == SleepState_Transferring) - { - LogSPS("Re-sleep delay passed; now checking for Sleep Proxy Servers"); - BeginSleepProcessing(m); - } - } - - // 4. See if we can answer any of our new local questions from the cache - for (i=0; m->NewQuestions && i<1000; i++) - { - if (m->NewQuestions->DelayAnswering && m->timenow - m->NewQuestions->DelayAnswering < 0) break; - AnswerNewQuestion(m); - } - if (i >= 1000) LogMsg("mDNS_Execute: AnswerNewQuestion exceeded loop limit"); - - // Make sure we deliver *all* local RMV events, and clear the corresponding rr->AnsweredLocalQ flags, *before* - // we begin generating *any* new ADD events in the m->NewLocalOnlyQuestions and m->NewLocalRecords loops below. - for (i=0; i<1000 && m->LocalRemoveEvents; i++) - { - m->LocalRemoveEvents = mDNSfalse; - m->CurrentRecord = m->ResourceRecords; - CheckRmvEventsForLocalRecords(m); - // Walk the LocalOnly records and deliver the RMV events - for (slot = 0; slot < AUTH_HASH_SLOTS; slot++) - for (ag = m->rrauth.rrauth_hash[slot]; ag; ag = ag->next) - { - m->CurrentRecord = ag->members; - if (m->CurrentRecord) CheckRmvEventsForLocalRecords(m); - } - } - - if (i >= 1000) LogMsg("mDNS_Execute: m->LocalRemoveEvents exceeded loop limit"); - - for (i=0; m->NewLocalOnlyQuestions && i<1000; i++) AnswerNewLocalOnlyQuestion(m); - if (i >= 1000) LogMsg("mDNS_Execute: AnswerNewLocalOnlyQuestion exceeded loop limit"); - - head = tail = mDNSNULL; - for (i=0; i<1000 && m->NewLocalRecords && m->NewLocalRecords != head; i++) - { - AuthRecord *rr = m->NewLocalRecords; - m->NewLocalRecords = m->NewLocalRecords->next; - if (LocalRecordReady(rr)) - { - debugf("mDNS_Execute: Delivering Add event with LocalAuthRecord %s", ARDisplayString(m, rr)); - AnswerAllLocalQuestionsWithLocalAuthRecord(m, rr, mDNStrue); - } - else if (!rr->next) - { - // If we have just one record that is not ready, we don't have to unlink and - // reinsert. As the NewLocalRecords will be NULL for this case, the loop will - // terminate and set the NewLocalRecords to rr. - debugf("mDNS_Execute: Just one LocalAuthRecord %s, breaking out of the loop early", ARDisplayString(m, rr)); - if (head != mDNSNULL || m->NewLocalRecords != mDNSNULL) - LogMsg("mDNS_Execute: ERROR!!: head %p, NewLocalRecords %p", head, m->NewLocalRecords); - - head = rr; - } - else - { - AuthRecord **p = &m->ResourceRecords; // Find this record in our list of active records - debugf("mDNS_Execute: Skipping LocalAuthRecord %s", ARDisplayString(m, rr)); - // if this is the first record we are skipping, move to the end of the list. - // if we have already skipped records before, append it at the end. - while (*p && *p != rr) p=&(*p)->next; - if (*p) *p = rr->next; // Cut this record from the list - else { LogMsg("mDNS_Execute: ERROR!! Cannot find record %s in ResourceRecords list", ARDisplayString(m, rr)); break; } - if (!head) - { - while (*p) p=&(*p)->next; - *p = rr; - head = tail = rr; - } - else - { - tail->next = rr; - tail = rr; - } - rr->next = mDNSNULL; - } - } - m->NewLocalRecords = head; - debugf("mDNS_Execute: Setting NewLocalRecords to %s", (head ? ARDisplayString(m, head) : "NULL")); - - if (i >= 1000) LogMsg("mDNS_Execute: m->NewLocalRecords exceeded loop limit"); - - // Check to see if we have any new LocalOnly/P2P records to examine for delivering - // to our local questions - if (m->NewLocalOnlyRecords) - { - m->NewLocalOnlyRecords = mDNSfalse; - for (slot = 0; slot < AUTH_HASH_SLOTS; slot++) - for (ag = m->rrauth.rrauth_hash[slot]; ag; ag = ag->next) - { - for (i=0; i<100 && ag->NewLocalOnlyRecords; i++) - { - AuthRecord *rr = ag->NewLocalOnlyRecords; - ag->NewLocalOnlyRecords = ag->NewLocalOnlyRecords->next; - // LocalOnly records should always be ready as they never probe - if (LocalRecordReady(rr)) - { - debugf("mDNS_Execute: Delivering Add event with LocalAuthRecord %s", ARDisplayString(m, rr)); - AnswerAllLocalQuestionsWithLocalAuthRecord(m, rr, mDNStrue); - } - else LogMsg("mDNS_Execute: LocalOnlyRecord %s not ready", ARDisplayString(m, rr)); - } - // We limit about 100 per AuthGroup that can be serviced at a time - if (i >= 100) LogMsg("mDNS_Execute: ag->NewLocalOnlyRecords exceeded loop limit"); - } - } - - // 5. Some questions may have picked a new DNS server and the cache may answer these questions now. - AnswerQuestionsForDNSServerChanges(m); - - // 6. See what packets we need to send - if (m->mDNSPlatformStatus != mStatus_NoError || (m->SleepState == SleepState_Sleeping)) - DiscardDeregistrations(m); - if (m->mDNSPlatformStatus == mStatus_NoError && (m->SuppressSending == 0 || m->timenow - m->SuppressSending >= 0)) - { - // If the platform code is ready, and we're not suppressing packet generation right now - // then send our responses, probes, and questions. - // We check the cache first, because there might be records close to expiring that trigger questions to refresh them. - // We send queries next, because there might be final-stage probes that complete their probing here, causing - // them to advance to announcing state, and we want those to be included in any announcements we send out. - // Finally, we send responses, including the previously mentioned records that just completed probing. - m->SuppressSending = 0; - - // 7. Send Query packets. This may cause some probing records to advance to announcing state - if (m->timenow - m->NextScheduledQuery >= 0 || m->timenow - m->NextScheduledProbe >= 0) SendQueries(m); - if (m->timenow - m->NextScheduledQuery >= 0) - { - DNSQuestion *q; - LogMsg("mDNS_Execute: SendQueries didn't send all its queries (%d - %d = %d) will try again in one second", - m->timenow, m->NextScheduledQuery, m->timenow - m->NextScheduledQuery); - m->NextScheduledQuery = m->timenow + mDNSPlatformOneSecond; - for (q = m->Questions; q && q != m->NewQuestions; q=q->next) - if (ActiveQuestion(q) && m->timenow - NextQSendTime(q) >= 0) - LogMsg("mDNS_Execute: SendQueries didn't send %##s (%s)", q->qname.c, DNSTypeName(q->qtype)); - } - if (m->timenow - m->NextScheduledProbe >= 0) - { - LogMsg("mDNS_Execute: SendQueries didn't send all its probes (%d - %d = %d) will try again in one second", - m->timenow, m->NextScheduledProbe, m->timenow - m->NextScheduledProbe); - m->NextScheduledProbe = m->timenow + mDNSPlatformOneSecond; - } - - // 8. Send Response packets, including probing records just advanced to announcing state - if (m->timenow - m->NextScheduledResponse >= 0) SendResponses(m); - if (m->timenow - m->NextScheduledResponse >= 0) - { - LogMsg("mDNS_Execute: SendResponses didn't send all its responses; will try again in one second"); - m->NextScheduledResponse = m->timenow + mDNSPlatformOneSecond; - } - } - - // Clear RandomDelay values, ready to pick a new different value next time - m->RandomQueryDelay = 0; - m->RandomReconfirmDelay = 0; - - if (m->NextScheduledStopTime && m->timenow - m->NextScheduledStopTime >= 0) TimeoutQuestions(m); -#ifndef UNICAST_DISABLED - if (m->NextSRVUpdate && m->timenow - m->NextSRVUpdate >= 0) UpdateAllSRVRecords(m); - if (m->timenow - m->NextScheduledNATOp >= 0) CheckNATMappings(m); - if (m->timenow - m->NextuDNSEvent >= 0) uDNS_Tasks(m); -#endif - } - - // Note about multi-threaded systems: - // On a multi-threaded system, some other thread could run right after the mDNS_Unlock(), - // performing mDNS API operations that change our next scheduled event time. - // - // On multi-threaded systems (like the current Windows implementation) that have a single main thread - // calling mDNS_Execute() (and other threads allowed to call mDNS API routines) it is the responsibility - // of the mDNSPlatformUnlock() routine to signal some kind of stateful condition variable that will - // signal whatever blocking primitive the main thread is using, so that it will wake up and execute one - // more iteration of its loop, and immediately call mDNS_Execute() again. The signal has to be stateful - // in the sense that if the main thread has not yet entered its blocking primitive, then as soon as it - // does, the state of the signal will be noticed, causing the blocking primitive to return immediately - // without blocking. This avoids the race condition between the signal from the other thread arriving - // just *before* or just *after* the main thread enters the blocking primitive. - // - // On multi-threaded systems (like the current Mac OS 9 implementation) that are entirely timer-driven, - // with no main mDNS_Execute() thread, it is the responsibility of the mDNSPlatformUnlock() routine to - // set the timer according to the m->NextScheduledEvent value, and then when the timer fires, the timer - // callback function should call mDNS_Execute() (and ignore the return value, which may already be stale - // by the time it gets to the timer callback function). - - mDNS_Unlock(m); // Calling mDNS_Unlock is what gives m->NextScheduledEvent its new value - return(m->NextScheduledEvent); - } - -mDNSlocal void SuspendLLQs(mDNS *m) - { - DNSQuestion *q; - for (q = m->Questions; q; q = q->next) - if (ActiveQuestion(q) && !mDNSOpaque16IsZero(q->TargetQID) && q->LongLived && q->state == LLQ_Established) - { q->ReqLease = 0; sendLLQRefresh(m, q); } - } - -mDNSlocal mDNSBool QuestionHasLocalAnswers(mDNS *const m, DNSQuestion *q) - { - AuthRecord *rr; - mDNSu32 slot; - AuthGroup *ag; - - slot = AuthHashSlot(&q->qname); - ag = AuthGroupForName(&m->rrauth, slot, q->qnamehash, &q->qname); - if (ag) - { - for (rr = ag->members; rr; rr=rr->next) - // Filter the /etc/hosts records - LocalOnly, Unique, A/AAAA/CNAME - if (LORecordAnswersAddressType(rr) && LocalOnlyRecordAnswersQuestion(rr, q)) - { - LogInfo("QuestionHasLocalAnswers: Question %p %##s (%s) has local answer %s", q, q->qname.c, DNSTypeName(q->qtype), ARDisplayString(m, rr)); - return mDNStrue; - } - } - return mDNSfalse; - } - -// ActivateUnicastQuery() is called from three places: -// 1. When a new question is created -// 2. On wake from sleep -// 3. When the DNS configuration changes -// In case 1 we don't want to mess with our established ThisQInterval and LastQTime (ScheduleImmediately is false) -// In cases 2 and 3 we do want to cause the question to be resent immediately (ScheduleImmediately is true) -mDNSlocal void ActivateUnicastQuery(mDNS *const m, DNSQuestion *const question, mDNSBool ScheduleImmediately) - { - // For now this AutoTunnel stuff is specific to Mac OS X. - // In the future, if there's demand, we may see if we can abstract it out cleanly into the platform layer -#if APPLE_OSX_mDNSResponder - // Even though BTMM client tunnels are only useful for AAAA queries, we need to treat v4 and v6 queries equally. - // Otherwise we can get the situation where the A query completes really fast (with an NXDOMAIN result) and the - // caller then gives up waiting for the AAAA result while we're still in the process of setting up the tunnel. - // To level the playing field, we block both A and AAAA queries while tunnel setup is in progress, and then - // returns results for both at the same time. If we are looking for the _autotunnel6 record, then skip this logic - // as this would trigger looking up _autotunnel6._autotunnel6 and end up failing the original query. - - if (RRTypeIsAddressType(question->qtype) && PrivateQuery(question) && - !SameDomainLabel(question->qname.c, (const mDNSu8 *)"\x0c_autotunnel6")&& question->QuestionCallback != AutoTunnelCallback) - { - question->NoAnswer = NoAnswer_Suspended; - AddNewClientTunnel(m, question); - return; - } -#endif // APPLE_OSX_mDNSResponder - - if (!question->DuplicateOf) - { - debugf("ActivateUnicastQuery: %##s %s%s%s", - question->qname.c, DNSTypeName(question->qtype), PrivateQuery(question) ? " (Private)" : "", ScheduleImmediately ? " ScheduleImmediately" : ""); - question->CNAMEReferrals = 0; - if (question->nta) { CancelGetZoneData(m, question->nta); question->nta = mDNSNULL; } - if (question->LongLived) - { - question->state = LLQ_InitialRequest; - question->id = zeroOpaque64; - question->servPort = zeroIPPort; - if (question->tcp) { DisposeTCPConn(question->tcp); question->tcp = mDNSNULL; } - } - // If the question has local answers, then we don't want answers from outside - if (ScheduleImmediately && !QuestionHasLocalAnswers(m, question)) - { - question->ThisQInterval = InitialQuestionInterval; - question->LastQTime = m->timenow - question->ThisQInterval; - SetNextQueryTime(m, question); - } - } - } - -// Caller should hold the lock -mDNSexport void mDNSCoreRestartAddressQueries(mDNS *const m, mDNSBool SearchDomainsChanged, FlushCache flushCacheRecords, - CallbackBeforeStartQuery BeforeStartCallback, void *context) - { - DNSQuestion *q; - DNSQuestion *restart = mDNSNULL; - - if (!m->mDNS_busy) LogMsg("mDNSCoreRestartAddressQueries: ERROR!! Lock not held"); - - // 1. Flush the cache records - if (flushCacheRecords) flushCacheRecords(m); - - // 2. Even though we may have purged the cache records above, before it can generate RMV event - // we are going to stop the question. Hence we need to deliver the RMV event before we - // stop the question. - // - // CurrentQuestion is used by RmvEventsForQuestion below. While delivering RMV events, the - // application callback can potentially stop the current question (detected by CurrentQuestion) or - // *any* other question which could be the next one that we may process here. RestartQuestion - // points to the "next" question which will be automatically advanced in mDNS_StopQuery_internal - // if the "next" question is stopped while the CurrentQuestion is stopped - - if (m->RestartQuestion) - LogMsg("mDNSCoreRestartAddressQueries: ERROR!! m->RestartQuestion already set: %##s (%s)", - m->RestartQuestion->qname.c, DNSTypeName(m->RestartQuestion->qtype)); - - m->RestartQuestion = m->Questions; - while (m->RestartQuestion) - { - q = m->RestartQuestion; - m->RestartQuestion = q->next; - // GetZoneData questions are referenced by other questions (original query that started the GetZoneData - // question) through their "nta" pointer. Normally when the original query stops, it stops the - // GetZoneData question and also frees the memory (See CancelGetZoneData). If we stop the GetZoneData - // question followed by the original query that refers to this GetZoneData question, we will end up - // freeing the GetZoneData question and then start the "freed" question at the end. - - if (IsGetZoneDataQuestion(q)) - { - DNSQuestion *refq = q->next; - LogInfo("mDNSCoreRestartAddressQueries: Skipping GetZoneDataQuestion %p %##s (%s)", q, q->qname.c, DNSTypeName(q->qtype)); - // debug stuff, we just try to find the referencing question and don't do much with it - while (refq) - { - if (q == &refq->nta->question) - { - LogInfo("mDNSCoreRestartAddressQueries: Question %p %##s (%s) referring to GetZoneDataQuestion %p, not stopping", refq, refq->qname.c, DNSTypeName(refq->qtype), q); - } - refq = refq->next; - } - continue; - } - - // This function is called when /etc/hosts changes and that could affect A, AAAA and CNAME queries - if (q->qtype != kDNSType_A && q->qtype != kDNSType_AAAA && q->qtype != kDNSType_CNAME) continue; - - // If the search domains did not change, then we restart all the queries. Otherwise, only - // for queries for which we "might" have appended search domains ("might" because we may - // find results before we apply search domains even though AppendSearchDomains is set to 1) - if (!SearchDomainsChanged || q->AppendSearchDomains) - { - // NOTE: CacheRecordRmvEventsForQuestion will not generate RMV events for queries that have non-zero - // LOAddressAnswers. Hence it is important that we call CacheRecordRmvEventsForQuestion before - // LocalRecordRmvEventsForQuestion (which decrements LOAddressAnswers). Let us say that - // /etc/hosts has an A Record for web.apple.com. Any queries for web.apple.com will be answered locally. - // But this can't prevent a CNAME/AAAA query to not to be sent on the wire. When it is sent on the wire, - // it could create cache entries. When we are restarting queries, we can't deliver the cache RMV events - // for the original query using these cache entries as ADDs were never delivered using these cache - // entries and hence this order is needed. - - // If the query is suppressed, the RMV events won't be delivered - if (!CacheRecordRmvEventsForQuestion(m, q)) { LogInfo("mDNSCoreRestartAddressQueries: Question deleted while delivering Cache Record RMV events"); continue; } - - // SuppressQuery status does not affect questions that are answered using local records - if (!LocalRecordRmvEventsForQuestion(m, q)) { LogInfo("mDNSCoreRestartAddressQueries: Question deleted while delivering Local Record RMV events"); continue; } - - LogInfo("mDNSCoreRestartAddressQueries: Stop question %p %##s (%s), AppendSearchDomains %d, qnameOrig %p", q, - q->qname.c, DNSTypeName(q->qtype), q->AppendSearchDomains, q->qnameOrig); - mDNS_StopQuery_internal(m, q); - // Reset state so that it looks like it was in the beginning i.e it should look at /etc/hosts, cache - // and then search domains should be appended. At the beginning, qnameOrig was NULL. - if (q->qnameOrig) - { - LogInfo("mDNSCoreRestartAddressQueries: qnameOrig %##s", q->qnameOrig); - AssignDomainName(&q->qname, q->qnameOrig); - mDNSPlatformMemFree(q->qnameOrig); - q->qnameOrig = mDNSNULL; - q->RetryWithSearchDomains = ApplySearchDomainsFirst(q) ? 1 : 0; - } - q->SearchListIndex = 0; - q->next = restart; - restart = q; - } - } - - // 3. Callback before we start the query - if (BeforeStartCallback) BeforeStartCallback(m, context); - - // 4. Restart all the stopped queries - while (restart) - { - q = restart; - restart = restart->next; - q->next = mDNSNULL; - LogInfo("mDNSCoreRestartAddressQueries: Start question %p %##s (%s)", q, q->qname.c, DNSTypeName(q->qtype)); - mDNS_StartQuery_internal(m, q); - } - } - -mDNSexport void mDNSCoreRestartQueries(mDNS *const m) - { - DNSQuestion *q; - -#ifndef UNICAST_DISABLED - // Retrigger all our uDNS questions - if (m->CurrentQuestion) - LogMsg("mDNSCoreRestartQueries: ERROR m->CurrentQuestion already set: %##s (%s)", - m->CurrentQuestion->qname.c, DNSTypeName(m->CurrentQuestion->qtype)); - m->CurrentQuestion = m->Questions; - while (m->CurrentQuestion) - { - q = m->CurrentQuestion; - m->CurrentQuestion = m->CurrentQuestion->next; - if (!mDNSOpaque16IsZero(q->TargetQID) && ActiveQuestion(q)) ActivateUnicastQuery(m, q, mDNStrue); - } -#endif - - // Retrigger all our mDNS questions - for (q = m->Questions; q; q=q->next) // Scan our list of questions - if (mDNSOpaque16IsZero(q->TargetQID) && ActiveQuestion(q)) - { - q->ThisQInterval = InitialQuestionInterval; // MUST be > zero for an active question - q->RequestUnicast = 2; // Set to 2 because is decremented once *before* we check it - q->LastQTime = m->timenow - q->ThisQInterval; - q->RecentAnswerPkts = 0; - ExpireDupSuppressInfo(q->DupSuppress, m->timenow); - m->NextScheduledQuery = m->timenow; - } - } - -// *************************************************************************** -#if COMPILER_LIKES_PRAGMA_MARK -#pragma mark - -#pragma mark - Power Management (Sleep/Wake) -#endif - -mDNSexport void mDNS_UpdateAllowSleep(mDNS *const m) - { -#ifndef IDLESLEEPCONTROL_DISABLED - mDNSBool allowSleep = mDNStrue; - char reason[128]; - - reason[0] = 0; - - if (m->SystemSleepOnlyIfWakeOnLAN) - { - // Don't sleep if we are a proxy for any services - if (m->ProxyRecords) - { - allowSleep = mDNSfalse; - mDNS_snprintf(reason, sizeof(reason), "sleep proxy for %d records", m->ProxyRecords); - LogInfo("Sleep disabled because we are proxying %d records", m->ProxyRecords); - } - - if (allowSleep && mDNSCoreHaveAdvertisedMulticastServices(m)) - { - // Scan the list of active interfaces - NetworkInterfaceInfo *intf; - for (intf = GetFirstActiveInterface(m->HostInterfaces); intf; intf = GetFirstActiveInterface(intf->next)) - { - if (intf->McastTxRx && !intf->Loopback) - { - // Disallow sleep if this interface doesn't support NetWake - if (!intf->NetWake) - { - allowSleep = mDNSfalse; - mDNS_snprintf(reason, sizeof(reason), "%s does not support NetWake", intf->ifname); - LogInfo("Sleep disabled because %s does not support NetWake", intf->ifname); - break; - } - - // Disallow sleep if there is no sleep proxy server - if (FindSPSInCache1(m, &intf->NetWakeBrowse, mDNSNULL, mDNSNULL) == mDNSNULL) - { - allowSleep = mDNSfalse; - mDNS_snprintf(reason, sizeof(reason), "%s does not support NetWake", intf->ifname); - LogInfo("Sleep disabled because %s has no sleep proxy", intf->ifname); - break; - } - } - } - } - } - - // Call the platform code to enable/disable sleep - mDNSPlatformSetAllowSleep(m, allowSleep, reason); -#endif /* !defined(IDLESLEEPCONTROL_DISABLED) */ - } - -mDNSlocal void SendSPSRegistrationForOwner(mDNS *const m, NetworkInterfaceInfo *const intf, const mDNSOpaque16 id, const OwnerOptData *const owner) - { - const int optspace = DNSOpt_Header_Space + DNSOpt_LeaseData_Space + DNSOpt_Owner_Space(&m->PrimaryMAC, &intf->MAC); - const int sps = intf->NextSPSAttempt / 3; - AuthRecord *rr; - - if (!intf->SPSAddr[sps].type) - { - intf->NextSPSAttemptTime = m->timenow + mDNSPlatformOneSecond; - if (m->NextScheduledSPRetry - intf->NextSPSAttemptTime > 0) - m->NextScheduledSPRetry = intf->NextSPSAttemptTime; - LogSPS("SendSPSRegistration: %s SPS %d (%d) %##s not yet resolved", intf->ifname, intf->NextSPSAttempt, sps, intf->NetWakeResolve[sps].qname.c); - goto exit; - } - - // Mark our mDNS records (not unicast records) for transfer to SPS - if (mDNSOpaque16IsZero(id)) - for (rr = m->ResourceRecords; rr; rr=rr->next) - if (rr->resrec.RecordType > kDNSRecordTypeDeregistering) - if (rr->resrec.InterfaceID == intf->InterfaceID || (!rr->resrec.InterfaceID && (rr->ForceMCast || IsLocalDomain(rr->resrec.name)))) - if (mDNSPlatformMemSame(owner, &rr->WakeUp, sizeof(*owner))) - rr->SendRNow = mDNSInterfaceMark; // mark it now - - while (1) - { - mDNSu8 *p = m->omsg.data; - // To comply with RFC 2782, PutResourceRecord suppresses name compression for SRV records in unicast updates. - // For now we follow that same logic for SPS registrations too. - // If we decide to compress SRV records in SPS registrations in the future, we can achieve that by creating our - // initial DNSMessage with h.flags set to zero, and then update it to UpdateReqFlags right before sending the packet. - InitializeDNSMessage(&m->omsg.h, mDNSOpaque16IsZero(id) ? mDNS_NewMessageID(m) : id, UpdateReqFlags); - - for (rr = m->ResourceRecords; rr; rr=rr->next) - if (rr->SendRNow || (!mDNSOpaque16IsZero(id) && !AuthRecord_uDNS(rr) && mDNSSameOpaque16(rr->updateid, id) && m->timenow - (rr->LastAPTime + rr->ThisAPInterval) >= 0)) - if (mDNSPlatformMemSame(owner, &rr->WakeUp, sizeof(*owner))) - { - mDNSu8 *newptr; - const mDNSu8 *const limit = m->omsg.data + (m->omsg.h.mDNS_numUpdates ? NormalMaxDNSMessageData : AbsoluteMaxDNSMessageData) - optspace; - if (rr->resrec.RecordType & kDNSRecordTypeUniqueMask) - rr->resrec.rrclass |= kDNSClass_UniqueRRSet; // Temporarily set the 'unique' bit so PutResourceRecord will set it - newptr = PutResourceRecordTTLWithLimit(&m->omsg, p, &m->omsg.h.mDNS_numUpdates, &rr->resrec, rr->resrec.rroriginalttl, limit); - rr->resrec.rrclass &= ~kDNSClass_UniqueRRSet; // Make sure to clear 'unique' bit back to normal state - if (!newptr) - LogSPS("SendSPSRegistration put %s FAILED %d/%d %s", intf->ifname, p - m->omsg.data, limit - m->omsg.data, ARDisplayString(m, rr)); - else - { - LogSPS("SendSPSRegistration put %s %s", intf->ifname, ARDisplayString(m, rr)); - rr->SendRNow = mDNSNULL; - rr->ThisAPInterval = mDNSPlatformOneSecond; - rr->LastAPTime = m->timenow; - rr->updateid = m->omsg.h.id; - if (m->NextScheduledResponse - (rr->LastAPTime + rr->ThisAPInterval) >= 0) - m->NextScheduledResponse = (rr->LastAPTime + rr->ThisAPInterval); - p = newptr; - } - } - - if (!m->omsg.h.mDNS_numUpdates) break; - else - { - AuthRecord opt; - mDNS_SetupResourceRecord(&opt, mDNSNULL, mDNSInterface_Any, kDNSType_OPT, kStandardTTL, kDNSRecordTypeKnownUnique, AuthRecordAny, mDNSNULL, mDNSNULL); - opt.resrec.rrclass = NormalMaxDNSMessageData; - opt.resrec.rdlength = sizeof(rdataOPT) * 2; // Two options in this OPT record - opt.resrec.rdestimate = sizeof(rdataOPT) * 2; - opt.resrec.rdata->u.opt[0].opt = kDNSOpt_Lease; - opt.resrec.rdata->u.opt[0].optlen = DNSOpt_LeaseData_Space - 4; - opt.resrec.rdata->u.opt[0].u.updatelease = DEFAULT_UPDATE_LEASE; - if (!owner->HMAC.l[0]) // If no owner data, - SetupOwnerOpt(m, intf, &opt.resrec.rdata->u.opt[1]); // use our own interface information - else // otherwise, use the owner data we were given - { - opt.resrec.rdata->u.opt[1].u.owner = *owner; - opt.resrec.rdata->u.opt[1].opt = kDNSOpt_Owner; - opt.resrec.rdata->u.opt[1].optlen = DNSOpt_Owner_Space(&owner->HMAC, &owner->IMAC) - 4; - } - LogSPS("SendSPSRegistration put %s %s", intf->ifname, ARDisplayString(m, &opt)); - p = PutResourceRecordTTLWithLimit(&m->omsg, p, &m->omsg.h.numAdditionals, &opt.resrec, opt.resrec.rroriginalttl, m->omsg.data + AbsoluteMaxDNSMessageData); - if (!p) - LogMsg("SendSPSRegistration: Failed to put OPT record (%d updates) %s", m->omsg.h.mDNS_numUpdates, ARDisplayString(m, &opt)); - else - { - mStatus err; - - LogSPS("SendSPSRegistration: Sending Update %s %d (%d) id %5d with %d records %d bytes to %#a:%d", intf->ifname, intf->NextSPSAttempt, sps, - mDNSVal16(m->omsg.h.id), m->omsg.h.mDNS_numUpdates, p - m->omsg.data, &intf->SPSAddr[sps], mDNSVal16(intf->SPSPort[sps])); - // if (intf->NextSPSAttempt < 5) m->omsg.h.flags = zeroID; // For simulating packet loss - err = mDNSSendDNSMessage(m, &m->omsg, p, intf->InterfaceID, mDNSNULL, &intf->SPSAddr[sps], intf->SPSPort[sps], mDNSNULL, mDNSNULL); - if (err) LogSPS("SendSPSRegistration: mDNSSendDNSMessage err %d", err); - if (err && intf->SPSAddr[sps].type == mDNSAddrType_IPv6 && intf->NetWakeResolve[sps].ThisQInterval == -1) - { - LogSPS("SendSPSRegistration %d %##s failed to send to IPv6 address; will try IPv4 instead", sps, intf->NetWakeResolve[sps].qname.c); - intf->NetWakeResolve[sps].qtype = kDNSType_A; - mDNS_StartQuery_internal(m, &intf->NetWakeResolve[sps]); - return; - } - } - } - } - - intf->NextSPSAttemptTime = m->timenow + mDNSPlatformOneSecond * 10; // If successful, update NextSPSAttemptTime - -exit: - if (mDNSOpaque16IsZero(id) && intf->NextSPSAttempt < 8) intf->NextSPSAttempt++; - } - -mDNSlocal mDNSBool RecordIsFirstOccurrenceOfOwner(mDNS *const m, const AuthRecord *const rr) - { - AuthRecord *ar; - for (ar = m->ResourceRecords; ar && ar != rr; ar=ar->next) - if (mDNSPlatformMemSame(&rr->WakeUp, &ar->WakeUp, sizeof(rr->WakeUp))) return mDNSfalse; - return mDNStrue; - } - -mDNSlocal void SendSPSRegistration(mDNS *const m, NetworkInterfaceInfo *const intf, const mDNSOpaque16 id) - { - AuthRecord *ar; - OwnerOptData owner = zeroOwner; - - SendSPSRegistrationForOwner(m, intf, id, &owner); - - for (ar = m->ResourceRecords; ar; ar=ar->next) - { - if (!mDNSPlatformMemSame(&owner, &ar->WakeUp, sizeof(owner)) && RecordIsFirstOccurrenceOfOwner(m, ar)) - { - owner = ar->WakeUp; - SendSPSRegistrationForOwner(m, intf, id, &owner); - } - } - } - -// RetrySPSRegistrations is called from SendResponses, with the lock held -mDNSlocal void RetrySPSRegistrations(mDNS *const m) - { - AuthRecord *rr; - NetworkInterfaceInfo *intf; - - // First make sure none of our interfaces' NextSPSAttemptTimes are inadvertently set to m->timenow + mDNSPlatformOneSecond * 10 - for (intf = GetFirstActiveInterface(m->HostInterfaces); intf; intf = GetFirstActiveInterface(intf->next)) - if (intf->NextSPSAttempt && intf->NextSPSAttemptTime == m->timenow + mDNSPlatformOneSecond * 10) - intf->NextSPSAttemptTime++; - - // Retry any record registrations that are due - for (rr = m->ResourceRecords; rr; rr=rr->next) - if (!AuthRecord_uDNS(rr) && !mDNSOpaque16IsZero(rr->updateid) && m->timenow - (rr->LastAPTime + rr->ThisAPInterval) >= 0) - for (intf = GetFirstActiveInterface(m->HostInterfaces); intf; intf = GetFirstActiveInterface(intf->next)) - if (!rr->resrec.InterfaceID || rr->resrec.InterfaceID == intf->InterfaceID) - { - LogSPS("RetrySPSRegistrations: %s", ARDisplayString(m, rr)); - SendSPSRegistration(m, intf, rr->updateid); - } - - // For interfaces where we did an SPS registration attempt, increment intf->NextSPSAttempt - for (intf = GetFirstActiveInterface(m->HostInterfaces); intf; intf = GetFirstActiveInterface(intf->next)) - if (intf->NextSPSAttempt && intf->NextSPSAttemptTime == m->timenow + mDNSPlatformOneSecond * 10 && intf->NextSPSAttempt < 8) - intf->NextSPSAttempt++; - } - -mDNSlocal void NetWakeResolve(mDNS *const m, DNSQuestion *question, const ResourceRecord *const answer, QC_result AddRecord) - { - NetworkInterfaceInfo *intf = (NetworkInterfaceInfo *)question->QuestionContext; - int sps = (int)(question - intf->NetWakeResolve); - (void)m; // Unused - LogSPS("NetWakeResolve: SPS: %d Add: %d %s", sps, AddRecord, RRDisplayString(m, answer)); - - if (!AddRecord) return; // Don't care about REMOVE events - if (answer->rrtype != question->qtype) return; // Don't care about CNAMEs - - // if (answer->rrtype == kDNSType_AAAA && sps == 0) return; // To test failing to resolve sleep proxy's address - - if (answer->rrtype == kDNSType_SRV) - { - // 1. Got the SRV record; now look up the target host's IPv6 link-local address - mDNS_StopQuery(m, question); - intf->SPSPort[sps] = answer->rdata->u.srv.port; - AssignDomainName(&question->qname, &answer->rdata->u.srv.target); - question->qtype = kDNSType_AAAA; - mDNS_StartQuery(m, question); - } - else if (answer->rrtype == kDNSType_AAAA && answer->rdlength == sizeof(mDNSv6Addr) && mDNSv6AddressIsLinkLocal(&answer->rdata->u.ipv6)) - { - // 2. Got the target host's IPv6 link-local address; record address and initiate an SPS registration if appropriate - mDNS_StopQuery(m, question); - question->ThisQInterval = -1; - intf->SPSAddr[sps].type = mDNSAddrType_IPv6; - intf->SPSAddr[sps].ip.v6 = answer->rdata->u.ipv6; - mDNS_Lock(m); - if (sps == intf->NextSPSAttempt/3) SendSPSRegistration(m, intf, zeroID); // If we're ready for this result, use it now - mDNS_Unlock(m); - } - else if (answer->rrtype == kDNSType_AAAA && answer->rdlength == 0) - { - // 3. Got negative response -- target host apparently has IPv6 disabled -- so try looking up the target host's IPv4 address(es) instead - mDNS_StopQuery(m, question); - LogSPS("NetWakeResolve: SPS %d %##s has no IPv6 address, will try IPv4 instead", sps, question->qname.c); - question->qtype = kDNSType_A; - mDNS_StartQuery(m, question); - } - else if (answer->rrtype == kDNSType_A && answer->rdlength == sizeof(mDNSv4Addr)) - { - // 4. Got an IPv4 address for the target host; record address and initiate an SPS registration if appropriate - mDNS_StopQuery(m, question); - question->ThisQInterval = -1; - intf->SPSAddr[sps].type = mDNSAddrType_IPv4; - intf->SPSAddr[sps].ip.v4 = answer->rdata->u.ipv4; - mDNS_Lock(m); - if (sps == intf->NextSPSAttempt/3) SendSPSRegistration(m, intf, zeroID); // If we're ready for this result, use it now - mDNS_Unlock(m); - } - } - -mDNSexport mDNSBool mDNSCoreHaveAdvertisedMulticastServices(mDNS *const m) - { - AuthRecord *rr; - for (rr = m->ResourceRecords; rr; rr=rr->next) - if (rr->resrec.rrtype == kDNSType_SRV && !AuthRecord_uDNS(rr) && !mDNSSameIPPort(rr->resrec.rdata->u.srv.port, DiscardPort)) - return mDNStrue; - return mDNSfalse; - } - -mDNSlocal void SendSleepGoodbyes(mDNS *const m) - { - AuthRecord *rr; - m->SleepState = SleepState_Sleeping; - -#ifndef UNICAST_DISABLED - SleepRecordRegistrations(m); // If we have no SPS, need to deregister our uDNS records -#endif /* UNICAST_DISABLED */ - - // Mark all the records we need to deregister and send them - for (rr = m->ResourceRecords; rr; rr=rr->next) - if (rr->resrec.RecordType == kDNSRecordTypeShared && rr->RequireGoodbye) - rr->ImmedAnswer = mDNSInterfaceMark; - SendResponses(m); - } - -// BeginSleepProcessing is called, with the lock held, from either mDNS_Execute or mDNSCoreMachineSleep -mDNSlocal void BeginSleepProcessing(mDNS *const m) - { - mDNSBool SendGoodbyes = mDNStrue; - const CacheRecord *sps[3] = { mDNSNULL }; - - m->NextScheduledSPRetry = m->timenow; - - if (!m->SystemWakeOnLANEnabled) LogSPS("BeginSleepProcessing: m->SystemWakeOnLANEnabled is false"); - else if (!mDNSCoreHaveAdvertisedMulticastServices(m)) LogSPS("BeginSleepProcessing: No advertised services"); - else // If we have at least one advertised service - { - NetworkInterfaceInfo *intf; - for (intf = GetFirstActiveInterface(m->HostInterfaces); intf; intf = GetFirstActiveInterface(intf->next)) - { - if (!intf->NetWake) LogSPS("BeginSleepProcessing: %-6s not capable of magic packet wakeup", intf->ifname); -#if APPLE_OSX_mDNSResponder - else if (ActivateLocalProxy(m, intf->ifname) == mStatus_NoError) - { - SendGoodbyes = mDNSfalse; - LogSPS("BeginSleepProcessing: %-6s using local proxy", intf->ifname); - // This will leave m->SleepState set to SleepState_Transferring, - // which is okay because with no outstanding resolves, or updates in flight, - // mDNSCoreReadyForSleep() will conclude correctly that all the updates have already completed - } -#endif // APPLE_OSX_mDNSResponder - else - { - FindSPSInCache(m, &intf->NetWakeBrowse, sps); - if (!sps[0]) LogSPS("BeginSleepProcessing: %-6s %#a No Sleep Proxy Server found (Next Browse Q in %d, interval %d)", - intf->ifname, &intf->ip, NextQSendTime(&intf->NetWakeBrowse) - m->timenow, intf->NetWakeBrowse.ThisQInterval); - else - { - int i; - SendGoodbyes = mDNSfalse; - intf->NextSPSAttempt = 0; - intf->NextSPSAttemptTime = m->timenow + mDNSPlatformOneSecond; - // Don't need to set m->NextScheduledSPRetry here because we already set "m->NextScheduledSPRetry = m->timenow" above - for (i=0; i<3; i++) - { -#if ForceAlerts - if (intf->SPSAddr[i].type) - { LogMsg("BeginSleepProcessing: %s %d intf->SPSAddr[i].type %d", intf->ifname, i, intf->SPSAddr[i].type); *(long*)0 = 0; } - if (intf->NetWakeResolve[i].ThisQInterval >= 0) - { LogMsg("BeginSleepProcessing: %s %d intf->NetWakeResolve[i].ThisQInterval %d", intf->ifname, i, intf->NetWakeResolve[i].ThisQInterval); *(long*)0 = 0; } -#endif - intf->SPSAddr[i].type = mDNSAddrType_None; - if (intf->NetWakeResolve[i].ThisQInterval >= 0) mDNS_StopQuery(m, &intf->NetWakeResolve[i]); - intf->NetWakeResolve[i].ThisQInterval = -1; - if (sps[i]) - { - LogSPS("BeginSleepProcessing: %-6s Found Sleep Proxy Server %d TTL %d %s", intf->ifname, i, sps[i]->resrec.rroriginalttl, CRDisplayString(m, sps[i])); - mDNS_SetupQuestion(&intf->NetWakeResolve[i], intf->InterfaceID, &sps[i]->resrec.rdata->u.name, kDNSType_SRV, NetWakeResolve, intf); - intf->NetWakeResolve[i].ReturnIntermed = mDNStrue; - mDNS_StartQuery_internal(m, &intf->NetWakeResolve[i]); - } - } - } - } - } - } - - if (SendGoodbyes) // If we didn't find even one Sleep Proxy - { - LogSPS("BeginSleepProcessing: Not registering with Sleep Proxy Server"); - SendSleepGoodbyes(m); - } - } - -// Call mDNSCoreMachineSleep(m, mDNStrue) when the machine is about to go to sleep. -// Call mDNSCoreMachineSleep(m, mDNSfalse) when the machine is has just woken up. -// Normally, the platform support layer below mDNSCore should call this, not the client layer above. -mDNSexport void mDNSCoreMachineSleep(mDNS *const m, mDNSBool sleep) - { - AuthRecord *rr; - - LogSPS("%s (old state %d) at %ld", sleep ? "Sleeping" : "Waking", m->SleepState, m->timenow); - - if (sleep && !m->SleepState) // Going to sleep - { - mDNS_Lock(m); - // If we're going to sleep, need to stop advertising that we're a Sleep Proxy Server - if (m->SPSSocket) - { - mDNSu8 oldstate = m->SPSState; - mDNS_DropLockBeforeCallback(); // mDNS_DeregisterService expects to be called without the lock held, so we emulate that here - m->SPSState = 2; - if (oldstate == 1) mDNS_DeregisterService(m, &m->SPSRecords); - mDNS_ReclaimLockAfterCallback(); - } - - m->SleepState = SleepState_Transferring; - if (m->SystemWakeOnLANEnabled && m->DelaySleep) - { - // If we just woke up moments ago, allow ten seconds for networking to stabilize before going back to sleep - LogSPS("mDNSCoreMachineSleep: Re-sleeping immediately after waking; will delay for %d ticks", m->DelaySleep - m->timenow); - m->SleepLimit = NonZeroTime(m->DelaySleep + mDNSPlatformOneSecond * 10); - } - else - { - m->DelaySleep = 0; - m->SleepLimit = NonZeroTime(m->timenow + mDNSPlatformOneSecond * 10); - BeginSleepProcessing(m); - } - -#ifndef UNICAST_DISABLED - SuspendLLQs(m); -#endif - mDNS_Unlock(m); - // RemoveAutoTunnel6Record needs to be called outside the lock, as it grabs the lock also. -#if APPLE_OSX_mDNSResponder - RemoveAutoTunnel6Record(m); -#endif - LogSPS("mDNSCoreMachineSleep: m->SleepState %d (%s) seq %d", m->SleepState, - m->SleepState == SleepState_Transferring ? "Transferring" : - m->SleepState == SleepState_Sleeping ? "Sleeping" : "?", m->SleepSeqNum); - } - else if (!sleep) // Waking up - { - mDNSu32 slot; - CacheGroup *cg; - CacheRecord *cr; - NetworkInterfaceInfo *intf; - - mDNS_Lock(m); - // Reset SleepLimit back to 0 now that we're awake again. - m->SleepLimit = 0; - - // If we were previously sleeping, but now we're not, increment m->SleepSeqNum to indicate that we're entering a new period of wakefulness - if (m->SleepState != SleepState_Awake) - { - m->SleepState = SleepState_Awake; - m->SleepSeqNum++; - // If the machine wakes and then immediately tries to sleep again (e.g. a maintenance wake) - // then we enforce a minimum delay of 16 seconds before we begin sleep processing. - // This is to allow time for the Ethernet link to come up, DHCP to get an address, mDNS to issue queries, etc., - // before we make our determination of whether there's a Sleep Proxy out there we should register with. - m->DelaySleep = NonZeroTime(m->timenow + mDNSPlatformOneSecond * 16); - } - - if (m->SPSState == 3) - { - m->SPSState = 0; - mDNSCoreBeSleepProxyServer_internal(m, m->SPSType, m->SPSPortability, m->SPSMarginalPower, m->SPSTotalPower); - } - - // In case we gave up waiting and went to sleep before we got an ack from the Sleep Proxy, - // on wake we go through our record list and clear updateid back to zero - for (rr = m->ResourceRecords; rr; rr=rr->next) rr->updateid = zeroID; - - // ... and the same for NextSPSAttempt - for (intf = GetFirstActiveInterface(m->HostInterfaces); intf; intf = GetFirstActiveInterface(intf->next)) intf->NextSPSAttempt = -1; - - // Restart unicast and multicast queries - mDNSCoreRestartQueries(m); - - // and reactivtate service registrations - m->NextSRVUpdate = NonZeroTime(m->timenow + mDNSPlatformOneSecond); - LogInfo("mDNSCoreMachineSleep waking: NextSRVUpdate in %d %d", m->NextSRVUpdate - m->timenow, m->timenow); - - // 2. Re-validate our cache records - FORALL_CACHERECORDS(slot, cg, cr) - mDNS_Reconfirm_internal(m, cr, kDefaultReconfirmTimeForWake); - - // 3. Retrigger probing and announcing for all our authoritative records - for (rr = m->ResourceRecords; rr; rr=rr->next) - if (AuthRecord_uDNS(rr)) - { - ActivateUnicastRegistration(m, rr); - } - else - { - if (rr->resrec.RecordType == kDNSRecordTypeVerified && !rr->DependentOn) rr->resrec.RecordType = kDNSRecordTypeUnique; - rr->ProbeCount = DefaultProbeCountForRecordType(rr->resrec.RecordType); - rr->AnnounceCount = InitialAnnounceCount; - rr->SendNSECNow = mDNSNULL; - InitializeLastAPTime(m, rr); - } - - // 4. Refresh NAT mappings - // We don't want to have to assume that all hardware can necessarily keep accurate - // track of passage of time while asleep, so on wake we refresh our NAT mappings - // We typically wake up with no interfaces active, so there's no need to rush to try to find our external address. - // When we get a network configuration change, mDNSMacOSXNetworkChanged calls uDNS_SetupDNSConfig, which calls - // mDNS_SetPrimaryInterfaceInfo, which then sets m->retryGetAddr to immediately request our external address from the NAT gateway. - m->retryIntervalGetAddr = NATMAP_INIT_RETRY; - m->retryGetAddr = m->timenow + mDNSPlatformOneSecond * 5; - LogInfo("mDNSCoreMachineSleep: retryGetAddr in %d %d", m->retryGetAddr - m->timenow, m->timenow); - RecreateNATMappings(m); - mDNS_Unlock(m); - } - } - -mDNSexport mDNSBool mDNSCoreReadyForSleep(mDNS *m, mDNSs32 now) - { - DNSQuestion *q; - AuthRecord *rr; - NetworkInterfaceInfo *intf; - - mDNS_Lock(m); - - if (m->DelaySleep) goto notready; - - // If we've not hit the sleep limit time, and it's not time for our next retry, we can skip these checks - if (m->SleepLimit - now > 0 && m->NextScheduledSPRetry - now > 0) goto notready; - - m->NextScheduledSPRetry = now + 0x40000000UL; - - // See if we might need to retransmit any lost Sleep Proxy Registrations - for (intf = GetFirstActiveInterface(m->HostInterfaces); intf; intf = GetFirstActiveInterface(intf->next)) - if (intf->NextSPSAttempt >= 0) - { - if (now - intf->NextSPSAttemptTime >= 0) - { - LogSPS("mDNSCoreReadyForSleep: retrying for %s SPS %d try %d", - intf->ifname, intf->NextSPSAttempt/3, intf->NextSPSAttempt); - SendSPSRegistration(m, intf, zeroID); - // Don't need to "goto notready" here, because if we do still have record registrations - // that have not been acknowledged yet, we'll catch that in the record list scan below. - } - else - if (m->NextScheduledSPRetry - intf->NextSPSAttemptTime > 0) - m->NextScheduledSPRetry = intf->NextSPSAttemptTime; - } - - // Scan list of interfaces, and see if we're still waiting for any sleep proxy resolves to complete - for (intf = GetFirstActiveInterface(m->HostInterfaces); intf; intf = GetFirstActiveInterface(intf->next)) - { - int sps = (intf->NextSPSAttempt == 0) ? 0 : (intf->NextSPSAttempt-1)/3; - if (intf->NetWakeResolve[sps].ThisQInterval >= 0) - { - LogSPS("mDNSCoreReadyForSleep: waiting for SPS Resolve %s %##s (%s)", - intf->ifname, intf->NetWakeResolve[sps].qname.c, DNSTypeName(intf->NetWakeResolve[sps].qtype)); - goto spsnotready; - } - } - - // Scan list of registered records - for (rr = m->ResourceRecords; rr; rr = rr->next) - if (!AuthRecord_uDNS(rr)) - if (!mDNSOpaque16IsZero(rr->updateid)) - { LogSPS("mDNSCoreReadyForSleep: waiting for SPS Update ID %d %s", mDNSVal16(rr->updateid), ARDisplayString(m,rr)); goto spsnotready; } - - // Scan list of private LLQs, and make sure they've all completed their handshake with the server - for (q = m->Questions; q; q = q->next) - if (!mDNSOpaque16IsZero(q->TargetQID) && q->LongLived && q->ReqLease == 0 && q->tcp) - { - LogSPS("mDNSCoreReadyForSleep: waiting for LLQ %##s (%s)", q->qname.c, DNSTypeName(q->qtype)); - goto notready; - } - - // Scan list of registered records - for (rr = m->ResourceRecords; rr; rr = rr->next) - if (AuthRecord_uDNS(rr)) - { - if (rr->state == regState_Refresh && rr->tcp) - { LogSPS("mDNSCoreReadyForSleep: waiting for Record Update ID %d %s", mDNSVal16(rr->updateid), ARDisplayString(m,rr)); goto notready; } - #if APPLE_OSX_mDNSResponder - if (!RecordReadyForSleep(m, rr)) { LogSPS("mDNSCoreReadyForSleep: waiting for %s", ARDisplayString(m, rr)); goto notready; } - #endif - } - - mDNS_Unlock(m); - return mDNStrue; - -spsnotready: - - // If we failed to complete sleep proxy registration within ten seconds, we give up on that - // and allow up to ten seconds more to complete wide-area deregistration instead - if (now - m->SleepLimit >= 0) - { - LogMsg("Failed to register with SPS, now sending goodbyes"); - - for (intf = GetFirstActiveInterface(m->HostInterfaces); intf; intf = GetFirstActiveInterface(intf->next)) - if (intf->NetWakeBrowse.ThisQInterval >= 0) - { - LogSPS("ReadyForSleep mDNS_DeactivateNetWake %s %##s (%s)", - intf->ifname, intf->NetWakeResolve[0].qname.c, DNSTypeName(intf->NetWakeResolve[0].qtype)); - mDNS_DeactivateNetWake_internal(m, intf); - } - - for (rr = m->ResourceRecords; rr; rr = rr->next) - if (!AuthRecord_uDNS(rr)) - if (!mDNSOpaque16IsZero(rr->updateid)) - { - LogSPS("ReadyForSleep clearing updateid for %s", ARDisplayString(m, rr)); - rr->updateid = zeroID; - } - - // We'd really like to allow up to ten seconds more here, - // but if we don't respond to the sleep notification within 30 seconds - // we'll be put back to sleep forcibly without the chance to schedule the next maintenance wake. - // Right now we wait 16 sec after wake for all the interfaces to come up, then we wait up to 10 seconds - // more for SPS resolves and record registrations to complete, which puts us at 26 seconds. - // If we allow just one more second to send our goodbyes, that puts us at 27 seconds. - m->SleepLimit = now + mDNSPlatformOneSecond * 1; - - SendSleepGoodbyes(m); - } - -notready: - mDNS_Unlock(m); - return mDNSfalse; - } - -mDNSexport mDNSs32 mDNSCoreIntervalToNextWake(mDNS *const m, mDNSs32 now) - { - AuthRecord *ar; - - // Even when we have no wake-on-LAN-capable interfaces, or we failed to find a sleep proxy, or we have other - // failure scenarios, we still want to wake up in at most 120 minutes, to see if the network environment has changed. - // E.g. we might wake up and find no wireless network because the base station got rebooted just at that moment, - // and if that happens we don't want to just give up and go back to sleep and never try again. - mDNSs32 e = now + (120 * 60 * mDNSPlatformOneSecond); // Sleep for at most 120 minutes - - NATTraversalInfo *nat; - for (nat = m->NATTraversals; nat; nat=nat->next) - if (nat->Protocol && nat->ExpiryTime && nat->ExpiryTime - now > mDNSPlatformOneSecond*4) - { - mDNSs32 t = nat->ExpiryTime - (nat->ExpiryTime - now) / 10; // Wake up when 90% of the way to the expiry time - if (e - t > 0) e = t; - LogSPS("ComputeWakeTime: %p %s Int %5d Ext %5d Err %d Retry %5d Interval %5d Expire %5d Wake %5d", - nat, nat->Protocol == NATOp_MapTCP ? "TCP" : "UDP", - mDNSVal16(nat->IntPort), mDNSVal16(nat->ExternalPort), nat->Result, - nat->retryPortMap ? (nat->retryPortMap - now) / mDNSPlatformOneSecond : 0, - nat->retryInterval / mDNSPlatformOneSecond, - nat->ExpiryTime ? (nat->ExpiryTime - now) / mDNSPlatformOneSecond : 0, - (t - now) / mDNSPlatformOneSecond); - } - - // This loop checks both the time we need to renew wide-area registrations, - // and the time we need to renew Sleep Proxy registrations - for (ar = m->ResourceRecords; ar; ar = ar->next) - if (ar->expire && ar->expire - now > mDNSPlatformOneSecond*4) - { - mDNSs32 t = ar->expire - (ar->expire - now) / 10; // Wake up when 90% of the way to the expiry time - if (e - t > 0) e = t; - LogSPS("ComputeWakeTime: %p Int %7d Next %7d Expire %7d Wake %7d %s", - ar, ar->ThisAPInterval / mDNSPlatformOneSecond, - (ar->LastAPTime + ar->ThisAPInterval - now) / mDNSPlatformOneSecond, - ar->expire ? (ar->expire - now) / mDNSPlatformOneSecond : 0, - (t - now) / mDNSPlatformOneSecond, ARDisplayString(m, ar)); - } - - return(e - now); - } - -// *************************************************************************** -#if COMPILER_LIKES_PRAGMA_MARK -#pragma mark - -#pragma mark - Packet Reception Functions -#endif - -#define MustSendRecord(RR) ((RR)->NR_AnswerTo || (RR)->NR_AdditionalTo) - -mDNSlocal mDNSu8 *GenerateUnicastResponse(const DNSMessage *const query, const mDNSu8 *const end, - const mDNSInterfaceID InterfaceID, mDNSBool LegacyQuery, DNSMessage *const response, AuthRecord *ResponseRecords) - { - mDNSu8 *responseptr = response->data; - const mDNSu8 *const limit = response->data + sizeof(response->data); - const mDNSu8 *ptr = query->data; - AuthRecord *rr; - mDNSu32 maxttl = 0x70000000; - int i; - - // Initialize the response fields so we can answer the questions - InitializeDNSMessage(&response->h, query->h.id, ResponseFlags); - - // *** - // *** 1. Write out the list of questions we are actually going to answer with this packet - // *** - if (LegacyQuery) - { - maxttl = kStaticCacheTTL; - for (i=0; ih.numQuestions; i++) // For each question... - { - DNSQuestion q; - ptr = getQuestion(query, ptr, end, InterfaceID, &q); // get the question... - if (!ptr) return(mDNSNULL); - - for (rr=ResponseRecords; rr; rr=rr->NextResponse) // and search our list of proposed answers - { - if (rr->NR_AnswerTo == ptr) // If we're going to generate a record answering this question - { // then put the question in the question section - responseptr = putQuestion(response, responseptr, limit, &q.qname, q.qtype, q.qclass); - if (!responseptr) { debugf("GenerateUnicastResponse: Ran out of space for questions!"); return(mDNSNULL); } - break; // break out of the ResponseRecords loop, and go on to the next question - } - } - } - - if (response->h.numQuestions == 0) { LogMsg("GenerateUnicastResponse: ERROR! Why no questions?"); return(mDNSNULL); } - } - - // *** - // *** 2. Write Answers - // *** - for (rr=ResponseRecords; rr; rr=rr->NextResponse) - if (rr->NR_AnswerTo) - { - mDNSu8 *p = PutResourceRecordTTL(response, responseptr, &response->h.numAnswers, &rr->resrec, - maxttl < rr->resrec.rroriginalttl ? maxttl : rr->resrec.rroriginalttl); - if (p) responseptr = p; - else { debugf("GenerateUnicastResponse: Ran out of space for answers!"); response->h.flags.b[0] |= kDNSFlag0_TC; } - } - - // *** - // *** 3. Write Additionals - // *** - for (rr=ResponseRecords; rr; rr=rr->NextResponse) - if (rr->NR_AdditionalTo && !rr->NR_AnswerTo) - { - mDNSu8 *p = PutResourceRecordTTL(response, responseptr, &response->h.numAdditionals, &rr->resrec, - maxttl < rr->resrec.rroriginalttl ? maxttl : rr->resrec.rroriginalttl); - if (p) responseptr = p; - else debugf("GenerateUnicastResponse: No more space for additionals"); - } - - return(responseptr); - } - -// AuthRecord *our is our Resource Record -// CacheRecord *pkt is the Resource Record from the response packet we've witnessed on the network -// Returns 0 if there is no conflict -// Returns +1 if there was a conflict and we won -// Returns -1 if there was a conflict and we lost and have to rename -mDNSlocal int CompareRData(const AuthRecord *const our, const CacheRecord *const pkt) - { - mDNSu8 ourdata[256], *ourptr = ourdata, *ourend; - mDNSu8 pktdata[256], *pktptr = pktdata, *pktend; - if (!our) { LogMsg("CompareRData ERROR: our is NULL"); return(+1); } - if (!pkt) { LogMsg("CompareRData ERROR: pkt is NULL"); return(+1); } - - ourend = putRData(mDNSNULL, ourdata, ourdata + sizeof(ourdata), &our->resrec); - pktend = putRData(mDNSNULL, pktdata, pktdata + sizeof(pktdata), &pkt->resrec); - while (ourptr < ourend && pktptr < pktend && *ourptr == *pktptr) { ourptr++; pktptr++; } - if (ourptr >= ourend && pktptr >= pktend) return(0); // If data identical, not a conflict - - if (ourptr >= ourend) return(-1); // Our data ran out first; We lost - if (pktptr >= pktend) return(+1); // Packet data ran out first; We won - if (*pktptr > *ourptr) return(-1); // Our data is numerically lower; We lost - if (*pktptr < *ourptr) return(+1); // Packet data is numerically lower; We won - - LogMsg("CompareRData ERROR: Invalid state"); - return(-1); - } - -// See if we have an authoritative record that's identical to this packet record, -// whose canonical DependentOn record is the specified master record. -// The DependentOn pointer is typically used for the TXT record of service registrations -// It indicates that there is no inherent conflict detection for the TXT record -// -- it depends on the SRV record to resolve name conflicts -// If we find any identical ResourceRecords in our authoritative list, then follow their DependentOn -// pointer chain (if any) to make sure we reach the canonical DependentOn record -// If the record has no DependentOn, then just return that record's pointer -// Returns NULL if we don't have any local RRs that are identical to the one from the packet -mDNSlocal mDNSBool MatchDependentOn(const mDNS *const m, const CacheRecord *const pktrr, const AuthRecord *const master) - { - const AuthRecord *r1; - for (r1 = m->ResourceRecords; r1; r1=r1->next) - { - if (IdenticalResourceRecord(&r1->resrec, &pktrr->resrec)) - { - const AuthRecord *r2 = r1; - while (r2->DependentOn) r2 = r2->DependentOn; - if (r2 == master) return(mDNStrue); - } - } - for (r1 = m->DuplicateRecords; r1; r1=r1->next) - { - if (IdenticalResourceRecord(&r1->resrec, &pktrr->resrec)) - { - const AuthRecord *r2 = r1; - while (r2->DependentOn) r2 = r2->DependentOn; - if (r2 == master) return(mDNStrue); - } - } - return(mDNSfalse); - } - -// Find the canonical RRSet pointer for this RR received in a packet. -// If we find any identical AuthRecord in our authoritative list, then follow its RRSet -// pointers (if any) to make sure we return the canonical member of this name/type/class -// Returns NULL if we don't have any local RRs that are identical to the one from the packet -mDNSlocal const AuthRecord *FindRRSet(const mDNS *const m, const CacheRecord *const pktrr) - { - const AuthRecord *rr; - for (rr = m->ResourceRecords; rr; rr=rr->next) - { - if (IdenticalResourceRecord(&rr->resrec, &pktrr->resrec)) - { - while (rr->RRSet && rr != rr->RRSet) rr = rr->RRSet; - return(rr); - } - } - return(mDNSNULL); - } - -// PacketRRConflict is called when we've received an RR (pktrr) which has the same name -// as one of our records (our) but different rdata. -// 1. If our record is not a type that's supposed to be unique, we don't care. -// 2a. If our record is marked as dependent on some other record for conflict detection, ignore this one. -// 2b. If the packet rr exactly matches one of our other RRs, and *that* record's DependentOn pointer -// points to our record, ignore this conflict (e.g. the packet record matches one of our -// TXT records, and that record is marked as dependent on 'our', its SRV record). -// 3. If we have some *other* RR that exactly matches the one from the packet, and that record and our record -// are members of the same RRSet, then this is not a conflict. -mDNSlocal mDNSBool PacketRRConflict(const mDNS *const m, const AuthRecord *const our, const CacheRecord *const pktrr) - { - // If not supposed to be unique, not a conflict - if (!(our->resrec.RecordType & kDNSRecordTypeUniqueMask)) return(mDNSfalse); - - // If a dependent record, not a conflict - if (our->DependentOn || MatchDependentOn(m, pktrr, our)) return(mDNSfalse); - else - { - // If the pktrr matches a member of ourset, not a conflict - const AuthRecord *ourset = our->RRSet ? our->RRSet : our; - const AuthRecord *pktset = FindRRSet(m, pktrr); - if (pktset == ourset) return(mDNSfalse); - - // For records we're proxying, where we don't know the full - // relationship between the records, having any matching record - // in our AuthRecords list is sufficient evidence of non-conflict - if (our->WakeUp.HMAC.l[0] && pktset) return(mDNSfalse); - } - - // Okay, this is a conflict - return(mDNStrue); - } - -// Note: ResolveSimultaneousProbe calls mDNS_Deregister_internal which can call a user callback, which may change -// the record list and/or question list. -// Any code walking either list must use the CurrentQuestion and/or CurrentRecord mechanism to protect against this. -mDNSlocal void ResolveSimultaneousProbe(mDNS *const m, const DNSMessage *const query, const mDNSu8 *const end, - DNSQuestion *q, AuthRecord *our) - { - int i; - const mDNSu8 *ptr = LocateAuthorities(query, end); - mDNSBool FoundUpdate = mDNSfalse; - - for (i = 0; i < query->h.numAuthorities; i++) - { - ptr = GetLargeResourceRecord(m, query, ptr, end, q->InterfaceID, kDNSRecordTypePacketAuth, &m->rec); - if (!ptr) break; - if (m->rec.r.resrec.RecordType != kDNSRecordTypePacketNegative && ResourceRecordAnswersQuestion(&m->rec.r.resrec, q)) - { - FoundUpdate = mDNStrue; - if (PacketRRConflict(m, our, &m->rec.r)) - { - int result = (int)our->resrec.rrclass - (int)m->rec.r.resrec.rrclass; - if (!result) result = (int)our->resrec.rrtype - (int)m->rec.r.resrec.rrtype; - if (!result) result = CompareRData(our, &m->rec.r); - if (result) - { - const char *const msg = (result < 0) ? "lost:" : (result > 0) ? "won: " : "tie: "; - LogMsg("ResolveSimultaneousProbe: %p Pkt Record: %08lX %s", q->InterfaceID, m->rec.r.resrec.rdatahash, CRDisplayString(m, &m->rec.r)); - LogMsg("ResolveSimultaneousProbe: %p Our Record %d %s %08lX %s", our->resrec.InterfaceID, our->ProbeCount, msg, our->resrec.rdatahash, ARDisplayString(m, our)); - } - // If we lost the tie-break for simultaneous probes, we don't immediately give up, because we might be seeing stale packets on the network. - // Instead we pause for one second, to give the other host (if real) a chance to establish its name, and then try probing again. - // If there really is another live host out there with the same name, it will answer our probes and we'll then rename. - if (result < 0) - { - m->SuppressProbes = NonZeroTime(m->timenow + mDNSPlatformOneSecond); - our->ProbeCount = DefaultProbeCountForTypeUnique; - our->AnnounceCount = InitialAnnounceCount; - InitializeLastAPTime(m, our); - goto exit; - } - } -#if 0 - else - { - LogMsg("ResolveSimultaneousProbe: %p Pkt Record: %08lX %s", q->InterfaceID, m->rec.r.resrec.rdatahash, CRDisplayString(m, &m->rec.r)); - LogMsg("ResolveSimultaneousProbe: %p Our Record %d ign: %08lX %s", our->resrec.InterfaceID, our->ProbeCount, our->resrec.rdatahash, ARDisplayString(m, our)); - } -#endif - } - m->rec.r.resrec.RecordType = 0; // Clear RecordType to show we're not still using it - } - if (!FoundUpdate) - LogInfo("ResolveSimultaneousProbe: %##s (%s): No Update Record found", our->resrec.name->c, DNSTypeName(our->resrec.rrtype)); -exit: - m->rec.r.resrec.RecordType = 0; // Clear RecordType to show we're not still using it - } - -mDNSlocal CacheRecord *FindIdenticalRecordInCache(const mDNS *const m, const ResourceRecord *const pktrr) - { - mDNSu32 slot = HashSlot(pktrr->name); - CacheGroup *cg = CacheGroupForRecord(m, slot, pktrr); - CacheRecord *rr; - mDNSBool match; - for (rr = cg ? cg->members : mDNSNULL; rr; rr=rr->next) - { - match = !pktrr->InterfaceID ? pktrr->rDNSServer == rr->resrec.rDNSServer : pktrr->InterfaceID == rr->resrec.InterfaceID; - if (match && IdenticalSameNameRecord(pktrr, &rr->resrec)) break; - } - return(rr); - } - -// Called from mDNSCoreReceiveUpdate when we get a sleep proxy registration request, -// to check our lists and discard any stale duplicates of this record we already have -mDNSlocal void ClearIdenticalProxyRecords(mDNS *const m, const OwnerOptData *const owner, AuthRecord *const thelist) - { - if (m->CurrentRecord) - LogMsg("ClearIdenticalProxyRecords ERROR m->CurrentRecord already set %s", ARDisplayString(m, m->CurrentRecord)); - m->CurrentRecord = thelist; - while (m->CurrentRecord) - { - AuthRecord *const rr = m->CurrentRecord; - if (m->rec.r.resrec.InterfaceID == rr->resrec.InterfaceID && mDNSSameEthAddress(&owner->HMAC, &rr->WakeUp.HMAC)) - if (IdenticalResourceRecord(&rr->resrec, &m->rec.r.resrec)) - { - LogSPS("ClearIdenticalProxyRecords: Removing %3d H-MAC %.6a I-MAC %.6a %d %d %s", - m->ProxyRecords, &rr->WakeUp.HMAC, &rr->WakeUp.IMAC, rr->WakeUp.seq, owner->seq, ARDisplayString(m, rr)); - rr->WakeUp.HMAC = zeroEthAddr; // Clear HMAC so that mDNS_Deregister_internal doesn't waste packets trying to wake this host - rr->RequireGoodbye = mDNSfalse; // and we don't want to send goodbye for it - mDNS_Deregister_internal(m, rr, mDNS_Dereg_normal); - SetSPSProxyListChanged(m->rec.r.resrec.InterfaceID); - } - // Mustn't advance m->CurrentRecord until *after* mDNS_Deregister_internal, because - // new records could have been added to the end of the list as a result of that call. - if (m->CurrentRecord == rr) // If m->CurrentRecord was not advanced for us, do it now - m->CurrentRecord = rr->next; - } - } - -// Called from ProcessQuery when we get an mDNS packet with an owner record in it -mDNSlocal void ClearProxyRecords(mDNS *const m, const OwnerOptData *const owner, AuthRecord *const thelist) - { - if (m->CurrentRecord) - LogMsg("ClearProxyRecords ERROR m->CurrentRecord already set %s", ARDisplayString(m, m->CurrentRecord)); - m->CurrentRecord = thelist; - while (m->CurrentRecord) - { - AuthRecord *const rr = m->CurrentRecord; - if (m->rec.r.resrec.InterfaceID == rr->resrec.InterfaceID && mDNSSameEthAddress(&owner->HMAC, &rr->WakeUp.HMAC)) - if (owner->seq != rr->WakeUp.seq || m->timenow - rr->TimeRcvd > mDNSPlatformOneSecond * 60) - { - if (rr->AddressProxy.type == mDNSAddrType_IPv6) - { - // We don't do this here because we know that the host is waking up at this point, so we don't send - // Unsolicited Neighbor Advertisements -- even Neighbor Advertisements agreeing with what the host should be - // saying itself -- because it can cause some IPv6 stacks to falsely conclude that there's an address conflict. - #if MDNS_USE_Unsolicited_Neighbor_Advertisements - LogSPS("NDP Announcement -- Releasing traffic for H-MAC %.6a I-MAC %.6a %s", - &rr->WakeUp.HMAC, &rr->WakeUp.IMAC, ARDisplayString(m,rr)); - SendNDP(m, NDP_Adv, NDP_Override, rr, &rr->AddressProxy.ip.v6, &rr->WakeUp.IMAC, &AllHosts_v6, &AllHosts_v6_Eth); - #endif - } - LogSPS("ClearProxyRecords: Removing %3d AC %2d %02X H-MAC %.6a I-MAC %.6a %d %d %s", - m->ProxyRecords, rr->AnnounceCount, rr->resrec.RecordType, - &rr->WakeUp.HMAC, &rr->WakeUp.IMAC, rr->WakeUp.seq, owner->seq, ARDisplayString(m, rr)); - if (rr->resrec.RecordType == kDNSRecordTypeDeregistering) rr->resrec.RecordType = kDNSRecordTypeShared; - rr->WakeUp.HMAC = zeroEthAddr; // Clear HMAC so that mDNS_Deregister_internal doesn't waste packets trying to wake this host - rr->RequireGoodbye = mDNSfalse; // and we don't want to send goodbye for it, since real host is now back and functional - mDNS_Deregister_internal(m, rr, mDNS_Dereg_normal); - SetSPSProxyListChanged(m->rec.r.resrec.InterfaceID); - } - // Mustn't advance m->CurrentRecord until *after* mDNS_Deregister_internal, because - // new records could have been added to the end of the list as a result of that call. - if (m->CurrentRecord == rr) // If m->CurrentRecord was not advanced for us, do it now - m->CurrentRecord = rr->next; - } - } - -// ProcessQuery examines a received query to see if we have any answers to give -mDNSlocal mDNSu8 *ProcessQuery(mDNS *const m, const DNSMessage *const query, const mDNSu8 *const end, - const mDNSAddr *srcaddr, const mDNSInterfaceID InterfaceID, mDNSBool LegacyQuery, mDNSBool QueryWasMulticast, - mDNSBool QueryWasLocalUnicast, DNSMessage *const response) - { - mDNSBool FromLocalSubnet = srcaddr && mDNS_AddressIsLocalSubnet(m, InterfaceID, srcaddr); - AuthRecord *ResponseRecords = mDNSNULL; - AuthRecord **nrp = &ResponseRecords; - CacheRecord *ExpectedAnswers = mDNSNULL; // Records in our cache we expect to see updated - CacheRecord **eap = &ExpectedAnswers; - DNSQuestion *DupQuestions = mDNSNULL; // Our questions that are identical to questions in this packet - DNSQuestion **dqp = &DupQuestions; - mDNSs32 delayresponse = 0; - mDNSBool SendLegacyResponse = mDNSfalse; - const mDNSu8 *ptr; - mDNSu8 *responseptr = mDNSNULL; - AuthRecord *rr; - int i; - - // *** - // *** 1. Look in Additional Section for an OPT record - // *** - ptr = LocateOptRR(query, end, DNSOpt_OwnerData_ID_Space); - if (ptr) - { - ptr = GetLargeResourceRecord(m, query, ptr, end, InterfaceID, kDNSRecordTypePacketAdd, &m->rec); - if (ptr && m->rec.r.resrec.RecordType != kDNSRecordTypePacketNegative && m->rec.r.resrec.rrtype == kDNSType_OPT) - { - const rdataOPT *opt; - const rdataOPT *const e = (const rdataOPT *)&m->rec.r.resrec.rdata->u.data[m->rec.r.resrec.rdlength]; - // Find owner sub-option(s). We verify that the MAC is non-zero, otherwise we could inadvertently - // delete all our own AuthRecords (which are identified by having zero MAC tags on them). - for (opt = &m->rec.r.resrec.rdata->u.opt[0]; opt < e; opt++) - if (opt->opt == kDNSOpt_Owner && opt->u.owner.vers == 0 && opt->u.owner.HMAC.l[0]) - { - ClearProxyRecords(m, &opt->u.owner, m->DuplicateRecords); - ClearProxyRecords(m, &opt->u.owner, m->ResourceRecords); - } - } - m->rec.r.resrec.RecordType = 0; // Clear RecordType to show we're not still using it - } - - // *** - // *** 2. Parse Question Section and mark potential answers - // *** - ptr = query->data; - for (i=0; ih.numQuestions; i++) // For each question... - { - mDNSBool QuestionNeedsMulticastResponse; - int NumAnswersForThisQuestion = 0; - AuthRecord *NSECAnswer = mDNSNULL; - DNSQuestion pktq, *q; - ptr = getQuestion(query, ptr, end, InterfaceID, &pktq); // get the question... - if (!ptr) goto exit; - - // The only queries that *need* a multicast response are: - // * Queries sent via multicast - // * from port 5353 - // * that don't have the kDNSQClass_UnicastResponse bit set - // These queries need multicast responses because other clients will: - // * suppress their own identical questions when they see these questions, and - // * expire their cache records if they don't see the expected responses - // For other queries, we may still choose to send the occasional multicast response anyway, - // to keep our neighbours caches warm, and for ongoing conflict detection. - QuestionNeedsMulticastResponse = QueryWasMulticast && !LegacyQuery && !(pktq.qclass & kDNSQClass_UnicastResponse); - // Clear the UnicastResponse flag -- don't want to confuse the rest of the code that follows later - pktq.qclass &= ~kDNSQClass_UnicastResponse; - - // Note: We use the m->CurrentRecord mechanism here because calling ResolveSimultaneousProbe - // can result in user callbacks which may change the record list and/or question list. - // Also note: we just mark potential answer records here, without trying to build the - // "ResponseRecords" list, because we don't want to risk user callbacks deleting records - // from that list while we're in the middle of trying to build it. - if (m->CurrentRecord) - LogMsg("ProcessQuery ERROR m->CurrentRecord already set %s", ARDisplayString(m, m->CurrentRecord)); - m->CurrentRecord = m->ResourceRecords; - while (m->CurrentRecord) - { - rr = m->CurrentRecord; - m->CurrentRecord = rr->next; - if (AnyTypeRecordAnswersQuestion(&rr->resrec, &pktq) && (QueryWasMulticast || QueryWasLocalUnicast || rr->AllowRemoteQuery)) - { - if (RRTypeAnswersQuestionType(&rr->resrec, pktq.qtype)) - { - if (rr->resrec.RecordType == kDNSRecordTypeUnique) - ResolveSimultaneousProbe(m, query, end, &pktq, rr); - else if (ResourceRecordIsValidAnswer(rr)) - { - NumAnswersForThisQuestion++; - // Note: We should check here if this is a probe-type query, and if so, generate an immediate - // unicast answer back to the source, because timeliness in answering probes is important. - - // Notes: - // NR_AnswerTo pointing into query packet means "answer via immediate legacy unicast" (may *also* choose to multicast) - // NR_AnswerTo == (mDNSu8*)~1 means "answer via delayed unicast" (to modern querier; may promote to multicast instead) - // NR_AnswerTo == (mDNSu8*)~0 means "definitely answer via multicast" (can't downgrade to unicast later) - // If we're not multicasting this record because the kDNSQClass_UnicastResponse bit was set, - // but the multicast querier is not on a matching subnet (e.g. because of overlaid subnets on one link) - // then we'll multicast it anyway (if we unicast, the receiver will ignore it because it has an apparently non-local source) - if (QuestionNeedsMulticastResponse || (!FromLocalSubnet && QueryWasMulticast && !LegacyQuery)) - { - // We only mark this question for sending if it is at least one second since the last time we multicast it - // on this interface. If it is more than a second, or LastMCInterface is different, then we may multicast it. - // This is to guard against the case where someone blasts us with queries as fast as they can. - if (m->timenow - (rr->LastMCTime + mDNSPlatformOneSecond) >= 0 || - (rr->LastMCInterface != mDNSInterfaceMark && rr->LastMCInterface != InterfaceID)) - rr->NR_AnswerTo = (mDNSu8*)~0; - } - else if (!rr->NR_AnswerTo) rr->NR_AnswerTo = LegacyQuery ? ptr : (mDNSu8*)~1; - } - } - else if ((rr->resrec.RecordType & kDNSRecordTypeActiveUniqueMask) && ResourceRecordIsValidAnswer(rr)) - { - // If we don't have any answers for this question, but we do own another record with the same name, - // then we'll want to mark it to generate an NSEC record on this interface - if (!NSECAnswer) NSECAnswer = rr; - } - } - } - - if (NumAnswersForThisQuestion == 0 && NSECAnswer) - { - NumAnswersForThisQuestion++; - NSECAnswer->SendNSECNow = InterfaceID; - m->NextScheduledResponse = m->timenow; - } - - // If we couldn't answer this question, someone else might be able to, - // so use random delay on response to reduce collisions - if (NumAnswersForThisQuestion == 0) delayresponse = mDNSPlatformOneSecond; // Divided by 50 = 20ms - -#if ENABLE_MULTI_PACKET_QUERY_SNOOPING - if (QuestionNeedsMulticastResponse) -#else - // We only do the following accelerated cache expiration and duplicate question suppression processing - // for non-truncated multicast queries with multicast responses. - // For any query generating a unicast response we don't do this because we can't assume we will see the response. - // For truncated queries we don't do this because a response we're expecting might be suppressed by a subsequent - // known-answer packet, and when there's packet loss we can't safely assume we'll receive *all* known-answer packets. - if (QuestionNeedsMulticastResponse && !(query->h.flags.b[0] & kDNSFlag0_TC)) -#endif - { - const mDNSu32 slot = HashSlot(&pktq.qname); - CacheGroup *cg = CacheGroupForName(m, slot, pktq.qnamehash, &pktq.qname); - CacheRecord *cr; - - // Make a list indicating which of our own cache records we expect to see updated as a result of this query - // Note: Records larger than 1K are not habitually multicast, so don't expect those to be updated -#if ENABLE_MULTI_PACKET_QUERY_SNOOPING - if (!(query->h.flags.b[0] & kDNSFlag0_TC)) -#endif - for (cr = cg ? cg->members : mDNSNULL; cr; cr=cr->next) - if (SameNameRecordAnswersQuestion(&cr->resrec, &pktq) && cr->resrec.rdlength <= SmallRecordLimit) - if (!cr->NextInKAList && eap != &cr->NextInKAList) - { - *eap = cr; - eap = &cr->NextInKAList; -#if ENABLE_MULTI_PACKET_QUERY_SNOOPING - if (cr->MPUnansweredQ == 0 || m->timenow - cr->MPLastUnansweredQT >= mDNSPlatformOneSecond) - { - // Although MPUnansweredQ is only really used for multi-packet query processing, - // we increment it for both single-packet and multi-packet queries, so that it stays in sync - // with the MPUnansweredKA value, which by necessity is incremented for both query types. - cr->MPUnansweredQ++; - cr->MPLastUnansweredQT = m->timenow; - cr->MPExpectingKA = mDNStrue; - } -#endif - } - - // Check if this question is the same as any of mine. - // We only do this for non-truncated queries. Right now it would be too complicated to try - // to keep track of duplicate suppression state between multiple packets, especially when we - // can't guarantee to receive all of the Known Answer packets that go with a particular query. -#if ENABLE_MULTI_PACKET_QUERY_SNOOPING - if (!(query->h.flags.b[0] & kDNSFlag0_TC)) -#endif - for (q = m->Questions; q; q=q->next) - if (!q->Target.type && ActiveQuestion(q) && m->timenow - q->LastQTxTime > mDNSPlatformOneSecond / 4) - if (!q->InterfaceID || q->InterfaceID == InterfaceID) - if (q->NextInDQList == mDNSNULL && dqp != &q->NextInDQList) - if (q->qtype == pktq.qtype && - q->qclass == pktq.qclass && - q->qnamehash == pktq.qnamehash && SameDomainName(&q->qname, &pktq.qname)) - { *dqp = q; dqp = &q->NextInDQList; } - } - } - - // *** - // *** 3. Now we can safely build the list of marked answers - // *** - for (rr = m->ResourceRecords; rr; rr=rr->next) // Now build our list of potential answers - if (rr->NR_AnswerTo) // If we marked the record... - AddRecordToResponseList(&nrp, rr, mDNSNULL); // ... add it to the list - - // *** - // *** 4. Add additional records - // *** - AddAdditionalsToResponseList(m, ResponseRecords, &nrp, InterfaceID); - - // *** - // *** 5. Parse Answer Section and cancel any records disallowed by Known-Answer list - // *** - for (i=0; ih.numAnswers; i++) // For each record in the query's answer section... - { - // Get the record... - CacheRecord *ourcacherr; - ptr = GetLargeResourceRecord(m, query, ptr, end, InterfaceID, kDNSRecordTypePacketAns, &m->rec); - if (!ptr) goto exit; - if (m->rec.r.resrec.RecordType != kDNSRecordTypePacketNegative) - { - // See if this Known-Answer suppresses any of our currently planned answers - for (rr=ResponseRecords; rr; rr=rr->NextResponse) - if (MustSendRecord(rr) && ShouldSuppressKnownAnswer(&m->rec.r, rr)) - { rr->NR_AnswerTo = mDNSNULL; rr->NR_AdditionalTo = mDNSNULL; } - - // See if this Known-Answer suppresses any previously scheduled answers (for multi-packet KA suppression) - for (rr=m->ResourceRecords; rr; rr=rr->next) - { - // If we're planning to send this answer on this interface, and only on this interface, then allow KA suppression - if (rr->ImmedAnswer == InterfaceID && ShouldSuppressKnownAnswer(&m->rec.r, rr)) - { - if (srcaddr->type == mDNSAddrType_IPv4) - { - if (mDNSSameIPv4Address(rr->v4Requester, srcaddr->ip.v4)) rr->v4Requester = zerov4Addr; - } - else if (srcaddr->type == mDNSAddrType_IPv6) - { - if (mDNSSameIPv6Address(rr->v6Requester, srcaddr->ip.v6)) rr->v6Requester = zerov6Addr; - } - if (mDNSIPv4AddressIsZero(rr->v4Requester) && mDNSIPv6AddressIsZero(rr->v6Requester)) - { - rr->ImmedAnswer = mDNSNULL; - rr->ImmedUnicast = mDNSfalse; - #if MDNS_LOG_ANSWER_SUPPRESSION_TIMES - LogMsg("Suppressed after%4d: %s", m->timenow - rr->ImmedAnswerMarkTime, ARDisplayString(m, rr)); - #endif - } - } - } - - ourcacherr = FindIdenticalRecordInCache(m, &m->rec.r.resrec); - - #if ENABLE_MULTI_PACKET_QUERY_SNOOPING - // See if this Known-Answer suppresses any answers we were expecting for our cache records. We do this always, - // even if the TC bit is not set (the TC bit will *not* be set in the *last* packet of a multi-packet KA list). - if (ourcacherr && ourcacherr->MPExpectingKA && m->timenow - ourcacherr->MPLastUnansweredQT < mDNSPlatformOneSecond) - { - ourcacherr->MPUnansweredKA++; - ourcacherr->MPExpectingKA = mDNSfalse; - } - #endif - - // Having built our ExpectedAnswers list from the questions in this packet, we then remove - // any records that are suppressed by the Known Answer list in this packet. - eap = &ExpectedAnswers; - while (*eap) - { - CacheRecord *cr = *eap; - if (cr->resrec.InterfaceID == InterfaceID && IdenticalResourceRecord(&m->rec.r.resrec, &cr->resrec)) - { *eap = cr->NextInKAList; cr->NextInKAList = mDNSNULL; } - else eap = &cr->NextInKAList; - } - - // See if this Known-Answer is a surprise to us. If so, we shouldn't suppress our own query. - if (!ourcacherr) - { - dqp = &DupQuestions; - while (*dqp) - { - DNSQuestion *q = *dqp; - if (ResourceRecordAnswersQuestion(&m->rec.r.resrec, q)) - { *dqp = q->NextInDQList; q->NextInDQList = mDNSNULL; } - else dqp = &q->NextInDQList; - } - } - } - m->rec.r.resrec.RecordType = 0; // Clear RecordType to show we're not still using it - } - - // *** - // *** 6. Cancel any additionals that were added because of now-deleted records - // *** - for (rr=ResponseRecords; rr; rr=rr->NextResponse) - if (rr->NR_AdditionalTo && !MustSendRecord(rr->NR_AdditionalTo)) - { rr->NR_AnswerTo = mDNSNULL; rr->NR_AdditionalTo = mDNSNULL; } - - // *** - // *** 7. Mark the send flags on the records we plan to send - // *** - for (rr=ResponseRecords; rr; rr=rr->NextResponse) - { - if (rr->NR_AnswerTo) - { - mDNSBool SendMulticastResponse = mDNSfalse; // Send modern multicast response - mDNSBool SendUnicastResponse = mDNSfalse; // Send modern unicast response (not legacy unicast response) - - // If it's been a while since we multicast this, then send a multicast response for conflict detection, etc. - if (m->timenow - (rr->LastMCTime + TicksTTL(rr)/4) >= 0) - { - SendMulticastResponse = mDNStrue; - // If this record was marked for modern (delayed) unicast response, then mark it as promoted to - // multicast response instead (don't want to end up ALSO setting SendUnicastResponse in the check below). - // If this record was marked for legacy unicast response, then we mustn't change the NR_AnswerTo value. - if (rr->NR_AnswerTo == (mDNSu8*)~1) rr->NR_AnswerTo = (mDNSu8*)~0; - } - - // If the client insists on a multicast response, then we'd better send one - if (rr->NR_AnswerTo == (mDNSu8*)~0) SendMulticastResponse = mDNStrue; - else if (rr->NR_AnswerTo == (mDNSu8*)~1) SendUnicastResponse = mDNStrue; - else if (rr->NR_AnswerTo) SendLegacyResponse = mDNStrue; - - if (SendMulticastResponse || SendUnicastResponse) - { -#if MDNS_LOG_ANSWER_SUPPRESSION_TIMES - rr->ImmedAnswerMarkTime = m->timenow; -#endif - m->NextScheduledResponse = m->timenow; - // If we're already planning to send this on another interface, just send it on all interfaces - if (rr->ImmedAnswer && rr->ImmedAnswer != InterfaceID) - rr->ImmedAnswer = mDNSInterfaceMark; - else - { - rr->ImmedAnswer = InterfaceID; // Record interface to send it on - if (SendUnicastResponse) rr->ImmedUnicast = mDNStrue; - if (srcaddr->type == mDNSAddrType_IPv4) - { - if (mDNSIPv4AddressIsZero(rr->v4Requester)) rr->v4Requester = srcaddr->ip.v4; - else if (!mDNSSameIPv4Address(rr->v4Requester, srcaddr->ip.v4)) rr->v4Requester = onesIPv4Addr; - } - else if (srcaddr->type == mDNSAddrType_IPv6) - { - if (mDNSIPv6AddressIsZero(rr->v6Requester)) rr->v6Requester = srcaddr->ip.v6; - else if (!mDNSSameIPv6Address(rr->v6Requester, srcaddr->ip.v6)) rr->v6Requester = onesIPv6Addr; - } - } - } - // If TC flag is set, it means we should expect that additional known answers may be coming in another packet, - // so we allow roughly half a second before deciding to reply (we've observed inter-packet delays of 100-200ms on 802.11) - // else, if record is a shared one, spread responses over 100ms to avoid implosion of simultaneous responses - // else, for a simple unique record reply, we can reply immediately; no need for delay - if (query->h.flags.b[0] & kDNSFlag0_TC) delayresponse = mDNSPlatformOneSecond * 20; // Divided by 50 = 400ms - else if (rr->resrec.RecordType == kDNSRecordTypeShared) delayresponse = mDNSPlatformOneSecond; // Divided by 50 = 20ms - } - else if (rr->NR_AdditionalTo && rr->NR_AdditionalTo->NR_AnswerTo == (mDNSu8*)~0) - { - // Since additional records are an optimization anyway, we only ever send them on one interface at a time - // If two clients on different interfaces do queries that invoke the same optional additional answer, - // then the earlier client is out of luck - rr->ImmedAdditional = InterfaceID; - // No need to set m->NextScheduledResponse here - // We'll send these additional records when we send them, or not, as the case may be - } - } - - // *** - // *** 8. If we think other machines are likely to answer these questions, set our packet suppression timer - // *** - if (delayresponse && (!m->SuppressSending || (m->SuppressSending - m->timenow) < (delayresponse + 49) / 50)) - { -#if MDNS_LOG_ANSWER_SUPPRESSION_TIMES - mDNSs32 oldss = m->SuppressSending; - if (oldss && delayresponse) - LogMsg("Current SuppressSending delay%5ld; require%5ld", m->SuppressSending - m->timenow, (delayresponse + 49) / 50); -#endif - // Pick a random delay: - // We start with the base delay chosen above (typically either 1 second or 20 seconds), - // and add a random value in the range 0-5 seconds (making 1-6 seconds or 20-25 seconds). - // This is an integer value, with resolution determined by the platform clock rate. - // We then divide that by 50 to get the delay value in ticks. We defer the division until last - // to get better results on platforms with coarse clock granularity (e.g. ten ticks per second). - // The +49 before dividing is to ensure we round up, not down, to ensure that even - // on platforms where the native clock rate is less than fifty ticks per second, - // we still guarantee that the final calculated delay is at least one platform tick. - // We want to make sure we don't ever allow the delay to be zero ticks, - // because if that happens we'll fail the Bonjour Conformance Test. - // Our final computed delay is 20-120ms for normal delayed replies, - // or 400-500ms in the case of multi-packet known-answer lists. - m->SuppressSending = m->timenow + (delayresponse + (mDNSs32)mDNSRandom((mDNSu32)mDNSPlatformOneSecond*5) + 49) / 50; - if (m->SuppressSending == 0) m->SuppressSending = 1; -#if MDNS_LOG_ANSWER_SUPPRESSION_TIMES - if (oldss && delayresponse) - LogMsg("Set SuppressSending to %5ld", m->SuppressSending - m->timenow); -#endif - } - - // *** - // *** 9. If query is from a legacy client, or from a new client requesting a unicast reply, then generate a unicast response too - // *** - if (SendLegacyResponse) - responseptr = GenerateUnicastResponse(query, end, InterfaceID, LegacyQuery, response, ResponseRecords); - -exit: - m->rec.r.resrec.RecordType = 0; // Clear RecordType to show we're not still using it - - // *** - // *** 10. Finally, clear our link chains ready for use next time - // *** - while (ResponseRecords) - { - rr = ResponseRecords; - ResponseRecords = rr->NextResponse; - rr->NextResponse = mDNSNULL; - rr->NR_AnswerTo = mDNSNULL; - rr->NR_AdditionalTo = mDNSNULL; - } - - while (ExpectedAnswers) - { - CacheRecord *cr = ExpectedAnswers; - ExpectedAnswers = cr->NextInKAList; - cr->NextInKAList = mDNSNULL; - - // For non-truncated queries, we can definitively say that we should expect - // to be seeing a response for any records still left in the ExpectedAnswers list - if (!(query->h.flags.b[0] & kDNSFlag0_TC)) - if (cr->UnansweredQueries == 0 || m->timenow - cr->LastUnansweredTime >= mDNSPlatformOneSecond) - { - cr->UnansweredQueries++; - cr->LastUnansweredTime = m->timenow; -#if ENABLE_MULTI_PACKET_QUERY_SNOOPING - if (cr->UnansweredQueries > 1) - debugf("ProcessQuery: (!TC) UAQ %lu MPQ %lu MPKA %lu %s", - cr->UnansweredQueries, cr->MPUnansweredQ, cr->MPUnansweredKA, CRDisplayString(m, cr)); -#endif - SetNextCacheCheckTimeForRecord(m, cr); - } - - // If we've seen multiple unanswered queries for this record, - // then mark it to expire in five seconds if we don't get a response by then. - if (cr->UnansweredQueries >= MaxUnansweredQueries) - { -#if ENABLE_MULTI_PACKET_QUERY_SNOOPING - // Only show debugging message if this record was not about to expire anyway - if (RRExpireTime(cr) - m->timenow > 4 * mDNSPlatformOneSecond) - debugf("ProcessQuery: (Max) UAQ %lu MPQ %lu MPKA %lu mDNS_Reconfirm() for %s", - cr->UnansweredQueries, cr->MPUnansweredQ, cr->MPUnansweredKA, CRDisplayString(m, cr)); -#endif - mDNS_Reconfirm_internal(m, cr, kDefaultReconfirmTimeForNoAnswer); - } -#if ENABLE_MULTI_PACKET_QUERY_SNOOPING - // Make a guess, based on the multi-packet query / known answer counts, whether we think we - // should have seen an answer for this. (We multiply MPQ by 4 and MPKA by 5, to allow for - // possible packet loss of up to 20% of the additional KA packets.) - else if (cr->MPUnansweredQ * 4 > cr->MPUnansweredKA * 5 + 8) - { - // We want to do this conservatively. - // If there are so many machines on the network that they have to use multi-packet known-answer lists, - // then we don't want them to all hit the network simultaneously with their final expiration queries. - // By setting the record to expire in four minutes, we achieve two things: - // (a) the 90-95% final expiration queries will be less bunched together - // (b) we allow some time for us to witness enough other failed queries that we don't have to do our own - mDNSu32 remain = (mDNSu32)(RRExpireTime(cr) - m->timenow) / 4; - if (remain > 240 * (mDNSu32)mDNSPlatformOneSecond) - remain = 240 * (mDNSu32)mDNSPlatformOneSecond; - - // Only show debugging message if this record was not about to expire anyway - if (RRExpireTime(cr) - m->timenow > 4 * mDNSPlatformOneSecond) - debugf("ProcessQuery: (MPQ) UAQ %lu MPQ %lu MPKA %lu mDNS_Reconfirm() for %s", - cr->UnansweredQueries, cr->MPUnansweredQ, cr->MPUnansweredKA, CRDisplayString(m, cr)); - - if (remain <= 60 * (mDNSu32)mDNSPlatformOneSecond) - cr->UnansweredQueries++; // Treat this as equivalent to one definite unanswered query - cr->MPUnansweredQ = 0; // Clear MPQ/MPKA statistics - cr->MPUnansweredKA = 0; - cr->MPExpectingKA = mDNSfalse; - - if (remain < kDefaultReconfirmTimeForNoAnswer) - remain = kDefaultReconfirmTimeForNoAnswer; - mDNS_Reconfirm_internal(m, cr, remain); - } -#endif - } - - while (DupQuestions) - { - DNSQuestion *q = DupQuestions; - DupQuestions = q->NextInDQList; - q->NextInDQList = mDNSNULL; - i = RecordDupSuppressInfo(q->DupSuppress, m->timenow, InterfaceID, srcaddr->type); - debugf("ProcessQuery: Recorded DSI for %##s (%s) on %p/%s %d", q->qname.c, DNSTypeName(q->qtype), InterfaceID, - srcaddr->type == mDNSAddrType_IPv4 ? "v4" : "v6", i); - } - - return(responseptr); - } - -mDNSlocal void mDNSCoreReceiveQuery(mDNS *const m, const DNSMessage *const msg, const mDNSu8 *const end, - const mDNSAddr *srcaddr, const mDNSIPPort srcport, const mDNSAddr *dstaddr, mDNSIPPort dstport, - const mDNSInterfaceID InterfaceID) - { - mDNSu8 *responseend = mDNSNULL; - mDNSBool QueryWasLocalUnicast = srcaddr && dstaddr && - !mDNSAddrIsDNSMulticast(dstaddr) && mDNS_AddressIsLocalSubnet(m, InterfaceID, srcaddr); - - if (!InterfaceID && dstaddr && mDNSAddrIsDNSMulticast(dstaddr)) - { - LogMsg("Ignoring Query from %#-15a:%-5d to %#-15a:%-5d on 0x%p with " - "%2d Question%s %2d Answer%s %2d Authorit%s %2d Additional%s %d bytes (Multicast, but no InterfaceID)", - srcaddr, mDNSVal16(srcport), dstaddr, mDNSVal16(dstport), InterfaceID, - msg->h.numQuestions, msg->h.numQuestions == 1 ? ", " : "s,", - msg->h.numAnswers, msg->h.numAnswers == 1 ? ", " : "s,", - msg->h.numAuthorities, msg->h.numAuthorities == 1 ? "y, " : "ies,", - msg->h.numAdditionals, msg->h.numAdditionals == 1 ? " " : "s", end - msg->data); - return; - } - - verbosedebugf("Received Query from %#-15a:%-5d to %#-15a:%-5d on 0x%p with " - "%2d Question%s %2d Answer%s %2d Authorit%s %2d Additional%s %d bytes", - srcaddr, mDNSVal16(srcport), dstaddr, mDNSVal16(dstport), InterfaceID, - msg->h.numQuestions, msg->h.numQuestions == 1 ? ", " : "s,", - msg->h.numAnswers, msg->h.numAnswers == 1 ? ", " : "s,", - msg->h.numAuthorities, msg->h.numAuthorities == 1 ? "y, " : "ies,", - msg->h.numAdditionals, msg->h.numAdditionals == 1 ? " " : "s", end - msg->data); - - responseend = ProcessQuery(m, msg, end, srcaddr, InterfaceID, - !mDNSSameIPPort(srcport, MulticastDNSPort), mDNSAddrIsDNSMulticast(dstaddr), QueryWasLocalUnicast, &m->omsg); - - if (responseend) // If responseend is non-null, that means we built a unicast response packet - { - debugf("Unicast Response: %d Question%s, %d Answer%s, %d Additional%s to %#-15a:%d on %p/%ld", - m->omsg.h.numQuestions, m->omsg.h.numQuestions == 1 ? "" : "s", - m->omsg.h.numAnswers, m->omsg.h.numAnswers == 1 ? "" : "s", - m->omsg.h.numAdditionals, m->omsg.h.numAdditionals == 1 ? "" : "s", - srcaddr, mDNSVal16(srcport), InterfaceID, srcaddr->type); - mDNSSendDNSMessage(m, &m->omsg, responseend, InterfaceID, mDNSNULL, srcaddr, srcport, mDNSNULL, mDNSNULL); - } - } - -#if 0 -mDNSlocal mDNSBool TrustedSource(const mDNS *const m, const mDNSAddr *const srcaddr) - { - DNSServer *s; - (void)m; // Unused - (void)srcaddr; // Unused - for (s = m->DNSServers; s; s = s->next) - if (mDNSSameAddress(srcaddr, &s->addr)) return(mDNStrue); - return(mDNSfalse); - } -#endif - -struct UDPSocket_struct - { - mDNSIPPort port; // MUST BE FIRST FIELD -- mDNSCoreReceive expects every UDPSocket_struct to begin with mDNSIPPort port - }; - -mDNSlocal DNSQuestion *ExpectingUnicastResponseForQuestion(const mDNS *const m, const mDNSIPPort port, const mDNSOpaque16 id, const DNSQuestion *const question, mDNSBool tcp) - { - DNSQuestion *q; - for (q = m->Questions; q; q=q->next) - { - if (!tcp && !q->LocalSocket) continue; - if (mDNSSameIPPort(tcp ? q->tcpSrcPort : q->LocalSocket->port, port) && - mDNSSameOpaque16(q->TargetQID, id) && - q->qtype == question->qtype && - q->qclass == question->qclass && - q->qnamehash == question->qnamehash && - SameDomainName(&q->qname, &question->qname)) - return(q); - } - return(mDNSNULL); - } - -// This function is called when we receive a unicast response. This could be the case of a unicast response from the -// DNS server or a response to the QU query. Hence, the cache record's InterfaceId can be both NULL or non-NULL (QU case) -mDNSlocal DNSQuestion *ExpectingUnicastResponseForRecord(mDNS *const m, - const mDNSAddr *const srcaddr, const mDNSBool SrcLocal, const mDNSIPPort port, const mDNSOpaque16 id, const CacheRecord *const rr, mDNSBool tcp) - { - DNSQuestion *q; - (void)id; - (void)srcaddr; - - for (q = m->Questions; q; q=q->next) - { - if (!q->DuplicateOf && ResourceRecordAnswersUnicastResponse(&rr->resrec, q)) - { - if (!mDNSOpaque16IsZero(q->TargetQID)) - { - debugf("ExpectingUnicastResponseForRecord msg->h.id %d q->TargetQID %d for %s", mDNSVal16(id), mDNSVal16(q->TargetQID), CRDisplayString(m, rr)); - - if (mDNSSameOpaque16(q->TargetQID, id)) - { - mDNSIPPort srcp; - if (!tcp) - { - srcp = q->LocalSocket ? q->LocalSocket->port : zeroIPPort; - } - else - { - srcp = q->tcpSrcPort; - } - if (mDNSSameIPPort(srcp, port)) return(q); - - // if (mDNSSameAddress(srcaddr, &q->Target)) return(mDNStrue); - // if (q->LongLived && mDNSSameAddress(srcaddr, &q->servAddr)) return(mDNStrue); Shouldn't need this now that we have LLQType checking - // if (TrustedSource(m, srcaddr)) return(mDNStrue); - LogInfo("WARNING: Ignoring suspect uDNS response for %##s (%s) [q->Target %#a:%d] from %#a:%d %s", - q->qname.c, DNSTypeName(q->qtype), &q->Target, mDNSVal16(srcp), srcaddr, mDNSVal16(port), CRDisplayString(m, rr)); - return(mDNSNULL); - } - } - else - { - if (SrcLocal && q->ExpectUnicastResp && (mDNSu32)(m->timenow - q->ExpectUnicastResp) < (mDNSu32)(mDNSPlatformOneSecond*2)) - return(q); - } - } - } - return(mDNSNULL); - } - -// Certain data types need more space for in-memory storage than their in-packet rdlength would imply -// Currently this applies only to rdata types containing more than one domainname, -// or types where the domainname is not the last item in the structure. -// In addition, NSEC currently requires less space for in-memory storage than its in-packet representation. -mDNSlocal mDNSu16 GetRDLengthMem(const ResourceRecord *const rr) - { - switch (rr->rrtype) - { - case kDNSType_SOA: return sizeof(rdataSOA); - case kDNSType_RP: return sizeof(rdataRP); - case kDNSType_PX: return sizeof(rdataPX); - case kDNSType_NSEC:return sizeof(rdataNSEC); - default: return rr->rdlength; - } - } - -mDNSexport CacheRecord *CreateNewCacheEntry(mDNS *const m, const mDNSu32 slot, CacheGroup *cg, mDNSs32 delay) - { - CacheRecord *rr = mDNSNULL; - mDNSu16 RDLength = GetRDLengthMem(&m->rec.r.resrec); - - if (!m->rec.r.resrec.InterfaceID) debugf("CreateNewCacheEntry %s", CRDisplayString(m, &m->rec.r)); - - //if (RDLength > InlineCacheRDSize) - // LogInfo("Rdata len %4d > InlineCacheRDSize %d %s", RDLength, InlineCacheRDSize, CRDisplayString(m, &m->rec.r)); - - if (!cg) cg = GetCacheGroup(m, slot, &m->rec.r.resrec); // If we don't have a CacheGroup for this name, make one now - if (cg) rr = GetCacheRecord(m, cg, RDLength); // Make a cache record, being careful not to recycle cg - if (!rr) NoCacheAnswer(m, &m->rec.r); - else - { - RData *saveptr = rr->resrec.rdata; // Save the rr->resrec.rdata pointer - *rr = m->rec.r; // Block copy the CacheRecord object - rr->resrec.rdata = saveptr; // Restore rr->resrec.rdata after the structure assignment - rr->resrec.name = cg->name; // And set rr->resrec.name to point into our CacheGroup header - rr->DelayDelivery = delay; - - // If this is an oversized record with external storage allocated, copy rdata to external storage - if (rr->resrec.rdata == (RData*)&rr->smallrdatastorage && RDLength > InlineCacheRDSize) - LogMsg("rr->resrec.rdata == &rr->rdatastorage but length > InlineCacheRDSize %##s", m->rec.r.resrec.name->c); - else if (rr->resrec.rdata != (RData*)&rr->smallrdatastorage && RDLength <= InlineCacheRDSize) - LogMsg("rr->resrec.rdata != &rr->rdatastorage but length <= InlineCacheRDSize %##s", m->rec.r.resrec.name->c); - if (RDLength > InlineCacheRDSize) - mDNSPlatformMemCopy(rr->resrec.rdata, m->rec.r.resrec.rdata, sizeofRDataHeader + RDLength); - - rr->next = mDNSNULL; // Clear 'next' pointer - *(cg->rrcache_tail) = rr; // Append this record to tail of cache slot list - cg->rrcache_tail = &(rr->next); // Advance tail pointer - - CacheRecordAdd(m, rr); // CacheRecordAdd calls SetNextCacheCheckTimeForRecord(m, rr); for us - } - return(rr); - } - -mDNSlocal void RefreshCacheRecord(mDNS *const m, CacheRecord *rr, mDNSu32 ttl) - { - rr->TimeRcvd = m->timenow; - rr->resrec.rroriginalttl = ttl; - rr->UnansweredQueries = 0; -#if ENABLE_MULTI_PACKET_QUERY_SNOOPING - rr->MPUnansweredQ = 0; - rr->MPUnansweredKA = 0; - rr->MPExpectingKA = mDNSfalse; -#endif - SetNextCacheCheckTimeForRecord(m, rr); - } - -mDNSexport void GrantCacheExtensions(mDNS *const m, DNSQuestion *q, mDNSu32 lease) - { - CacheRecord *rr; - const mDNSu32 slot = HashSlot(&q->qname); - CacheGroup *cg = CacheGroupForName(m, slot, q->qnamehash, &q->qname); - for (rr = cg ? cg->members : mDNSNULL; rr; rr=rr->next) - if (rr->CRActiveQuestion == q) - { - //LogInfo("GrantCacheExtensions: new lease %d / %s", lease, CRDisplayString(m, rr)); - RefreshCacheRecord(m, rr, lease); - } - } - -mDNSlocal mDNSu32 GetEffectiveTTL(const uDNS_LLQType LLQType, mDNSu32 ttl) // TTL in seconds - { - if (LLQType == uDNS_LLQ_Entire) ttl = kLLQ_DefLease; - else if (LLQType == uDNS_LLQ_Events) - { - // If the TTL is -1 for uDNS LLQ event packet, that means "remove" - if (ttl == 0xFFFFFFFF) ttl = 0; - else ttl = kLLQ_DefLease; - } - else // else not LLQ (standard uDNS response) - { - // The TTL is already capped to a maximum value in GetLargeResourceRecord, but just to be extra safe we - // also do this check here to make sure we can't get overflow below when we add a quarter to the TTL - if (ttl > 0x60000000UL / mDNSPlatformOneSecond) ttl = 0x60000000UL / mDNSPlatformOneSecond; - - // Adjustment factor to avoid race condition: - // Suppose real record as TTL of 3600, and our local caching server has held it for 3500 seconds, so it returns an aged TTL of 100. - // If we do our normal refresh at 80% of the TTL, our local caching server will return 20 seconds, so we'll do another - // 80% refresh after 16 seconds, and then the server will return 4 seconds, and so on, in the fashion of Zeno's paradox. - // To avoid this, we extend the record's effective TTL to give it a little extra grace period. - // We adjust the 100 second TTL to 126. This means that when we do our 80% query at 101 seconds, - // the cached copy at our local caching server will already have expired, so the server will be forced - // to fetch a fresh copy from the authoritative server, and then return a fresh record with the full TTL of 3600 seconds. - ttl += ttl/4 + 2; - - // For mDNS, TTL zero means "delete this record" - // For uDNS, TTL zero means: this data is true at this moment, but don't cache it. - // For the sake of network efficiency, we impose a minimum effective TTL of 15 seconds. - // This means that we'll do our 80, 85, 90, 95% queries at 12.00, 12.75, 13.50, 14.25 seconds - // respectively, and then if we get no response, delete the record from the cache at 15 seconds. - // This gives the server up to three seconds to respond between when we send our 80% query at 12 seconds - // and when we delete the record at 15 seconds. Allowing cache lifetimes less than 15 seconds would - // (with the current code) result in the server having even less than three seconds to respond - // before we deleted the record and reported a "remove" event to any active questions. - // Furthermore, with the current code, if we were to allow a TTL of less than 2 seconds - // then things really break (e.g. we end up making a negative cache entry). - // In the future we may want to revisit this and consider properly supporting non-cached (TTL=0) uDNS answers. - if (ttl < 15) ttl = 15; - } - - return ttl; - } - -// Note: mDNSCoreReceiveResponse calls mDNS_Deregister_internal which can call a user callback, which may change -// the record list and/or question list. -// Any code walking either list must use the CurrentQuestion and/or CurrentRecord mechanism to protect against this. -// InterfaceID non-NULL tells us the interface this multicast response was received on -// InterfaceID NULL tells us this was a unicast response -// dstaddr NULL tells us we received this over an outgoing TCP connection we made -mDNSlocal void mDNSCoreReceiveResponse(mDNS *const m, - const DNSMessage *const response, const mDNSu8 *end, - const mDNSAddr *srcaddr, const mDNSIPPort srcport, const mDNSAddr *dstaddr, mDNSIPPort dstport, - const mDNSInterfaceID InterfaceID) - { - int i; - mDNSBool ResponseMCast = dstaddr && mDNSAddrIsDNSMulticast(dstaddr); - mDNSBool ResponseSrcLocal = !srcaddr || mDNS_AddressIsLocalSubnet(m, InterfaceID, srcaddr); - DNSQuestion *llqMatch = mDNSNULL; - uDNS_LLQType LLQType = uDNS_recvLLQResponse(m, response, end, srcaddr, srcport, &llqMatch); - - // "(CacheRecord*)1" is a special (non-zero) end-of-list marker - // We use this non-zero marker so that records in our CacheFlushRecords list will always have NextInCFList - // set non-zero, and that tells GetCacheEntity() that they're not, at this moment, eligible for recycling. - CacheRecord *CacheFlushRecords = (CacheRecord*)1; - CacheRecord **cfp = &CacheFlushRecords; - - // All records in a DNS response packet are treated as equally valid statements of truth. If we want - // to guard against spoof responses, then the only credible protection against that is cryptographic - // security, e.g. DNSSEC., not worring about which section in the spoof packet contained the record - int firstauthority = response->h.numAnswers; - int firstadditional = firstauthority + response->h.numAuthorities; - int totalrecords = firstadditional + response->h.numAdditionals; - const mDNSu8 *ptr = response->data; - DNSServer *uDNSServer = mDNSNULL; - - debugf("Received Response from %#-15a addressed to %#-15a on %p with " - "%2d Question%s %2d Answer%s %2d Authorit%s %2d Additional%s %d bytes LLQType %d", - srcaddr, dstaddr, InterfaceID, - response->h.numQuestions, response->h.numQuestions == 1 ? ", " : "s,", - response->h.numAnswers, response->h.numAnswers == 1 ? ", " : "s,", - response->h.numAuthorities, response->h.numAuthorities == 1 ? "y, " : "ies,", - response->h.numAdditionals, response->h.numAdditionals == 1 ? " " : "s", end - response->data, LLQType); - - // According to RFC 2181 - // When a DNS client receives a reply with TC - // set, it should ignore that response, and query again, using a - // mechanism, such as a TCP connection, that will permit larger replies. - // It feels wrong to be throwing away data after the network went to all the trouble of delivering it to us, but - // delivering some records of the RRSet first and then the remainder a couple of milliseconds later was causing - // failures in our Microsoft Active Directory client, which expects to get the entire set of answers at once. - // Can't bind to Active Directory - // In addition, if the client immediately canceled its query after getting the initial partial response, then we'll - // abort our TCP connection, and not complete the operation, and end up with an incomplete RRSet in our cache. - // Next time there's a query for this RRSet we'll see answers in our cache, and assume we have the whole RRSet already, - // and not even do the TCP query. - // Accordingly, if we get a uDNS reply with kDNSFlag0_TC set, we bail out and wait for the TCP response containing the entire RRSet. - if (!InterfaceID && (response->h.flags.b[0] & kDNSFlag0_TC)) return; - - if (LLQType == uDNS_LLQ_Ignore) return; - - // 1. We ignore questions (if any) in mDNS response packets - // 2. If this is an LLQ response, we handle it much the same - // 3. If we get a uDNS UDP response with the TC (truncated) bit set, then we can't treat this - // answer as being the authoritative complete RRSet, and respond by deleting all other - // matching cache records that don't appear in this packet. - // Otherwise, this is a authoritative uDNS answer, so arrange for any stale records to be purged - if (ResponseMCast || LLQType == uDNS_LLQ_Events || (response->h.flags.b[0] & kDNSFlag0_TC)) - ptr = LocateAnswers(response, end); - // Otherwise, for one-shot queries, any answers in our cache that are not also contained - // in this response packet are immediately deemed to be invalid. - else - { - mDNSu8 rcode = (mDNSu8)(response->h.flags.b[1] & kDNSFlag1_RC_Mask); - mDNSBool failure = !(rcode == kDNSFlag1_RC_NoErr || rcode == kDNSFlag1_RC_NXDomain || rcode == kDNSFlag1_RC_NotAuth); - mDNSBool returnEarly = mDNSfalse; - // We could possibly combine this with the similar loop at the end of this function -- - // instead of tagging cache records here and then rescuing them if we find them in the answer section, - // we could instead use the "m->PktNum" mechanism to tag each cache record with the packet number in - // which it was received (or refreshed), and then at the end if we find any cache records which - // answer questions in this packet's question section, but which aren't tagged with this packet's - // packet number, then we deduce they are old and delete them - for (i = 0; i < response->h.numQuestions && ptr && ptr < end; i++) - { - DNSQuestion q, *qptr = mDNSNULL; - ptr = getQuestion(response, ptr, end, InterfaceID, &q); - if (ptr && (qptr = ExpectingUnicastResponseForQuestion(m, dstport, response->h.id, &q, !dstaddr))) - { - if (!failure) - { - CacheRecord *rr; - const mDNSu32 slot = HashSlot(&q.qname); - CacheGroup *cg = CacheGroupForName(m, slot, q.qnamehash, &q.qname); - for (rr = cg ? cg->members : mDNSNULL; rr; rr=rr->next) - if (SameNameRecordAnswersQuestion(&rr->resrec, qptr)) - { - debugf("uDNS marking %p %##s (%s) %p %s", q.InterfaceID, q.qname.c, DNSTypeName(q.qtype), - rr->resrec.InterfaceID, CRDisplayString(m, rr)); - // Don't want to disturb rroriginalttl here, because code below might need it for the exponential backoff doubling algorithm - rr->TimeRcvd = m->timenow - TicksTTL(rr) - 1; - rr->UnansweredQueries = MaxUnansweredQueries; - } - } - else - { - if (qptr) - { - LogInfo("mDNSCoreReceiveResponse: Server %p responded with code %d to query %##s (%s)", qptr->qDNSServer, rcode, q.qname.c, DNSTypeName(q.qtype)); - PenalizeDNSServer(m, qptr); - } - returnEarly = mDNStrue; - } - } - } - if (returnEarly) - { - LogInfo("Ignoring %2d Answer%s %2d Authorit%s %2d Additional%s", - response->h.numAnswers, response->h.numAnswers == 1 ? ", " : "s,", - response->h.numAuthorities, response->h.numAuthorities == 1 ? "y, " : "ies,", - response->h.numAdditionals, response->h.numAdditionals == 1 ? "" : "s"); - // not goto exit because we won't have any CacheFlushRecords and we do not want to - // generate negative cache entries (we want to query the next server) - return; - } - } - - for (i = 0; i < totalrecords && ptr && ptr < end; i++) - { - // All responses sent via LL multicast are acceptable for caching - // All responses received over our outbound TCP connections are acceptable for caching - mDNSBool AcceptableResponse = ResponseMCast || !dstaddr || LLQType; - // (Note that just because we are willing to cache something, that doesn't necessarily make it a trustworthy answer - // to any specific question -- any code reading records from the cache needs to make that determination for itself.) - - const mDNSu8 RecordType = - (i < firstauthority ) ? (mDNSu8)kDNSRecordTypePacketAns : - (i < firstadditional) ? (mDNSu8)kDNSRecordTypePacketAuth : (mDNSu8)kDNSRecordTypePacketAdd; - ptr = GetLargeResourceRecord(m, response, ptr, end, InterfaceID, RecordType, &m->rec); - if (!ptr) goto exit; // Break out of the loop and clean up our CacheFlushRecords list before exiting - if (m->rec.r.resrec.RecordType == kDNSRecordTypePacketNegative) { m->rec.r.resrec.RecordType = 0; continue; } - - // Don't want to cache OPT or TSIG pseudo-RRs - if (m->rec.r.resrec.rrtype == kDNSType_TSIG) { m->rec.r.resrec.RecordType = 0; continue; } - if (m->rec.r.resrec.rrtype == kDNSType_OPT) - { - const rdataOPT *opt; - const rdataOPT *const e = (const rdataOPT *)&m->rec.r.resrec.rdata->u.data[m->rec.r.resrec.rdlength]; - // Find owner sub-option(s). We verify that the MAC is non-zero, otherwise we could inadvertently - // delete all our own AuthRecords (which are identified by having zero MAC tags on them). - for (opt = &m->rec.r.resrec.rdata->u.opt[0]; opt < e; opt++) - if (opt->opt == kDNSOpt_Owner && opt->u.owner.vers == 0 && opt->u.owner.HMAC.l[0]) - { - ClearProxyRecords(m, &opt->u.owner, m->DuplicateRecords); - ClearProxyRecords(m, &opt->u.owner, m->ResourceRecords); - } - m->rec.r.resrec.RecordType = 0; - continue; - } - - // if a CNAME record points to itself, then don't add it to the cache - if ((m->rec.r.resrec.rrtype == kDNSType_CNAME) && SameDomainName(m->rec.r.resrec.name, &m->rec.r.resrec.rdata->u.name)) - { - LogInfo("mDNSCoreReceiveResponse: CNAME loop domain name %##s", m->rec.r.resrec.name->c); - m->rec.r.resrec.RecordType = 0; - continue; - } - - // When we receive uDNS LLQ responses, we assume a long cache lifetime -- - // In the case of active LLQs, we'll get remove events when the records actually do go away - // In the case of polling LLQs, we assume the record remains valid until the next poll - if (!mDNSOpaque16IsZero(response->h.id)) - m->rec.r.resrec.rroriginalttl = GetEffectiveTTL(LLQType, m->rec.r.resrec.rroriginalttl); - - // If response was not sent via LL multicast, - // then see if it answers a recent query of ours, which would also make it acceptable for caching. - if (!ResponseMCast) - { - if (LLQType) - { - // For Long Lived queries that are both sent over UDP and Private TCP, LLQType is set. - // Even though it is AcceptableResponse, we need a matching DNSServer pointer for the - // queries to get ADD/RMV events. To lookup the question, we can't use - // ExpectingUnicastResponseForRecord as the port numbers don't match. uDNS_recvLLQRespose - // has already matched the question using the 64 bit Id in the packet and we use that here. - - if (llqMatch != mDNSNULL) m->rec.r.resrec.rDNSServer = uDNSServer = llqMatch->qDNSServer; - } - else if (!AcceptableResponse || !dstaddr) - { - // For responses that come over TCP (Responses that can't fit within UDP) or TLS (Private queries - // that are not long lived e.g., AAAA lookup in a Private domain), it is indicated by !dstaddr. - // Even though it is AcceptableResponse, we still need a DNSServer pointer for the resource records that - // we create. - - DNSQuestion *q = ExpectingUnicastResponseForRecord(m, srcaddr, ResponseSrcLocal, dstport, response->h.id, &m->rec.r, !dstaddr); - - // Intialize the DNS server on the resource record which will now filter what questions we answer with - // this record. - // - // We could potentially lookup the DNS server based on the source address, but that may not work always - // and that's why ExpectingUnicastResponseForRecord does not try to verify whether the response came - // from the DNS server that queried. We follow the same logic here. If we can find a matching quetion based - // on the "id" and "source port", then this response answers the question and assume the response - // came from the same DNS server that we sent the query to. - - if (q != mDNSNULL) - { - AcceptableResponse = mDNStrue; - if (!InterfaceID) - { - debugf("mDNSCoreReceiveResponse: InterfaceID %p %##s (%s)", q->InterfaceID, q->qname.c, DNSTypeName(q->qtype)); - m->rec.r.resrec.rDNSServer = uDNSServer = q->qDNSServer; - } - } - else - { - // If we can't find a matching question, we need to see whether we have seen records earlier that matched - // the question. The code below does that. So, make this record unacceptable for now - if (!InterfaceID) - { - debugf("mDNSCoreReceiveResponse: Can't find question for record name %##s", m->rec.r.resrec.name->c); - AcceptableResponse = mDNSfalse; - } - } - } - } - - // 1. Check that this packet resource record does not conflict with any of ours - if (mDNSOpaque16IsZero(response->h.id) && m->rec.r.resrec.rrtype != kDNSType_NSEC) - { - if (m->CurrentRecord) - LogMsg("mDNSCoreReceiveResponse ERROR m->CurrentRecord already set %s", ARDisplayString(m, m->CurrentRecord)); - m->CurrentRecord = m->ResourceRecords; - while (m->CurrentRecord) - { - AuthRecord *rr = m->CurrentRecord; - m->CurrentRecord = rr->next; - // We accept all multicast responses, and unicast responses resulting from queries we issued - // For other unicast responses, this code accepts them only for responses with an - // (apparently) local source address that pertain to a record of our own that's in probing state - if (!AcceptableResponse && !(ResponseSrcLocal && rr->resrec.RecordType == kDNSRecordTypeUnique)) continue; - - if (PacketRRMatchesSignature(&m->rec.r, rr)) // If interface, name, type (if shared record) and class match... - { - // ... check to see if type and rdata are identical - if (IdenticalSameNameRecord(&m->rec.r.resrec, &rr->resrec)) - { - // If the RR in the packet is identical to ours, just check they're not trying to lower the TTL on us - if (m->rec.r.resrec.rroriginalttl >= rr->resrec.rroriginalttl/2 || m->SleepState) - { - // If we were planning to send on this -- and only this -- interface, then we don't need to any more - if (rr->ImmedAnswer == InterfaceID) { rr->ImmedAnswer = mDNSNULL; rr->ImmedUnicast = mDNSfalse; } - } - else - { - if (rr->ImmedAnswer == mDNSNULL) { rr->ImmedAnswer = InterfaceID; m->NextScheduledResponse = m->timenow; } - else if (rr->ImmedAnswer != InterfaceID) { rr->ImmedAnswer = mDNSInterfaceMark; m->NextScheduledResponse = m->timenow; } - } - } - // else, the packet RR has different type or different rdata -- check to see if this is a conflict - else if (m->rec.r.resrec.rroriginalttl > 0 && PacketRRConflict(m, rr, &m->rec.r)) - { - LogInfo("mDNSCoreReceiveResponse: Pkt Record: %08lX %s", m->rec.r.resrec.rdatahash, CRDisplayString(m, &m->rec.r)); - LogInfo("mDNSCoreReceiveResponse: Our Record: %08lX %s", rr-> resrec.rdatahash, ARDisplayString(m, rr)); - - // If this record is marked DependentOn another record for conflict detection purposes, - // then *that* record has to be bumped back to probing state to resolve the conflict - if (rr->DependentOn) - { - while (rr->DependentOn) rr = rr->DependentOn; - LogInfo("mDNSCoreReceiveResponse: Dep Record: %08lX %s", rr-> resrec.rdatahash, ARDisplayString(m, rr)); - } - - // If we've just whacked this record's ProbeCount, don't need to do it again - if (rr->ProbeCount > DefaultProbeCountForTypeUnique) - LogInfo("mDNSCoreReceiveResponse: Already reset to Probing: %s", ARDisplayString(m, rr)); - else if (rr->ProbeCount == DefaultProbeCountForTypeUnique) - LogMsg("mDNSCoreReceiveResponse: Ignoring response received before we even began probing: %s", ARDisplayString(m, rr)); - else - { - LogMsg("mDNSCoreReceiveResponse: Received from %#a:%d %s", srcaddr, mDNSVal16(srcport), CRDisplayString(m, &m->rec.r)); - // If we'd previously verified this record, put it back to probing state and try again - if (rr->resrec.RecordType == kDNSRecordTypeVerified) - { - LogMsg("mDNSCoreReceiveResponse: Resetting to Probing: %s", ARDisplayString(m, rr)); - rr->resrec.RecordType = kDNSRecordTypeUnique; - // We set ProbeCount to one more than the usual value so we know we've already touched this record. - // This is because our single probe for "example-name.local" could yield a response with (say) two A records and - // three AAAA records in it, and we don't want to call RecordProbeFailure() five times and count that as five conflicts. - // This special value is recognised and reset to DefaultProbeCountForTypeUnique in SendQueries(). - rr->ProbeCount = DefaultProbeCountForTypeUnique + 1; - rr->AnnounceCount = InitialAnnounceCount; - InitializeLastAPTime(m, rr); - RecordProbeFailure(m, rr); // Repeated late conflicts also cause us to back off to the slower probing rate - } - // If we're probing for this record, we just failed - else if (rr->resrec.RecordType == kDNSRecordTypeUnique) - { - LogMsg("mDNSCoreReceiveResponse: ProbeCount %d; will deregister %s", rr->ProbeCount, ARDisplayString(m, rr)); - mDNS_Deregister_internal(m, rr, mDNS_Dereg_conflict); - } - // We assumed this record must be unique, but we were wrong. (e.g. There are two mDNSResponders on the - // same machine giving different answers for the reverse mapping record, or there are two machines on the - // network using the same IP address.) This is simply a misconfiguration, and there's nothing we can do - // to fix it -- e.g. it's not our job to be trying to change the machine's IP address. We just discard our - // record to avoid continued conflicts (as we do for a conflict on our Unique records) and get on with life. - else if (rr->resrec.RecordType == kDNSRecordTypeKnownUnique) - { - LogMsg("mDNSCoreReceiveResponse: Unexpected conflict discarding %s", ARDisplayString(m, rr)); - mDNS_Deregister_internal(m, rr, mDNS_Dereg_conflict); - } - else - LogMsg("mDNSCoreReceiveResponse: Unexpected record type %X %s", rr->resrec.RecordType, ARDisplayString(m, rr)); - } - } - // Else, matching signature, different type or rdata, but not a considered a conflict. - // If the packet record has the cache-flush bit set, then we check to see if we - // have any record(s) of the same type that we should re-assert to rescue them - // (see note about "multi-homing and bridged networks" at the end of this function). - else if (m->rec.r.resrec.rrtype == rr->resrec.rrtype) - if ((m->rec.r.resrec.RecordType & kDNSRecordTypePacketUniqueMask) && m->timenow - rr->LastMCTime > mDNSPlatformOneSecond/2) - { rr->ImmedAnswer = mDNSInterfaceMark; m->NextScheduledResponse = m->timenow; } - } - } - } - - if (!AcceptableResponse) - { - const CacheRecord *cr; - for (cr = CacheFlushRecords; cr != (CacheRecord*)1; cr = cr->NextInCFList) - { - domainname *target = GetRRDomainNameTarget(&cr->resrec); - // When we issue a query for A record, the response might contain both a CNAME and A records. Only the CNAME would - // match the question and we already created a cache entry in the previous pass of this loop. Now when we process - // the A record, it does not match the question because the record name here is the CNAME. Hence we try to - // match with the previous records to make it an AcceptableResponse. We have to be careful about setting the - // DNSServer value that we got in the previous pass. This can happen for other record types like SRV also. - - if (target && cr->resrec.rdatahash == m->rec.r.resrec.namehash && SameDomainName(target, m->rec.r.resrec.name)) - { - debugf("mDNSCoreReceiveResponse: Found a matching entry for %##s in the CacheFlushRecords", m->rec.r.resrec.name->c); - AcceptableResponse = mDNStrue; - m->rec.r.resrec.rDNSServer = uDNSServer; - break; - } - } - } - - // 2. See if we want to add this packet resource record to our cache - // We only try to cache answers if we have a cache to put them in - // Also, we ignore any apparent attempts at cache poisoning unicast to us that do not answer any outstanding active query - if (!AcceptableResponse) LogInfo("mDNSCoreReceiveResponse ignoring %s", CRDisplayString(m, &m->rec.r)); - if (m->rrcache_size && AcceptableResponse) - { - const mDNSu32 slot = HashSlot(m->rec.r.resrec.name); - CacheGroup *cg = CacheGroupForRecord(m, slot, &m->rec.r.resrec); - CacheRecord *rr; - - // 2a. Check if this packet resource record is already in our cache - for (rr = cg ? cg->members : mDNSNULL; rr; rr=rr->next) - { - mDNSBool match = !InterfaceID ? m->rec.r.resrec.rDNSServer == rr->resrec.rDNSServer : rr->resrec.InterfaceID == InterfaceID; - // If we found this exact resource record, refresh its TTL - if (match && IdenticalSameNameRecord(&m->rec.r.resrec, &rr->resrec)) - { - if (m->rec.r.resrec.rdlength > InlineCacheRDSize) - verbosedebugf("Found record size %5d interface %p already in cache: %s", - m->rec.r.resrec.rdlength, InterfaceID, CRDisplayString(m, &m->rec.r)); - - if (m->rec.r.resrec.RecordType & kDNSRecordTypePacketUniqueMask) - { - // If this packet record has the kDNSClass_UniqueRRSet flag set, then add it to our cache flushing list - if (rr->NextInCFList == mDNSNULL && cfp != &rr->NextInCFList && LLQType != uDNS_LLQ_Events) - { *cfp = rr; cfp = &rr->NextInCFList; *cfp = (CacheRecord*)1; } - - // If this packet record is marked unique, and our previous cached copy was not, then fix it - if (!(rr->resrec.RecordType & kDNSRecordTypePacketUniqueMask)) - { - DNSQuestion *q; - for (q = m->Questions; q; q=q->next) if (ResourceRecordAnswersQuestion(&rr->resrec, q)) q->UniqueAnswers++; - rr->resrec.RecordType = m->rec.r.resrec.RecordType; - } - } - - if (!SameRDataBody(&m->rec.r.resrec, &rr->resrec.rdata->u, SameDomainNameCS)) - { - // If the rdata of the packet record differs in name capitalization from the record in our cache - // then mDNSPlatformMemSame will detect this. In this case, throw the old record away, so that clients get - // a 'remove' event for the record with the old capitalization, and then an 'add' event for the new one. - // mDNS -F returns the same domain multiple times with different casing - rr->resrec.rroriginalttl = 0; - rr->TimeRcvd = m->timenow; - rr->UnansweredQueries = MaxUnansweredQueries; - SetNextCacheCheckTimeForRecord(m, rr); - LogInfo("Discarding due to domainname case change old: %s", CRDisplayString(m,rr)); - LogInfo("Discarding due to domainname case change new: %s", CRDisplayString(m,&m->rec.r)); - LogInfo("Discarding due to domainname case change in %d slot %3d in %d %d", - NextCacheCheckEvent(rr) - m->timenow, slot, m->rrcache_nextcheck[slot] - m->timenow, m->NextCacheCheck - m->timenow); - // DO NOT break out here -- we want to continue as if we never found it - } - else if (m->rec.r.resrec.rroriginalttl > 0) - { - DNSQuestion *q; - //if (rr->resrec.rroriginalttl == 0) LogMsg("uDNS rescuing %s", CRDisplayString(m, rr)); - RefreshCacheRecord(m, rr, m->rec.r.resrec.rroriginalttl); - - // We have to reset the question interval to MaxQuestionInterval so that we don't keep - // polling the network once we get a valid response back. For the first time when a new - // cache entry is created, AnswerCurrentQuestionWithResourceRecord does that. - // Subsequently, if we reissue questions from within the mDNSResponder e.g., DNS server - // configuration changed, without flushing the cache, we reset the question interval here. - // Currently, we do this for for both multicast and unicast questions as long as the record - // type is unique. For unicast, resource record is always unique and for multicast it is - // true for records like A etc. but not for PTR. - if (rr->resrec.RecordType & kDNSRecordTypePacketUniqueMask) - { - for (q = m->Questions; q; q=q->next) - { - if (!q->DuplicateOf && !q->LongLived && - ActiveQuestion(q) && ResourceRecordAnswersQuestion(&rr->resrec, q)) - { - ResetQuestionState(m, q); - debugf("mDNSCoreReceiveResponse: Set MaxQuestionInterval for %p %##s (%s)", q, q->qname.c, DNSTypeName(q->qtype)); - break; // Why break here? Aren't there other questions we might want to look at?-- SC July 2010 - } - } - } - break; - } - else - { - // If the packet TTL is zero, that means we're deleting this record. - // To give other hosts on the network a chance to protest, we push the deletion - // out one second into the future. Also, we set UnansweredQueries to MaxUnansweredQueries. - // Otherwise, we'll do final queries for this record at 80% and 90% of its apparent - // lifetime (800ms and 900ms from now) which is a pointless waste of network bandwidth. - // If record's current expiry time is more than a second from now, we set it to expire in one second. - // If the record is already going to expire in less than one second anyway, we leave it alone -- - // we don't want to let the goodbye packet *extend* the record's lifetime in our cache. - debugf("DE for %s", CRDisplayString(m, rr)); - if (RRExpireTime(rr) - m->timenow > mDNSPlatformOneSecond) - { - rr->resrec.rroriginalttl = 1; - rr->TimeRcvd = m->timenow; - rr->UnansweredQueries = MaxUnansweredQueries; - SetNextCacheCheckTimeForRecord(m, rr); - } - break; - } - } - } - - // If packet resource record not in our cache, add it now - // (unless it is just a deletion of a record we never had, in which case we don't care) - if (!rr && m->rec.r.resrec.rroriginalttl > 0) - { - const mDNSBool AddToCFList = (m->rec.r.resrec.RecordType & kDNSRecordTypePacketUniqueMask) && (LLQType != uDNS_LLQ_Events); - const mDNSs32 delay = AddToCFList ? NonZeroTime(m->timenow + mDNSPlatformOneSecond) : - CheckForSoonToExpireRecords(m, m->rec.r.resrec.name, m->rec.r.resrec.namehash, slot); - // If unique, assume we may have to delay delivery of this 'add' event. - // Below, where we walk the CacheFlushRecords list, we either call CacheRecordDeferredAdd() - // to immediately to generate answer callbacks, or we call ScheduleNextCacheCheckTime() - // to schedule an mDNS_Execute task at the appropriate time. - rr = CreateNewCacheEntry(m, slot, cg, delay); - if (rr) - { - if (AddToCFList) { *cfp = rr; cfp = &rr->NextInCFList; *cfp = (CacheRecord*)1; } - else if (rr->DelayDelivery) ScheduleNextCacheCheckTime(m, slot, rr->DelayDelivery); - } - } - } - m->rec.r.resrec.RecordType = 0; // Clear RecordType to show we're not still using it - } - -exit: - m->rec.r.resrec.RecordType = 0; // Clear RecordType to show we're not still using it - - // If we've just received one or more records with their cache flush bits set, - // then scan that cache slot to see if there are any old stale records we need to flush - while (CacheFlushRecords != (CacheRecord*)1) - { - CacheRecord *r1 = CacheFlushRecords, *r2; - const mDNSu32 slot = HashSlot(r1->resrec.name); - const CacheGroup *cg = CacheGroupForRecord(m, slot, &r1->resrec); - CacheFlushRecords = CacheFlushRecords->NextInCFList; - r1->NextInCFList = mDNSNULL; - - // Look for records in the cache with the same signature as this new one with the cache flush - // bit set, and either (a) if they're fresh, just make sure the whole RRSet has the same TTL - // (as required by DNS semantics) or (b) if they're old, mark them for deletion in one second. - // We make these TTL adjustments *only* for records that still have *more* than one second - // remaining to live. Otherwise, a record that we tagged for deletion half a second ago - // (and now has half a second remaining) could inadvertently get its life extended, by either - // (a) if we got an explicit goodbye packet half a second ago, the record would be considered - // "fresh" and would be incorrectly resurrected back to the same TTL as the rest of the RRSet, - // or (b) otherwise, the record would not be fully resurrected, but would be reset to expire - // in one second, thereby inadvertently delaying its actual expiration, instead of hastening it. - // If this were to happen repeatedly, the record's expiration could be deferred indefinitely. - // To avoid this, we need to ensure that the cache flushing operation will only act to - // *decrease* a record's remaining lifetime, never *increase* it. - for (r2 = cg ? cg->members : mDNSNULL; r2; r2=r2->next) - // For Unicast (null InterfaceID) the DNSservers should also match - if ((r1->resrec.InterfaceID == r2->resrec.InterfaceID) && - (r1->resrec.InterfaceID || (r1->resrec.rDNSServer == r2->resrec.rDNSServer)) && - r1->resrec.rrtype == r2->resrec.rrtype && - r1->resrec.rrclass == r2->resrec.rrclass) - { - // If record is recent, just ensure the whole RRSet has the same TTL (as required by DNS semantics) - // else, if record is old, mark it to be flushed - if (m->timenow - r2->TimeRcvd < mDNSPlatformOneSecond && RRExpireTime(r2) - m->timenow > mDNSPlatformOneSecond) - { - // If we find mismatched TTLs in an RRSet, correct them. - // We only do this for records with a TTL of 2 or higher. It's possible to have a - // goodbye announcement with the cache flush bit set (or a case-change on record rdata, - // which we treat as a goodbye followed by an addition) and in that case it would be - // inappropriate to synchronize all the other records to a TTL of 0 (or 1). - // We suppress the message for the specific case of correcting from 240 to 60 for type TXT, - // because certain early Bonjour devices are known to have this specific mismatch, and - // there's no point filling syslog with messages about something we already know about. - // We also don't log this for uDNS responses, since a caching name server is obliged - // to give us an aged TTL to correct for how long it has held the record, - // so our received TTLs are expected to vary in that case - if (r2->resrec.rroriginalttl != r1->resrec.rroriginalttl && r1->resrec.rroriginalttl > 1) - { - if (!(r2->resrec.rroriginalttl == 240 && r1->resrec.rroriginalttl == 60 && r2->resrec.rrtype == kDNSType_TXT) && - mDNSOpaque16IsZero(response->h.id)) - LogInfo("Correcting TTL from %4d to %4d for %s", - r2->resrec.rroriginalttl, r1->resrec.rroriginalttl, CRDisplayString(m, r2)); - r2->resrec.rroriginalttl = r1->resrec.rroriginalttl; - } - r2->TimeRcvd = m->timenow; - } - else // else, if record is old, mark it to be flushed - { - verbosedebugf("Cache flush new %p age %d expire in %d %s", r1, m->timenow - r1->TimeRcvd, RRExpireTime(r1) - m->timenow, CRDisplayString(m, r1)); - verbosedebugf("Cache flush old %p age %d expire in %d %s", r2, m->timenow - r2->TimeRcvd, RRExpireTime(r2) - m->timenow, CRDisplayString(m, r2)); - // We set stale records to expire in one second. - // This gives the owner a chance to rescue it if necessary. - // This is important in the case of multi-homing and bridged networks: - // Suppose host X is on Ethernet. X then connects to an AirPort base station, which happens to be - // bridged onto the same Ethernet. When X announces its AirPort IP address with the cache-flush bit - // set, the AirPort packet will be bridged onto the Ethernet, and all other hosts on the Ethernet - // will promptly delete their cached copies of the (still valid) Ethernet IP address record. - // By delaying the deletion by one second, we give X a change to notice that this bridging has - // happened, and re-announce its Ethernet IP address to rescue it from deletion from all our caches. - - // We set UnansweredQueries to MaxUnansweredQueries to avoid expensive and unnecessary - // final expiration queries for this record. - - // If a record is deleted twice, first with an explicit DE record, then a second time by virtue of the cache - // flush bit on the new record replacing it, then we allow the record to be deleted immediately, without the usual - // one-second grace period. This improves responsiveness for mDNS_Update(), as used for things like iChat status updates. - // Updating TXT records is too slow - // We check for "rroriginalttl == 1" because we want to include records tagged by the "packet TTL is zero" check above, - // which sets rroriginalttl to 1, but not records tagged by the rdata case-change check, which sets rroriginalttl to 0. - if (r2->TimeRcvd == m->timenow && r2->resrec.rroriginalttl == 1 && r2->UnansweredQueries == MaxUnansweredQueries) - { - LogInfo("Cache flush for DE record %s", CRDisplayString(m, r2)); - r2->resrec.rroriginalttl = 0; - } - else if (RRExpireTime(r2) - m->timenow > mDNSPlatformOneSecond) - { - // We only set a record to expire in one second if it currently has *more* than a second to live - // If it's already due to expire in a second or less, we just leave it alone - r2->resrec.rroriginalttl = 1; - r2->UnansweredQueries = MaxUnansweredQueries; - r2->TimeRcvd = m->timenow - 1; - // We use (m->timenow - 1) instead of m->timenow, because we use that to identify records - // that we marked for deletion via an explicit DE record - } - } - SetNextCacheCheckTimeForRecord(m, r2); - } - - if (r1->DelayDelivery) // If we were planning to delay delivery of this record, see if we still need to - { - r1->DelayDelivery = CheckForSoonToExpireRecords(m, r1->resrec.name, r1->resrec.namehash, slot); - // If no longer delaying, deliver answer now, else schedule delivery for the appropriate time - if (!r1->DelayDelivery) CacheRecordDeferredAdd(m, r1); - else ScheduleNextCacheCheckTime(m, slot, r1->DelayDelivery); - } - } - - // See if we need to generate negative cache entries for unanswered unicast questions - ptr = response->data; - for (i = 0; i < response->h.numQuestions && ptr && ptr < end; i++) - { - DNSQuestion q; - DNSQuestion *qptr = mDNSNULL; - ptr = getQuestion(response, ptr, end, InterfaceID, &q); - if (ptr && (qptr = ExpectingUnicastResponseForQuestion(m, dstport, response->h.id, &q, !dstaddr))) - { - CacheRecord *rr, *neg = mDNSNULL; - mDNSu32 slot = HashSlot(&q.qname); - CacheGroup *cg = CacheGroupForName(m, slot, q.qnamehash, &q.qname); - for (rr = cg ? cg->members : mDNSNULL; rr; rr=rr->next) - if (SameNameRecordAnswersQuestion(&rr->resrec, qptr)) - { - // 1. If we got a fresh answer to this query, then don't need to generate a negative entry - if (RRExpireTime(rr) - m->timenow > 0) break; - // 2. If we already had a negative entry, keep track of it so we can resurrect it instead of creating a new one - if (rr->resrec.RecordType == kDNSRecordTypePacketNegative) neg = rr; - } - // When we're doing parallel unicast and multicast queries for dot-local names (for supporting Microsoft - // Active Directory sites) we don't want to waste memory making negative cache entries for all the unicast answers. - // Otherwise we just fill up our cache with negative entries for just about every single multicast name we ever look up - // (since the Microsoft Active Directory server is going to assert that pretty much every single multicast name doesn't exist). - // This is not only a waste of memory, but there's also the problem of those negative entries confusing us later -- e.g. we - // suppress sending our mDNS query packet because we think we already have a valid (negative) answer to that query in our cache. - // The one exception is that we *DO* want to make a negative cache entry for "local. SOA", for the (common) case where we're - // *not* on a Microsoft Active Directory network, and there is no authoritative server for "local". Note that this is not - // in conflict with the mDNS spec, because that spec says, "Multicast DNS Zones have no SOA record," so it's okay to cache - // negative answers for "local. SOA" from a uDNS server, because the mDNS spec already says that such records do not exist :-) - if (!InterfaceID && q.qtype != kDNSType_SOA && IsLocalDomain(&q.qname)) - { - // If we did not find a positive answer and we can append search domains to this question, - // generate a negative response (without creating a cache entry) to append search domains. - if (qptr->AppendSearchDomains && !rr) - { - LogInfo("mDNSCoreReceiveResponse: Generate negative response for %##s (%s)", q.qname.c, DNSTypeName(q.qtype)); - m->CurrentQuestion = qptr; - GenerateNegativeResponse(m); - m->CurrentQuestion = mDNSNULL; - } - else LogInfo("mDNSCoreReceiveResponse: Skipping check to see if we need to generate a negative cache entry for %##s (%s)", q.qname.c, DNSTypeName(q.qtype)); - } - else - { - if (!rr) - { - // We start off assuming a negative caching TTL of 60 seconds - // but then look to see if we can find an SOA authority record to tell us a better value we should be using - mDNSu32 negttl = 60; - int repeat = 0; - const domainname *name = &q.qname; - mDNSu32 hash = q.qnamehash; - - // Special case for our special Microsoft Active Directory "local SOA" check. - // Some cheap home gateways don't include an SOA record in the authority section when - // they send negative responses, so we don't know how long to cache the negative result. - // Because we don't want to keep hitting the root name servers with our query to find - // if we're on a network using Microsoft Active Directory using "local" as a private - // internal top-level domain, we make sure to cache the negative result for at least one day. - if (q.qtype == kDNSType_SOA && SameDomainName(&q.qname, &localdomain)) negttl = 60 * 60 * 24; - - // If we're going to make (or update) a negative entry, then look for the appropriate TTL from the SOA record - if (response->h.numAuthorities && (ptr = LocateAuthorities(response, end)) != mDNSNULL) - { - ptr = GetLargeResourceRecord(m, response, ptr, end, InterfaceID, kDNSRecordTypePacketAuth, &m->rec); - if (ptr && m->rec.r.resrec.RecordType != kDNSRecordTypePacketNegative && m->rec.r.resrec.rrtype == kDNSType_SOA) - { - const rdataSOA *const soa = (const rdataSOA *)m->rec.r.resrec.rdata->u.data; - mDNSu32 ttl_s = soa->min; - // We use the lesser of the SOA.MIN field and the SOA record's TTL, *except* - // for the SOA record for ".", where the record is reported as non-cacheable - // (TTL zero) for some reason, so in this case we just take the SOA record's TTL as-is - if (ttl_s > m->rec.r.resrec.rroriginalttl && m->rec.r.resrec.name->c[0]) - ttl_s = m->rec.r.resrec.rroriginalttl; - if (negttl < ttl_s) negttl = ttl_s; - - // Special check for SOA queries: If we queried for a.b.c.d.com, and got no answer, - // with an Authority Section SOA record for d.com, then this is a hint that the authority - // is d.com, and consequently SOA records b.c.d.com and c.d.com don't exist either. - // To do this we set the repeat count so the while loop below will make a series of negative cache entries for us - if (q.qtype == kDNSType_SOA) - { - int qcount = CountLabels(&q.qname); - int scount = CountLabels(m->rec.r.resrec.name); - if (qcount - 1 > scount) - if (SameDomainName(SkipLeadingLabels(&q.qname, qcount - scount), m->rec.r.resrec.name)) - repeat = qcount - 1 - scount; - } - } - m->rec.r.resrec.RecordType = 0; // Clear RecordType to show we're not still using it - } - - // If we already had a negative entry in the cache, then we double our existing negative TTL. This is to avoid - // the case where the record doesn't exist (e.g. particularly for things like our lb._dns-sd._udp. query), - // and the server returns no SOA record (or an SOA record with a small MIN TTL) so we assume a TTL - // of 60 seconds, and we end up polling the server every minute for a record that doesn't exist. - // With this fix in place, when this happens, we double the effective TTL each time (up to one hour), - // so that we back off our polling rate and don't keep hitting the server continually. - if (neg) - { - if (negttl < neg->resrec.rroriginalttl * 2) - negttl = neg->resrec.rroriginalttl * 2; - if (negttl > 3600) - negttl = 3600; - } - - negttl = GetEffectiveTTL(LLQType, negttl); // Add 25% grace period if necessary - - // If we already had a negative cache entry just update it, else make one or more new negative cache entries - if (neg) - { - debugf("mDNSCoreReceiveResponse: Renewing negative TTL from %d to %d %s", neg->resrec.rroriginalttl, negttl, CRDisplayString(m, neg)); - RefreshCacheRecord(m, neg, negttl); - // When we created the cache for the first time and answered the question, the question's - // interval was set to MaxQuestionInterval. If the cache is about to expire and we are resending - // the queries, the interval should still be at MaxQuestionInterval. If the query is being - // restarted (setting it to InitialQuestionInterval) for other reasons e.g., wakeup, - // we should reset its question interval here to MaxQuestionInterval. - ResetQuestionState(m, qptr); - } - else while (1) - { - debugf("mDNSCoreReceiveResponse making negative cache entry TTL %d for %##s (%s)", negttl, name->c, DNSTypeName(q.qtype)); - MakeNegativeCacheRecord(m, &m->rec.r, name, hash, q.qtype, q.qclass, negttl, mDNSInterface_Any, qptr->qDNSServer); - CreateNewCacheEntry(m, slot, cg, 0); // We never need any delivery delay for these generated negative cache records - m->rec.r.resrec.RecordType = 0; // Clear RecordType to show we're not still using it - if (!repeat) break; - repeat--; - name = (const domainname *)(name->c + 1 + name->c[0]); - hash = DomainNameHashValue(name); - slot = HashSlot(name); - cg = CacheGroupForName(m, slot, hash, name); - } - } - } - } - } - } - -// ScheduleWakeup causes all proxy records with WakeUp.HMAC matching mDNSEthAddr 'e' to be deregistered, causing -// multiple wakeup magic packets to be sent if appropriate, and all records to be ultimately freed after a few seconds. -// ScheduleWakeup is called on mDNS record conflicts, ARP conflicts, NDP conflicts, or reception of trigger traffic -// that warrants waking the sleeping host. -// ScheduleWakeup must be called with the lock held (ScheduleWakeupForList uses mDNS_Deregister_internal) - -mDNSlocal void ScheduleWakeupForList(mDNS *const m, mDNSInterfaceID InterfaceID, mDNSEthAddr *e, AuthRecord *const thelist) - { - // We don't need to use the m->CurrentRecord mechanism here because the target HMAC is nonzero, - // so all we're doing is marking the record to generate a few wakeup packets - AuthRecord *rr; - if (!e->l[0]) { LogMsg("ScheduleWakeupForList ERROR: Target HMAC is zero"); return; } - for (rr = thelist; rr; rr = rr->next) - if (rr->resrec.InterfaceID == InterfaceID && rr->resrec.RecordType != kDNSRecordTypeDeregistering && mDNSSameEthAddress(&rr->WakeUp.HMAC, e)) - { - LogInfo("ScheduleWakeupForList: Scheduling wakeup packets for %s", ARDisplayString(m, rr)); - mDNS_Deregister_internal(m, rr, mDNS_Dereg_normal); - } - } - -mDNSlocal void ScheduleWakeup(mDNS *const m, mDNSInterfaceID InterfaceID, mDNSEthAddr *e) - { - if (!e->l[0]) { LogMsg("ScheduleWakeup ERROR: Target HMAC is zero"); return; } - ScheduleWakeupForList(m, InterfaceID, e, m->DuplicateRecords); - ScheduleWakeupForList(m, InterfaceID, e, m->ResourceRecords); - } - -mDNSlocal void SPSRecordCallback(mDNS *const m, AuthRecord *const ar, mStatus result) - { - if (result && result != mStatus_MemFree) - LogInfo("SPS Callback %d %s", result, ARDisplayString(m, ar)); - - if (result == mStatus_NameConflict) - { - mDNS_Lock(m); - LogMsg("%-7s Conflicting mDNS -- waking %.6a %s", InterfaceNameForID(m, ar->resrec.InterfaceID), &ar->WakeUp.HMAC, ARDisplayString(m, ar)); - if (ar->WakeUp.HMAC.l[0]) - { - SendWakeup(m, ar->resrec.InterfaceID, &ar->WakeUp.IMAC, &ar->WakeUp.password); // Send one wakeup magic packet - ScheduleWakeup(m, ar->resrec.InterfaceID, &ar->WakeUp.HMAC); // Schedule all other records with the same owner to be woken - } - mDNS_Unlock(m); - } - - if (result == mStatus_NameConflict || result == mStatus_MemFree) - { - m->ProxyRecords--; - mDNSPlatformMemFree(ar); - mDNS_UpdateAllowSleep(m); - } - } - -mDNSlocal void mDNSCoreReceiveUpdate(mDNS *const m, - const DNSMessage *const msg, const mDNSu8 *end, - const mDNSAddr *srcaddr, const mDNSIPPort srcport, const mDNSAddr *dstaddr, mDNSIPPort dstport, - const mDNSInterfaceID InterfaceID) - { - int i; - AuthRecord opt; - mDNSu8 *p = m->omsg.data; - OwnerOptData owner = zeroOwner; // Need to zero this, so we'll know if this Update packet was missing its Owner option - mDNSu32 updatelease = 0; - const mDNSu8 *ptr; - - LogSPS("Received Update from %#-15a:%-5d to %#-15a:%-5d on 0x%p with " - "%2d Question%s %2d Answer%s %2d Authorit%s %2d Additional%s %d bytes", - srcaddr, mDNSVal16(srcport), dstaddr, mDNSVal16(dstport), InterfaceID, - msg->h.numQuestions, msg->h.numQuestions == 1 ? ", " : "s,", - msg->h.numAnswers, msg->h.numAnswers == 1 ? ", " : "s,", - msg->h.numAuthorities, msg->h.numAuthorities == 1 ? "y, " : "ies,", - msg->h.numAdditionals, msg->h.numAdditionals == 1 ? " " : "s", end - msg->data); - - if (!InterfaceID || !m->SPSSocket || !mDNSSameIPPort(dstport, m->SPSSocket->port)) return; - - if (mDNS_PacketLoggingEnabled) - DumpPacket(m, mStatus_NoError, mDNSfalse, "UDP", srcaddr, srcport, dstaddr, dstport, msg, end); - - ptr = LocateOptRR(msg, end, DNSOpt_LeaseData_Space + DNSOpt_OwnerData_ID_Space); - if (ptr) - { - ptr = GetLargeResourceRecord(m, msg, ptr, end, 0, kDNSRecordTypePacketAdd, &m->rec); - if (ptr && m->rec.r.resrec.RecordType != kDNSRecordTypePacketNegative && m->rec.r.resrec.rrtype == kDNSType_OPT) - { - const rdataOPT *o; - const rdataOPT *const e = (const rdataOPT *)&m->rec.r.resrec.rdata->u.data[m->rec.r.resrec.rdlength]; - for (o = &m->rec.r.resrec.rdata->u.opt[0]; o < e; o++) - { - if (o->opt == kDNSOpt_Lease) updatelease = o->u.updatelease; - else if (o->opt == kDNSOpt_Owner && o->u.owner.vers == 0) owner = o->u.owner; - } - } - m->rec.r.resrec.RecordType = 0; // Clear RecordType to show we're not still using it - } - - InitializeDNSMessage(&m->omsg.h, msg->h.id, UpdateRespFlags); - - if (!updatelease || !owner.HMAC.l[0]) - { - static int msgs = 0; - if (msgs < 100) - { - msgs++; - LogMsg("Refusing sleep proxy registration from %#a:%d:%s%s", srcaddr, mDNSVal16(srcport), - !updatelease ? " No lease" : "", !owner.HMAC.l[0] ? " No owner" : ""); - } - m->omsg.h.flags.b[1] |= kDNSFlag1_RC_FormErr; - } - else if (m->ProxyRecords + msg->h.mDNS_numUpdates > MAX_PROXY_RECORDS) - { - static int msgs = 0; - if (msgs < 100) - { - msgs++; - LogMsg("Refusing sleep proxy registration from %#a:%d: Too many records %d + %d = %d > %d", srcaddr, mDNSVal16(srcport), - m->ProxyRecords, msg->h.mDNS_numUpdates, m->ProxyRecords + msg->h.mDNS_numUpdates, MAX_PROXY_RECORDS); - } - m->omsg.h.flags.b[1] |= kDNSFlag1_RC_Refused; - } - else - { - LogSPS("Received Update for H-MAC %.6a I-MAC %.6a Password %.6a seq %d", &owner.HMAC, &owner.IMAC, &owner.password, owner.seq); - - if (updatelease > 24 * 60 * 60) - updatelease = 24 * 60 * 60; - - if (updatelease > 0x40000000UL / mDNSPlatformOneSecond) - updatelease = 0x40000000UL / mDNSPlatformOneSecond; - - ptr = LocateAuthorities(msg, end); - for (i = 0; i < msg->h.mDNS_numUpdates && ptr && ptr < end; i++) - { - ptr = GetLargeResourceRecord(m, msg, ptr, end, InterfaceID, kDNSRecordTypePacketAuth, &m->rec); - if (ptr && m->rec.r.resrec.RecordType != kDNSRecordTypePacketNegative) - { - mDNSu16 RDLengthMem = GetRDLengthMem(&m->rec.r.resrec); - AuthRecord *ar = mDNSPlatformMemAllocate(sizeof(AuthRecord) - sizeof(RDataBody) + RDLengthMem); - if (!ar) { m->omsg.h.flags.b[1] |= kDNSFlag1_RC_Refused; break; } - else - { - mDNSu8 RecordType = m->rec.r.resrec.RecordType & kDNSRecordTypePacketUniqueMask ? kDNSRecordTypeUnique : kDNSRecordTypeShared; - m->rec.r.resrec.rrclass &= ~kDNSClass_UniqueRRSet; - ClearIdenticalProxyRecords(m, &owner, m->DuplicateRecords); // Make sure we don't have any old stale duplicates of this record - ClearIdenticalProxyRecords(m, &owner, m->ResourceRecords); - mDNS_SetupResourceRecord(ar, mDNSNULL, InterfaceID, m->rec.r.resrec.rrtype, m->rec.r.resrec.rroriginalttl, RecordType, AuthRecordAny, SPSRecordCallback, ar); - AssignDomainName(&ar->namestorage, m->rec.r.resrec.name); - ar->resrec.rdlength = GetRDLength(&m->rec.r.resrec, mDNSfalse); - ar->resrec.rdata->MaxRDLength = RDLengthMem; - mDNSPlatformMemCopy(ar->resrec.rdata->u.data, m->rec.r.resrec.rdata->u.data, RDLengthMem); - ar->ForceMCast = mDNStrue; - ar->WakeUp = owner; - if (m->rec.r.resrec.rrtype == kDNSType_PTR) - { - mDNSs32 t = ReverseMapDomainType(m->rec.r.resrec.name); - if (t == mDNSAddrType_IPv4) GetIPv4FromName(&ar->AddressProxy, m->rec.r.resrec.name); - else if (t == mDNSAddrType_IPv6) GetIPv6FromName(&ar->AddressProxy, m->rec.r.resrec.name); - debugf("mDNSCoreReceiveUpdate: PTR %d %d %#a %s", t, ar->AddressProxy.type, &ar->AddressProxy, ARDisplayString(m, ar)); - if (ar->AddressProxy.type) SetSPSProxyListChanged(InterfaceID); - } - ar->TimeRcvd = m->timenow; - ar->TimeExpire = m->timenow + updatelease * mDNSPlatformOneSecond; - if (m->NextScheduledSPS - ar->TimeExpire > 0) - m->NextScheduledSPS = ar->TimeExpire; - mDNS_Register_internal(m, ar); - // Unsolicited Neighbor Advertisements (RFC 2461 Section 7.2.6) give us fast address cache updating, - // but some older IPv6 clients get confused by them, so for now we don't send them. Without Unsolicited - // Neighbor Advertisements we have to rely on Neighbor Unreachability Detection instead, which is slower. - // Given this, we'll do our best to wake for existing IPv6 connections, but we don't want to encourage - // new ones for sleeping clients, so we'll we send deletions for our SPS clients' AAAA records. - if (m->KnownBugs & mDNS_KnownBug_LimitedIPv6) - if (ar->resrec.rrtype == kDNSType_AAAA) ar->resrec.rroriginalttl = 0; - m->ProxyRecords++; - mDNS_UpdateAllowSleep(m); - LogSPS("SPS Registered %4d %X %s", m->ProxyRecords, RecordType, ARDisplayString(m,ar)); - } - } - m->rec.r.resrec.RecordType = 0; // Clear RecordType to show we're not still using it - } - - if (m->omsg.h.flags.b[1] & kDNSFlag1_RC_Mask) - { - LogMsg("Refusing sleep proxy registration from %#a:%d: Out of memory", srcaddr, mDNSVal16(srcport)); - ClearProxyRecords(m, &owner, m->DuplicateRecords); - ClearProxyRecords(m, &owner, m->ResourceRecords); - } - else - { - mDNS_SetupResourceRecord(&opt, mDNSNULL, mDNSInterface_Any, kDNSType_OPT, kStandardTTL, kDNSRecordTypeKnownUnique, AuthRecordAny, mDNSNULL, mDNSNULL); - opt.resrec.rrclass = NormalMaxDNSMessageData; - opt.resrec.rdlength = sizeof(rdataOPT); // One option in this OPT record - opt.resrec.rdestimate = sizeof(rdataOPT); - opt.resrec.rdata->u.opt[0].opt = kDNSOpt_Lease; - opt.resrec.rdata->u.opt[0].u.updatelease = updatelease; - p = PutResourceRecordTTLWithLimit(&m->omsg, p, &m->omsg.h.numAdditionals, &opt.resrec, opt.resrec.rroriginalttl, m->omsg.data + AbsoluteMaxDNSMessageData); - } - } - - if (p) mDNSSendDNSMessage(m, &m->omsg, p, InterfaceID, m->SPSSocket, srcaddr, srcport, mDNSNULL, mDNSNULL); - } - -mDNSlocal void mDNSCoreReceiveUpdateR(mDNS *const m, const DNSMessage *const msg, const mDNSu8 *end, const mDNSInterfaceID InterfaceID) - { - if (InterfaceID) - { - mDNSu32 updatelease = 60 * 60; // If SPS fails to indicate lease time, assume one hour - const mDNSu8 *ptr = LocateOptRR(msg, end, DNSOpt_LeaseData_Space); - if (ptr) - { - ptr = GetLargeResourceRecord(m, msg, ptr, end, 0, kDNSRecordTypePacketAdd, &m->rec); - if (ptr && m->rec.r.resrec.RecordType != kDNSRecordTypePacketNegative && m->rec.r.resrec.rrtype == kDNSType_OPT) - { - const rdataOPT *o; - const rdataOPT *const e = (const rdataOPT *)&m->rec.r.resrec.rdata->u.data[m->rec.r.resrec.rdlength]; - for (o = &m->rec.r.resrec.rdata->u.opt[0]; o < e; o++) - if (o->opt == kDNSOpt_Lease) - { - updatelease = o->u.updatelease; - LogSPS("Sleep Proxy granted lease time %4d seconds", updatelease); - } - } - m->rec.r.resrec.RecordType = 0; // Clear RecordType to show we're not still using it - } - - if (m->CurrentRecord) - LogMsg("mDNSCoreReceiveUpdateR ERROR m->CurrentRecord already set %s", ARDisplayString(m, m->CurrentRecord)); - m->CurrentRecord = m->ResourceRecords; - while (m->CurrentRecord) - { - AuthRecord *const rr = m->CurrentRecord; - if (rr->resrec.InterfaceID == InterfaceID || (!rr->resrec.InterfaceID && (rr->ForceMCast || IsLocalDomain(rr->resrec.name)))) - if (mDNSSameOpaque16(rr->updateid, msg->h.id)) - { - rr->updateid = zeroID; - rr->expire = NonZeroTime(m->timenow + updatelease * mDNSPlatformOneSecond); - LogSPS("Sleep Proxy %s record %5d %s", rr->WakeUp.HMAC.l[0] ? "transferred" : "registered", updatelease, ARDisplayString(m,rr)); - if (rr->WakeUp.HMAC.l[0]) - { - rr->WakeUp.HMAC = zeroEthAddr; // Clear HMAC so that mDNS_Deregister_internal doesn't waste packets trying to wake this host - rr->RequireGoodbye = mDNSfalse; // and we don't want to send goodbye for it - mDNS_Deregister_internal(m, rr, mDNS_Dereg_normal); - } - } - // Mustn't advance m->CurrentRecord until *after* mDNS_Deregister_internal, because - // new records could have been added to the end of the list as a result of that call. - if (m->CurrentRecord == rr) // If m->CurrentRecord was not advanced for us, do it now - m->CurrentRecord = rr->next; - } - } - // If we were waiting to go to sleep, then this SPS registration or wide-area record deletion - // may have been the thing we were waiting for, so schedule another check to see if we can sleep now. - if (m->SleepLimit) m->NextScheduledSPRetry = m->timenow; - } - -mDNSexport void MakeNegativeCacheRecord(mDNS *const m, CacheRecord *const cr, - const domainname *const name, const mDNSu32 namehash, const mDNSu16 rrtype, const mDNSu16 rrclass, mDNSu32 ttl_seconds, mDNSInterfaceID InterfaceID, DNSServer *dnsserver) - { - if (cr == &m->rec.r && m->rec.r.resrec.RecordType) - { - LogMsg("MakeNegativeCacheRecord: m->rec appears to be already in use for %s", CRDisplayString(m, &m->rec.r)); -#if ForceAlerts - *(long*)0 = 0; -#endif - } - - // Create empty resource record - cr->resrec.RecordType = kDNSRecordTypePacketNegative; - cr->resrec.InterfaceID = InterfaceID; - cr->resrec.rDNSServer = dnsserver; - cr->resrec.name = name; // Will be updated to point to cg->name when we call CreateNewCacheEntry - cr->resrec.rrtype = rrtype; - cr->resrec.rrclass = rrclass; - cr->resrec.rroriginalttl = ttl_seconds; - cr->resrec.rdlength = 0; - cr->resrec.rdestimate = 0; - cr->resrec.namehash = namehash; - cr->resrec.rdatahash = 0; - cr->resrec.rdata = (RData*)&cr->smallrdatastorage; - cr->resrec.rdata->MaxRDLength = 0; - - cr->NextInKAList = mDNSNULL; - cr->TimeRcvd = m->timenow; - cr->DelayDelivery = 0; - cr->NextRequiredQuery = m->timenow; - cr->LastUsed = m->timenow; - cr->CRActiveQuestion = mDNSNULL; - cr->UnansweredQueries = 0; - cr->LastUnansweredTime = 0; -#if ENABLE_MULTI_PACKET_QUERY_SNOOPING - cr->MPUnansweredQ = 0; - cr->MPLastUnansweredQT = 0; - cr->MPUnansweredKA = 0; - cr->MPExpectingKA = mDNSfalse; -#endif - cr->NextInCFList = mDNSNULL; - } - -mDNSexport void mDNSCoreReceive(mDNS *const m, void *const pkt, const mDNSu8 *const end, - const mDNSAddr *const srcaddr, const mDNSIPPort srcport, const mDNSAddr *dstaddr, const mDNSIPPort dstport, - const mDNSInterfaceID InterfaceID) - { - mDNSInterfaceID ifid = InterfaceID; - DNSMessage *msg = (DNSMessage *)pkt; - const mDNSu8 StdQ = kDNSFlag0_QR_Query | kDNSFlag0_OP_StdQuery; - const mDNSu8 StdR = kDNSFlag0_QR_Response | kDNSFlag0_OP_StdQuery; - const mDNSu8 UpdQ = kDNSFlag0_QR_Query | kDNSFlag0_OP_Update; - const mDNSu8 UpdR = kDNSFlag0_QR_Response | kDNSFlag0_OP_Update; - mDNSu8 QR_OP; - mDNSu8 *ptr = mDNSNULL; - mDNSBool TLS = (dstaddr == (mDNSAddr *)1); // For debug logs: dstaddr = 0 means TCP; dstaddr = 1 means TLS - if (TLS) dstaddr = mDNSNULL; - -#ifndef UNICAST_DISABLED - if (mDNSSameAddress(srcaddr, &m->Router)) - { -#ifdef _LEGACY_NAT_TRAVERSAL_ - if (mDNSSameIPPort(srcport, SSDPPort) || (m->SSDPSocket && mDNSSameIPPort(dstport, m->SSDPSocket->port))) - { - mDNS_Lock(m); - LNT_ConfigureRouterInfo(m, InterfaceID, pkt, (mDNSu16)(end - (mDNSu8 *)pkt)); - mDNS_Unlock(m); - return; - } -#endif - if (mDNSSameIPPort(srcport, NATPMPPort)) - { - mDNS_Lock(m); - uDNS_ReceiveNATPMPPacket(m, InterfaceID, pkt, (mDNSu16)(end - (mDNSu8 *)pkt)); - mDNS_Unlock(m); - return; - } - } -#ifdef _LEGACY_NAT_TRAVERSAL_ - else if (m->SSDPSocket && mDNSSameIPPort(dstport, m->SSDPSocket->port)) { debugf("Ignoring SSDP response from %#a:%d", srcaddr, mDNSVal16(srcport)); return; } -#endif - -#endif - if ((unsigned)(end - (mDNSu8 *)pkt) < sizeof(DNSMessageHeader)) - { - LogMsg("DNS Message from %#a:%d to %#a:%d length %d too short", srcaddr, mDNSVal16(srcport), dstaddr, mDNSVal16(dstport), end - (mDNSu8 *)pkt); - return; - } - QR_OP = (mDNSu8)(msg->h.flags.b[0] & kDNSFlag0_QROP_Mask); - // Read the integer parts which are in IETF byte-order (MSB first, LSB second) - ptr = (mDNSu8 *)&msg->h.numQuestions; - msg->h.numQuestions = (mDNSu16)((mDNSu16)ptr[0] << 8 | ptr[1]); - msg->h.numAnswers = (mDNSu16)((mDNSu16)ptr[2] << 8 | ptr[3]); - msg->h.numAuthorities = (mDNSu16)((mDNSu16)ptr[4] << 8 | ptr[5]); - msg->h.numAdditionals = (mDNSu16)((mDNSu16)ptr[6] << 8 | ptr[7]); - - if (!m) { LogMsg("mDNSCoreReceive ERROR m is NULL"); return; } - - // We use zero addresses and all-ones addresses at various places in the code to indicate special values like "no address" - // If we accept and try to process a packet with zero or all-ones source address, that could really mess things up - if (srcaddr && !mDNSAddressIsValid(srcaddr)) { debugf("mDNSCoreReceive ignoring packet from %#a", srcaddr); return; } - - mDNS_Lock(m); - m->PktNum++; -#ifndef UNICAST_DISABLED - if (!dstaddr || (!mDNSAddressIsAllDNSLinkGroup(dstaddr) && (QR_OP == StdR || QR_OP == UpdR))) - if (!mDNSOpaque16IsZero(msg->h.id)) // uDNS_ReceiveMsg only needs to get real uDNS responses, not "QU" mDNS responses - { - ifid = mDNSInterface_Any; - if (mDNS_PacketLoggingEnabled) - DumpPacket(m, mStatus_NoError, mDNSfalse, TLS ? "TLS" : !dstaddr ? "TCP" : "UDP", srcaddr, srcport, dstaddr, dstport, msg, end); - uDNS_ReceiveMsg(m, msg, end, srcaddr, srcport); - // Note: mDNSCore also needs to get access to received unicast responses - } -#endif - if (QR_OP == StdQ) mDNSCoreReceiveQuery (m, msg, end, srcaddr, srcport, dstaddr, dstport, ifid); - else if (QR_OP == StdR) mDNSCoreReceiveResponse(m, msg, end, srcaddr, srcport, dstaddr, dstport, ifid); - else if (QR_OP == UpdQ) mDNSCoreReceiveUpdate (m, msg, end, srcaddr, srcport, dstaddr, dstport, InterfaceID); - else if (QR_OP == UpdR) mDNSCoreReceiveUpdateR (m, msg, end, InterfaceID); - else - { - LogMsg("Unknown DNS packet type %02X%02X from %#-15a:%-5d to %#-15a:%-5d length %d on %p (ignored)", - msg->h.flags.b[0], msg->h.flags.b[1], srcaddr, mDNSVal16(srcport), dstaddr, mDNSVal16(dstport), end - (mDNSu8 *)pkt, InterfaceID); - if (mDNS_LoggingEnabled) - { - int i = 0; - while (iTarget.type == mDNSAddrType_None && (B)->Target.type == mDNSAddrType_None) || \ - (mDNSSameAddress(&(A)->Target, &(B)->Target) && mDNSSameIPPort((A)->TargetPort, (B)->TargetPort))) - -// Note: We explicitly disallow making a public query be a duplicate of a private one. This is to avoid the -// circular deadlock where a client does a query for something like "dns-sd -Q _dns-query-tls._tcp.company.com SRV" -// and we have a key for company.com, so we try to locate the private query server for company.com, which necessarily entails -// doing a standard DNS query for the _dns-query-tls._tcp SRV record for company.com. If we make the latter (public) query -// a duplicate of the former (private) query, then it will block forever waiting for an answer that will never come. -// -// We keep SuppressUnusable questions separate so that we can return a quick response to them and not get blocked behind -// the queries that are not marked SuppressUnusable. But if the query is not suppressed, they are treated the same as -// non-SuppressUnusable questions. This should be fine as the goal of SuppressUnusable is to return quickly only if it -// is suppressed. If it is not suppressed, we do try all the DNS servers for valid answers like any other question. -// The main reason for this design is that cache entries point to a *single* question and that question is responsible -// for keeping the cache fresh as long as it is active. Having multiple active question for a single cache entry -// breaks this design principle. - -// If IsLLQ(Q) is true, it means the question is both: -// (a) long-lived and -// (b) being performed by a unicast DNS long-lived query (either full LLQ, or polling) -// for multicast questions, we don't want to treat LongLived as anything special -#define IsLLQ(Q) ((Q)->LongLived && !mDNSOpaque16IsZero((Q)->TargetQID)) - -mDNSlocal DNSQuestion *FindDuplicateQuestion(const mDNS *const m, const DNSQuestion *const question) - { - DNSQuestion *q; - // Note: A question can only be marked as a duplicate of one that occurs *earlier* in the list. - // This prevents circular references, where two questions are each marked as a duplicate of the other. - // Accordingly, we break out of the loop when we get to 'question', because there's no point searching - // further in the list. - for (q = m->Questions; q && q != question; q=q->next) // Scan our list for another question - if (q->InterfaceID == question->InterfaceID && // with the same InterfaceID, - SameQTarget(q, question) && // and same unicast/multicast target settings - q->qtype == question->qtype && // type, - q->qclass == question->qclass && // class, - IsLLQ(q) == IsLLQ(question) && // and long-lived status matches - (!q->AuthInfo || question->AuthInfo) && // to avoid deadlock, don't make public query dup of a private one - (q->SuppressQuery == question->SuppressQuery) && // Questions that are suppressed/not suppressed - q->qnamehash == question->qnamehash && - SameDomainName(&q->qname, &question->qname)) // and name - return(q); - return(mDNSNULL); - } - -// This is called after a question is deleted, in case other identical questions were being suppressed as duplicates -mDNSlocal void UpdateQuestionDuplicates(mDNS *const m, DNSQuestion *const question) - { - DNSQuestion *q; - DNSQuestion *first = mDNSNULL; - - // This is referring to some other question as duplicate. No other question can refer to this - // question as a duplicate. - if (question->DuplicateOf) - { - LogInfo("UpdateQuestionDuplicates: question %p %##s (%s) duplicate of %p %##s (%s)", - question, question->qname.c, DNSTypeName(question->qtype), - question->DuplicateOf, question->DuplicateOf->qname.c, DNSTypeName(question->DuplicateOf->qtype)); - return; - } - - for (q = m->Questions; q; q=q->next) // Scan our list of questions - if (q->DuplicateOf == question) // To see if any questions were referencing this as their duplicate - { - q->DuplicateOf = first; - if (!first) - { - first = q; - // If q used to be a duplicate, but now is not, - // then inherit the state from the question that's going away - q->LastQTime = question->LastQTime; - q->ThisQInterval = question->ThisQInterval; - q->ExpectUnicastResp = question->ExpectUnicastResp; - q->LastAnswerPktNum = question->LastAnswerPktNum; - q->RecentAnswerPkts = question->RecentAnswerPkts; - q->RequestUnicast = question->RequestUnicast; - q->LastQTxTime = question->LastQTxTime; - q->CNAMEReferrals = question->CNAMEReferrals; - q->nta = question->nta; - q->servAddr = question->servAddr; - q->servPort = question->servPort; - q->qDNSServer = question->qDNSServer; - q->validDNSServers = question->validDNSServers; - q->unansweredQueries = question->unansweredQueries; - q->noServerResponse = question->noServerResponse; - q->triedAllServersOnce = question->triedAllServersOnce; - - q->TargetQID = question->TargetQID; - q->LocalSocket = question->LocalSocket; - - q->state = question->state; - // q->tcp = question->tcp; - q->ReqLease = question->ReqLease; - q->expire = question->expire; - q->ntries = question->ntries; - q->id = question->id; - - question->LocalSocket = mDNSNULL; - question->nta = mDNSNULL; // If we've got a GetZoneData in progress, transfer it to the newly active question - // question->tcp = mDNSNULL; - - if (q->LocalSocket) - debugf("UpdateQuestionDuplicates transferred LocalSocket pointer for %##s (%s)", q->qname.c, DNSTypeName(q->qtype)); - - if (q->nta) - { - LogInfo("UpdateQuestionDuplicates transferred nta pointer for %##s (%s)", q->qname.c, DNSTypeName(q->qtype)); - q->nta->ZoneDataContext = q; - } - - // Need to work out how to safely transfer this state too -- appropriate context pointers need to be updated or the code will crash - if (question->tcp) LogInfo("UpdateQuestionDuplicates did not transfer tcp pointer"); - - if (question->state == LLQ_Established) - { - LogInfo("UpdateQuestionDuplicates transferred LLQ state for %##s (%s)", q->qname.c, DNSTypeName(q->qtype)); - question->state = 0; // Must zero question->state, or mDNS_StopQuery_internal will clean up and cancel our LLQ from the server - } - - SetNextQueryTime(m,q); - } - } - } - -mDNSexport McastResolver *mDNS_AddMcastResolver(mDNS *const m, const domainname *d, const mDNSInterfaceID interface, mDNSu32 timeout) - { - McastResolver **p = &m->McastResolvers; - McastResolver *tmp = mDNSNULL; - - if (!d) d = (const domainname *)""; - - LogInfo("mDNS_AddMcastResolver: Adding %##s, InterfaceID %p, timeout %u", d->c, interface, timeout); - - if (m->mDNS_busy != m->mDNS_reentrancy+1) - LogMsg("mDNS_AddMcastResolver: Lock not held! mDNS_busy (%ld) mDNS_reentrancy (%ld)", m->mDNS_busy, m->mDNS_reentrancy); - - while (*p) // Check if we already have this {interface, domain} tuple registered - { - if ((*p)->interface == interface && SameDomainName(&(*p)->domain, d)) - { - if (!((*p)->flags & DNSServer_FlagDelete)) LogMsg("Note: Mcast Resolver domain %##s (%p) registered more than once", d->c, interface); - (*p)->flags &= ~DNSServer_FlagDelete; - tmp = *p; - *p = tmp->next; - tmp->next = mDNSNULL; - } - else - p=&(*p)->next; - } - - if (tmp) *p = tmp; // move to end of list, to ensure ordering from platform layer - else - { - // allocate, add to list - *p = mDNSPlatformMemAllocate(sizeof(**p)); - if (!*p) LogMsg("mDNS_AddMcastResolver: ERROR!! - malloc"); - else - { - (*p)->interface = interface; - (*p)->flags = DNSServer_FlagNew; - (*p)->timeout = timeout; - AssignDomainName(&(*p)->domain, d); - (*p)->next = mDNSNULL; - } - } - return(*p); - } - -mDNSinline mDNSs32 PenaltyTimeForServer(mDNS *m, DNSServer *server) - { - mDNSs32 ptime = 0; - if (server->penaltyTime != 0) - { - ptime = server->penaltyTime - m->timenow; - if (ptime < 0) - { - // This should always be a positive value between 0 and DNSSERVER_PENALTY_TIME - // If it does not get reset in ResetDNSServerPenalties for some reason, we do it - // here - LogMsg("PenaltyTimeForServer: PenaltyTime negative %d, (server penaltyTime %d, timenow %d) resetting the penalty", - ptime, server->penaltyTime, m->timenow); - server->penaltyTime = 0; - ptime = 0; - } - } - return ptime; - } - -//Checks to see whether the newname is a better match for the name, given the best one we have -//seen so far (given in bestcount). -//Returns -1 if the newname is not a better match -//Returns 0 if the newname is the same as the old match -//Returns 1 if the newname is a better match -mDNSlocal int BetterMatchForName(const domainname *name, int namecount, const domainname *newname, int newcount, - int bestcount) - { - // If the name contains fewer labels than the new server's domain or the new name - // contains fewer labels than the current best, then it can't possibly be a better match - if (namecount < newcount || newcount < bestcount) return -1; - - // If there is no match, return -1 and the caller will skip this newname for - // selection - // - // If we find a match and the number of labels is the same as bestcount, then - // we return 0 so that the caller can do additional logic to pick one of - // the best based on some other factors e.g., penaltyTime - // - // If we find a match and the number of labels is more than bestcount, then we - // return 1 so that the caller can pick this over the old one. - // - // Note: newcount can either be equal or greater than bestcount beause of the - // check above. - - if (SameDomainName(SkipLeadingLabels(name, namecount - newcount), newname)) - return bestcount == newcount ? 0 : 1; - else - return -1; - } - -// Normally, we have McastResolvers for .local, in-addr.arpa and ip6.arpa. But there -// can be queries that can forced to multicast (ForceMCast) even though they don't end in these -// names. In that case, we give a default timeout of 5 seconds -#define DEFAULT_MCAST_TIMEOUT 5 -mDNSlocal mDNSu32 GetTimeoutForMcastQuestion(mDNS *m, DNSQuestion *question) - { - McastResolver *curmatch = mDNSNULL; - int bestmatchlen = -1, namecount = CountLabels(&question->qname); - McastResolver *curr; - int bettermatch, currcount; - for (curr = m->McastResolvers; curr; curr = curr->next) - { - currcount = CountLabels(&curr->domain); - bettermatch = BetterMatchForName(&question->qname, namecount, &curr->domain, currcount, bestmatchlen); - // Take the first best match. If there are multiple equally good matches (bettermatch = 0), we take - // the timeout value from the first one - if (bettermatch == 1) - { - curmatch = curr; - bestmatchlen = currcount; - } - } - LogInfo("GetTimeoutForMcastQuestion: question %##s curmatch %p, Timeout %d", question->qname.c, curmatch, - curmatch ? curmatch->timeout : DEFAULT_MCAST_TIMEOUT); - return ( curmatch ? curmatch->timeout : DEFAULT_MCAST_TIMEOUT); - } - -// Returns true if it is a Domain Enumeration Query -mDNSexport mDNSBool DomainEnumQuery(const domainname *qname) - { - const mDNSu8 *mDNS_DEQLabels[] = { (const mDNSu8 *)"\001b", (const mDNSu8 *)"\002db", (const mDNSu8 *)"\002lb", - (const mDNSu8 *)"\001r", (const mDNSu8 *)"\002dr", (const mDNSu8 *)mDNSNULL, }; - const domainname *d = qname; - const mDNSu8 *label; - int i = 0; - - // We need at least 3 labels (DEQ prefix) + one more label to make a meaningful DE query - if (CountLabels(qname) < 4) { debugf("DomainEnumQuery: question %##s, not enough labels", qname->c); return mDNSfalse; } - - label = (const mDNSu8 *)d; - while (mDNS_DEQLabels[i] != (const mDNSu8 *)mDNSNULL) - { - if (SameDomainLabel(mDNS_DEQLabels[i], label)) {debugf("DomainEnumQuery: DEQ %##s, label1 match", qname->c); break;} - i++; - } - if (mDNS_DEQLabels[i] == (const mDNSu8 *)mDNSNULL) - { - debugf("DomainEnumQuery: Not a DEQ %##s, label1 mismatch", qname->c); - return mDNSfalse; - } - debugf("DomainEnumQuery: DEQ %##s, label1 match", qname->c); - - // CountLabels already verified the number of labels - d = (const domainname *)(d->c + 1 + d->c[0]); // Second Label - label = (const mDNSu8 *)d; - if (!SameDomainLabel(label, (const mDNSu8 *)"\007_dns-sd")) - { - debugf("DomainEnumQuery: Not a DEQ %##s, label2 mismatch", qname->c); - return(mDNSfalse); - } - debugf("DomainEnumQuery: DEQ %##s, label2 match", qname->c); - - d = (const domainname *)(d->c + 1 + d->c[0]); // Third Label - label = (const mDNSu8 *)d; - if (!SameDomainLabel(label, (const mDNSu8 *)"\004_udp")) - { - debugf("DomainEnumQuery: Not a DEQ %##s, label3 mismatch", qname->c); - return(mDNSfalse); - } - debugf("DomainEnumQuery: DEQ %##s, label3 match", qname->c); - - debugf("DomainEnumQuery: Question %##s is a Domain Enumeration query", qname->c); - - return mDNStrue; - } - -// Sets all the Valid DNS servers for a question -mDNSexport mDNSu32 SetValidDNSServers(mDNS *m, DNSQuestion *question) - { - DNSServer *curmatch = mDNSNULL; - int bestmatchlen = -1, namecount = CountLabels(&question->qname); - DNSServer *curr; - int bettermatch, currcount; - int index = 0; - mDNSu32 timeout = 0; - mDNSBool DEQuery; - - question->validDNSServers = zeroOpaque64; - DEQuery = DomainEnumQuery(&question->qname); - for (curr = m->DNSServers; curr; curr = curr->next) - { - debugf("SetValidDNSServers: Parsing DNS server Address %#a (Domain %##s), Scope: %d", &curr->addr, curr->domain.c, curr->scoped); - // skip servers that will soon be deleted - if (curr->flags & DNSServer_FlagDelete) - { debugf("SetValidDNSServers: Delete set for index %d, DNS server %#a (Domain %##s), scoped %d", index, &curr->addr, curr->domain.c, curr->scoped); continue; } - - // This happens normally when you unplug the interface where we reset the interfaceID to mDNSInterface_Any for all - // the DNS servers whose scope match the interfaceID. Few seconds later, we also receive the updated DNS configuration. - // But any questions that has mDNSInterface_Any scope that are started/restarted before we receive the update - // (e.g., CheckSuppressUnusableQuestions is called when interfaces are deregistered with the core) should not - // match the scoped entries by mistake. - // - // Note: DNS configuration change will help pick the new dns servers but currently it does not affect the timeout - - if (curr->scoped && curr->interface == mDNSInterface_Any) - { debugf("SetValidDNSServers: Scoped DNS server %#a (Domain %##s) with Interface Any", &curr->addr, curr->domain.c); continue; } - - currcount = CountLabels(&curr->domain); - if ((!DEQuery || !curr->cellIntf) && - ((!curr->scoped && (!question->InterfaceID || (question->InterfaceID == mDNSInterface_Unicast))) || - (curr->interface == question->InterfaceID))) - { - bettermatch = BetterMatchForName(&question->qname, namecount, &curr->domain, currcount, bestmatchlen); - - // If we found a better match (bettermatch == 1) then clear all the bits - // corresponding to the old DNSServers that we have may set before and start fresh. - // If we find an equal match, then include that DNSServer also by setting the corresponding - // bit - if ((bettermatch == 1) || (bettermatch == 0)) - { - curmatch = curr; - bestmatchlen = currcount; - if (bettermatch) { debugf("SetValidDNSServers: Resetting all the bits"); question->validDNSServers = zeroOpaque64; timeout = 0; } - debugf("SetValidDNSServers: question %##s Setting the bit for DNS server Address %#a (Domain %##s), Scoped:%d index %d," - " Timeout %d, interface %p", question->qname.c, &curr->addr, curr->domain.c, curr->scoped, index, curr->timeout, - curr->interface); - timeout += curr->timeout; - if (DEQuery) debugf("DomainEnumQuery: Question %##s, DNSServer %#a, cell %d", question->qname.c, &curr->addr, curr->cellIntf); - bit_set_opaque64(question->validDNSServers, index); - } - } - index++; - } - question->noServerResponse = 0; - - debugf("SetValidDNSServers: ValidDNSServer bits 0x%x%x for question %p %##s (%s)", - question->validDNSServers.l[1], question->validDNSServers.l[0], question, question->qname.c, DNSTypeName(question->qtype)); - // If there are no matching resolvers, then use the default value to timeout - return (timeout ? timeout : DEFAULT_UDNS_TIMEOUT); - } - -// Get the Best server that matches a name. If you find penalized servers, look for the one -// that will come out of the penalty box soon -mDNSlocal DNSServer *GetBestServer(mDNS *m, const domainname *name, mDNSInterfaceID InterfaceID, mDNSOpaque64 validBits, int *selected, mDNSBool nameMatch) - { - DNSServer *curmatch = mDNSNULL; - int bestmatchlen = -1, namecount = name ? CountLabels(name) : 0; - DNSServer *curr; - mDNSs32 bestPenaltyTime, currPenaltyTime; - int bettermatch, currcount; - int index = 0; - int currindex = -1; - - debugf("GetBestServer: ValidDNSServer bits 0x%x%x", validBits.l[1], validBits.l[0]); - bestPenaltyTime = DNSSERVER_PENALTY_TIME + 1; - for (curr = m->DNSServers; curr; curr = curr->next) - { - // skip servers that will soon be deleted - if (curr->flags & DNSServer_FlagDelete) - { debugf("GetBestServer: Delete set for index %d, DNS server %#a (Domain %##s), scoped %d", index, &curr->addr, curr->domain.c, curr->scoped); continue; } - - // Check if this is a valid DNSServer - if (!bit_get_opaque64(validBits, index)) { debugf("GetBestServer: continuing for index %d", index); index++; continue; } - - currcount = CountLabels(&curr->domain); - currPenaltyTime = PenaltyTimeForServer(m, curr); - - debugf("GetBestServer: Address %#a (Domain %##s), PenaltyTime(abs) %d, PenaltyTime(rel) %d", - &curr->addr, curr->domain.c, curr->penaltyTime, currPenaltyTime); - - // If there are multiple best servers for a given question, we will pick the first one - // if none of them are penalized. If some of them are penalized in that list, we pick - // the least penalized one. BetterMatchForName walks through all best matches and - // "currPenaltyTime < bestPenaltyTime" check lets us either pick the first best server - // in the list when there are no penalized servers and least one among them - // when there are some penalized servers - // - // Notes on InterfaceID matching: - // - // 1) A DNSServer entry may have an InterfaceID but the scoped flag may not be set. This - // is the old way of specifying an InterfaceID option for DNSServer. We recoginize these - // entries by "scoped" being false. These are like any other unscoped entries except that - // if it is picked e.g., domain match, when the packet is sent out later, the packet will - // be sent out on that interface. Theese entries can be matched by either specifying a - // zero InterfaceID or non-zero InterfaceID on the question. Specifying an InterfaceID on - // the question will cause an extra check on matching the InterfaceID on the question - // against the DNSServer. - // - // 2) A DNSServer may also have both scoped set and InterfaceID non-NULL. This - // is the new way of specifying an InterfaceID option for DNSServer. These will be considered - // only when the question has non-zero interfaceID. - - if ((!curr->scoped && !InterfaceID) || (curr->interface == InterfaceID)) - { - - // If we know that all the names are already equally good matches, then skip calling BetterMatchForName. - // This happens when we initially walk all the DNS servers and set the validity bit on the question. - // Actually we just need PenaltyTime match, but for the sake of readability we just skip the expensive - // part and still do some redundant steps e.g., InterfaceID match - - if (nameMatch) bettermatch = BetterMatchForName(name, namecount, &curr->domain, currcount, bestmatchlen); - else bettermatch = 0; - - // If we found a better match (bettermatch == 1) then we don't need to - // compare penalty times. But if we found an equal match, then we compare - // the penalty times to pick a better match - - if ((bettermatch == 1) || ((bettermatch == 0) && currPenaltyTime < bestPenaltyTime)) - { currindex = index; curmatch = curr; bestmatchlen = currcount; bestPenaltyTime = currPenaltyTime; } - } - index++; - } - if (selected) *selected = currindex; - return curmatch; - } - -// Look up a DNS Server, matching by name and InterfaceID -mDNSexport DNSServer *GetServerForName(mDNS *m, const domainname *name, mDNSInterfaceID InterfaceID) - { - DNSServer *curmatch = mDNSNULL; - char *ifname = mDNSNULL; // for logging purposes only - mDNSOpaque64 allValid; - - if ((InterfaceID == mDNSInterface_Unicast) || (InterfaceID == mDNSInterface_LocalOnly)) - InterfaceID = mDNSNULL; - - if (InterfaceID) ifname = InterfaceNameForID(m, InterfaceID); - - // By passing in all ones, we make sure that every DNS server is considered - allValid.l[0] = allValid.l[1] = 0xFFFFFFFF; - - curmatch = GetBestServer(m, name, InterfaceID, allValid, mDNSNULL, mDNStrue); - - if (curmatch != mDNSNULL) - LogInfo("GetServerForName: DNS server %#a:%d (Penalty Time Left %d) (Scope %s:%p) found for name %##s", &curmatch->addr, - mDNSVal16(curmatch->port), (curmatch->penaltyTime ? (curmatch->penaltyTime - m->timenow) : 0), ifname ? ifname : "None", - InterfaceID, name); - else - LogInfo("GetServerForName: no DNS server (Scope %s:%p) found for name %##s", ifname ? ifname : "None", InterfaceID, name); - - return(curmatch); - } - -// Look up a DNS Server for a question within its valid DNSServer bits -mDNSexport DNSServer *GetServerForQuestion(mDNS *m, DNSQuestion *question) - { - DNSServer *curmatch = mDNSNULL; - char *ifname = mDNSNULL; // for logging purposes only - mDNSInterfaceID InterfaceID = question->InterfaceID; - const domainname *name = &question->qname; - int currindex; - - if ((InterfaceID == mDNSInterface_Unicast) || (InterfaceID == mDNSInterface_LocalOnly)) - InterfaceID = mDNSNULL; - - if (InterfaceID) ifname = InterfaceNameForID(m, InterfaceID); - - if (!mDNSOpaque64IsZero(&question->validDNSServers)) - { - curmatch = GetBestServer(m, name, InterfaceID, question->validDNSServers, &currindex, mDNSfalse); - if (currindex != -1) bit_clr_opaque64(question->validDNSServers, currindex); - } - - if (curmatch != mDNSNULL) - LogInfo("GetServerForQuestion: %p DNS server %#a:%d (Penalty Time Left %d) (Scope %s:%p) found for name %##s (%s)", question, &curmatch->addr, - mDNSVal16(curmatch->port), (curmatch->penaltyTime ? (curmatch->penaltyTime - m->timenow) : 0), ifname ? ifname : "None", - InterfaceID, name, DNSTypeName(question->qtype)); - else - LogInfo("GetServerForQuestion: %p no DNS server (Scope %s:%p) found for name %##s (%s)", question, ifname ? ifname : "None", InterfaceID, name, DNSTypeName(question->qtype)); - - return(curmatch); - } - - -#define ValidQuestionTarget(Q) (((Q)->Target.type == mDNSAddrType_IPv4 || (Q)->Target.type == mDNSAddrType_IPv6) && \ - (mDNSSameIPPort((Q)->TargetPort, UnicastDNSPort) || mDNSSameIPPort((Q)->TargetPort, MulticastDNSPort))) - -// Called in normal client context (lock not held) -mDNSlocal void LLQNATCallback(mDNS *m, NATTraversalInfo *n) - { - DNSQuestion *q; - (void)n; // Unused - mDNS_Lock(m); - LogInfo("LLQNATCallback external address:port %.4a:%u, NAT result %d", &n->ExternalAddress, mDNSVal16(n->ExternalPort), n->Result); - for (q = m->Questions; q; q=q->next) - if (ActiveQuestion(q) && !mDNSOpaque16IsZero(q->TargetQID) && q->LongLived) - startLLQHandshake(m, q); // If ExternalPort is zero, will do StartLLQPolling instead -#if APPLE_OSX_mDNSResponder - UpdateAutoTunnelDomainStatuses(m); -#endif - mDNS_Unlock(m); - } - -mDNSlocal mDNSBool ShouldSuppressQuery(mDNS *const m, domainname *qname, mDNSu16 qtype, mDNSInterfaceID InterfaceID) - { - NetworkInterfaceInfo *i; - mDNSs32 iptype; - DomainAuthInfo *AuthInfo; - - if (qtype == kDNSType_A) iptype = mDNSAddrType_IPv4; - else if (qtype == kDNSType_AAAA) iptype = mDNSAddrType_IPv6; - else { LogInfo("ShouldSuppressQuery: Query not suppressed for %##s, qtype %s, not A/AAAA type", qname, DNSTypeName(qtype)); return mDNSfalse; } - - // We still want the ability to be able to listen to the local services and hence - // don't fail .local requests. We always have a loopback interface which we don't - // check here. - if (InterfaceID != mDNSInterface_Unicast && IsLocalDomain(qname)) { LogInfo("ShouldSuppressQuery: Query not suppressed for %##s, qtype %s, Local question", qname, DNSTypeName(qtype)); return mDNSfalse; } - - // Skip Private domains as we have special addresses to get the hosts in the Private domain - AuthInfo = GetAuthInfoForName_internal(m, qname); - if (AuthInfo && !AuthInfo->deltime && AuthInfo->AutoTunnel) - { LogInfo("ShouldSuppressQuery: Query not suppressed for %##s, qtype %s, Private Domain", qname, DNSTypeName(qtype)); return mDNSfalse; } - - // Match on Type, Address and InterfaceID - // - // Check whether we are looking for a name that ends in .local, then presence of a link-local - // address on the interface is sufficient. - for (i = m->HostInterfaces; i; i = i->next) - { - if (i->ip.type != iptype) continue; - - if (!InterfaceID || (InterfaceID == mDNSInterface_LocalOnly) || (InterfaceID == mDNSInterface_P2P) || - (InterfaceID == mDNSInterface_Unicast) || (i->InterfaceID == InterfaceID)) - { - if (iptype == mDNSAddrType_IPv4 && !mDNSv4AddressIsLoopback(&i->ip.ip.v4) && !mDNSv4AddressIsLinkLocal(&i->ip.ip.v4)) - { - LogInfo("ShouldSuppressQuery: Query not suppressed for %##s, qtype %s, Local Address %.4a found", qname, DNSTypeName(qtype), - &i->ip.ip.v4); - return mDNSfalse; - } - else if (iptype == mDNSAddrType_IPv6 && - !mDNSv6AddressIsLoopback(&i->ip.ip.v6) && - !mDNSv6AddressIsLinkLocal(&i->ip.ip.v6) && - !mDNSSameIPv6Address(i->ip.ip.v6, m->AutoTunnelHostAddr) && - !mDNSSameIPv6Address(i->ip.ip.v6, m->AutoTunnelRelayAddrOut)) - { - LogInfo("ShouldSuppressQuery: Query not suppressed for %##s, qtype %s, Local Address %.16a found", qname, DNSTypeName(qtype), - &i->ip.ip.v6); - return mDNSfalse; - } - } - } - LogInfo("ShouldSuppressQuery: Query suppressed for %##s, qtype %s, because no matching interface found", qname, DNSTypeName(qtype)); - return mDNStrue; - } - -mDNSlocal void CacheRecordRmvEventsForCurrentQuestion(mDNS *const m, DNSQuestion *q) - { - CacheRecord *rr; - mDNSu32 slot; - CacheGroup *cg; - - slot = HashSlot(&q->qname); - cg = CacheGroupForName(m, slot, q->qnamehash, &q->qname); - for (rr = cg ? cg->members : mDNSNULL; rr; rr=rr->next) - { - // Don't deliver RMV events for negative records - if (rr->resrec.RecordType == kDNSRecordTypePacketNegative) - { - LogInfo("CacheRecordRmvEventsForCurrentQuestion: CacheRecord %s Suppressing RMV events for question %p %##s (%s), CRActiveQuestion %p, CurrentAnswers %d", - CRDisplayString(m, rr), q, q->qname.c, DNSTypeName(q->qtype), rr->CRActiveQuestion, q->CurrentAnswers); - continue; - } - - if (SameNameRecordAnswersQuestion(&rr->resrec, q)) - { - LogInfo("CacheRecordRmvEventsForCurrentQuestion: Calling AnswerCurrentQuestionWithResourceRecord (RMV) for question %##s using resource record %s LocalAnswers %d", - q->qname.c, CRDisplayString(m, rr), q->LOAddressAnswers); - - q->CurrentAnswers--; - if (rr->resrec.rdlength > SmallRecordLimit) q->LargeAnswers--; - if (rr->resrec.RecordType & kDNSRecordTypePacketUniqueMask) q->UniqueAnswers--; - - if (rr->CRActiveQuestion == q) - { - DNSQuestion *qptr; - // If this was the active question for this cache entry, it was the one that was - // responsible for keeping the cache entry fresh when the cache entry was reaching - // its expiry. We need to handover the responsibility to someone else. Otherwise, - // when the cache entry is about to expire, we won't find an active question - // (pointed by CRActiveQuestion) to refresh the cache. - for (qptr = m->Questions; qptr; qptr=qptr->next) - if (qptr != q && ActiveQuestion(qptr) && ResourceRecordAnswersQuestion(&rr->resrec, qptr)) - break; - - if (qptr) - LogInfo("CacheRecordRmvEventsForCurrentQuestion: Updating CRActiveQuestion to %p for cache record %s, " - "Original question CurrentAnswers %d, new question CurrentAnswers %d, SuppressUnusable %d, SuppressQuery %d", - qptr, CRDisplayString(m,rr), q->CurrentAnswers, qptr->CurrentAnswers, qptr->SuppressUnusable, qptr->SuppressQuery); - - rr->CRActiveQuestion = qptr; // Question used to be active; new value may or may not be null - if (!qptr) m->rrcache_active--; // If no longer active, decrement rrcache_active count - } - AnswerCurrentQuestionWithResourceRecord(m, rr, QC_rmv); - if (m->CurrentQuestion != q) break; // If callback deleted q, then we're finished here - } - } - } - -mDNSlocal mDNSBool IsQuestionNew(mDNS *const m, DNSQuestion *question) - { - DNSQuestion *q; - for (q = m->NewQuestions; q; q = q->next) - if (q == question) return mDNStrue; - return mDNSfalse; - } - -mDNSlocal mDNSBool LocalRecordRmvEventsForQuestion(mDNS *const m, DNSQuestion *q) - { - AuthRecord *rr; - mDNSu32 slot; - AuthGroup *ag; - - if (m->CurrentQuestion) - LogMsg("LocalRecordRmvEventsForQuestion: ERROR m->CurrentQuestion already set: %##s (%s)", - m->CurrentQuestion->qname.c, DNSTypeName(m->CurrentQuestion->qtype)); - - if (IsQuestionNew(m, q)) - { - LogInfo("LocalRecordRmvEventsForQuestion: New Question %##s (%s)", q->qname.c, DNSTypeName(q->qtype)); - return mDNStrue; - } - m->CurrentQuestion = q; - slot = AuthHashSlot(&q->qname); - ag = AuthGroupForName(&m->rrauth, slot, q->qnamehash, &q->qname); - if (ag) - { - for (rr = ag->members; rr; rr=rr->next) - // Filter the /etc/hosts records - LocalOnly, Unique, A/AAAA/CNAME - if (LORecordAnswersAddressType(rr) && LocalOnlyRecordAnswersQuestion(rr, q)) - { - LogInfo("LocalRecordRmvEventsForQuestion: Delivering possible Rmv events with record %s", - ARDisplayString(m, rr)); - if (q->CurrentAnswers <= 0 || q->LOAddressAnswers <= 0) - { - LogMsg("LocalRecordRmvEventsForQuestion: ERROR!! CurrentAnswers or LOAddressAnswers is zero %p %##s" - " (%s) CurrentAnswers %d, LOAddressAnswers %d", q, q->qname.c, DNSTypeName(q->qtype), - q->CurrentAnswers, q->LOAddressAnswers); - continue; - } - AnswerLocalQuestionWithLocalAuthRecord(m, rr, QC_rmv); // MUST NOT dereference q again - if (m->CurrentQuestion != q) { m->CurrentQuestion = mDNSNULL; return mDNSfalse; } - } - } - m->CurrentQuestion = mDNSNULL; - return mDNStrue; - } - -// Returns false if the question got deleted while delivering the RMV events -// The caller should handle the case -mDNSlocal mDNSBool CacheRecordRmvEventsForQuestion(mDNS *const m, DNSQuestion *q) - { - if (m->CurrentQuestion) - LogMsg("CacheRecordRmvEventsForQuestion: ERROR m->CurrentQuestion already set: %##s (%s)", - m->CurrentQuestion->qname.c, DNSTypeName(m->CurrentQuestion->qtype)); - - // If it is a new question, we have not delivered any ADD events yet. So, don't deliver RMV events. - // If this question was answered using local auth records, then you can't deliver RMVs using cache - if (!IsQuestionNew(m, q) && !q->LOAddressAnswers) - { - m->CurrentQuestion = q; - CacheRecordRmvEventsForCurrentQuestion(m, q); - if (m->CurrentQuestion != q) { m->CurrentQuestion = mDNSNULL; return mDNSfalse; } - m->CurrentQuestion = mDNSNULL; - } - else { LogInfo("CacheRecordRmvEventsForQuestion: Question %p %##s (%s) is a new question", q, q->qname.c, DNSTypeName(q->qtype)); } - return mDNStrue; - } - -// The caller should hold the lock -mDNSexport void CheckSuppressUnusableQuestions(mDNS *const m) - { - DNSQuestion *q; - DNSQuestion *restart = mDNSNULL; - - // We look through all questions including new questions. During network change events, - // we potentially restart questions here in this function that ends up as new questions, - // which may be suppressed at this instance. Before it is handled we get another network - // event that changes the status e.g., address becomes available. If we did not process - // new questions, we would never change its SuppressQuery status. - // - // CurrentQuestion is used by RmvEventsForQuestion below. While delivering RMV events, the - // application callback can potentially stop the current question (detected by CurrentQuestion) or - // *any* other question which could be the next one that we may process here. RestartQuestion - // points to the "next" question which will be automatically advanced in mDNS_StopQuery_internal - // if the "next" question is stopped while the CurrentQuestion is stopped - if (m->RestartQuestion) - LogMsg("CheckSuppressUnusableQuestions: ERROR!! m->RestartQuestion already set: %##s (%s)", - m->RestartQuestion->qname.c, DNSTypeName(m->RestartQuestion->qtype)); - m->RestartQuestion = m->Questions; - while (m->RestartQuestion) - { - q = m->RestartQuestion; - m->RestartQuestion = q->next; - if (!mDNSOpaque16IsZero(q->TargetQID) && q->SuppressUnusable) - { - mDNSBool old = q->SuppressQuery; - q->SuppressQuery = ShouldSuppressQuery(m, &q->qname, q->qtype, q->InterfaceID); - if (q->SuppressQuery != old) - { - // NOTE: CacheRecordRmvEventsForQuestion will not generate RMV events for queries that have non-zero - // LOddressAnswers. Hence it is important that we call CacheRecordRmvEventsForQuestion before - // LocalRecordRmvEventsForQuestion (which decrements LOAddressAnswers) - - if (q->SuppressQuery) - { - // Previously it was not suppressed, Generate RMV events for the ADDs that we might have delivered before - // followed by a negative cache response. Temporarily turn off suppression so that - // AnswerCurrentQuestionWithResourceRecord can answer the question - q->SuppressQuery = mDNSfalse; - if (!CacheRecordRmvEventsForQuestion(m, q)) { LogInfo("CheckSuppressUnusableQuestions: Question deleted while delivering RMV events"); continue; } - q->SuppressQuery = mDNStrue; - } - - // SuppressUnusable does not affect questions that are answered from the local records (/etc/hosts) - // and SuppressQuery status does not mean anything for these questions. As we are going to stop the - // question below, we need to deliver the RMV events so that the ADDs that will be delivered during - // the restart will not be a duplicate ADD - if (!LocalRecordRmvEventsForQuestion(m, q)) { LogInfo("CheckSuppressUnusableQuestions: Question deleted while delivering RMV events"); continue; } - - // There are two cases here. - // - // 1. Previously it was suppressed and now it is not suppressed, restart the question so - // that it will start as a new question. Note that we can't just call ActivateUnicastQuery - // because when we get the response, if we had entries in the cache already, it will not answer - // this question if the cache entry did not change. Hence, we need to restart - // the query so that it can be answered from the cache. - // - // 2. Previously it was not suppressed and now it is suppressed. We need to restart the questions - // so that we redo the duplicate checks in mDNS_StartQuery_internal. A SuppressUnusable question - // is a duplicate of non-SuppressUnusable question if it is not suppressed (SuppressQuery is false). - // A SuppressUnusable question is not a duplicate of non-SuppressUnusable question if it is suppressed - // (SuppressQuery is true). The reason for this is that when a question is suppressed, we want an - // immediate response and not want to be blocked behind a question that is querying DNS servers. When - // the question is not suppressed, we don't want two active questions sending packets on the wire. - // This affects both efficiency and also the current design where there is only one active question - // pointed to from a cache entry. - // - // We restart queries in a two step process by first calling stop and build a temporary list which we - // will restart at the end. The main reason for the two step process is to handle duplicate questions. - // If there are duplicate questions, calling stop inherits the values from another question on the list (which - // will soon become the real question) including q->ThisQInterval which might be zero if it was - // suppressed before. At the end when we have restarted all questions, none of them is active as each - // inherits from one another and we need to reactivate one of the questions here which is a little hacky. - // - // It is much cleaner and less error prone to build a list of questions and restart at the end. - - LogInfo("CheckSuppressUnusableQuestions: Stop question %p %##s (%s)", q, q->qname.c, DNSTypeName(q->qtype)); - mDNS_StopQuery_internal(m, q); - q->next = restart; - restart = q; - } - } - } - while (restart) - { - q = restart; - restart = restart->next; - q->next = mDNSNULL; - LogInfo("CheckSuppressUnusableQuestions: Start question %p %##s (%s)", q, q->qname.c, DNSTypeName(q->qtype)); - mDNS_StartQuery_internal(m, q); - } - } - -mDNSexport mStatus mDNS_StartQuery_internal(mDNS *const m, DNSQuestion *const question) - { - if (question->Target.type && !ValidQuestionTarget(question)) - { - LogMsg("mDNS_StartQuery_internal: Warning! Target.type = %ld port = %u (Client forgot to initialize before calling mDNS_StartQuery? for question %##s)", - question->Target.type, mDNSVal16(question->TargetPort), question->qname.c); - question->Target.type = mDNSAddrType_None; - } - - if (!question->Target.type) question->TargetPort = zeroIPPort; // If no question->Target specified clear TargetPort - - question->TargetQID = -#ifndef UNICAST_DISABLED - (question->Target.type || Question_uDNS(question)) ? mDNS_NewMessageID(m) : -#endif // UNICAST_DISABLED - zeroID; - - debugf("mDNS_StartQuery: %##s (%s)", question->qname.c, DNSTypeName(question->qtype)); - - if (m->rrcache_size == 0) // Can't do queries if we have no cache space allocated - return(mStatus_NoCache); - else - { - int i; - DNSQuestion **q; - - if (!ValidateDomainName(&question->qname)) - { - LogMsg("Attempt to start query with invalid qname %##s (%s)", question->qname.c, DNSTypeName(question->qtype)); - return(mStatus_Invalid); - } - - // Note: It important that new questions are appended at the *end* of the list, not prepended at the start - q = &m->Questions; - if (question->InterfaceID == mDNSInterface_LocalOnly || question->InterfaceID == mDNSInterface_P2P) q = &m->LocalOnlyQuestions; - while (*q && *q != question) q=&(*q)->next; - - if (*q) - { - LogMsg("Error! Tried to add a question %##s (%s) %p that's already in the active list", - question->qname.c, DNSTypeName(question->qtype), question); - return(mStatus_AlreadyRegistered); - } - - *q = question; - - // If this question is referencing a specific interface, verify it exists - if (question->InterfaceID && question->InterfaceID != mDNSInterface_LocalOnly && question->InterfaceID != mDNSInterface_Unicast && question->InterfaceID != mDNSInterface_P2P) - { - NetworkInterfaceInfo *intf = FirstInterfaceForID(m, question->InterfaceID); - if (!intf) - LogMsg("Note: InterfaceID %p for question %##s (%s) not currently found in active interface list", - question->InterfaceID, question->qname.c, DNSTypeName(question->qtype)); - } - - // Note: In the case where we already have the answer to this question in our cache, that may be all the client - // wanted, and they may immediately cancel their question. In this case, sending an actual query on the wire would - // be a waste. For that reason, we schedule our first query to go out in half a second (InitialQuestionInterval). - // If AnswerNewQuestion() finds that we have *no* relevant answers currently in our cache, then it will accelerate - // that to go out immediately. - question->next = mDNSNULL; - question->qnamehash = DomainNameHashValue(&question->qname); // MUST do this before FindDuplicateQuestion() - question->DelayAnswering = CheckForSoonToExpireRecords(m, &question->qname, question->qnamehash, HashSlot(&question->qname)); - question->LastQTime = m->timenow; - question->ThisQInterval = InitialQuestionInterval; // MUST be > zero for an active question - question->ExpectUnicastResp = 0; - question->LastAnswerPktNum = m->PktNum; - question->RecentAnswerPkts = 0; - question->CurrentAnswers = 0; - question->LargeAnswers = 0; - question->UniqueAnswers = 0; - question->LOAddressAnswers = 0; - question->FlappingInterface1 = mDNSNULL; - question->FlappingInterface2 = mDNSNULL; - // Must do AuthInfo and SuppressQuery before calling FindDuplicateQuestion() - question->AuthInfo = GetAuthInfoForQuestion(m, question); - if (question->SuppressUnusable) - question->SuppressQuery = ShouldSuppressQuery(m, &question->qname, question->qtype, question->InterfaceID); - else - question->SuppressQuery = 0; - question->DuplicateOf = FindDuplicateQuestion(m, question); - question->NextInDQList = mDNSNULL; - question->SendQNow = mDNSNULL; - question->SendOnAll = mDNSfalse; - question->RequestUnicast = 0; - question->LastQTxTime = m->timenow; - question->CNAMEReferrals = 0; - - // We'll create our question->LocalSocket on demand, if needed. - // We won't need one for duplicate questions, or from questions answered immediately out of the cache. - // We also don't need one for LLQs because (when we're using NAT) we want them all to share a single - // NAT mapping for receiving inbound add/remove events. - question->LocalSocket = mDNSNULL; - question->deliverAddEvents = mDNSfalse; - question->qDNSServer = mDNSNULL; - question->unansweredQueries = 0; - question->nta = mDNSNULL; - question->servAddr = zeroAddr; - question->servPort = zeroIPPort; - question->tcp = mDNSNULL; - question->NoAnswer = NoAnswer_Normal; - - question->state = LLQ_InitialRequest; - question->ReqLease = 0; - question->expire = 0; - question->ntries = 0; - question->id = zeroOpaque64; - question->validDNSServers = zeroOpaque64; - question->triedAllServersOnce = 0; - question->noServerResponse = 0; - question->StopTime = 0; - if (question->WakeOnResolve) - { - question->WakeOnResolveCount = InitialWakeOnResolveCount; - mDNS_PurgeBeforeResolve(m, question); - } - else - question->WakeOnResolveCount = 0; - - if (question->DuplicateOf) question->AuthInfo = question->DuplicateOf->AuthInfo; - - for (i=0; iDupSuppress[i].InterfaceID = mDNSNULL; - - debugf("mDNS_StartQuery: Question %##s (%s) Interface %p Now %d Send in %d Answer in %d (%p) %s (%p)", - question->qname.c, DNSTypeName(question->qtype), question->InterfaceID, m->timenow, - NextQSendTime(question) - m->timenow, - question->DelayAnswering ? question->DelayAnswering - m->timenow : 0, - question, question->DuplicateOf ? "duplicate of" : "not duplicate", question->DuplicateOf); - - if (question->DelayAnswering) - LogInfo("mDNS_StartQuery_internal: Delaying answering for %d ticks while cache stabilizes for %##s (%s)", - question->DelayAnswering - m->timenow, question->qname.c, DNSTypeName(question->qtype)); - - if (question->InterfaceID == mDNSInterface_LocalOnly || question->InterfaceID == mDNSInterface_P2P) - { - if (!m->NewLocalOnlyQuestions) m->NewLocalOnlyQuestions = question; - } - else - { - if (!m->NewQuestions) m->NewQuestions = question; - - // If the question's id is non-zero, then it's Wide Area - // MUST NOT do this Wide Area setup until near the end of - // mDNS_StartQuery_internal -- this code may itself issue queries (e.g. SOA, - // NS, etc.) and if we haven't finished setting up our own question and setting - // m->NewQuestions if necessary then we could end up recursively re-entering - // this routine with the question list data structures in an inconsistent state. - if (!mDNSOpaque16IsZero(question->TargetQID)) - { - // Duplicate questions should have the same DNSServers so that when we find - // a matching resource record, all of them get the answers. Calling GetServerForQuestion - // for the duplicate question may get a different DNS server from the original question - mDNSu32 timeout = SetValidDNSServers(m, question); - // We set the timeout whenever mDNS_StartQuery_internal is called. This means if we have - // a networking change/search domain change that calls this function again we keep - // reinitializing the timeout value which means it may never timeout. If this becomes - // a common case in the future, we can easily fix this by adding extra state that - // indicates that we have already set the StopTime. - if (question->TimeoutQuestion) - question->StopTime = NonZeroTime(m->timenow + timeout * mDNSPlatformOneSecond); - if (question->DuplicateOf) - { - question->validDNSServers = question->DuplicateOf->validDNSServers; - question->qDNSServer = question->DuplicateOf->qDNSServer; - LogInfo("mDNS_StartQuery_internal: Duplicate question %p (%p) %##s (%s), Timeout %d, DNS Server %#a:%d", - question, question->DuplicateOf, question->qname.c, DNSTypeName(question->qtype), timeout, - question->qDNSServer ? &question->qDNSServer->addr : mDNSNULL, - mDNSVal16(question->qDNSServer ? question->qDNSServer->port : zeroIPPort)); - } - else - { - question->qDNSServer = GetServerForQuestion(m, question); - LogInfo("mDNS_StartQuery_internal: question %p %##s (%s) Timeout %d, DNS Server %#a:%d", - question, question->qname.c, DNSTypeName(question->qtype), timeout, - question->qDNSServer ? &question->qDNSServer->addr : mDNSNULL, - mDNSVal16(question->qDNSServer ? question->qDNSServer->port : zeroIPPort)); - } - ActivateUnicastQuery(m, question, mDNSfalse); - - // If long-lived query, and we don't have our NAT mapping active, start it now - if (question->LongLived && !m->LLQNAT.clientContext) - { - m->LLQNAT.Protocol = NATOp_MapUDP; - m->LLQNAT.IntPort = m->UnicastPort4; - m->LLQNAT.RequestedPort = m->UnicastPort4; - m->LLQNAT.clientCallback = LLQNATCallback; - m->LLQNAT.clientContext = (void*)1; // Means LLQ NAT Traversal is active - mDNS_StartNATOperation_internal(m, &m->LLQNAT); - } - -#if APPLE_OSX_mDNSResponder - if (question->LongLived) - UpdateAutoTunnelDomainStatuses(m); -#endif - - } - else - { - if (question->TimeoutQuestion) - question->StopTime = NonZeroTime(m->timenow + GetTimeoutForMcastQuestion(m, question) * mDNSPlatformOneSecond); - } - if (question->StopTime) SetNextQueryStopTime(m, question); - SetNextQueryTime(m,question); - } - - return(mStatus_NoError); - } - } - -// CancelGetZoneData is an internal routine (i.e. must be called with the lock already held) -mDNSexport void CancelGetZoneData(mDNS *const m, ZoneData *nta) - { - debugf("CancelGetZoneData %##s (%s)", nta->question.qname.c, DNSTypeName(nta->question.qtype)); - // This function may be called anytime to free the zone information.The question may or may not have stopped. - // If it was already stopped, mDNS_StopQuery_internal would have set q->ThisQInterval to -1 and should not - // call it again - if (nta->question.ThisQInterval != -1) - { - mDNS_StopQuery_internal(m, &nta->question); - if (nta->question.ThisQInterval != -1) - LogMsg("CancelGetZoneData: Question %##s (%s) ThisQInterval %d not -1", nta->question.qname.c, DNSTypeName(nta->question.qtype), nta->question.ThisQInterval); - } - mDNSPlatformMemFree(nta); - } - -mDNSexport mStatus mDNS_StopQuery_internal(mDNS *const m, DNSQuestion *const question) - { - const mDNSu32 slot = HashSlot(&question->qname); - CacheGroup *cg = CacheGroupForName(m, slot, question->qnamehash, &question->qname); - CacheRecord *rr; - DNSQuestion **qp = &m->Questions; - - //LogInfo("mDNS_StopQuery_internal %##s (%s)", question->qname.c, DNSTypeName(question->qtype)); - - if (question->InterfaceID == mDNSInterface_LocalOnly || question->InterfaceID == mDNSInterface_P2P) qp = &m->LocalOnlyQuestions; - while (*qp && *qp != question) qp=&(*qp)->next; - if (*qp) *qp = (*qp)->next; - else - { -#if !ForceAlerts - if (question->ThisQInterval >= 0) // Only log error message if the query was supposed to be active -#endif - LogMsg("mDNS_StopQuery_internal: Question %##s (%s) not found in active list", - question->qname.c, DNSTypeName(question->qtype)); -#if ForceAlerts - *(long*)0 = 0; -#endif - return(mStatus_BadReferenceErr); - } - - // Take care to cut question from list *before* calling UpdateQuestionDuplicates - UpdateQuestionDuplicates(m, question); - // But don't trash ThisQInterval until afterwards. - question->ThisQInterval = -1; - - // If there are any cache records referencing this as their active question, then see if there is any - // other question that is also referencing them, else their CRActiveQuestion needs to get set to NULL. - for (rr = cg ? cg->members : mDNSNULL; rr; rr=rr->next) - { - if (rr->CRActiveQuestion == question) - { - DNSQuestion *q; - // Checking for ActiveQuestion filters questions that are suppressed also - // as suppressed questions are not active - for (q = m->Questions; q; q=q->next) // Scan our list of questions - if (ActiveQuestion(q) && ResourceRecordAnswersQuestion(&rr->resrec, q)) - break; - if (q) - debugf("mDNS_StopQuery_internal: Updating CRActiveQuestion to %p for cache record %s, Original question CurrentAnswers %d, new question " - "CurrentAnswers %d, SuppressQuery %d", q, CRDisplayString(m,rr), question->CurrentAnswers, q->CurrentAnswers, q->SuppressQuery); - rr->CRActiveQuestion = q; // Question used to be active; new value may or may not be null - if (!q) m->rrcache_active--; // If no longer active, decrement rrcache_active count - } - } - - // If we just deleted the question that CacheRecordAdd() or CacheRecordRmv() is about to look at, - // bump its pointer forward one question. - if (m->CurrentQuestion == question) - { - debugf("mDNS_StopQuery_internal: Just deleted the currently active question: %##s (%s)", - question->qname.c, DNSTypeName(question->qtype)); - m->CurrentQuestion = question->next; - } - - if (m->NewQuestions == question) - { - debugf("mDNS_StopQuery_internal: Just deleted a new question that wasn't even answered yet: %##s (%s)", - question->qname.c, DNSTypeName(question->qtype)); - m->NewQuestions = question->next; - } - - if (m->NewLocalOnlyQuestions == question) m->NewLocalOnlyQuestions = question->next; - - if (m->RestartQuestion == question) - { - LogMsg("mDNS_StopQuery_internal: Just deleted the current restart question: %##s (%s)", - question->qname.c, DNSTypeName(question->qtype)); - m->RestartQuestion = question->next; - } - - // Take care not to trash question->next until *after* we've updated m->CurrentQuestion and m->NewQuestions - question->next = mDNSNULL; - - // LogMsg("mDNS_StopQuery_internal: Question %##s (%s) removed", question->qname.c, DNSTypeName(question->qtype)); - - // And finally, cancel any associated GetZoneData operation that's still running. - // Must not do this until last, because there's a good chance the GetZoneData question is the next in the list, - // so if we delete it earlier in this routine, we could find that our "question->next" pointer above is already - // invalid before we even use it. By making sure that we update m->CurrentQuestion and m->NewQuestions if necessary - // *first*, then they're all ready to be updated a second time if necessary when we cancel our GetZoneData query. - if (question->tcp) { DisposeTCPConn(question->tcp); question->tcp = mDNSNULL; } - if (question->LocalSocket) { mDNSPlatformUDPClose(question->LocalSocket); question->LocalSocket = mDNSNULL; } - if (!mDNSOpaque16IsZero(question->TargetQID) && question->LongLived) - { - // Scan our list to see if any more wide-area LLQs remain. If not, stop our NAT Traversal. - DNSQuestion *q; - for (q = m->Questions; q; q=q->next) - if (!mDNSOpaque16IsZero(q->TargetQID) && q->LongLived) break; - if (!q) - { - if (!m->LLQNAT.clientContext) // Should never happen, but just in case... - LogMsg("mDNS_StopQuery ERROR LLQNAT.clientContext NULL"); - else - { - LogInfo("Stopping LLQNAT"); - mDNS_StopNATOperation_internal(m, &m->LLQNAT); - m->LLQNAT.clientContext = mDNSNULL; // Means LLQ NAT Traversal not running - } - } - - // If necessary, tell server it can delete this LLQ state - if (question->state == LLQ_Established) - { - question->ReqLease = 0; - sendLLQRefresh(m, question); - // If we need need to make a TCP connection to cancel the LLQ, that's going to take a little while. - // We clear the tcp->question backpointer so that when the TCP connection completes, it doesn't - // crash trying to access our cancelled question, but we don't cancel the TCP operation itself -- - // we let that run out its natural course and complete asynchronously. - if (question->tcp) - { - question->tcp->question = mDNSNULL; - question->tcp = mDNSNULL; - } - } -#if APPLE_OSX_mDNSResponder - UpdateAutoTunnelDomainStatuses(m); -#endif - } - // wait until we send the refresh above which needs the nta - if (question->nta) { CancelGetZoneData(m, question->nta); question->nta = mDNSNULL; } - - return(mStatus_NoError); - } - -mDNSexport mStatus mDNS_StartQuery(mDNS *const m, DNSQuestion *const question) - { - mStatus status; - mDNS_Lock(m); - status = mDNS_StartQuery_internal(m, question); - mDNS_Unlock(m); - return(status); - } - -mDNSexport mStatus mDNS_StopQuery(mDNS *const m, DNSQuestion *const question) - { - mStatus status; - mDNS_Lock(m); - status = mDNS_StopQuery_internal(m, question); - mDNS_Unlock(m); - return(status); - } - -// Note that mDNS_StopQueryWithRemoves() does not currently implement the full generality of the other APIs -// Specifically, question callbacks invoked as a result of this call cannot themselves make API calls. -// We invoke the callback without using mDNS_DropLockBeforeCallback/mDNS_ReclaimLockAfterCallback -// specifically to catch and report if the client callback does try to make API calls -mDNSexport mStatus mDNS_StopQueryWithRemoves(mDNS *const m, DNSQuestion *const question) - { - mStatus status; - DNSQuestion *qq; - mDNS_Lock(m); - - // Check if question is new -- don't want to give remove events for a question we haven't even answered yet - for (qq = m->NewQuestions; qq; qq=qq->next) if (qq == question) break; - - status = mDNS_StopQuery_internal(m, question); - if (status == mStatus_NoError && !qq) - { - const CacheRecord *rr; - const mDNSu32 slot = HashSlot(&question->qname); - CacheGroup *const cg = CacheGroupForName(m, slot, question->qnamehash, &question->qname); - LogInfo("Generating terminal removes for %##s (%s)", question->qname.c, DNSTypeName(question->qtype)); - for (rr = cg ? cg->members : mDNSNULL; rr; rr=rr->next) - if (rr->resrec.RecordType != kDNSRecordTypePacketNegative && SameNameRecordAnswersQuestion(&rr->resrec, question)) - { - // Don't use mDNS_DropLockBeforeCallback() here, since we don't allow API calls - if (question->QuestionCallback) - question->QuestionCallback(m, question, &rr->resrec, mDNSfalse); - } - } - mDNS_Unlock(m); - return(status); - } - -mDNSexport mStatus mDNS_Reconfirm(mDNS *const m, CacheRecord *const cr) - { - mStatus status; - mDNS_Lock(m); - status = mDNS_Reconfirm_internal(m, cr, kDefaultReconfirmTimeForNoAnswer); - if (status == mStatus_NoError) ReconfirmAntecedents(m, cr->resrec.name, cr->resrec.namehash, 0); - mDNS_Unlock(m); - return(status); - } - -mDNSexport mStatus mDNS_ReconfirmByValue(mDNS *const m, ResourceRecord *const rr) - { - mStatus status = mStatus_BadReferenceErr; - CacheRecord *cr; - mDNS_Lock(m); - cr = FindIdenticalRecordInCache(m, rr); - debugf("mDNS_ReconfirmByValue: %p %s", cr, RRDisplayString(m, rr)); - if (cr) status = mDNS_Reconfirm_internal(m, cr, kDefaultReconfirmTimeForNoAnswer); - if (status == mStatus_NoError) ReconfirmAntecedents(m, cr->resrec.name, cr->resrec.namehash, 0); - mDNS_Unlock(m); - return(status); - } - -mDNSlocal mStatus mDNS_StartBrowse_internal(mDNS *const m, DNSQuestion *const question, - const domainname *const srv, const domainname *const domain, - const mDNSInterfaceID InterfaceID, mDNSBool ForceMCast, mDNSQuestionCallback *Callback, void *Context) - { - question->InterfaceID = InterfaceID; - question->Target = zeroAddr; - question->qtype = kDNSType_PTR; - question->qclass = kDNSClass_IN; - question->LongLived = mDNStrue; - question->ExpectUnique = mDNSfalse; - question->ForceMCast = ForceMCast; - question->ReturnIntermed = mDNSfalse; - question->SuppressUnusable = mDNSfalse; - question->SearchListIndex = 0; - question->AppendSearchDomains = 0; - question->RetryWithSearchDomains = mDNSfalse; - question->TimeoutQuestion = 0; - question->WakeOnResolve = 0; - question->qnameOrig = mDNSNULL; - question->QuestionCallback = Callback; - question->QuestionContext = Context; - if (!ConstructServiceName(&question->qname, mDNSNULL, srv, domain)) return(mStatus_BadParamErr); - - return(mDNS_StartQuery_internal(m, question)); - } - -mDNSexport mStatus mDNS_StartBrowse(mDNS *const m, DNSQuestion *const question, - const domainname *const srv, const domainname *const domain, - const mDNSInterfaceID InterfaceID, mDNSBool ForceMCast, mDNSQuestionCallback *Callback, void *Context) - { - mStatus status; - mDNS_Lock(m); - status = mDNS_StartBrowse_internal(m, question, srv, domain, InterfaceID, ForceMCast, Callback, Context); - mDNS_Unlock(m); - return(status); - } - -mDNSlocal mDNSBool MachineHasActiveIPv6(mDNS *const m) - { - NetworkInterfaceInfo *intf; - for (intf = m->HostInterfaces; intf; intf = intf->next) - if (intf->ip.type == mDNSAddrType_IPv6) return(mDNStrue); - return(mDNSfalse); - } - -mDNSlocal void FoundServiceInfoSRV(mDNS *const m, DNSQuestion *question, const ResourceRecord *const answer, QC_result AddRecord) - { - ServiceInfoQuery *query = (ServiceInfoQuery *)question->QuestionContext; - mDNSBool PortChanged = !mDNSSameIPPort(query->info->port, answer->rdata->u.srv.port); - if (!AddRecord) return; - if (answer->rrtype != kDNSType_SRV) return; - - query->info->port = answer->rdata->u.srv.port; - - // If this is our first answer, then set the GotSRV flag and start the address query - if (!query->GotSRV) - { - query->GotSRV = mDNStrue; - query->qAv4.InterfaceID = answer->InterfaceID; - AssignDomainName(&query->qAv4.qname, &answer->rdata->u.srv.target); - query->qAv6.InterfaceID = answer->InterfaceID; - AssignDomainName(&query->qAv6.qname, &answer->rdata->u.srv.target); - mDNS_StartQuery(m, &query->qAv4); - // Only do the AAAA query if this machine actually has IPv6 active - if (MachineHasActiveIPv6(m)) mDNS_StartQuery(m, &query->qAv6); - } - // If this is not our first answer, only re-issue the address query if the target host name has changed - else if ((query->qAv4.InterfaceID != query->qSRV.InterfaceID && query->qAv4.InterfaceID != answer->InterfaceID) || - !SameDomainName(&query->qAv4.qname, &answer->rdata->u.srv.target)) - { - mDNS_StopQuery(m, &query->qAv4); - if (query->qAv6.ThisQInterval >= 0) mDNS_StopQuery(m, &query->qAv6); - if (SameDomainName(&query->qAv4.qname, &answer->rdata->u.srv.target) && !PortChanged) - { - // If we get here, it means: - // 1. This is not our first SRV answer - // 2. The interface ID is different, but the target host and port are the same - // This implies that we're seeing the exact same SRV record on more than one interface, so we should - // make our address queries at least as broad as the original SRV query so that we catch all the answers. - query->qAv4.InterfaceID = query->qSRV.InterfaceID; // Will be mDNSInterface_Any, or a specific interface - query->qAv6.InterfaceID = query->qSRV.InterfaceID; - } - else - { - query->qAv4.InterfaceID = answer->InterfaceID; - AssignDomainName(&query->qAv4.qname, &answer->rdata->u.srv.target); - query->qAv6.InterfaceID = answer->InterfaceID; - AssignDomainName(&query->qAv6.qname, &answer->rdata->u.srv.target); - } - debugf("FoundServiceInfoSRV: Restarting address queries for %##s (%s)", query->qAv4.qname.c, DNSTypeName(query->qAv4.qtype)); - mDNS_StartQuery(m, &query->qAv4); - // Only do the AAAA query if this machine actually has IPv6 active - if (MachineHasActiveIPv6(m)) mDNS_StartQuery(m, &query->qAv6); - } - else if (query->ServiceInfoQueryCallback && query->GotADD && query->GotTXT && PortChanged) - { - if (++query->Answers >= 100) - debugf("**** WARNING **** Have given %lu answers for %##s (SRV) %##s %u", - query->Answers, query->qSRV.qname.c, answer->rdata->u.srv.target.c, - mDNSVal16(answer->rdata->u.srv.port)); - query->ServiceInfoQueryCallback(m, query); - } - // CAUTION: MUST NOT do anything more with query after calling query->Callback(), because the client's - // callback function is allowed to do anything, including deleting this query and freeing its memory. - } - -mDNSlocal void FoundServiceInfoTXT(mDNS *const m, DNSQuestion *question, const ResourceRecord *const answer, QC_result AddRecord) - { - ServiceInfoQuery *query = (ServiceInfoQuery *)question->QuestionContext; - if (!AddRecord) return; - if (answer->rrtype != kDNSType_TXT) return; - if (answer->rdlength > sizeof(query->info->TXTinfo)) return; - - query->GotTXT = mDNStrue; - query->info->TXTlen = answer->rdlength; - query->info->TXTinfo[0] = 0; // In case answer->rdlength is zero - mDNSPlatformMemCopy(query->info->TXTinfo, answer->rdata->u.txt.c, answer->rdlength); - - verbosedebugf("FoundServiceInfoTXT: %##s GotADD=%d", query->info->name.c, query->GotADD); - - // CAUTION: MUST NOT do anything more with query after calling query->Callback(), because the client's - // callback function is allowed to do anything, including deleting this query and freeing its memory. - if (query->ServiceInfoQueryCallback && query->GotADD) - { - if (++query->Answers >= 100) - debugf("**** WARNING **** have given %lu answers for %##s (TXT) %#s...", - query->Answers, query->qSRV.qname.c, answer->rdata->u.txt.c); - query->ServiceInfoQueryCallback(m, query); - } - } - -mDNSlocal void FoundServiceInfo(mDNS *const m, DNSQuestion *question, const ResourceRecord *const answer, QC_result AddRecord) - { - ServiceInfoQuery *query = (ServiceInfoQuery *)question->QuestionContext; - //LogInfo("FoundServiceInfo %d %s", AddRecord, RRDisplayString(m, answer)); - if (!AddRecord) return; - - if (answer->rrtype == kDNSType_A) - { - query->info->ip.type = mDNSAddrType_IPv4; - query->info->ip.ip.v4 = answer->rdata->u.ipv4; - } - else if (answer->rrtype == kDNSType_AAAA) - { - query->info->ip.type = mDNSAddrType_IPv6; - query->info->ip.ip.v6 = answer->rdata->u.ipv6; - } - else - { - debugf("FoundServiceInfo: answer %##s type %d (%s) unexpected", answer->name->c, answer->rrtype, DNSTypeName(answer->rrtype)); - return; - } - - query->GotADD = mDNStrue; - query->info->InterfaceID = answer->InterfaceID; - - verbosedebugf("FoundServiceInfo v%ld: %##s GotTXT=%d", query->info->ip.type, query->info->name.c, query->GotTXT); - - // CAUTION: MUST NOT do anything more with query after calling query->Callback(), because the client's - // callback function is allowed to do anything, including deleting this query and freeing its memory. - if (query->ServiceInfoQueryCallback && query->GotTXT) - { - if (++query->Answers >= 100) - debugf(answer->rrtype == kDNSType_A ? - "**** WARNING **** have given %lu answers for %##s (A) %.4a" : - "**** WARNING **** have given %lu answers for %##s (AAAA) %.16a", - query->Answers, query->qSRV.qname.c, &answer->rdata->u.data); - query->ServiceInfoQueryCallback(m, query); - } - } - -// On entry, the client must have set the name and InterfaceID fields of the ServiceInfo structure -// If the query is not interface-specific, then InterfaceID may be zero -// Each time the Callback is invoked, the remainder of the fields will have been filled in -// In addition, InterfaceID will be updated to give the interface identifier corresponding to that response -mDNSexport mStatus mDNS_StartResolveService(mDNS *const m, - ServiceInfoQuery *query, ServiceInfo *info, mDNSServiceInfoQueryCallback *Callback, void *Context) - { - mStatus status; - mDNS_Lock(m); - - query->qSRV.ThisQInterval = -1; // So that mDNS_StopResolveService() knows whether to cancel this question - query->qSRV.InterfaceID = info->InterfaceID; - query->qSRV.Target = zeroAddr; - AssignDomainName(&query->qSRV.qname, &info->name); - query->qSRV.qtype = kDNSType_SRV; - query->qSRV.qclass = kDNSClass_IN; - query->qSRV.LongLived = mDNSfalse; - query->qSRV.ExpectUnique = mDNStrue; - query->qSRV.ForceMCast = mDNSfalse; - query->qSRV.ReturnIntermed = mDNSfalse; - query->qSRV.SuppressUnusable = mDNSfalse; - query->qSRV.SearchListIndex = 0; - query->qSRV.AppendSearchDomains = 0; - query->qSRV.RetryWithSearchDomains = mDNSfalse; - query->qSRV.TimeoutQuestion = 0; - query->qSRV.WakeOnResolve = 0; - query->qSRV.qnameOrig = mDNSNULL; - query->qSRV.QuestionCallback = FoundServiceInfoSRV; - query->qSRV.QuestionContext = query; - - query->qTXT.ThisQInterval = -1; // So that mDNS_StopResolveService() knows whether to cancel this question - query->qTXT.InterfaceID = info->InterfaceID; - query->qTXT.Target = zeroAddr; - AssignDomainName(&query->qTXT.qname, &info->name); - query->qTXT.qtype = kDNSType_TXT; - query->qTXT.qclass = kDNSClass_IN; - query->qTXT.LongLived = mDNSfalse; - query->qTXT.ExpectUnique = mDNStrue; - query->qTXT.ForceMCast = mDNSfalse; - query->qTXT.ReturnIntermed = mDNSfalse; - query->qTXT.SuppressUnusable = mDNSfalse; - query->qTXT.SearchListIndex = 0; - query->qTXT.AppendSearchDomains = 0; - query->qTXT.RetryWithSearchDomains = mDNSfalse; - query->qTXT.TimeoutQuestion = 0; - query->qTXT.WakeOnResolve = 0; - query->qTXT.qnameOrig = mDNSNULL; - query->qTXT.QuestionCallback = FoundServiceInfoTXT; - query->qTXT.QuestionContext = query; - - query->qAv4.ThisQInterval = -1; // So that mDNS_StopResolveService() knows whether to cancel this question - query->qAv4.InterfaceID = info->InterfaceID; - query->qAv4.Target = zeroAddr; - query->qAv4.qname.c[0] = 0; - query->qAv4.qtype = kDNSType_A; - query->qAv4.qclass = kDNSClass_IN; - query->qAv4.LongLived = mDNSfalse; - query->qAv4.ExpectUnique = mDNStrue; - query->qAv4.ForceMCast = mDNSfalse; - query->qAv4.ReturnIntermed = mDNSfalse; - query->qAv4.SuppressUnusable = mDNSfalse; - query->qAv4.SearchListIndex = 0; - query->qAv4.AppendSearchDomains = 0; - query->qAv4.RetryWithSearchDomains = mDNSfalse; - query->qAv4.TimeoutQuestion = 0; - query->qAv4.WakeOnResolve = 0; - query->qAv4.qnameOrig = mDNSNULL; - query->qAv4.QuestionCallback = FoundServiceInfo; - query->qAv4.QuestionContext = query; - - query->qAv6.ThisQInterval = -1; // So that mDNS_StopResolveService() knows whether to cancel this question - query->qAv6.InterfaceID = info->InterfaceID; - query->qAv6.Target = zeroAddr; - query->qAv6.qname.c[0] = 0; - query->qAv6.qtype = kDNSType_AAAA; - query->qAv6.qclass = kDNSClass_IN; - query->qAv6.LongLived = mDNSfalse; - query->qAv6.ExpectUnique = mDNStrue; - query->qAv6.ForceMCast = mDNSfalse; - query->qAv6.ReturnIntermed = mDNSfalse; - query->qAv6.SuppressUnusable = mDNSfalse; - query->qAv6.SearchListIndex = 0; - query->qAv6.AppendSearchDomains = 0; - query->qAv6.RetryWithSearchDomains = mDNSfalse; - query->qAv6.TimeoutQuestion = 0; - query->qAv6.WakeOnResolve = 0; - query->qAv6.qnameOrig = mDNSNULL; - query->qAv6.QuestionCallback = FoundServiceInfo; - query->qAv6.QuestionContext = query; - - query->GotSRV = mDNSfalse; - query->GotTXT = mDNSfalse; - query->GotADD = mDNSfalse; - query->Answers = 0; - - query->info = info; - query->ServiceInfoQueryCallback = Callback; - query->ServiceInfoQueryContext = Context; - -// info->name = Must already be set up by client -// info->interface = Must already be set up by client - info->ip = zeroAddr; - info->port = zeroIPPort; - info->TXTlen = 0; - - // We use mDNS_StartQuery_internal here because we're already holding the lock - status = mDNS_StartQuery_internal(m, &query->qSRV); - if (status == mStatus_NoError) status = mDNS_StartQuery_internal(m, &query->qTXT); - if (status != mStatus_NoError) mDNS_StopResolveService(m, query); - - mDNS_Unlock(m); - return(status); - } - -mDNSexport void mDNS_StopResolveService (mDNS *const m, ServiceInfoQuery *q) - { - mDNS_Lock(m); - // We use mDNS_StopQuery_internal here because we're already holding the lock - if (q->qSRV.ThisQInterval >= 0) mDNS_StopQuery_internal(m, &q->qSRV); - if (q->qTXT.ThisQInterval >= 0) mDNS_StopQuery_internal(m, &q->qTXT); - if (q->qAv4.ThisQInterval >= 0) mDNS_StopQuery_internal(m, &q->qAv4); - if (q->qAv6.ThisQInterval >= 0) mDNS_StopQuery_internal(m, &q->qAv6); - mDNS_Unlock(m); - } - -mDNSexport mStatus mDNS_GetDomains(mDNS *const m, DNSQuestion *const question, mDNS_DomainType DomainType, const domainname *dom, - const mDNSInterfaceID InterfaceID, mDNSQuestionCallback *Callback, void *Context) - { - question->InterfaceID = InterfaceID; - question->Target = zeroAddr; - question->qtype = kDNSType_PTR; - question->qclass = kDNSClass_IN; - question->LongLived = mDNSfalse; - question->ExpectUnique = mDNSfalse; - question->ForceMCast = mDNSfalse; - question->ReturnIntermed = mDNSfalse; - question->SuppressUnusable = mDNSfalse; - question->SearchListIndex = 0; - question->AppendSearchDomains = 0; - question->RetryWithSearchDomains = mDNSfalse; - question->TimeoutQuestion = 0; - question->WakeOnResolve = 0; - question->qnameOrig = mDNSNULL; - question->QuestionCallback = Callback; - question->QuestionContext = Context; - if (DomainType > mDNS_DomainTypeMax) return(mStatus_BadParamErr); - if (!MakeDomainNameFromDNSNameString(&question->qname, mDNS_DomainTypeNames[DomainType])) return(mStatus_BadParamErr); - if (!dom) dom = &localdomain; - if (!AppendDomainName(&question->qname, dom)) return(mStatus_BadParamErr); - return(mDNS_StartQuery(m, question)); - } - -// *************************************************************************** -#if COMPILER_LIKES_PRAGMA_MARK -#pragma mark - -#pragma mark - Responder Functions -#endif - -mDNSexport mStatus mDNS_Register(mDNS *const m, AuthRecord *const rr) - { - mStatus status; - mDNS_Lock(m); - status = mDNS_Register_internal(m, rr); - mDNS_Unlock(m); - return(status); - } - -mDNSexport mStatus mDNS_Update(mDNS *const m, AuthRecord *const rr, mDNSu32 newttl, - const mDNSu16 newrdlength, RData *const newrdata, mDNSRecordUpdateCallback *Callback) - { - if (!ValidateRData(rr->resrec.rrtype, newrdlength, newrdata)) - { - LogMsg("Attempt to update record with invalid rdata: %s", GetRRDisplayString_rdb(&rr->resrec, &newrdata->u, m->MsgBuffer)); - return(mStatus_Invalid); - } - - mDNS_Lock(m); - - // If TTL is unspecified, leave TTL unchanged - if (newttl == 0) newttl = rr->resrec.rroriginalttl; - - // If we already have an update queued up which has not gone through yet, give the client a chance to free that memory - if (rr->NewRData) - { - RData *n = rr->NewRData; - rr->NewRData = mDNSNULL; // Clear the NewRData pointer ... - if (rr->UpdateCallback) - rr->UpdateCallback(m, rr, n, rr->newrdlength); // ...and let the client free this memory, if necessary - } - - rr->NewRData = newrdata; - rr->newrdlength = newrdlength; - rr->UpdateCallback = Callback; - -#ifndef UNICAST_DISABLED - if (rr->ARType != AuthRecordLocalOnly && rr->ARType != AuthRecordP2P && !IsLocalDomain(rr->resrec.name)) - { - mStatus status = uDNS_UpdateRecord(m, rr); - // The caller frees the memory on error, don't retain stale pointers - if (status != mStatus_NoError) { rr->NewRData = mDNSNULL; rr->newrdlength = 0; } - mDNS_Unlock(m); - return(status); - } -#endif - - if (RRLocalOnly(rr) || (rr->resrec.rroriginalttl == newttl && - rr->resrec.rdlength == newrdlength && mDNSPlatformMemSame(rr->resrec.rdata->u.data, newrdata->u.data, newrdlength))) - CompleteRDataUpdate(m, rr); - else - { - rr->AnnounceCount = InitialAnnounceCount; - InitializeLastAPTime(m, rr); - while (rr->NextUpdateCredit && m->timenow - rr->NextUpdateCredit >= 0) GrantUpdateCredit(rr); - if (!rr->UpdateBlocked && rr->UpdateCredits) rr->UpdateCredits--; - if (!rr->NextUpdateCredit) rr->NextUpdateCredit = NonZeroTime(m->timenow + kUpdateCreditRefreshInterval); - if (rr->AnnounceCount > rr->UpdateCredits + 1) rr->AnnounceCount = (mDNSu8)(rr->UpdateCredits + 1); - if (rr->UpdateCredits <= 5) - { - mDNSu32 delay = 6 - rr->UpdateCredits; // Delay 1 second, then 2, then 3, etc. up to 6 seconds maximum - if (!rr->UpdateBlocked) rr->UpdateBlocked = NonZeroTime(m->timenow + (mDNSs32)delay * mDNSPlatformOneSecond); - rr->ThisAPInterval *= 4; - rr->LastAPTime = rr->UpdateBlocked - rr->ThisAPInterval; - LogMsg("Excessive update rate for %##s; delaying announcement by %ld second%s", - rr->resrec.name->c, delay, delay > 1 ? "s" : ""); - } - rr->resrec.rroriginalttl = newttl; - } - - mDNS_Unlock(m); - return(mStatus_NoError); - } - -// Note: mDNS_Deregister calls mDNS_Deregister_internal which can call a user callback, which may change -// the record list and/or question list. -// Any code walking either list must use the CurrentQuestion and/or CurrentRecord mechanism to protect against this. -mDNSexport mStatus mDNS_Deregister(mDNS *const m, AuthRecord *const rr) - { - mStatus status; - mDNS_Lock(m); - status = mDNS_Deregister_internal(m, rr, mDNS_Dereg_normal); - mDNS_Unlock(m); - return(status); - } - -// Circular reference: AdvertiseInterface references mDNS_HostNameCallback, which calls mDNS_SetFQDN, which call AdvertiseInterface -mDNSlocal void mDNS_HostNameCallback(mDNS *const m, AuthRecord *const rr, mStatus result); - -mDNSlocal NetworkInterfaceInfo *FindFirstAdvertisedInterface(mDNS *const m) - { - NetworkInterfaceInfo *intf; - for (intf = m->HostInterfaces; intf; intf = intf->next) - if (intf->Advertise) break; - return(intf); - } - -mDNSlocal void AdvertiseInterface(mDNS *const m, NetworkInterfaceInfo *set) - { - char buffer[MAX_REVERSE_MAPPING_NAME]; - NetworkInterfaceInfo *primary = FindFirstAdvertisedInterface(m); - if (!primary) primary = set; // If no existing advertised interface, this new NetworkInterfaceInfo becomes our new primary - - // Send dynamic update for non-linklocal IPv4 Addresses - mDNS_SetupResourceRecord(&set->RR_A, mDNSNULL, set->InterfaceID, kDNSType_A, kHostNameTTL, kDNSRecordTypeUnique, AuthRecordAny, mDNS_HostNameCallback, set); - mDNS_SetupResourceRecord(&set->RR_PTR, mDNSNULL, set->InterfaceID, kDNSType_PTR, kHostNameTTL, kDNSRecordTypeKnownUnique, AuthRecordAny, mDNSNULL, mDNSNULL); - mDNS_SetupResourceRecord(&set->RR_HINFO, mDNSNULL, set->InterfaceID, kDNSType_HINFO, kHostNameTTL, kDNSRecordTypeUnique, AuthRecordAny, mDNSNULL, mDNSNULL); - -#if ANSWER_REMOTE_HOSTNAME_QUERIES - set->RR_A .AllowRemoteQuery = mDNStrue; - set->RR_PTR .AllowRemoteQuery = mDNStrue; - set->RR_HINFO.AllowRemoteQuery = mDNStrue; -#endif - // 1. Set up Address record to map from host name ("foo.local.") to IP address - // 2. Set up reverse-lookup PTR record to map from our address back to our host name - AssignDomainName(&set->RR_A.namestorage, &m->MulticastHostname); - if (set->ip.type == mDNSAddrType_IPv4) - { - set->RR_A.resrec.rrtype = kDNSType_A; - set->RR_A.resrec.rdata->u.ipv4 = set->ip.ip.v4; - // Note: This is reverse order compared to a normal dotted-decimal IP address, so we can't use our customary "%.4a" format code - mDNS_snprintf(buffer, sizeof(buffer), "%d.%d.%d.%d.in-addr.arpa.", - set->ip.ip.v4.b[3], set->ip.ip.v4.b[2], set->ip.ip.v4.b[1], set->ip.ip.v4.b[0]); - } - else if (set->ip.type == mDNSAddrType_IPv6) - { - int i; - set->RR_A.resrec.rrtype = kDNSType_AAAA; - set->RR_A.resrec.rdata->u.ipv6 = set->ip.ip.v6; - for (i = 0; i < 16; i++) - { - static const char hexValues[] = "0123456789ABCDEF"; - buffer[i * 4 ] = hexValues[set->ip.ip.v6.b[15 - i] & 0x0F]; - buffer[i * 4 + 1] = '.'; - buffer[i * 4 + 2] = hexValues[set->ip.ip.v6.b[15 - i] >> 4]; - buffer[i * 4 + 3] = '.'; - } - mDNS_snprintf(&buffer[64], sizeof(buffer)-64, "ip6.arpa."); - } - - MakeDomainNameFromDNSNameString(&set->RR_PTR.namestorage, buffer); - set->RR_PTR.AutoTarget = Target_AutoHost; // Tell mDNS that the target of this PTR is to be kept in sync with our host name - set->RR_PTR.ForceMCast = mDNStrue; // This PTR points to our dot-local name, so don't ever try to write it into a uDNS server - - set->RR_A.RRSet = &primary->RR_A; // May refer to self - - mDNS_Register_internal(m, &set->RR_A); - mDNS_Register_internal(m, &set->RR_PTR); - - if (!NO_HINFO && m->HIHardware.c[0] > 0 && m->HISoftware.c[0] > 0 && m->HIHardware.c[0] + m->HISoftware.c[0] <= 254) - { - mDNSu8 *p = set->RR_HINFO.resrec.rdata->u.data; - AssignDomainName(&set->RR_HINFO.namestorage, &m->MulticastHostname); - set->RR_HINFO.DependentOn = &set->RR_A; - mDNSPlatformMemCopy(p, &m->HIHardware, 1 + (mDNSu32)m->HIHardware.c[0]); - p += 1 + (int)p[0]; - mDNSPlatformMemCopy(p, &m->HISoftware, 1 + (mDNSu32)m->HISoftware.c[0]); - mDNS_Register_internal(m, &set->RR_HINFO); - } - else - { - debugf("Not creating HINFO record: platform support layer provided no information"); - set->RR_HINFO.resrec.RecordType = kDNSRecordTypeUnregistered; - } - } - -mDNSlocal void DeadvertiseInterface(mDNS *const m, NetworkInterfaceInfo *set) - { - NetworkInterfaceInfo *intf; - - // If we still have address records referring to this one, update them - NetworkInterfaceInfo *primary = FindFirstAdvertisedInterface(m); - AuthRecord *A = primary ? &primary->RR_A : mDNSNULL; - for (intf = m->HostInterfaces; intf; intf = intf->next) - if (intf->RR_A.RRSet == &set->RR_A) - intf->RR_A.RRSet = A; - - // Unregister these records. - // When doing the mDNS_Exit processing, we first call DeadvertiseInterface for each interface, so by the time the platform - // support layer gets to call mDNS_DeregisterInterface, the address and PTR records have already been deregistered for it. - // Also, in the event of a name conflict, one or more of our records will have been forcibly deregistered. - // To avoid unnecessary and misleading warning messages, we check the RecordType before calling mDNS_Deregister_internal(). - if (set->RR_A. resrec.RecordType) mDNS_Deregister_internal(m, &set->RR_A, mDNS_Dereg_normal); - if (set->RR_PTR. resrec.RecordType) mDNS_Deregister_internal(m, &set->RR_PTR, mDNS_Dereg_normal); - if (set->RR_HINFO.resrec.RecordType) mDNS_Deregister_internal(m, &set->RR_HINFO, mDNS_Dereg_normal); - } - -mDNSexport void mDNS_SetFQDN(mDNS *const m) - { - domainname newmname; - NetworkInterfaceInfo *intf; - AuthRecord *rr; - newmname.c[0] = 0; - - if (!AppendDomainLabel(&newmname, &m->hostlabel)) { LogMsg("ERROR: mDNS_SetFQDN: Cannot create MulticastHostname"); return; } - if (!AppendLiteralLabelString(&newmname, "local")) { LogMsg("ERROR: mDNS_SetFQDN: Cannot create MulticastHostname"); return; } - - mDNS_Lock(m); - - if (SameDomainNameCS(&m->MulticastHostname, &newmname)) debugf("mDNS_SetFQDN - hostname unchanged"); - else - { - AssignDomainName(&m->MulticastHostname, &newmname); - - // 1. Stop advertising our address records on all interfaces - for (intf = m->HostInterfaces; intf; intf = intf->next) - if (intf->Advertise) DeadvertiseInterface(m, intf); - - // 2. Start advertising our address records using the new name - for (intf = m->HostInterfaces; intf; intf = intf->next) - if (intf->Advertise) AdvertiseInterface(m, intf); - } - - // 3. Make sure that any AutoTarget SRV records (and the like) get updated - for (rr = m->ResourceRecords; rr; rr=rr->next) if (rr->AutoTarget) SetTargetToHostName(m, rr); - for (rr = m->DuplicateRecords; rr; rr=rr->next) if (rr->AutoTarget) SetTargetToHostName(m, rr); - - mDNS_Unlock(m); - } - -mDNSlocal void mDNS_HostNameCallback(mDNS *const m, AuthRecord *const rr, mStatus result) - { - (void)rr; // Unused parameter - - #if MDNS_DEBUGMSGS - { - char *msg = "Unknown result"; - if (result == mStatus_NoError) msg = "Name registered"; - else if (result == mStatus_NameConflict) msg = "Name conflict"; - debugf("mDNS_HostNameCallback: %##s (%s) %s (%ld)", rr->resrec.name->c, DNSTypeName(rr->resrec.rrtype), msg, result); - } - #endif - - if (result == mStatus_NoError) - { - // Notify the client that the host name is successfully registered - if (m->MainCallback) - m->MainCallback(m, mStatus_NoError); - } - else if (result == mStatus_NameConflict) - { - domainlabel oldlabel = m->hostlabel; - - // 1. First give the client callback a chance to pick a new name - if (m->MainCallback) - m->MainCallback(m, mStatus_NameConflict); - - // 2. If the client callback didn't do it, add (or increment) an index ourselves - // This needs to be case-INSENSITIVE compare, because we need to know that the name has been changed so as to - // remedy the conflict, and a name that differs only in capitalization will just suffer the exact same conflict again. - if (SameDomainLabel(m->hostlabel.c, oldlabel.c)) - IncrementLabelSuffix(&m->hostlabel, mDNSfalse); - - // 3. Generate the FQDNs from the hostlabel, - // and make sure all SRV records, etc., are updated to reference our new hostname - mDNS_SetFQDN(m); - LogMsg("Local Hostname %#s.local already in use; will try %#s.local instead", oldlabel.c, m->hostlabel.c); - } - else if (result == mStatus_MemFree) - { - // .local hostnames do not require goodbyes - we ignore the MemFree (which is sent directly by - // mDNS_Deregister_internal), and allow the caller to deallocate immediately following mDNS_DeadvertiseInterface - debugf("mDNS_HostNameCallback: MemFree (ignored)"); - } - else - LogMsg("mDNS_HostNameCallback: Unknown error %d for registration of record %s", result, rr->resrec.name->c); - } - -mDNSlocal void UpdateInterfaceProtocols(mDNS *const m, NetworkInterfaceInfo *active) - { - NetworkInterfaceInfo *intf; - active->IPv4Available = mDNSfalse; - active->IPv6Available = mDNSfalse; - for (intf = m->HostInterfaces; intf; intf = intf->next) - if (intf->InterfaceID == active->InterfaceID) - { - if (intf->ip.type == mDNSAddrType_IPv4 && intf->McastTxRx) active->IPv4Available = mDNStrue; - if (intf->ip.type == mDNSAddrType_IPv6 && intf->McastTxRx) active->IPv6Available = mDNStrue; - } - } - -mDNSlocal void RestartRecordGetZoneData(mDNS * const m) - { - AuthRecord *rr; - LogInfo("RestartRecordGetZoneData: ResourceRecords"); - for (rr = m->ResourceRecords; rr; rr=rr->next) - if (AuthRecord_uDNS(rr) && rr->state != regState_NoTarget) - { - debugf("RestartRecordGetZoneData: StartGetZoneData for %##s", rr->resrec.name->c); - // Zero out the updateid so that if we have a pending response from the server, it won't - // be accepted as a valid response. If we accept the response, we might free the new "nta" - if (rr->nta) { rr->updateid = zeroID; CancelGetZoneData(m, rr->nta); } - rr->nta = StartGetZoneData(m, rr->resrec.name, ZoneServiceUpdate, RecordRegistrationGotZoneData, rr); - } - } - -mDNSlocal void InitializeNetWakeState(mDNS *const m, NetworkInterfaceInfo *set) - { - int i; - set->NetWakeBrowse.ThisQInterval = -1; - for (i=0; i<3; i++) - { - set->NetWakeResolve[i].ThisQInterval = -1; - set->SPSAddr[i].type = mDNSAddrType_None; - } - set->NextSPSAttempt = -1; - set->NextSPSAttemptTime = m->timenow; - } - -mDNSexport void mDNS_ActivateNetWake_internal(mDNS *const m, NetworkInterfaceInfo *set) - { - NetworkInterfaceInfo *p = m->HostInterfaces; - while (p && p != set) p=p->next; - if (!p) { LogMsg("mDNS_ActivateNetWake_internal: NetworkInterfaceInfo %p not found in active list", set); return; } - - if (set->InterfaceActive) - { - LogSPS("ActivateNetWake for %s (%#a)", set->ifname, &set->ip); - mDNS_StartBrowse_internal(m, &set->NetWakeBrowse, &SleepProxyServiceType, &localdomain, set->InterfaceID, mDNSfalse, m->SPSBrowseCallback, set); - } - } - -mDNSexport void mDNS_DeactivateNetWake_internal(mDNS *const m, NetworkInterfaceInfo *set) - { - NetworkInterfaceInfo *p = m->HostInterfaces; - while (p && p != set) p=p->next; - if (!p) { LogMsg("mDNS_DeactivateNetWake_internal: NetworkInterfaceInfo %p not found in active list", set); return; } - - if (set->NetWakeBrowse.ThisQInterval >= 0) - { - int i; - LogSPS("DeactivateNetWake for %s (%#a)", set->ifname, &set->ip); - - // Stop our browse and resolve operations - mDNS_StopQuery_internal(m, &set->NetWakeBrowse); - for (i=0; i<3; i++) if (set->NetWakeResolve[i].ThisQInterval >= 0) mDNS_StopQuery_internal(m, &set->NetWakeResolve[i]); - - // Make special call to the browse callback to let it know it can to remove all records for this interface - if (m->SPSBrowseCallback) - { - mDNS_DropLockBeforeCallback(); // Allow client to legally make mDNS API calls from the callback - m->SPSBrowseCallback(m, &set->NetWakeBrowse, mDNSNULL, mDNSfalse); - mDNS_ReclaimLockAfterCallback(); // Decrement mDNS_reentrancy to block mDNS API calls again - } - - // Reset our variables back to initial state, so we're ready for when NetWake is turned back on - // (includes resetting NetWakeBrowse.ThisQInterval back to -1) - InitializeNetWakeState(m, set); - } - } - -mDNSexport mStatus mDNS_RegisterInterface(mDNS *const m, NetworkInterfaceInfo *set, mDNSBool flapping) - { - AuthRecord *rr; - mDNSBool FirstOfType = mDNStrue; - NetworkInterfaceInfo **p = &m->HostInterfaces; - - if (!set->InterfaceID) - { LogMsg("mDNS_RegisterInterface: Error! Tried to register a NetworkInterfaceInfo %#a with zero InterfaceID", &set->ip); return(mStatus_Invalid); } - - if (!mDNSAddressIsValidNonZero(&set->mask)) - { LogMsg("mDNS_RegisterInterface: Error! Tried to register a NetworkInterfaceInfo %#a with invalid mask %#a", &set->ip, &set->mask); return(mStatus_Invalid); } - - mDNS_Lock(m); - - // Assume this interface will be active now, unless we find a duplicate already in the list - set->InterfaceActive = mDNStrue; - set->IPv4Available = (mDNSu8)(set->ip.type == mDNSAddrType_IPv4 && set->McastTxRx); - set->IPv6Available = (mDNSu8)(set->ip.type == mDNSAddrType_IPv6 && set->McastTxRx); - - InitializeNetWakeState(m, set); - - // Scan list to see if this InterfaceID is already represented - while (*p) - { - if (*p == set) - { - LogMsg("mDNS_RegisterInterface: Error! Tried to register a NetworkInterfaceInfo that's already in the list"); - mDNS_Unlock(m); - return(mStatus_AlreadyRegistered); - } - - if ((*p)->InterfaceID == set->InterfaceID) - { - // This InterfaceID already represented by a different interface in the list, so mark this instance inactive for now - set->InterfaceActive = mDNSfalse; - if (set->ip.type == (*p)->ip.type) FirstOfType = mDNSfalse; - if (set->ip.type == mDNSAddrType_IPv4 && set->McastTxRx) (*p)->IPv4Available = mDNStrue; - if (set->ip.type == mDNSAddrType_IPv6 && set->McastTxRx) (*p)->IPv6Available = mDNStrue; - } - - p=&(*p)->next; - } - - set->next = mDNSNULL; - *p = set; - - if (set->Advertise) - AdvertiseInterface(m, set); - - LogInfo("mDNS_RegisterInterface: InterfaceID %p %s (%#a) %s", set->InterfaceID, set->ifname, &set->ip, - set->InterfaceActive ? - "not represented in list; marking active and retriggering queries" : - "already represented in list; marking inactive for now"); - - if (set->NetWake) mDNS_ActivateNetWake_internal(m, set); - - // In early versions of OS X the IPv6 address remains on an interface even when the interface is turned off, - // giving the false impression that there's an active representative of this interface when there really isn't. - // Therefore, when registering an interface, we want to re-trigger our questions and re-probe our Resource Records, - // even if we believe that we previously had an active representative of this interface. - if (set->McastTxRx && (FirstOfType || set->InterfaceActive)) - { - DNSQuestion *q; - // Normally, after an interface comes up, we pause half a second before beginning probing. - // This is to guard against cases where there's rapid interface changes, where we could be confused by - // seeing packets we ourselves sent just moments ago (perhaps when this interface had a different address) - // which are then echoed back after a short delay by some Ethernet switches and some 802.11 base stations. - // We don't want to do a probe, and then see a stale echo of an announcement we ourselves sent, - // and think it's a conflicting answer to our probe. - // In the case of a flapping interface, we pause for five seconds, and reduce the announcement count to one packet. - const mDNSs32 probedelay = flapping ? mDNSPlatformOneSecond * 5 : mDNSPlatformOneSecond / 2; - const mDNSu8 numannounce = flapping ? (mDNSu8)1 : InitialAnnounceCount; - - // Use a small amount of randomness: - // In the case of a network administrator turning on an Ethernet hub so that all the - // connected machines establish link at exactly the same time, we don't want them all - // to go and hit the network with identical queries at exactly the same moment. - // We set a random delay of up to InitialQuestionInterval (1/3 second). - // We must *never* set m->SuppressSending to more than that (or set it repeatedly in a way - // that causes mDNSResponder to remain in a prolonged state of SuppressSending, because - // suppressing packet sending for more than about 1/3 second can cause protocol correctness - // to start to break down (e.g. we don't answer probes fast enough, and get name conflicts). - // See mDNS: m->SuppressSending set too enthusiastically - if (!m->SuppressSending) m->SuppressSending = m->timenow + (mDNSs32)mDNSRandom((mDNSu32)InitialQuestionInterval); - - if (flapping) LogMsg("mDNS_RegisterInterface: Frequent transitions for interface %s (%#a)", set->ifname, &set->ip); - - LogInfo("mDNS_RegisterInterface: %s (%#a) probedelay %d", set->ifname, &set->ip, probedelay); - if (m->SuppressProbes == 0 || - m->SuppressProbes - NonZeroTime(m->timenow + probedelay) < 0) - m->SuppressProbes = NonZeroTime(m->timenow + probedelay); - - // Include OWNER option in packets for 60 seconds after connecting to the network. Setting - // it here also handles the wake up case as the network link comes UP after waking causing - // us to reconnect to the network. If we do this as part of the wake up code, it is possible - // that the network link comes UP after 60 seconds and we never set the OWNER option - m->AnnounceOwner = NonZeroTime(m->timenow + 60 * mDNSPlatformOneSecond); - LogInfo("mDNS_RegisterInterface: Setting AnnounceOwner"); - - for (q = m->Questions; q; q=q->next) // Scan our list of questions - if (mDNSOpaque16IsZero(q->TargetQID)) - if (!q->InterfaceID || q->InterfaceID == set->InterfaceID) // If non-specific Q, or Q on this specific interface, - { // then reactivate this question - // If flapping, delay between first and second queries is nine seconds instead of one second - mDNSBool dodelay = flapping && (q->FlappingInterface1 == set->InterfaceID || q->FlappingInterface2 == set->InterfaceID); - mDNSs32 initial = dodelay ? InitialQuestionInterval * QuestionIntervalStep2 : InitialQuestionInterval; - mDNSs32 qdelay = dodelay ? mDNSPlatformOneSecond * 5 : 0; - if (dodelay) LogInfo("No cache records expired for %##s (%s); okay to delay questions a little", q->qname.c, DNSTypeName(q->qtype)); - - if (!q->ThisQInterval || q->ThisQInterval > initial) - { - q->ThisQInterval = initial; - q->RequestUnicast = 2; // Set to 2 because is decremented once *before* we check it - } - q->LastQTime = m->timenow - q->ThisQInterval + qdelay; - q->RecentAnswerPkts = 0; - SetNextQueryTime(m,q); - } - - // For all our non-specific authoritative resource records (and any dormant records specific to this interface) - // we now need them to re-probe if necessary, and then re-announce. - for (rr = m->ResourceRecords; rr; rr=rr->next) - if (!AuthRecord_uDNS(rr)) - if (!rr->resrec.InterfaceID || rr->resrec.InterfaceID == set->InterfaceID) - { - if (rr->resrec.RecordType == kDNSRecordTypeVerified && !rr->DependentOn) rr->resrec.RecordType = kDNSRecordTypeUnique; - rr->ProbeCount = DefaultProbeCountForRecordType(rr->resrec.RecordType); - if (rr->AnnounceCount < numannounce) rr->AnnounceCount = numannounce; - rr->SendNSECNow = mDNSNULL; - InitializeLastAPTime(m, rr); - } - } - - RestartRecordGetZoneData(m); - - CheckSuppressUnusableQuestions(m); - - mDNS_UpdateAllowSleep(m); - - mDNS_Unlock(m); - return(mStatus_NoError); - } - -// Note: mDNS_DeregisterInterface calls mDNS_Deregister_internal which can call a user callback, which may change -// the record list and/or question list. -// Any code walking either list must use the CurrentQuestion and/or CurrentRecord mechanism to protect against this. -mDNSexport void mDNS_DeregisterInterface(mDNS *const m, NetworkInterfaceInfo *set, mDNSBool flapping) - { - NetworkInterfaceInfo **p = &m->HostInterfaces; - mDNSBool revalidate = mDNSfalse; - - mDNS_Lock(m); - - // Find this record in our list - while (*p && *p != set) p=&(*p)->next; - if (!*p) { debugf("mDNS_DeregisterInterface: NetworkInterfaceInfo not found in list"); mDNS_Unlock(m); return; } - - mDNS_DeactivateNetWake_internal(m, set); - - // Unlink this record from our list - *p = (*p)->next; - set->next = mDNSNULL; - - if (!set->InterfaceActive) - { - // If this interface not the active member of its set, update the v4/v6Available flags for the active member - NetworkInterfaceInfo *intf; - for (intf = m->HostInterfaces; intf; intf = intf->next) - if (intf->InterfaceActive && intf->InterfaceID == set->InterfaceID) - UpdateInterfaceProtocols(m, intf); - } - else - { - NetworkInterfaceInfo *intf = FirstInterfaceForID(m, set->InterfaceID); - if (intf) - { - LogInfo("mDNS_DeregisterInterface: Another representative of InterfaceID %p %s (%#a) exists;" - " making it active", set->InterfaceID, set->ifname, &set->ip); - if (intf->InterfaceActive) - LogMsg("mDNS_DeregisterInterface: ERROR intf->InterfaceActive already set for %s (%#a)", set->ifname, &set->ip); - intf->InterfaceActive = mDNStrue; - UpdateInterfaceProtocols(m, intf); - - if (intf->NetWake) mDNS_ActivateNetWake_internal(m, intf); - - // See if another representative *of the same type* exists. If not, we mave have gone from - // dual-stack to v6-only (or v4-only) so we need to reconfirm which records are still valid. - for (intf = m->HostInterfaces; intf; intf = intf->next) - if (intf->InterfaceID == set->InterfaceID && intf->ip.type == set->ip.type) - break; - if (!intf) revalidate = mDNStrue; - } - else - { - mDNSu32 slot; - CacheGroup *cg; - CacheRecord *rr; - DNSQuestion *q; - DNSServer *s; - - LogInfo("mDNS_DeregisterInterface: Last representative of InterfaceID %p %s (%#a) deregistered;" - " marking questions etc. dormant", set->InterfaceID, set->ifname, &set->ip); - - if (set->McastTxRx && flapping) - LogMsg("DeregisterInterface: Frequent transitions for interface %s (%#a)", set->ifname, &set->ip); - - // 1. Deactivate any questions specific to this interface, and tag appropriate questions - // so that mDNS_RegisterInterface() knows how swiftly it needs to reactivate them - for (q = m->Questions; q; q=q->next) - { - if (q->InterfaceID == set->InterfaceID) q->ThisQInterval = 0; - if (!q->InterfaceID || q->InterfaceID == set->InterfaceID) - { - q->FlappingInterface2 = q->FlappingInterface1; - q->FlappingInterface1 = set->InterfaceID; // Keep history of the last two interfaces to go away - } - } - - // 2. Flush any cache records received on this interface - revalidate = mDNSfalse; // Don't revalidate if we're flushing the records - FORALL_CACHERECORDS(slot, cg, rr) - if (rr->resrec.InterfaceID == set->InterfaceID) - { - // If this interface is deemed flapping, - // postpone deleting the cache records in case the interface comes back again - if (set->McastTxRx && flapping) - { - // For a flapping interface we want these record to go away after 30 seconds - mDNS_Reconfirm_internal(m, rr, kDefaultReconfirmTimeForFlappingInterface); - // We set UnansweredQueries = MaxUnansweredQueries so we don't waste time doing any queries for them -- - // if the interface does come back, any relevant questions will be reactivated anyway - rr->UnansweredQueries = MaxUnansweredQueries; - } - else - mDNS_PurgeCacheResourceRecord(m, rr); - } - - // 3. Any DNS servers specific to this interface are now unusable - for (s = m->DNSServers; s; s = s->next) - if (s->interface == set->InterfaceID) - { - s->interface = mDNSInterface_Any; - s->teststate = DNSServer_Disabled; - } - } - } - - // If we were advertising on this interface, deregister those address and reverse-lookup records now - if (set->Advertise) DeadvertiseInterface(m, set); - - // If we have any cache records received on this interface that went away, then re-verify them. - // In some versions of OS X the IPv6 address remains on an interface even when the interface is turned off, - // giving the false impression that there's an active representative of this interface when there really isn't. - // Don't need to do this when shutting down, because *all* interfaces are about to go away - if (revalidate && !m->ShutdownTime) - { - mDNSu32 slot; - CacheGroup *cg; - CacheRecord *rr; - FORALL_CACHERECORDS(slot, cg, rr) - if (rr->resrec.InterfaceID == set->InterfaceID) - mDNS_Reconfirm_internal(m, rr, kDefaultReconfirmTimeForFlappingInterface); - } - - CheckSuppressUnusableQuestions(m); - - mDNS_UpdateAllowSleep(m); - - mDNS_Unlock(m); - } - -mDNSlocal void ServiceCallback(mDNS *const m, AuthRecord *const rr, mStatus result) - { - ServiceRecordSet *sr = (ServiceRecordSet *)rr->RecordContext; - (void)m; // Unused parameter - - #if MDNS_DEBUGMSGS - { - char *msg = "Unknown result"; - if (result == mStatus_NoError) msg = "Name Registered"; - else if (result == mStatus_NameConflict) msg = "Name Conflict"; - else if (result == mStatus_MemFree) msg = "Memory Free"; - debugf("ServiceCallback: %##s (%s) %s (%d)", rr->resrec.name->c, DNSTypeName(rr->resrec.rrtype), msg, result); - } - #endif - - // Only pass on the NoError acknowledgement for the SRV record (when it finishes probing) - if (result == mStatus_NoError && rr != &sr->RR_SRV) return; - - // If we got a name conflict on either SRV or TXT, forcibly deregister this service, and record that we did that - if (result == mStatus_NameConflict) - { - sr->Conflict = mDNStrue; // Record that this service set had a conflict - mDNS_DeregisterService(m, sr); // Unlink the records from our list - return; - } - - if (result == mStatus_MemFree) - { - // If the SRV/TXT/PTR records, or the _services._dns-sd._udp record, or any of the subtype PTR records, - // are still in the process of deregistering, don't pass on the NameConflict/MemFree message until - // every record is finished cleaning up. - mDNSu32 i; - ExtraResourceRecord *e = sr->Extras; - - if (sr->RR_SRV.resrec.RecordType != kDNSRecordTypeUnregistered) return; - if (sr->RR_TXT.resrec.RecordType != kDNSRecordTypeUnregistered) return; - if (sr->RR_PTR.resrec.RecordType != kDNSRecordTypeUnregistered) return; - if (sr->RR_ADV.resrec.RecordType != kDNSRecordTypeUnregistered) return; - for (i=0; iNumSubTypes; i++) if (sr->SubTypes[i].resrec.RecordType != kDNSRecordTypeUnregistered) return; - - while (e) - { - if (e->r.resrec.RecordType != kDNSRecordTypeUnregistered) return; - e = e->next; - } - - // If this ServiceRecordSet was forcibly deregistered, and now its memory is ready for reuse, - // then we can now report the NameConflict to the client - if (sr->Conflict) result = mStatus_NameConflict; - - } - - LogInfo("ServiceCallback: All records %s for %##s", (result == mStatus_MemFree ? "Unregistered": "Registered"), sr->RR_PTR.resrec.name->c); - // CAUTION: MUST NOT do anything more with sr after calling sr->Callback(), because the client's callback - // function is allowed to do anything, including deregistering this service and freeing its memory. - if (sr->ServiceCallback) - sr->ServiceCallback(m, sr, result); - } - -mDNSlocal void NSSCallback(mDNS *const m, AuthRecord *const rr, mStatus result) - { - ServiceRecordSet *sr = (ServiceRecordSet *)rr->RecordContext; - if (sr->ServiceCallback) - sr->ServiceCallback(m, sr, result); - } - -// Note: -// Name is first label of domain name (any dots in the name are actual dots, not label separators) -// Type is service type (e.g. "_ipp._tcp.") -// Domain is fully qualified domain name (i.e. ending with a null label) -// We always register a TXT, even if it is empty (so that clients are not -// left waiting forever looking for a nonexistent record.) -// If the host parameter is mDNSNULL or the root domain (ASCII NUL), -// then the default host name (m->MulticastHostname) is automatically used -// If the optional target host parameter is set, then the storage it points to must remain valid for the lifetime of the service registration -mDNSexport mStatus mDNS_RegisterService(mDNS *const m, ServiceRecordSet *sr, - const domainlabel *const name, const domainname *const type, const domainname *const domain, - const domainname *const host, mDNSIPPort port, const mDNSu8 txtinfo[], mDNSu16 txtlen, - AuthRecord *SubTypes, mDNSu32 NumSubTypes, - mDNSInterfaceID InterfaceID, mDNSServiceCallback Callback, void *Context, mDNSu32 flags) - { - mStatus err; - mDNSu32 i; - mDNSu32 hostTTL; - AuthRecType artype; - mDNSu8 recordType = (flags & regFlagKnownUnique) ? kDNSRecordTypeKnownUnique : kDNSRecordTypeUnique; - - sr->ServiceCallback = Callback; - sr->ServiceContext = Context; - sr->Conflict = mDNSfalse; - - sr->Extras = mDNSNULL; - sr->NumSubTypes = NumSubTypes; - sr->SubTypes = SubTypes; - - if (InterfaceID == mDNSInterface_LocalOnly) - artype = AuthRecordLocalOnly; - else if (InterfaceID == mDNSInterface_P2P) - artype = AuthRecordP2P; - else if ((InterfaceID == mDNSInterface_Any) && (flags & regFlagIncludeP2P)) - artype = AuthRecordAnyIncludeP2P; - else - artype = AuthRecordAny; - - // Initialize the AuthRecord objects to sane values - // Need to initialize everything correctly *before* making the decision whether to do a RegisterNoSuchService and bail out - mDNS_SetupResourceRecord(&sr->RR_ADV, mDNSNULL, InterfaceID, kDNSType_PTR, kStandardTTL, kDNSRecordTypeAdvisory, artype, ServiceCallback, sr); - mDNS_SetupResourceRecord(&sr->RR_PTR, mDNSNULL, InterfaceID, kDNSType_PTR, kStandardTTL, kDNSRecordTypeShared, artype, ServiceCallback, sr); - - if (SameDomainName(type, (const domainname *) "\x4" "_ubd" "\x4" "_tcp")) - hostTTL = kHostNameSmallTTL; - else - hostTTL = kHostNameTTL; - - mDNS_SetupResourceRecord(&sr->RR_SRV, mDNSNULL, InterfaceID, kDNSType_SRV, hostTTL, recordType, artype, ServiceCallback, sr); - mDNS_SetupResourceRecord(&sr->RR_TXT, mDNSNULL, InterfaceID, kDNSType_TXT, kStandardTTL, kDNSRecordTypeUnique, artype, ServiceCallback, sr); - - // If port number is zero, that means the client is really trying to do a RegisterNoSuchService - if (mDNSIPPortIsZero(port)) - return(mDNS_RegisterNoSuchService(m, &sr->RR_SRV, name, type, domain, mDNSNULL, InterfaceID, NSSCallback, sr, (flags & regFlagIncludeP2P))); - - // If the client is registering an oversized TXT record, - // it is the client's responsibility to alloate a ServiceRecordSet structure that is large enough for it - if (sr->RR_TXT.resrec.rdata->MaxRDLength < txtlen) - sr->RR_TXT.resrec.rdata->MaxRDLength = txtlen; - - // Set up the record names - // For now we only create an advisory record for the main type, not for subtypes - // We need to gain some operational experience before we decide if there's a need to create them for subtypes too - if (ConstructServiceName(&sr->RR_ADV.namestorage, (const domainlabel*)"\x09_services", (const domainname*)"\x07_dns-sd\x04_udp", domain) == mDNSNULL) - return(mStatus_BadParamErr); - if (ConstructServiceName(&sr->RR_PTR.namestorage, mDNSNULL, type, domain) == mDNSNULL) return(mStatus_BadParamErr); - if (ConstructServiceName(&sr->RR_SRV.namestorage, name, type, domain) == mDNSNULL) return(mStatus_BadParamErr); - AssignDomainName(&sr->RR_TXT.namestorage, sr->RR_SRV.resrec.name); - - // 1. Set up the ADV record rdata to advertise our service type - AssignDomainName(&sr->RR_ADV.resrec.rdata->u.name, sr->RR_PTR.resrec.name); - - // 2. Set up the PTR record rdata to point to our service name - // We set up two additionals, so when a client asks for this PTR we automatically send the SRV and the TXT too - // Note: uDNS registration code assumes that Additional1 points to the SRV record - AssignDomainName(&sr->RR_PTR.resrec.rdata->u.name, sr->RR_SRV.resrec.name); - sr->RR_PTR.Additional1 = &sr->RR_SRV; - sr->RR_PTR.Additional2 = &sr->RR_TXT; - - // 2a. Set up any subtype PTRs to point to our service name - // If the client is using subtypes, it is the client's responsibility to have - // already set the first label of the record name to the subtype being registered - for (i=0; iSubTypes[i].resrec.name); - st.c[1+st.c[0]] = 0; // Only want the first label, not the whole FQDN (particularly for mDNS_RenameAndReregisterService()) - AppendDomainName(&st, type); - mDNS_SetupResourceRecord(&sr->SubTypes[i], mDNSNULL, InterfaceID, kDNSType_PTR, kStandardTTL, kDNSRecordTypeShared, artype, ServiceCallback, sr); - if (ConstructServiceName(&sr->SubTypes[i].namestorage, mDNSNULL, &st, domain) == mDNSNULL) return(mStatus_BadParamErr); - AssignDomainName(&sr->SubTypes[i].resrec.rdata->u.name, &sr->RR_SRV.namestorage); - sr->SubTypes[i].Additional1 = &sr->RR_SRV; - sr->SubTypes[i].Additional2 = &sr->RR_TXT; - } - - // 3. Set up the SRV record rdata. - sr->RR_SRV.resrec.rdata->u.srv.priority = 0; - sr->RR_SRV.resrec.rdata->u.srv.weight = 0; - sr->RR_SRV.resrec.rdata->u.srv.port = port; - - // Setting AutoTarget tells DNS that the target of this SRV is to be automatically kept in sync with our host name - if (host && host->c[0]) AssignDomainName(&sr->RR_SRV.resrec.rdata->u.srv.target, host); - else { sr->RR_SRV.AutoTarget = Target_AutoHost; sr->RR_SRV.resrec.rdata->u.srv.target.c[0] = '\0'; } - - // 4. Set up the TXT record rdata, - // and set DependentOn because we're depending on the SRV record to find and resolve conflicts for us - // Note: uDNS registration code assumes that DependentOn points to the SRV record - if (txtinfo == mDNSNULL) sr->RR_TXT.resrec.rdlength = 0; - else if (txtinfo != sr->RR_TXT.resrec.rdata->u.txt.c) - { - sr->RR_TXT.resrec.rdlength = txtlen; - if (sr->RR_TXT.resrec.rdlength > sr->RR_TXT.resrec.rdata->MaxRDLength) return(mStatus_BadParamErr); - mDNSPlatformMemCopy(sr->RR_TXT.resrec.rdata->u.txt.c, txtinfo, txtlen); - } - sr->RR_TXT.DependentOn = &sr->RR_SRV; - - mDNS_Lock(m); - // It is important that we register SRV first. uDNS assumes that SRV is registered first so - // that if the SRV cannot find a target, rest of the records that belong to this service - // will not be activated. - err = mDNS_Register_internal(m, &sr->RR_SRV); - // If we can't register the SRV record due to errors, bail out. It has not been inserted in - // any list and hence no need to deregister. We could probably do similar checks for other - // records below and bail out. For now, this seems to be sufficient to address rdar://9304275 - if (err) - { - mDNS_Unlock(m); - return err; - } - if (!err) err = mDNS_Register_internal(m, &sr->RR_TXT); - // We register the RR_PTR last, because we want to be sure that in the event of a forced call to - // mDNS_StartExit, the RR_PTR will be the last one to be forcibly deregistered, since that is what triggers - // the mStatus_MemFree callback to ServiceCallback, which in turn passes on the mStatus_MemFree back to - // the client callback, which is then at liberty to free the ServiceRecordSet memory at will. We need to - // make sure we've deregistered all our records and done any other necessary cleanup before that happens. - if (!err) err = mDNS_Register_internal(m, &sr->RR_ADV); - for (i=0; iSubTypes[i]); - if (!err) err = mDNS_Register_internal(m, &sr->RR_PTR); - - mDNS_Unlock(m); - - if (err) mDNS_DeregisterService(m, sr); - return(err); - } - -mDNSexport mStatus mDNS_AddRecordToService(mDNS *const m, ServiceRecordSet *sr, - ExtraResourceRecord *extra, RData *rdata, mDNSu32 ttl, mDNSu32 includeP2P) - { - ExtraResourceRecord **e; - mStatus status; - AuthRecType artype; - mDNSInterfaceID InterfaceID = sr->RR_PTR.resrec.InterfaceID; - - if (InterfaceID == mDNSInterface_LocalOnly) - artype = AuthRecordLocalOnly; - if (InterfaceID == mDNSInterface_P2P) - artype = AuthRecordP2P; - else if ((InterfaceID == mDNSInterface_Any) && includeP2P) - artype = AuthRecordAnyIncludeP2P; - else - artype = AuthRecordAny; - - extra->next = mDNSNULL; - mDNS_SetupResourceRecord(&extra->r, rdata, sr->RR_PTR.resrec.InterfaceID, - extra->r.resrec.rrtype, ttl, kDNSRecordTypeUnique, artype, ServiceCallback, sr); - AssignDomainName(&extra->r.namestorage, sr->RR_SRV.resrec.name); - - mDNS_Lock(m); - e = &sr->Extras; - while (*e) e = &(*e)->next; - - if (ttl == 0) ttl = kStandardTTL; - - extra->r.DependentOn = &sr->RR_SRV; - - debugf("mDNS_AddRecordToService adding record to %##s %s %d", - extra->r.resrec.name->c, DNSTypeName(extra->r.resrec.rrtype), extra->r.resrec.rdlength); - - status = mDNS_Register_internal(m, &extra->r); - if (status == mStatus_NoError) *e = extra; - - mDNS_Unlock(m); - return(status); - } - -mDNSexport mStatus mDNS_RemoveRecordFromService(mDNS *const m, ServiceRecordSet *sr, ExtraResourceRecord *extra, - mDNSRecordCallback MemFreeCallback, void *Context) - { - ExtraResourceRecord **e; - mStatus status; - - mDNS_Lock(m); - e = &sr->Extras; - while (*e && *e != extra) e = &(*e)->next; - if (!*e) - { - debugf("mDNS_RemoveRecordFromService failed to remove record from %##s", extra->r.resrec.name->c); - status = mStatus_BadReferenceErr; - } - else - { - debugf("mDNS_RemoveRecordFromService removing record from %##s", extra->r.resrec.name->c); - extra->r.RecordCallback = MemFreeCallback; - extra->r.RecordContext = Context; - *e = (*e)->next; - status = mDNS_Deregister_internal(m, &extra->r, mDNS_Dereg_normal); - } - mDNS_Unlock(m); - return(status); - } - -mDNSexport mStatus mDNS_RenameAndReregisterService(mDNS *const m, ServiceRecordSet *const sr, const domainlabel *newname) - { - // Note: Don't need to use mDNS_Lock(m) here, because this code is just using public routines - // mDNS_RegisterService() and mDNS_AddRecordToService(), which do the right locking internally. - domainlabel name1, name2; - domainname type, domain; - const domainname *host = sr->RR_SRV.AutoTarget ? mDNSNULL : &sr->RR_SRV.resrec.rdata->u.srv.target; - ExtraResourceRecord *extras = sr->Extras; - mStatus err; - - DeconstructServiceName(sr->RR_SRV.resrec.name, &name1, &type, &domain); - if (!newname) - { - name2 = name1; - IncrementLabelSuffix(&name2, mDNStrue); - newname = &name2; - } - - if (SameDomainName(&domain, &localdomain)) - debugf("%##s service renamed from \"%#s\" to \"%#s\"", type.c, name1.c, newname->c); - else debugf("%##s service (domain %##s) renamed from \"%#s\" to \"%#s\"",type.c, domain.c, name1.c, newname->c); - - err = mDNS_RegisterService(m, sr, newname, &type, &domain, - host, sr->RR_SRV.resrec.rdata->u.srv.port, sr->RR_TXT.resrec.rdata->u.txt.c, sr->RR_TXT.resrec.rdlength, - sr->SubTypes, sr->NumSubTypes, - sr->RR_PTR.resrec.InterfaceID, sr->ServiceCallback, sr->ServiceContext, 0); - - // mDNS_RegisterService() just reset sr->Extras to NULL. - // Fortunately we already grabbed ourselves a copy of this pointer (above), so we can now run - // through the old list of extra records, and re-add them to our freshly created service registration - while (!err && extras) - { - ExtraResourceRecord *e = extras; - extras = extras->next; - err = mDNS_AddRecordToService(m, sr, e, e->r.resrec.rdata, e->r.resrec.rroriginalttl, 0); - } - - return(err); - } - -// Note: mDNS_DeregisterService calls mDNS_Deregister_internal which can call a user callback, -// which may change the record list and/or question list. -// Any code walking either list must use the CurrentQuestion and/or CurrentRecord mechanism to protect against this. -mDNSexport mStatus mDNS_DeregisterService_drt(mDNS *const m, ServiceRecordSet *sr, mDNS_Dereg_type drt) - { - // If port number is zero, that means this was actually registered using mDNS_RegisterNoSuchService() - if (mDNSIPPortIsZero(sr->RR_SRV.resrec.rdata->u.srv.port)) return(mDNS_DeregisterNoSuchService(m, &sr->RR_SRV)); - - if (sr->RR_PTR.resrec.RecordType == kDNSRecordTypeUnregistered) - { - debugf("Service set for %##s already deregistered", sr->RR_SRV.resrec.name->c); - return(mStatus_BadReferenceErr); - } - else if (sr->RR_PTR.resrec.RecordType == kDNSRecordTypeDeregistering) - { - LogInfo("Service set for %##s already in the process of deregistering", sr->RR_SRV.resrec.name->c); - // Avoid race condition: - // If a service gets a conflict, then we set the Conflict flag to tell us to generate - // an mStatus_NameConflict message when we get the mStatus_MemFree for our PTR record. - // If the client happens to deregister the service in the middle of that process, then - // we clear the flag back to the normal state, so that we deliver a plain mStatus_MemFree - // instead of incorrectly promoting it to mStatus_NameConflict. - // This race condition is exposed particularly when the conformance test generates - // a whole batch of simultaneous conflicts across a range of services all advertised - // using the same system default name, and if we don't take this precaution then - // we end up incrementing m->nicelabel multiple times instead of just once. - // Bug when auto-renaming Computer Name after name collision - sr->Conflict = mDNSfalse; - return(mStatus_NoError); - } - else - { - mDNSu32 i; - mStatus status; - ExtraResourceRecord *e; - mDNS_Lock(m); - e = sr->Extras; - - // We use mDNS_Dereg_repeat because, in the event of a collision, some or all of the - // SRV, TXT, or Extra records could have already been automatically deregistered, and that's okay - mDNS_Deregister_internal(m, &sr->RR_SRV, mDNS_Dereg_repeat); - mDNS_Deregister_internal(m, &sr->RR_TXT, mDNS_Dereg_repeat); - - mDNS_Deregister_internal(m, &sr->RR_ADV, drt); - - // We deregister all of the extra records, but we leave the sr->Extras list intact - // in case the client wants to do a RenameAndReregister and reinstate the registration - while (e) - { - mDNS_Deregister_internal(m, &e->r, mDNS_Dereg_repeat); - e = e->next; - } - - for (i=0; iNumSubTypes; i++) - mDNS_Deregister_internal(m, &sr->SubTypes[i], drt); - - status = mDNS_Deregister_internal(m, &sr->RR_PTR, drt); - mDNS_Unlock(m); - return(status); - } - } - -// Create a registration that asserts that no such service exists with this name. -// This can be useful where there is a given function is available through several protocols. -// For example, a printer called "Stuart's Printer" may implement printing via the "pdl-datastream" and "IPP" -// protocols, but not via "LPR". In this case it would be prudent for the printer to assert the non-existence of an -// "LPR" service called "Stuart's Printer". Without this precaution, another printer than offers only "LPR" printing -// could inadvertently advertise its service under the same name "Stuart's Printer", which might be confusing for users. -mDNSexport mStatus mDNS_RegisterNoSuchService(mDNS *const m, AuthRecord *const rr, - const domainlabel *const name, const domainname *const type, const domainname *const domain, - const domainname *const host, - const mDNSInterfaceID InterfaceID, mDNSRecordCallback Callback, void *Context, mDNSBool includeP2P) - { - AuthRecType artype; - - if (InterfaceID == mDNSInterface_LocalOnly) - artype = AuthRecordLocalOnly; - else if (InterfaceID == mDNSInterface_P2P) - artype = AuthRecordP2P; - else if ((InterfaceID == mDNSInterface_Any) && includeP2P) - artype = AuthRecordAnyIncludeP2P; - else - artype = AuthRecordAny; - - mDNS_SetupResourceRecord(rr, mDNSNULL, InterfaceID, kDNSType_SRV, kHostNameTTL, kDNSRecordTypeUnique, artype, Callback, Context); - if (ConstructServiceName(&rr->namestorage, name, type, domain) == mDNSNULL) return(mStatus_BadParamErr); - rr->resrec.rdata->u.srv.priority = 0; - rr->resrec.rdata->u.srv.weight = 0; - rr->resrec.rdata->u.srv.port = zeroIPPort; - if (host && host->c[0]) AssignDomainName(&rr->resrec.rdata->u.srv.target, host); - else rr->AutoTarget = Target_AutoHost; - return(mDNS_Register(m, rr)); - } - -mDNSexport mStatus mDNS_AdvertiseDomains(mDNS *const m, AuthRecord *rr, - mDNS_DomainType DomainType, const mDNSInterfaceID InterfaceID, char *domname) - { - AuthRecType artype; - - if (InterfaceID == mDNSInterface_LocalOnly) - artype = AuthRecordLocalOnly; - else if (InterfaceID == mDNSInterface_P2P) - artype = AuthRecordP2P; - else - artype = AuthRecordAny; - mDNS_SetupResourceRecord(rr, mDNSNULL, InterfaceID, kDNSType_PTR, kStandardTTL, kDNSRecordTypeShared, artype, mDNSNULL, mDNSNULL); - if (!MakeDomainNameFromDNSNameString(&rr->namestorage, mDNS_DomainTypeNames[DomainType])) return(mStatus_BadParamErr); - if (!MakeDomainNameFromDNSNameString(&rr->resrec.rdata->u.name, domname)) return(mStatus_BadParamErr); - return(mDNS_Register(m, rr)); - } - -mDNSlocal mDNSBool mDNS_IdUsedInResourceRecordsList(mDNS * const m, mDNSOpaque16 id) - { - AuthRecord *r; - for (r = m->ResourceRecords; r; r=r->next) if (mDNSSameOpaque16(id, r->updateid)) return mDNStrue; - return mDNSfalse; - } - -mDNSlocal mDNSBool mDNS_IdUsedInQuestionsList(mDNS * const m, mDNSOpaque16 id) - { - DNSQuestion *q; - for (q = m->Questions; q; q=q->next) if (mDNSSameOpaque16(id, q->TargetQID)) return mDNStrue; - return mDNSfalse; - } - -mDNSexport mDNSOpaque16 mDNS_NewMessageID(mDNS * const m) - { - mDNSOpaque16 id; - int i; - - for (i=0; i<10; i++) - { - id = mDNSOpaque16fromIntVal(1 + (mDNSu16)mDNSRandom(0xFFFE)); - if (!mDNS_IdUsedInResourceRecordsList(m, id) && !mDNS_IdUsedInQuestionsList(m, id)) break; - } - - debugf("mDNS_NewMessageID: %5d", mDNSVal16(id)); - - return id; - } - -// *************************************************************************** -#if COMPILER_LIKES_PRAGMA_MARK -#pragma mark - -#pragma mark - Sleep Proxy Server -#endif - -mDNSlocal void RestartARPProbing(mDNS *const m, AuthRecord *const rr) - { - // If we see an ARP from a machine we think is sleeping, then either - // (i) the machine has woken, or - // (ii) it's just a stray old packet from before the machine slept - // To handle the second case, we reset ProbeCount, so we'll suppress our own answers for a while, to avoid - // generating ARP conflicts with a waking machine, and set rr->LastAPTime so we'll start probing again in 10 seconds. - // If the machine has just woken then we'll discard our records when we see the first new mDNS probe from that machine. - // If it was a stray old packet, then after 10 seconds we'll probe again and then start answering ARPs again. In this case we *do* - // need to send new ARP Announcements, because the owner's ARP broadcasts will have updated neighboring ARP caches, so we need to - // re-assert our (temporary) ownership of that IP address in order to receive subsequent packets addressed to that IPv4 address. - - rr->resrec.RecordType = kDNSRecordTypeUnique; - rr->ProbeCount = DefaultProbeCountForTypeUnique; - - // If we haven't started announcing yet (and we're not already in ten-second-delay mode) the machine is probably - // still going to sleep, so we just reset rr->ProbeCount so we'll continue probing until it stops responding. - // If we *have* started announcing, the machine is probably in the process of waking back up, so in that case - // we're more cautious and we wait ten seconds before probing it again. We do this because while waking from - // sleep, some network interfaces tend to lose or delay inbound packets, and without this delay, if the waking machine - // didn't answer our three probes within three seconds then we'd announce and cause it an unnecessary address conflict. - if (rr->AnnounceCount == InitialAnnounceCount && m->timenow - rr->LastAPTime >= 0) - InitializeLastAPTime(m, rr); - else - { - rr->AnnounceCount = InitialAnnounceCount; - rr->ThisAPInterval = mDNSPlatformOneSecond; - rr->LastAPTime = m->timenow + mDNSPlatformOneSecond * 9; // Send first packet at rr->LastAPTime + rr->ThisAPInterval, i.e. 10 seconds from now - SetNextAnnounceProbeTime(m, rr); - } - } - -mDNSlocal void mDNSCoreReceiveRawARP(mDNS *const m, const ARP_EthIP *const arp, const mDNSInterfaceID InterfaceID) - { - static const mDNSOpaque16 ARP_op_request = { { 0, 1 } }; - AuthRecord *rr; - NetworkInterfaceInfo *intf = FirstInterfaceForID(m, InterfaceID); - if (!intf) return; - - mDNS_Lock(m); - - // Pass 1: - // Process ARP Requests and Probes (but not Announcements), and generate an ARP Reply if necessary. - // We also process ARPs from our own kernel (and 'answer' them by injecting a local ARP table entry) - // We ignore ARP Announcements here -- Announcements are not questions, they're assertions, so we don't need to answer them. - // The times we might need to react to an ARP Announcement are: - // (i) as an indication that the host in question has not gone to sleep yet (so we should delay beginning to proxy for it) or - // (ii) if it's a conflicting Announcement from another host - // -- and we check for these in Pass 2 below. - if (mDNSSameOpaque16(arp->op, ARP_op_request) && !mDNSSameIPv4Address(arp->spa, arp->tpa)) - { - for (rr = m->ResourceRecords; rr; rr=rr->next) - if (rr->resrec.InterfaceID == InterfaceID && rr->resrec.RecordType != kDNSRecordTypeDeregistering && - rr->AddressProxy.type == mDNSAddrType_IPv4 && mDNSSameIPv4Address(rr->AddressProxy.ip.v4, arp->tpa)) - { - static const char msg1[] = "ARP Req from owner -- re-probing"; - static const char msg2[] = "Ignoring ARP Request from "; - static const char msg3[] = "Creating Local ARP Cache entry "; - static const char msg4[] = "Answering ARP Request from "; - const char *const msg = mDNSSameEthAddress(&arp->sha, &rr->WakeUp.IMAC) ? msg1 : - (rr->AnnounceCount == InitialAnnounceCount) ? msg2 : - mDNSSameEthAddress(&arp->sha, &intf->MAC) ? msg3 : msg4; - LogSPS("%-7s %s %.6a %.4a for %.4a -- H-MAC %.6a I-MAC %.6a %s", - intf->ifname, msg, &arp->sha, &arp->spa, &arp->tpa, &rr->WakeUp.HMAC, &rr->WakeUp.IMAC, ARDisplayString(m, rr)); - if (msg == msg1) RestartARPProbing(m, rr); - else if (msg == msg3) mDNSPlatformSetLocalAddressCacheEntry(m, &rr->AddressProxy, &rr->WakeUp.IMAC, InterfaceID); - else if (msg == msg4) SendARP(m, 2, rr, &arp->tpa, &arp->sha, &arp->spa, &arp->sha); - } - } - - // Pass 2: - // For all types of ARP packet we check the Sender IP address to make sure it doesn't conflict with any AddressProxy record we're holding. - // (Strictly speaking we're only checking Announcement/Request/Reply packets, since ARP Probes have zero Sender IP address, - // so by definition (and by design) they can never conflict with any real (i.e. non-zero) IP address). - // We ignore ARPs we sent ourselves (Sender MAC address is our MAC address) because our own proxy ARPs do not constitute a conflict that we need to handle. - // If we see an apparently conflicting ARP, we check the sender hardware address: - // If the sender hardware address is the original owner this is benign, so we just suppress our own proxy answering for a while longer. - // If the sender hardware address is *not* the original owner, then this is a conflict, and we need to wake the sleeping machine to handle it. - if (mDNSSameEthAddress(&arp->sha, &intf->MAC)) - debugf("ARP from self for %.4a", &arp->tpa); - else - { - if (!mDNSSameIPv4Address(arp->spa, zerov4Addr)) - for (rr = m->ResourceRecords; rr; rr=rr->next) - if (rr->resrec.InterfaceID == InterfaceID && rr->resrec.RecordType != kDNSRecordTypeDeregistering && - rr->AddressProxy.type == mDNSAddrType_IPv4 && mDNSSameIPv4Address(rr->AddressProxy.ip.v4, arp->spa)) - { - RestartARPProbing(m, rr); - if (mDNSSameEthAddress(&arp->sha, &rr->WakeUp.IMAC)) - LogSPS("%-7s ARP %s from owner %.6a %.4a for %-15.4a -- re-starting probing for %s", intf->ifname, - mDNSSameIPv4Address(arp->spa, arp->tpa) ? "Announcement " : mDNSSameOpaque16(arp->op, ARP_op_request) ? "Request " : "Response ", - &arp->sha, &arp->spa, &arp->tpa, ARDisplayString(m, rr)); - else - { - LogMsg("%-7s Conflicting ARP from %.6a %.4a for %.4a -- waking H-MAC %.6a I-MAC %.6a %s", intf->ifname, - &arp->sha, &arp->spa, &arp->tpa, &rr->WakeUp.HMAC, &rr->WakeUp.IMAC, ARDisplayString(m, rr)); - ScheduleWakeup(m, rr->resrec.InterfaceID, &rr->WakeUp.HMAC); - } - } - } - - mDNS_Unlock(m); - } - -/* -// Option 1 is Source Link Layer Address Option -// Option 2 is Target Link Layer Address Option -mDNSlocal const mDNSEthAddr *GetLinkLayerAddressOption(const IPv6NDP *const ndp, const mDNSu8 *const end, mDNSu8 op) - { - const mDNSu8 *options = (mDNSu8 *)(ndp+1); - while (options < end) - { - debugf("NDP Option %02X len %2d %d", options[0], options[1], end - options); - if (options[0] == op && options[1] == 1) return (const mDNSEthAddr*)(options+2); - options += options[1] * 8; - } - return mDNSNULL; - } -*/ - -mDNSlocal void mDNSCoreReceiveRawND(mDNS *const m, const mDNSEthAddr *const sha, const mDNSv6Addr *spa, - const IPv6NDP *const ndp, const mDNSu8 *const end, const mDNSInterfaceID InterfaceID) - { - AuthRecord *rr; - NetworkInterfaceInfo *intf = FirstInterfaceForID(m, InterfaceID); - if (!intf) return; - - mDNS_Lock(m); - - // Pass 1: Process Neighbor Solicitations, and generate a Neighbor Advertisement if necessary. - if (ndp->type == NDP_Sol) - { - //const mDNSEthAddr *const sha = GetLinkLayerAddressOption(ndp, end, NDP_SrcLL); - (void)end; - for (rr = m->ResourceRecords; rr; rr=rr->next) - if (rr->resrec.InterfaceID == InterfaceID && rr->resrec.RecordType != kDNSRecordTypeDeregistering && - rr->AddressProxy.type == mDNSAddrType_IPv6 && mDNSSameIPv6Address(rr->AddressProxy.ip.v6, ndp->target)) - { - static const char msg1[] = "NDP Req from owner -- re-probing"; - static const char msg2[] = "Ignoring NDP Request from "; - static const char msg3[] = "Creating Local NDP Cache entry "; - static const char msg4[] = "Answering NDP Request from "; - static const char msg5[] = "Answering NDP Probe from "; - const char *const msg = sha && mDNSSameEthAddress(sha, &rr->WakeUp.IMAC) ? msg1 : - (rr->AnnounceCount == InitialAnnounceCount) ? msg2 : - sha && mDNSSameEthAddress(sha, &intf->MAC) ? msg3 : - spa && mDNSIPv6AddressIsZero(*spa) ? msg4 : msg5; - LogSPS("%-7s %s %.6a %.16a for %.16a -- H-MAC %.6a I-MAC %.6a %s", - intf->ifname, msg, sha, spa, &ndp->target, &rr->WakeUp.HMAC, &rr->WakeUp.IMAC, ARDisplayString(m, rr)); - if (msg == msg1) RestartARPProbing(m, rr); - else if (msg == msg3) - { - if (!(m->KnownBugs & mDNS_KnownBug_LimitedIPv6)) - mDNSPlatformSetLocalAddressCacheEntry(m, &rr->AddressProxy, &rr->WakeUp.IMAC, InterfaceID); - } - else if (msg == msg4) SendNDP(m, NDP_Adv, NDP_Solicited, rr, &ndp->target, mDNSNULL, spa, sha ); - else if (msg == msg5) SendNDP(m, NDP_Adv, 0, rr, &ndp->target, mDNSNULL, &AllHosts_v6, &AllHosts_v6_Eth); - } - } - - // Pass 2: For all types of NDP packet we check the Sender IP address to make sure it doesn't conflict with any AddressProxy record we're holding. - if (mDNSSameEthAddress(sha, &intf->MAC)) - debugf("NDP from self for %.16a", &ndp->target); - else - { - // For Neighbor Advertisements we check the Target address field, not the actual IPv6 source address. - // When a machine has both link-local and routable IPv6 addresses, it may send NDP packets making assertions - // about its routable IPv6 address, using its link-local address as the source address for all NDP packets. - // Hence it is the NDP target address we care about, not the actual packet source address. - if (ndp->type == NDP_Adv) spa = &ndp->target; - if (!mDNSSameIPv6Address(*spa, zerov6Addr)) - for (rr = m->ResourceRecords; rr; rr=rr->next) - if (rr->resrec.InterfaceID == InterfaceID && rr->resrec.RecordType != kDNSRecordTypeDeregistering && - rr->AddressProxy.type == mDNSAddrType_IPv6 && mDNSSameIPv6Address(rr->AddressProxy.ip.v6, *spa)) - { - RestartARPProbing(m, rr); - if (mDNSSameEthAddress(sha, &rr->WakeUp.IMAC)) - LogSPS("%-7s NDP %s from owner %.6a %.16a for %.16a -- re-starting probing for %s", intf->ifname, - ndp->type == NDP_Sol ? "Solicitation " : "Advertisement", sha, spa, &ndp->target, ARDisplayString(m, rr)); - else - { - LogMsg("%-7s Conflicting NDP from %.6a %.16a for %.16a -- waking H-MAC %.6a I-MAC %.6a %s", intf->ifname, - sha, spa, &ndp->target, &rr->WakeUp.HMAC, &rr->WakeUp.IMAC, ARDisplayString(m, rr)); - ScheduleWakeup(m, rr->resrec.InterfaceID, &rr->WakeUp.HMAC); - } - } - } - - mDNS_Unlock(m); - } - -mDNSlocal void mDNSCoreReceiveRawTransportPacket(mDNS *const m, const mDNSEthAddr *const sha, const mDNSAddr *const src, const mDNSAddr *const dst, const mDNSu8 protocol, - const mDNSu8 *const p, const TransportLayerPacket *const t, const mDNSu8 *const end, const mDNSInterfaceID InterfaceID, const mDNSu16 len) - { - const mDNSIPPort port = (protocol == 0x06) ? t->tcp.dst : (protocol == 0x11) ? t->udp.dst : zeroIPPort; - mDNSBool wake = mDNSfalse; - - switch (protocol) - { - #define XX wake ? "Received" : "Ignoring", end-p - case 0x01: LogSPS("Ignoring %d-byte ICMP from %#a to %#a", end-p, src, dst); - break; - - case 0x06: { - #define SSH_AsNumber 22 - static const mDNSIPPort SSH = { { SSH_AsNumber >> 8, SSH_AsNumber & 0xFF } }; - - // Plan to wake if - // (a) RST is not set, AND - // (b) packet is SYN, SYN+FIN, or plain data packet (no SYN or FIN). We won't wake for FIN alone. - wake = (!(t->tcp.flags & 4) && (t->tcp.flags & 3) != 1); - - // For now, to reduce spurious wakeups, we wake only for TCP SYN, - // except for ssh connections, where we'll wake for plain data packets too - if (!mDNSSameIPPort(port, SSH) && !(t->tcp.flags & 2)) wake = mDNSfalse; - - LogSPS("%s %d-byte TCP from %#a:%d to %#a:%d%s%s%s", XX, - src, mDNSVal16(t->tcp.src), dst, mDNSVal16(port), - (t->tcp.flags & 2) ? " SYN" : "", - (t->tcp.flags & 1) ? " FIN" : "", - (t->tcp.flags & 4) ? " RST" : ""); - } - break; - - case 0x11: { - #define ARD_AsNumber 3283 - static const mDNSIPPort ARD = { { ARD_AsNumber >> 8, ARD_AsNumber & 0xFF } }; - const mDNSu16 udplen = (mDNSu16)((mDNSu16)t->bytes[4] << 8 | t->bytes[5]); // Length *including* 8-byte UDP header - if (udplen >= sizeof(UDPHeader)) - { - const mDNSu16 datalen = udplen - sizeof(UDPHeader); - wake = mDNStrue; - - // For Back to My Mac UDP port 4500 (IPSEC) packets, we do some special handling - if (mDNSSameIPPort(port, IPSECPort)) - { - // Specifically ignore NAT keepalive packets - if (datalen == 1 && end >= &t->bytes[9] && t->bytes[8] == 0xFF) wake = mDNSfalse; - else - { - // Skip over the Non-ESP Marker if present - const mDNSBool NonESP = (end >= &t->bytes[12] && t->bytes[8] == 0 && t->bytes[9] == 0 && t->bytes[10] == 0 && t->bytes[11] == 0); - const IKEHeader *const ike = (IKEHeader *)(t + (NonESP ? 12 : 8)); - const mDNSu16 ikelen = datalen - (NonESP ? 4 : 0); - if (ikelen >= sizeof(IKEHeader) && end >= ((mDNSu8 *)ike) + sizeof(IKEHeader)) - if ((ike->Version & 0x10) == 0x10) - { - // ExchangeType == 5 means 'Informational' - // ExchangeType == 34 means 'IKE_SA_INIT' - if (ike->ExchangeType == 5 || ike->ExchangeType == 34) wake = mDNSfalse; - LogSPS("%s %d-byte IKE ExchangeType %d", XX, ike->ExchangeType); - } - } - } - - // For now, because we haven't yet worked out a clean elegant way to do this, we just special-case the - // Apple Remote Desktop port number -- we ignore all packets to UDP 3283 (the "Net Assistant" port), - // except for Apple Remote Desktop's explicit manual wakeup packet, which looks like this: - // UDP header (8 bytes) - // Payload: 13 88 00 6a 41 4e 41 20 (8 bytes) ffffffffffff (6 bytes) 16xMAC (96 bytes) = 110 bytes total - if (mDNSSameIPPort(port, ARD)) wake = (datalen >= 110 && end >= &t->bytes[10] && t->bytes[8] == 0x13 && t->bytes[9] == 0x88); - - LogSPS("%s %d-byte UDP from %#a:%d to %#a:%d", XX, src, mDNSVal16(t->udp.src), dst, mDNSVal16(port)); - } - } - break; - - case 0x3A: if (&t->bytes[len] <= end) - { - mDNSu16 checksum = IPv6CheckSum(&src->ip.v6, &dst->ip.v6, protocol, t->bytes, len); - if (!checksum) mDNSCoreReceiveRawND(m, sha, &src->ip.v6, &t->ndp, &t->bytes[len], InterfaceID); - else LogInfo("IPv6CheckSum bad %04X %02X%02X from %#a to %#a", checksum, t->bytes[2], t->bytes[3], src, dst); - } - break; - - default: LogSPS("Ignoring %d-byte IP packet unknown protocol %d from %#a to %#a", end-p, protocol, src, dst); - break; - } - - if (wake) - { - AuthRecord *rr, *r2; - - mDNS_Lock(m); - for (rr = m->ResourceRecords; rr; rr=rr->next) - if (rr->resrec.InterfaceID == InterfaceID && - rr->resrec.RecordType != kDNSRecordTypeDeregistering && - rr->AddressProxy.type && mDNSSameAddress(&rr->AddressProxy, dst)) - { - const mDNSu8 *const tp = (protocol == 6) ? (const mDNSu8 *)"\x4_tcp" : (const mDNSu8 *)"\x4_udp"; - for (r2 = m->ResourceRecords; r2; r2=r2->next) - if (r2->resrec.InterfaceID == InterfaceID && mDNSSameEthAddress(&r2->WakeUp.HMAC, &rr->WakeUp.HMAC) && - r2->resrec.RecordType != kDNSRecordTypeDeregistering && - r2->resrec.rrtype == kDNSType_SRV && mDNSSameIPPort(r2->resrec.rdata->u.srv.port, port) && - SameDomainLabel(ThirdLabel(r2->resrec.name)->c, tp)) - break; - if (!r2 && mDNSSameIPPort(port, IPSECPort)) r2 = rr; // So that we wake for BTMM IPSEC packets, even without a matching SRV record - if (r2) - { - LogMsg("Waking host at %s %#a H-MAC %.6a I-MAC %.6a for %s", - InterfaceNameForID(m, rr->resrec.InterfaceID), dst, &rr->WakeUp.HMAC, &rr->WakeUp.IMAC, ARDisplayString(m, r2)); - ScheduleWakeup(m, rr->resrec.InterfaceID, &rr->WakeUp.HMAC); - } - else - LogSPS("Sleeping host at %s %#a %.6a has no service on %#s %d", - InterfaceNameForID(m, rr->resrec.InterfaceID), dst, &rr->WakeUp.HMAC, tp, mDNSVal16(port)); - } - mDNS_Unlock(m); - } - } - -mDNSexport void mDNSCoreReceiveRawPacket(mDNS *const m, const mDNSu8 *const p, const mDNSu8 *const end, const mDNSInterfaceID InterfaceID) - { - static const mDNSOpaque16 Ethertype_ARP = { { 0x08, 0x06 } }; // Ethertype 0x0806 = ARP - static const mDNSOpaque16 Ethertype_IPv4 = { { 0x08, 0x00 } }; // Ethertype 0x0800 = IPv4 - static const mDNSOpaque16 Ethertype_IPv6 = { { 0x86, 0xDD } }; // Ethertype 0x86DD = IPv6 - static const mDNSOpaque16 ARP_hrd_eth = { { 0x00, 0x01 } }; // Hardware address space (Ethernet = 1) - static const mDNSOpaque16 ARP_pro_ip = { { 0x08, 0x00 } }; // Protocol address space (IP = 0x0800) - - // Note: BPF guarantees that the NETWORK LAYER header will be word aligned, not the link-layer header. - // In other words, we can safely assume that pkt below (ARP, IPv4 or IPv6) is properly word aligned, - // but if pkt is 4-byte aligned, that necessarily means that eth CANNOT also be 4-byte aligned - // since it points to a an address 14 bytes before pkt. - const EthernetHeader *const eth = (const EthernetHeader *)p; - const NetworkLayerPacket *const pkt = (const NetworkLayerPacket *)(eth+1); - mDNSAddr src, dst; - #define RequiredCapLen(P) ((P)==0x01 ? 4 : (P)==0x06 ? 20 : (P)==0x11 ? 8 : (P)==0x3A ? 24 : 0) - - // Is ARP? Length must be at least 14 + 28 = 42 bytes - if (end >= p+42 && mDNSSameOpaque16(eth->ethertype, Ethertype_ARP) && mDNSSameOpaque16(pkt->arp.hrd, ARP_hrd_eth) && mDNSSameOpaque16(pkt->arp.pro, ARP_pro_ip)) - mDNSCoreReceiveRawARP(m, &pkt->arp, InterfaceID); - // Is IPv4 with zero fragmentation offset? Length must be at least 14 + 20 = 34 bytes - else if (end >= p+34 && mDNSSameOpaque16(eth->ethertype, Ethertype_IPv4) && (pkt->v4.flagsfrags.b[0] & 0x1F) == 0 && pkt->v4.flagsfrags.b[1] == 0) - { - const mDNSu8 *const trans = p + 14 + (pkt->v4.vlen & 0xF) * 4; - debugf("Got IPv4 %02X from %.4a to %.4a", pkt->v4.protocol, &pkt->v4.src, &pkt->v4.dst); - src.type = mDNSAddrType_IPv4; src.ip.v4 = pkt->v4.src; - dst.type = mDNSAddrType_IPv4; dst.ip.v4 = pkt->v4.dst; - if (end >= trans + RequiredCapLen(pkt->v4.protocol)) - mDNSCoreReceiveRawTransportPacket(m, ð->src, &src, &dst, pkt->v4.protocol, p, (TransportLayerPacket*)trans, end, InterfaceID, 0); - } - // Is IPv6? Length must be at least 14 + 28 = 42 bytes - else if (end >= p+54 && mDNSSameOpaque16(eth->ethertype, Ethertype_IPv6)) - { - const mDNSu8 *const trans = p + 54; - debugf("Got IPv6 %02X from %.16a to %.16a", pkt->v6.pro, &pkt->v6.src, &pkt->v6.dst); - src.type = mDNSAddrType_IPv6; src.ip.v6 = pkt->v6.src; - dst.type = mDNSAddrType_IPv6; dst.ip.v6 = pkt->v6.dst; - if (end >= trans + RequiredCapLen(pkt->v6.pro)) - mDNSCoreReceiveRawTransportPacket(m, ð->src, &src, &dst, pkt->v6.pro, p, (TransportLayerPacket*)trans, end, InterfaceID, - (mDNSu16)pkt->bytes[4] << 8 | pkt->bytes[5]); - } - } - -mDNSlocal void ConstructSleepProxyServerName(mDNS *const m, domainlabel *name) - { - name->c[0] = (mDNSu8)mDNS_snprintf((char*)name->c+1, 62, "%d-%d-%d-%d %#s", - m->SPSType, m->SPSPortability, m->SPSMarginalPower, m->SPSTotalPower, &m->nicelabel); - } - -mDNSlocal void SleepProxyServerCallback(mDNS *const m, ServiceRecordSet *const srs, mStatus result) - { - if (result == mStatus_NameConflict) - mDNS_RenameAndReregisterService(m, srs, mDNSNULL); - else if (result == mStatus_MemFree) - { - if (m->SleepState) - m->SPSState = 3; - else - { - m->SPSState = (mDNSu8)(m->SPSSocket != mDNSNULL); - if (m->SPSState) - { - domainlabel name; - ConstructSleepProxyServerName(m, &name); - mDNS_RegisterService(m, srs, - &name, &SleepProxyServiceType, &localdomain, - mDNSNULL, m->SPSSocket->port, // Host, port - (mDNSu8 *)"", 1, // TXT data, length - mDNSNULL, 0, // Subtypes (none) - mDNSInterface_Any, // Interface ID - SleepProxyServerCallback, mDNSNULL, 0); // Callback, context, flags - } - LogSPS("Sleep Proxy Server %#s %s", srs->RR_SRV.resrec.name->c, m->SPSState ? "started" : "stopped"); - } - } - } - -// Called with lock held -mDNSexport void mDNSCoreBeSleepProxyServer_internal(mDNS *const m, mDNSu8 sps, mDNSu8 port, mDNSu8 marginalpower, mDNSu8 totpower) - { - // This routine uses mDNS_DeregisterService and calls SleepProxyServerCallback, so we execute in user callback context - mDNS_DropLockBeforeCallback(); - - // If turning off SPS, close our socket - // (Do this first, BEFORE calling mDNS_DeregisterService below) - if (!sps && m->SPSSocket) { mDNSPlatformUDPClose(m->SPSSocket); m->SPSSocket = mDNSNULL; } - - // If turning off, or changing type, deregister old name - if (m->SPSState == 1 && sps != m->SPSType) - { m->SPSState = 2; mDNS_DeregisterService_drt(m, &m->SPSRecords, sps ? mDNS_Dereg_rapid : mDNS_Dereg_normal); } - - // Record our new SPS parameters - m->SPSType = sps; - m->SPSPortability = port; - m->SPSMarginalPower = marginalpower; - m->SPSTotalPower = totpower; - - // If turning on, open socket and advertise service - if (sps) - { - if (!m->SPSSocket) - { - m->SPSSocket = mDNSPlatformUDPSocket(m, zeroIPPort); - if (!m->SPSSocket) { LogMsg("mDNSCoreBeSleepProxyServer: Failed to allocate SPSSocket"); goto fail; } - } - if (m->SPSState == 0) SleepProxyServerCallback(m, &m->SPSRecords, mStatus_MemFree); - } - else if (m->SPSState) - { - LogSPS("mDNSCoreBeSleepProxyServer turning off from state %d; will wake clients", m->SPSState); - m->NextScheduledSPS = m->timenow; - } -fail: - mDNS_ReclaimLockAfterCallback(); - } - -// *************************************************************************** -#if COMPILER_LIKES_PRAGMA_MARK -#pragma mark - -#pragma mark - Startup and Shutdown -#endif - -mDNSlocal void mDNS_GrowCache_internal(mDNS *const m, CacheEntity *storage, mDNSu32 numrecords) - { - if (storage && numrecords) - { - mDNSu32 i; - debugf("Adding cache storage for %d more records (%d bytes)", numrecords, numrecords*sizeof(CacheEntity)); - for (i=0; irrcache_free; - m->rrcache_free = storage; - m->rrcache_size += numrecords; - } - } - -mDNSexport void mDNS_GrowCache(mDNS *const m, CacheEntity *storage, mDNSu32 numrecords) - { - mDNS_Lock(m); - mDNS_GrowCache_internal(m, storage, numrecords); - mDNS_Unlock(m); - } - -mDNSexport mStatus mDNS_Init(mDNS *const m, mDNS_PlatformSupport *const p, - CacheEntity *rrcachestorage, mDNSu32 rrcachesize, - mDNSBool AdvertiseLocalAddresses, mDNSCallback *Callback, void *Context) - { - mDNSu32 slot; - mDNSs32 timenow; - mStatus result; - - if (!rrcachestorage) rrcachesize = 0; - - m->p = p; - m->KnownBugs = 0; - m->CanReceiveUnicastOn5353 = mDNSfalse; // Assume we can't receive unicasts on 5353, unless platform layer tells us otherwise - m->AdvertiseLocalAddresses = AdvertiseLocalAddresses; - m->DivertMulticastAdvertisements = mDNSfalse; - m->mDNSPlatformStatus = mStatus_Waiting; - m->UnicastPort4 = zeroIPPort; - m->UnicastPort6 = zeroIPPort; - m->PrimaryMAC = zeroEthAddr; - m->MainCallback = Callback; - m->MainContext = Context; - m->rec.r.resrec.RecordType = 0; - - // For debugging: To catch and report locking failures - m->mDNS_busy = 0; - m->mDNS_reentrancy = 0; - m->ShutdownTime = 0; - m->lock_rrcache = 0; - m->lock_Questions = 0; - m->lock_Records = 0; - - // Task Scheduling variables - result = mDNSPlatformTimeInit(); - if (result != mStatus_NoError) return(result); - m->timenow_adjust = (mDNSs32)mDNSRandom(0xFFFFFFFF); - timenow = mDNS_TimeNow_NoLock(m); - - m->timenow = 0; // MUST only be set within mDNS_Lock/mDNS_Unlock section - m->timenow_last = timenow; - m->NextScheduledEvent = timenow; - m->SuppressSending = timenow; - m->NextCacheCheck = timenow + 0x78000000; - m->NextScheduledQuery = timenow + 0x78000000; - m->NextScheduledProbe = timenow + 0x78000000; - m->NextScheduledResponse = timenow + 0x78000000; - m->NextScheduledNATOp = timenow + 0x78000000; - m->NextScheduledSPS = timenow + 0x78000000; - m->NextScheduledStopTime = timenow + 0x78000000; - m->RandomQueryDelay = 0; - m->RandomReconfirmDelay = 0; - m->PktNum = 0; - m->LocalRemoveEvents = mDNSfalse; - m->SleepState = SleepState_Awake; - m->SleepSeqNum = 0; - m->SystemWakeOnLANEnabled = mDNSfalse; - m->AnnounceOwner = NonZeroTime(timenow + 60 * mDNSPlatformOneSecond); - m->DelaySleep = 0; - m->SleepLimit = 0; - - // These fields only required for mDNS Searcher... - m->Questions = mDNSNULL; - m->NewQuestions = mDNSNULL; - m->CurrentQuestion = mDNSNULL; - m->LocalOnlyQuestions = mDNSNULL; - m->NewLocalOnlyQuestions = mDNSNULL; - m->RestartQuestion = mDNSNULL; - m->rrcache_size = 0; - m->rrcache_totalused = 0; - m->rrcache_active = 0; - m->rrcache_report = 10; - m->rrcache_free = mDNSNULL; - - for (slot = 0; slot < CACHE_HASH_SLOTS; slot++) - { - m->rrcache_hash[slot] = mDNSNULL; - m->rrcache_nextcheck[slot] = timenow + 0x78000000;; - } - - mDNS_GrowCache_internal(m, rrcachestorage, rrcachesize); - m->rrauth.rrauth_free = mDNSNULL; - - for (slot = 0; slot < AUTH_HASH_SLOTS; slot++) - m->rrauth.rrauth_hash[slot] = mDNSNULL; - - // Fields below only required for mDNS Responder... - m->hostlabel.c[0] = 0; - m->nicelabel.c[0] = 0; - m->MulticastHostname.c[0] = 0; - m->HIHardware.c[0] = 0; - m->HISoftware.c[0] = 0; - m->ResourceRecords = mDNSNULL; - m->DuplicateRecords = mDNSNULL; - m->NewLocalRecords = mDNSNULL; - m->NewLocalOnlyRecords = mDNSfalse; - m->CurrentRecord = mDNSNULL; - m->HostInterfaces = mDNSNULL; - m->ProbeFailTime = 0; - m->NumFailedProbes = 0; - m->SuppressProbes = 0; - -#ifndef UNICAST_DISABLED - m->NextuDNSEvent = timenow + 0x78000000; - m->NextSRVUpdate = timenow + 0x78000000; - - m->DNSServers = mDNSNULL; - - m->Router = zeroAddr; - m->AdvertisedV4 = zeroAddr; - m->AdvertisedV6 = zeroAddr; - - m->AuthInfoList = mDNSNULL; - - m->ReverseMap.ThisQInterval = -1; - m->StaticHostname.c[0] = 0; - m->FQDN.c[0] = 0; - m->Hostnames = mDNSNULL; - m->AutoTunnelHostAddr.b[0] = 0; - m->AutoTunnelHostAddrActive = mDNSfalse; - m->AutoTunnelLabel.c[0] = 0; - - m->StartWABQueries = mDNSfalse; - m->RegisterAutoTunnel6 = mDNStrue; - - // NAT traversal fields - m->NATTraversals = mDNSNULL; - m->CurrentNATTraversal = mDNSNULL; - m->retryIntervalGetAddr = 0; // delta between time sent and retry - m->retryGetAddr = timenow + 0x78000000; // absolute time when we retry - m->ExternalAddress = zerov4Addr; - - m->NATMcastRecvskt = mDNSNULL; - m->LastNATupseconds = 0; - m->LastNATReplyLocalTime = timenow; - m->LastNATMapResultCode = NATErr_None; - - m->UPnPInterfaceID = 0; - m->SSDPSocket = mDNSNULL; - m->SSDPWANPPPConnection = mDNSfalse; - m->UPnPRouterPort = zeroIPPort; - m->UPnPSOAPPort = zeroIPPort; - m->UPnPRouterURL = mDNSNULL; - m->UPnPWANPPPConnection = mDNSfalse; - m->UPnPSOAPURL = mDNSNULL; - m->UPnPRouterAddressString = mDNSNULL; - m->UPnPSOAPAddressString = mDNSNULL; - m->SPSType = 0; - m->SPSPortability = 0; - m->SPSMarginalPower = 0; - m->SPSTotalPower = 0; - m->SPSState = 0; - m->SPSProxyListChanged = mDNSNULL; - m->SPSSocket = mDNSNULL; - m->SPSBrowseCallback = mDNSNULL; - m->ProxyRecords = 0; - -#endif - -#if APPLE_OSX_mDNSResponder - m->TunnelClients = mDNSNULL; - -#if ! NO_WCF - CHECK_WCF_FUNCTION(WCFConnectionNew) - { - m->WCF = WCFConnectionNew(); - if (!m->WCF) { LogMsg("WCFConnectionNew failed"); return -1; } - } -#endif - -#endif - - result = mDNSPlatformInit(m); - -#ifndef UNICAST_DISABLED - // It's better to do this *after* the platform layer has set up the - // interface list and security credentials - uDNS_SetupDNSConfig(m); // Get initial DNS configuration -#endif - - return(result); - } - -mDNSexport void mDNS_ConfigChanged(mDNS *const m) - { - if (m->SPSState == 1) - { - domainlabel name, newname; - domainname type, domain; - DeconstructServiceName(m->SPSRecords.RR_SRV.resrec.name, &name, &type, &domain); - ConstructSleepProxyServerName(m, &newname); - if (!SameDomainLabelCS(name.c, newname.c)) - { - LogSPS("Renaming SPS from '%#s' to '%#s'", name.c, newname.c); - // When SleepProxyServerCallback gets the mStatus_MemFree message, - // it will reregister the service under the new name - m->SPSState = 2; - mDNS_DeregisterService_drt(m, &m->SPSRecords, mDNS_Dereg_rapid); - } - } - - if (m->MainCallback) - m->MainCallback(m, mStatus_ConfigChanged); - } - -mDNSlocal void DynDNSHostNameCallback(mDNS *const m, AuthRecord *const rr, mStatus result) - { - (void)m; // unused - debugf("NameStatusCallback: result %d for registration of name %##s", result, rr->resrec.name->c); - mDNSPlatformDynDNSHostNameStatusChanged(rr->resrec.name, result); - } - -mDNSlocal void PurgeOrReconfirmCacheRecord(mDNS *const m, CacheRecord *cr, const DNSServer * const ptr, mDNSBool lameduck) - { - mDNSBool purge = cr->resrec.RecordType == kDNSRecordTypePacketNegative || - cr->resrec.rrtype == kDNSType_A || - cr->resrec.rrtype == kDNSType_AAAA || - cr->resrec.rrtype == kDNSType_SRV; - - (void) lameduck; - (void) ptr; - debugf("PurgeOrReconfirmCacheRecord: %s cache record due to %s server %p %#a:%d (%##s): %s", - purge ? "purging" : "reconfirming", - lameduck ? "lame duck" : "new", - ptr, &ptr->addr, mDNSVal16(ptr->port), ptr->domain.c, CRDisplayString(m, cr)); - - if (purge) - { - LogInfo("PurgeorReconfirmCacheRecord: Purging Resourcerecord %s, RecordType %x", CRDisplayString(m, cr), cr->resrec.RecordType); - mDNS_PurgeCacheResourceRecord(m, cr); - } - else - { - LogInfo("PurgeorReconfirmCacheRecord: Reconfirming Resourcerecord %s, RecordType %x", CRDisplayString(m, cr), cr->resrec.RecordType); - mDNS_Reconfirm_internal(m, cr, kDefaultReconfirmTimeForNoAnswer); - } - } - -mDNSlocal void mDNS_PurgeBeforeResolve(mDNS *const m, DNSQuestion *q) - { - const mDNSu32 slot = HashSlot(&q->qname); - CacheGroup *const cg = CacheGroupForName(m, slot, q->qnamehash, &q->qname); - CacheRecord *rp; - - for (rp = cg ? cg->members : mDNSNULL; rp; rp = rp->next) - { - if (SameNameRecordAnswersQuestion(&rp->resrec, q)) - { - LogInfo("mDNS_PurgeBeforeResolve: Flushing %s", CRDisplayString(m, rp)); - mDNS_PurgeCacheResourceRecord(m, rp); - } - } - } - -mDNSlocal void CacheRecordResetDNSServer(mDNS *const m, DNSQuestion *q, DNSServer *new) - { - const mDNSu32 slot = HashSlot(&q->qname); - CacheGroup *const cg = CacheGroupForName(m, slot, q->qnamehash, &q->qname); - CacheRecord *rp; - mDNSBool found = mDNSfalse; - mDNSBool foundNew = mDNSfalse; - DNSServer *old = q->qDNSServer; - mDNSBool newQuestion = IsQuestionNew(m, q); - DNSQuestion *qptr; - - // This function is called when the DNSServer is updated to the new question. There may already be - // some cache entries matching the old DNSServer and/or new DNSServer. There are four cases. In the - // following table, "Yes" denotes that a cache entry was found for old/new DNSServer. - // - // old DNSServer new DNSServer - // - // Case 1 Yes Yes - // Case 2 No Yes - // Case 3 Yes No - // Case 4 No No - // - // Case 1: There are cache entries for both old and new DNSServer. We handle this case by simply - // expiring the old Cache entries, deliver a RMV event (if an ADD event was delivered before) - // followed by the ADD event of the cache entries corresponding to the new server. This - // case happens when we pick a DNSServer, issue a query and get a valid response and create - // cache entries after which it stops responding. Another query (non-duplicate) picks a different - // DNSServer and creates identical cache entries (perhaps through records in Additional records). - // Now if the first one expires and tries to pick the new DNSServer (the original DNSServer - // is not responding) we will find cache entries corresponding to both DNSServers. - // - // Case 2: There are no cache entries for the old DNSServer but there are some for the new DNSServer. - // This means we should deliver an ADD event. Normally ADD events are delivered by - // AnswerNewQuestion if it is a new question. So, we check to see if it is a new question - // and if so, leave it to AnswerNewQuestion to deliver it. Otherwise, we use - // AnswerQuestionsForDNSServerChanges to deliver the ADD event. This case happens when a - // question picks a DNS server for which AnswerNewQuestion could not deliver an answer even - // though there were potential cache entries but DNSServer did not match. Now when we - // pick a new DNSServer, those cache entries may answer this question. - // - // Case 3: There are the cache entries for the old DNSServer but none for the new. We just move - // the old cache entries to point to the new DNSServer and the caller is expected to - // do a purge or reconfirm to delete or validate the RDATA. We don't need to do anything - // special for delivering ADD events, as it should have been done/will be done by - // AnswerNewQuestion. This case happens when we picked a DNSServer, sent the query and - // got a response and the cache is expired now and we are reissuing the question but the - // original DNSServer does not respond. - // - // Case 4: There are no cache entries either for the old or for the new DNSServer. There is nothing - // much we can do here. - // - // Case 2 and 3 are the most common while case 4 is possible when no DNSServers are working. Case 1 - // is relatively less likely to happen in practice - - // Temporarily set the DNSServer to look for the matching records for the new DNSServer. - q->qDNSServer = new; - for (rp = cg ? cg->members : mDNSNULL; rp; rp = rp->next) - { - if (SameNameRecordAnswersQuestion(&rp->resrec, q)) - { - LogInfo("CacheRecordResetDNSServer: Found cache record %##s for new DNSServer address: %#a", rp->resrec.name->c, - (rp->resrec.rDNSServer != mDNSNULL ? &rp->resrec.rDNSServer->addr : mDNSNULL)); - foundNew = mDNStrue; - break; - } - } - q->qDNSServer = old; - - for (rp = cg ? cg->members : mDNSNULL; rp; rp = rp->next) - { - if (SameNameRecordAnswersQuestion(&rp->resrec, q)) - { - // Case1 - found = mDNStrue; - if (foundNew) - { - LogInfo("CacheRecordResetDNSServer: Flushing Resourcerecord %##s, before:%#a, after:%#a", rp->resrec.name->c, - (rp->resrec.rDNSServer != mDNSNULL ? &rp->resrec.rDNSServer->addr : mDNSNULL), - (new != mDNSNULL ? &new->addr : mDNSNULL)); - mDNS_PurgeCacheResourceRecord(m, rp); - if (newQuestion) - { - // "q" is not a duplicate question. If it is a newQuestion, then the CRActiveQuestion can't be - // possibly set as it is set only when we deliver the ADD event to the question. - if (rp->CRActiveQuestion != mDNSNULL) - { - LogMsg("CacheRecordResetDNSServer: ERROR!!: CRActiveQuestion %p set, current question %p, name %##s", rp->CRActiveQuestion, q, q->qname.c); - rp->CRActiveQuestion = mDNSNULL; - } - // if this is a new question, then we never delivered an ADD yet, so don't deliver the RMV. - continue; - } - } - LogInfo("CacheRecordResetDNSServer: resetting cache record %##s DNSServer address before:%#a," - " after:%#a, CRActiveQuestion %p", rp->resrec.name->c, (rp->resrec.rDNSServer != mDNSNULL ? - &rp->resrec.rDNSServer->addr : mDNSNULL), (new != mDNSNULL ? &new->addr : mDNSNULL), - rp->CRActiveQuestion); - // Though we set it to the new DNS server, the caller is *assumed* to do either a purge - // or reconfirm or send out questions to the "new" server to verify whether the cached - // RDATA is valid - rp->resrec.rDNSServer = new; - } - } - - // Case 1 and Case 2 - if ((found && foundNew) || (!found && foundNew)) - { - if (newQuestion) - LogInfo("CacheRecordResetDNSServer: deliverAddEvents not set for question %p %##s (%s)", q, q->qname.c, DNSTypeName(q->qtype)); - else if (QuerySuppressed(q)) - LogInfo("CacheRecordResetDNSServer: deliverAddEvents not set for suppressed question %p %##s (%s)", q, q->qname.c, DNSTypeName(q->qtype)); - else - { - LogInfo("CacheRecordResetDNSServer: deliverAddEvents set for %p %##s (%s)", q, q->qname.c, DNSTypeName(q->qtype)); - q->deliverAddEvents = mDNStrue; - for (qptr = q->next; qptr; qptr = qptr->next) - if (qptr->DuplicateOf == q) qptr->deliverAddEvents = mDNStrue; - } - return; - } - - // Case 3 and Case 4 - return; - } - -mDNSexport void DNSServerChangeForQuestion(mDNS *const m, DNSQuestion *q, DNSServer *new) - { - DNSQuestion *qptr; - - // 1. Whenever we change the DNS server, we change the message identifier also so that response - // from the old server is not accepted as a response from the new server but only messages - // from the new server are accepted as valid responses. We do it irrespective of whether "new" - // is NULL or not. It is possible that we send two queries, no responses, pick a new DNS server - // which is NULL and now the response comes back and will try to penalize the DNS server which - // is NULL. By setting the messageID here, we will not accept that as a valid response. - - q->TargetQID = mDNS_NewMessageID(m); - - // 2. Move the old cache records to point them at the new DNSServer so that we can deliver the ADD/RMV events - // appropriately. At any point in time, we want all the cache records point only to one DNSServer for a given - // question. "DNSServer" here is the DNSServer object and not the DNS server itself. It is possible to - // have the same DNS server address in two objects, one scoped and another not scoped. But, the cache is per - // DNSServer object. By maintaining the question and the cache entries point to the same DNSServer - // always, the cache maintenance and delivery of ADD/RMV events becomes simpler. - // - // CacheRecordResetDNSServer should be called only once for the non-duplicate question as once the cache - // entries are moved to point to the new DNSServer, we don't need to call it for the duplicate question - // and it is wrong to call for the duplicate question as it's decision to mark deliverAddevents will be - // incorrect. - - if (q->DuplicateOf) - LogMsg("DNSServerChangeForQuestion: ERROR: Called for duplicate question %##s", q->qname.c); - else - CacheRecordResetDNSServer(m, q, new); - - // 3. Make sure all the duplicate questions point to the same DNSServer so that delivery - // of events for all of them are consistent. Duplicates for a question are always inserted - // after in the list. - q->qDNSServer = new; - for (qptr = q->next ; qptr; qptr = qptr->next) - { - if (qptr->DuplicateOf == q) { qptr->validDNSServers = q->validDNSServers; qptr->qDNSServer = new; } - } - } - -mDNSexport mStatus uDNS_SetupDNSConfig(mDNS *const m) - { - mDNSu32 slot; - CacheGroup *cg; - CacheRecord *cr; - - mDNSAddr v4, v6, r; - domainname fqdn; - DNSServer *ptr, **p = &m->DNSServers; - const DNSServer *oldServers = m->DNSServers; - DNSQuestion *q; - McastResolver *mr, **mres = &m->McastResolvers; - - debugf("uDNS_SetupDNSConfig: entry"); - - // Let the platform layer get the current DNS information - // The m->StartWABQueries is set when we get the first domain enumeration query (no need to hit the network - // with domain enumeration queries until we actually need that information). Even if it is not set, we still - // need to setup the search domains so that we can append them to queries that need them. - - uDNS_SetupSearchDomains(m, m->StartWABQueries ? UDNS_START_WAB_QUERY : 0); - - mDNS_Lock(m); - - for (ptr = m->DNSServers; ptr; ptr = ptr->next) - { - ptr->penaltyTime = 0; - ptr->flags |= DNSServer_FlagDelete; - } - - // We handle the mcast resolvers here itself as mDNSPlatformSetDNSConfig looks at - // mcast resolvers. Today we get both mcast and ucast configuration using the same - // API - for (mr = m->McastResolvers; mr; mr = mr->next) - mr->flags |= McastResolver_FlagDelete; - - mDNSPlatformSetDNSConfig(m, mDNStrue, mDNSfalse, &fqdn, mDNSNULL, mDNSNULL); - - // For now, we just delete the mcast resolvers. We don't deal with cache or - // questions here. Neither question nor cache point to mcast resolvers. Questions - // do inherit the timeout values from mcast resolvers. But we don't bother - // affecting them as they never change. - while (*mres) - { - if (((*mres)->flags & DNSServer_FlagDelete) != 0) - { - mr = *mres; - *mres = (*mres)->next; - debugf("uDNS_SetupDNSConfig: Deleting mcast resolver %##s", mr, mr->domain.c); - mDNSPlatformMemFree(mr); - } - else - { - (*mres)->flags &= ~McastResolver_FlagNew; - mres = &(*mres)->next; - } - } - - // Mark the records to be flushed that match a new resolver. We need to do this before - // we walk the questions below where we change the DNSServer pointer of the cache - // record - FORALL_CACHERECORDS(slot, cg, cr) - { - if (cr->resrec.InterfaceID) continue; - - // We just mark them for purge or reconfirm. We can't affect the DNSServer pointer - // here as the code below that calls CacheRecordResetDNSServer relies on this - // - // The new DNSServer may be a scoped or non-scoped one. We use the active question's - // InterfaceID for looking up the right DNS server - ptr = GetServerForName(m, cr->resrec.name, cr->CRActiveQuestion ? cr->CRActiveQuestion->InterfaceID : mDNSNULL); - - // Purge or Reconfirm if this cache entry would use the new DNS server - if (ptr && (ptr != cr->resrec.rDNSServer)) - { - // As the DNSServers for this cache record is not the same anymore, we don't - // want any new questions to pick this old value - if (cr->CRActiveQuestion == mDNSNULL) - { - LogInfo("uDNS_SetupDNSConfig: Purging Resourcerecord %s", CRDisplayString(m, cr)); - mDNS_PurgeCacheResourceRecord(m, cr); - } - else - { - LogInfo("uDNS_SetupDNSConfig: Purging/Reconfirming Resourcerecord %s", CRDisplayString(m, cr)); - PurgeOrReconfirmCacheRecord(m, cr, ptr, mDNSfalse); - } - } - } - // Update our qDNSServer pointers before we go and free the DNSServer object memory - for (q = m->Questions; q; q=q->next) - if (!mDNSOpaque16IsZero(q->TargetQID)) - { - DNSServer *s, *t; - DNSQuestion *qptr; - if (q->DuplicateOf) continue; - SetValidDNSServers(m, q); - q->triedAllServersOnce = 0; - s = GetServerForQuestion(m, q); - t = q->qDNSServer; - if (t != s) - { - // If DNS Server for this question has changed, reactivate it - debugf("uDNS_SetupDNSConfig: Updating DNS Server from %p %#a:%d (%##s) to %p %#a:%d (%##s) for %##s (%s)", - t, t ? &t->addr : mDNSNULL, mDNSVal16(t ? t->port : zeroIPPort), t ? t->domain.c : (mDNSu8*)"", - s, s ? &s->addr : mDNSNULL, mDNSVal16(s ? s->port : zeroIPPort), s ? s->domain.c : (mDNSu8*)"", - q->qname.c, DNSTypeName(q->qtype)); - - // After we reset the DNSServer pointer on the cache records here, three things could happen: - // - // 1) The query gets sent out and when the actual response comes back later it is possible - // that the response has the same RDATA, in which case we update our cache entry. - // If the response is different, then the entry will expire and a new entry gets added. - // For the latter case to generate a RMV followed by ADD events, we need to reset the DNS - // server here to match the question and the cache record. - // - // 2) We might have marked the cache entries for purge above and for us to be able to generate the RMV - // events for the questions, the DNSServer on the question should match the Cache Record - // - // 3) We might have marked the cache entries for reconfirm above, for which we send the query out which is - // the same as the first case above. - - DNSServerChangeForQuestion(m, q, s); - q->unansweredQueries = 0; - // We still need to pick a new DNSServer for the questions that have been - // suppressed, but it is wrong to activate the query as DNS server change - // could not possibly change the status of SuppressUnusable questions - if (!QuerySuppressed(q)) - { - debugf("uDNS_SetupDNSConfig: Activating query %p %##s (%s)", q, q->qname.c, DNSTypeName(q->qtype)); - ActivateUnicastQuery(m, q, mDNStrue); - // ActivateUnicastQuery is called for duplicate questions also as it does something - // special for AutoTunnel questions - for (qptr = q->next ; qptr; qptr = qptr->next) - { - if (qptr->DuplicateOf == q) ActivateUnicastQuery(m, qptr, mDNStrue); - } - } - } - else - { - debugf("uDNS_SetupDNSConfig: Not Updating DNS server question %p %##s (%s) DNS server %#a:%d %p %d", - q, q->qname.c, DNSTypeName(q->qtype), t ? &t->addr : mDNSNULL, mDNSVal16(t ? t->port : zeroIPPort), q->DuplicateOf, q->SuppressUnusable); - for (qptr = q->next ; qptr; qptr = qptr->next) - if (qptr->DuplicateOf == q) { qptr->validDNSServers = q->validDNSServers; qptr->qDNSServer = q->qDNSServer; } - } - } - - while (*p) - { - if (((*p)->flags & DNSServer_FlagDelete) != 0) - { - // Scan our cache, looking for uDNS records that we would have queried this server for. - // We reconfirm any records that match, because in this world of split DNS, firewalls, etc. - // different DNS servers can give different answers to the same question. - ptr = *p; - FORALL_CACHERECORDS(slot, cg, cr) - { - if (cr->resrec.InterfaceID) continue; - if (cr->resrec.rDNSServer == ptr) - { - // If we don't have an active question for this cache record, neither Purge can - // generate RMV events nor Reconfirm can send queries out. Just set the DNSServer - // pointer on the record NULL so that we don't point to freed memory (We might dereference - // DNSServer pointers from resource record for logging purposes). - // - // If there is an active question, point to its DNSServer as long as it does not point to the - // freed one. We already went through the questions above and made them point at either the - // new server or NULL if there is no server and also affected the cache entries that match - // this question. Hence, whenever we hit a resource record with a DNSServer that is just - // about to be deleted, we should never have an active question. The code below just tries to - // be careful logging messages if we ever hit this case. - - if (cr->CRActiveQuestion) - { - DNSQuestion *qptr = cr->CRActiveQuestion; - if (qptr->qDNSServer == mDNSNULL) - LogMsg("uDNS_SetupDNSConfig: Cache Record %s match: Active question %##s (%s) with DNSServer Address NULL, Server to be deleted %#a", - CRDisplayString(m, cr), qptr->qname.c, DNSTypeName(qptr->qtype), &ptr->addr); - else - LogMsg("uDNS_SetupDNSConfig: Cache Record %s match: Active question %##s (%s) DNSServer Address %#a, Server to be deleted %#a", - CRDisplayString(m, cr), qptr->qname.c, DNSTypeName(qptr->qtype), &qptr->qDNSServer->addr, &ptr->addr); - - if (qptr->qDNSServer == ptr) - { - qptr->validDNSServers = zeroOpaque64; - qptr->qDNSServer = mDNSNULL; - cr->resrec.rDNSServer = mDNSNULL; - } - else - { - cr->resrec.rDNSServer = qptr->qDNSServer; - } - } - else - { - LogInfo("uDNS_SetupDNSConfig: Cache Record %##s has no Active question, Record's DNSServer Address %#a, Server to be deleted %#a", - cr->resrec.name, &cr->resrec.rDNSServer->addr, &ptr->addr); - cr->resrec.rDNSServer = mDNSNULL; - } - - PurgeOrReconfirmCacheRecord(m, cr, ptr, mDNStrue); - } - } - *p = (*p)->next; - debugf("uDNS_SetupDNSConfig: Deleting server %p %#a:%d (%##s)", ptr, &ptr->addr, mDNSVal16(ptr->port), ptr->domain.c); - mDNSPlatformMemFree(ptr); - NumUnicastDNSServers--; - } - else - { - (*p)->flags &= ~DNSServer_FlagNew; - p = &(*p)->next; - } - } - - // If we now have no DNS servers at all and we used to have some, then immediately purge all unicast cache records (including for LLQs). - // This is important for giving prompt remove events when the user disconnects the Ethernet cable or turns off wireless. - // Otherwise, stale data lingers for 5-10 seconds, which is not the user-experience people expect from Bonjour. - // Similarly, if we now have some DNS servers and we used to have none, we want to purge any fake negative results we may have generated. - if ((m->DNSServers != mDNSNULL) != (oldServers != mDNSNULL)) - { - int count = 0; - FORALL_CACHERECORDS(slot, cg, cr) if (!cr->resrec.InterfaceID) { mDNS_PurgeCacheResourceRecord(m, cr); count++; } - LogInfo("uDNS_SetupDNSConfig: %s available; purged %d unicast DNS records from cache", - m->DNSServers ? "DNS server became" : "No DNS servers", count); - - // Force anything that needs to get zone data to get that information again - RestartRecordGetZoneData(m); - } - - // Did our FQDN change? - if (!SameDomainName(&fqdn, &m->FQDN)) - { - if (m->FQDN.c[0]) mDNS_RemoveDynDNSHostName(m, &m->FQDN); - - AssignDomainName(&m->FQDN, &fqdn); - - if (m->FQDN.c[0]) - { - mDNSPlatformDynDNSHostNameStatusChanged(&m->FQDN, 1); - mDNS_AddDynDNSHostName(m, &m->FQDN, DynDNSHostNameCallback, mDNSNULL); - } - } - - mDNS_Unlock(m); - - // handle router and primary interface changes - v4 = v6 = r = zeroAddr; - v4.type = r.type = mDNSAddrType_IPv4; - - if (mDNSPlatformGetPrimaryInterface(m, &v4, &v6, &r) == mStatus_NoError && !mDNSv4AddressIsLinkLocal(&v4.ip.v4)) - { - mDNS_SetPrimaryInterfaceInfo(m, - !mDNSIPv4AddressIsZero(v4.ip.v4) ? &v4 : mDNSNULL, - !mDNSIPv6AddressIsZero(v6.ip.v6) ? &v6 : mDNSNULL, - !mDNSIPv4AddressIsZero(r .ip.v4) ? &r : mDNSNULL); - } - else - { - mDNS_SetPrimaryInterfaceInfo(m, mDNSNULL, mDNSNULL, mDNSNULL); - if (m->FQDN.c[0]) mDNSPlatformDynDNSHostNameStatusChanged(&m->FQDN, 1); // Set status to 1 to indicate temporary failure - } - - debugf("uDNS_SetupDNSConfig: number of unicast DNS servers %d", NumUnicastDNSServers); - return mStatus_NoError; - } - -mDNSexport void mDNSCoreInitComplete(mDNS *const m, mStatus result) - { - m->mDNSPlatformStatus = result; - if (m->MainCallback) - { - mDNS_Lock(m); - mDNS_DropLockBeforeCallback(); // Allow client to legally make mDNS API calls from the callback - m->MainCallback(m, mStatus_NoError); - mDNS_ReclaimLockAfterCallback(); // Decrement mDNS_reentrancy to block mDNS API calls again - mDNS_Unlock(m); - } - } - -mDNSlocal void DeregLoop(mDNS *const m, AuthRecord *const start) - { - m->CurrentRecord = start; - while (m->CurrentRecord) - { - AuthRecord *rr = m->CurrentRecord; - LogInfo("DeregLoop: %s deregistration for %p %02X %s", - (rr->resrec.RecordType != kDNSRecordTypeDeregistering) ? "Initiating " : "Accelerating", - rr, rr->resrec.RecordType, ARDisplayString(m, rr)); - if (rr->resrec.RecordType != kDNSRecordTypeDeregistering) - mDNS_Deregister_internal(m, rr, mDNS_Dereg_rapid); - else if (rr->AnnounceCount > 1) - { - rr->AnnounceCount = 1; - rr->LastAPTime = m->timenow - rr->ThisAPInterval; - } - // Mustn't advance m->CurrentRecord until *after* mDNS_Deregister_internal, because - // new records could have been added to the end of the list as a result of that call. - if (m->CurrentRecord == rr) // If m->CurrentRecord was not advanced for us, do it now - m->CurrentRecord = rr->next; - } - } - -mDNSexport void mDNS_StartExit(mDNS *const m) - { - NetworkInterfaceInfo *intf; - AuthRecord *rr; - - mDNS_Lock(m); - - LogInfo("mDNS_StartExit"); - m->ShutdownTime = NonZeroTime(m->timenow + mDNSPlatformOneSecond * 5); - - mDNSCoreBeSleepProxyServer_internal(m, 0, 0, 0, 0); - -#if APPLE_OSX_mDNSResponder -#if ! NO_WCF - CHECK_WCF_FUNCTION(WCFConnectionDealloc) - { - if (m->WCF) WCFConnectionDealloc((WCFConnection *)m->WCF); - } -#endif -#endif - -#ifndef UNICAST_DISABLED - { - SearchListElem *s; - SuspendLLQs(m); - // Don't need to do SleepRecordRegistrations() here - // because we deregister all records and services later in this routine - while (m->Hostnames) mDNS_RemoveDynDNSHostName(m, &m->Hostnames->fqdn); - - // For each member of our SearchList, deregister any records it may have created, and cut them from the list. - // Otherwise they'll be forcibly deregistered for us (without being cut them from the appropriate list) - // and we may crash because the list still contains dangling pointers. - for (s = SearchList; s; s = s->next) - while (s->AuthRecs) - { - ARListElem *dereg = s->AuthRecs; - s->AuthRecs = s->AuthRecs->next; - mDNS_Deregister_internal(m, &dereg->ar, mDNS_Dereg_normal); // Memory will be freed in the FreeARElemCallback - } - } -#endif - - for (intf = m->HostInterfaces; intf; intf = intf->next) - if (intf->Advertise) - DeadvertiseInterface(m, intf); - - // Shut down all our active NAT Traversals - while (m->NATTraversals) - { - NATTraversalInfo *t = m->NATTraversals; - mDNS_StopNATOperation_internal(m, t); // This will cut 't' from the list, thereby advancing m->NATTraversals in the process - - // After stopping the NAT Traversal, we zero out the fields. - // This has particularly important implications for our AutoTunnel records -- - // when we deregister our AutoTunnel records below, we don't want their mStatus_MemFree - // handlers to just turn around and attempt to re-register those same records. - // Clearing t->ExternalPort/t->RequestedPort will cause the mStatus_MemFree callback handlers - // to not do this. - t->ExternalAddress = zerov4Addr; - t->ExternalPort = zeroIPPort; - t->RequestedPort = zeroIPPort; - t->Lifetime = 0; - t->Result = mStatus_NoError; - } - - // Make sure there are nothing but deregistering records remaining in the list - if (m->CurrentRecord) - LogMsg("mDNS_StartExit: ERROR m->CurrentRecord already set %s", ARDisplayString(m, m->CurrentRecord)); - - // We're in the process of shutting down, so queries, etc. are no longer available. - // Consequently, determining certain information, e.g. the uDNS update server's IP - // address, will not be possible. The records on the main list are more likely to - // already contain such information, so we deregister the duplicate records first. - LogInfo("mDNS_StartExit: Deregistering duplicate resource records"); - DeregLoop(m, m->DuplicateRecords); - LogInfo("mDNS_StartExit: Deregistering resource records"); - DeregLoop(m, m->ResourceRecords); - - // If we scheduled a response to send goodbye packets, we set NextScheduledResponse to now. Normally when deregistering records, - // we allow up to 100ms delay (to help improve record grouping) but when shutting down we don't want any such delay. - if (m->NextScheduledResponse - m->timenow < mDNSPlatformOneSecond) - { - m->NextScheduledResponse = m->timenow; - m->SuppressSending = 0; - } - - if (m->ResourceRecords) LogInfo("mDNS_StartExit: Sending final record deregistrations"); - else LogInfo("mDNS_StartExit: No deregistering records remain"); - - for (rr = m->DuplicateRecords; rr; rr = rr->next) - LogMsg("mDNS_StartExit: Should not still have Duplicate Records remaining: %02X %s", rr->resrec.RecordType, ARDisplayString(m, rr)); - - // If any deregistering records remain, send their deregistration announcements before we exit - if (m->mDNSPlatformStatus != mStatus_NoError) DiscardDeregistrations(m); - - mDNS_Unlock(m); - - LogInfo("mDNS_StartExit: done"); - } - -mDNSexport void mDNS_FinalExit(mDNS *const m) - { - mDNSu32 rrcache_active = 0; - mDNSu32 rrcache_totalused = 0; - mDNSu32 slot; - AuthRecord *rr; - - LogInfo("mDNS_FinalExit: mDNSPlatformClose"); - mDNSPlatformClose(m); - - rrcache_totalused = m->rrcache_totalused; - for (slot = 0; slot < CACHE_HASH_SLOTS; slot++) - { - while (m->rrcache_hash[slot]) - { - CacheGroup *cg = m->rrcache_hash[slot]; - while (cg->members) - { - CacheRecord *cr = cg->members; - cg->members = cg->members->next; - if (cr->CRActiveQuestion) rrcache_active++; - ReleaseCacheRecord(m, cr); - } - cg->rrcache_tail = &cg->members; - ReleaseCacheGroup(m, &m->rrcache_hash[slot]); - } - } - debugf("mDNS_FinalExit: RR Cache was using %ld records, %lu active", rrcache_totalused, rrcache_active); - if (rrcache_active != m->rrcache_active) - LogMsg("*** ERROR *** rrcache_active %lu != m->rrcache_active %lu", rrcache_active, m->rrcache_active); - - for (rr = m->ResourceRecords; rr; rr = rr->next) - LogMsg("mDNS_FinalExit failed to send goodbye for: %p %02X %s", rr, rr->resrec.RecordType, ARDisplayString(m, rr)); - - LogInfo("mDNS_FinalExit: done"); - } diff -Nru qtcreator-2.5.0/src/tools/mdnssd/mDNSDebug.c qtcreator-2.5.2/src/tools/mdnssd/mDNSDebug.c --- qtcreator-2.5.0/src/tools/mdnssd/mDNSDebug.c 2012-05-09 12:13:19.000000000 +0000 +++ qtcreator-2.5.2/src/tools/mdnssd/mDNSDebug.c 1970-01-01 00:00:00.000000000 +0000 @@ -1,93 +0,0 @@ -/* -*- Mode: C; tab-width: 4 -*- - * - * Copyright (c) 2003 Apple Computer, Inc. All rights reserved. - * - * Licensed under the Apache License, Version 2.0 (the "License"); - * you may not use this file except in compliance with the License. - * You may obtain a copy of the License at - * - * http://www.apache.org/licenses/LICENSE-2.0 - * - * Unless required by applicable law or agreed to in writing, software - * distributed under the License is distributed on an "AS IS" BASIS, - * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. - * See the License for the specific language governing permissions and - * limitations under the License. - - File: mDNSDebug.c - - Contains: Implementation of debugging utilities. Requires a POSIX environment. - - Version: 1.0 - - */ - -#include "mDNSDebug.h" - -#include - -#if defined(WIN32) || defined(EFI32) || defined(EFI64) || defined(EFIX64) -// Need to add Windows/EFI syslog support here -#define LOG_PID 0x01 -#define LOG_CONS 0x02 -#define LOG_PERROR 0x20 -#else -#include -#endif - -#include "mDNSEmbeddedAPI.h" - -mDNSexport int mDNS_LoggingEnabled = 0; -mDNSexport int mDNS_PacketLoggingEnabled = 0; - -#if MDNS_DEBUGMSGS -mDNSexport int mDNS_DebugMode = mDNStrue; -#else -mDNSexport int mDNS_DebugMode = mDNSfalse; -#endif - -// Note, this uses mDNS_vsnprintf instead of standard "vsnprintf", because mDNS_vsnprintf knows -// how to print special data types like IP addresses and length-prefixed domain names -#if MDNS_DEBUGMSGS > 1 -mDNSexport void verbosedebugf_(const char *format, ...) - { - char buffer[512]; - va_list ptr; - va_start(ptr,format); - buffer[mDNS_vsnprintf(buffer, sizeof(buffer), format, ptr)] = 0; - va_end(ptr); - mDNSPlatformWriteDebugMsg(buffer); - } -#endif - -// Log message with default "mDNSResponder" ident string at the start -mDNSlocal void LogMsgWithLevelv(mDNSLogLevel_t logLevel, const char *format, va_list ptr) - { - char buffer[512]; - buffer[mDNS_vsnprintf((char *)buffer, sizeof(buffer), format, ptr)] = 0; - mDNSPlatformWriteLogMsg(ProgramName, buffer, logLevel); - } - -#define LOG_HELPER_BODY(L) \ - { \ - va_list ptr; \ - va_start(ptr,format); \ - LogMsgWithLevelv(L, format, ptr); \ - va_end(ptr); \ - } - -// see mDNSDebug.h -#if !MDNS_HAS_VA_ARG_MACROS -void LogMsg_(const char *format, ...) LOG_HELPER_BODY(MDNS_LOG_MSG) -void LogOperation_(const char *format, ...) LOG_HELPER_BODY(MDNS_LOG_OPERATION) -void LogSPS_(const char *format, ...) LOG_HELPER_BODY(MDNS_LOG_SPS) -void LogInfo_(const char *format, ...) LOG_HELPER_BODY(MDNS_LOG_INFO) -#endif - -#if MDNS_DEBUGMSGS -void debugf_(const char *format, ...) LOG_HELPER_BODY(MDNS_LOG_DEBUG) -#endif - -// Log message with default "mDNSResponder" ident string at the start -mDNSexport void LogMsgWithLevel(mDNSLogLevel_t logLevel, const char *format, ...) - LOG_HELPER_BODY(logLevel) diff -Nru qtcreator-2.5.0/src/tools/mdnssd/mDNSDebug.h qtcreator-2.5.2/src/tools/mdnssd/mDNSDebug.h --- qtcreator-2.5.0/src/tools/mdnssd/mDNSDebug.h 2012-05-09 12:13:19.000000000 +0000 +++ qtcreator-2.5.2/src/tools/mdnssd/mDNSDebug.h 1970-01-01 00:00:00.000000000 +0000 @@ -1,164 +0,0 @@ -/* -*- Mode: C; tab-width: 4 -*- - * - * Copyright (c) 2002-2003 Apple Computer, Inc. All rights reserved. - * - * Licensed under the Apache License, Version 2.0 (the "License"); - * you may not use this file except in compliance with the License. - * You may obtain a copy of the License at - * - * http://www.apache.org/licenses/LICENSE-2.0 - * - * Unless required by applicable law or agreed to in writing, software - * distributed under the License is distributed on an "AS IS" BASIS, - * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. - * See the License for the specific language governing permissions and - * limitations under the License. - */ - -#ifndef __mDNSDebug_h -#define __mDNSDebug_h - -// Set MDNS_DEBUGMSGS to 0 to optimize debugf() calls out of the compiled code -// Set MDNS_DEBUGMSGS to 1 to generate normal debugging messages -// Set MDNS_DEBUGMSGS to 2 to generate verbose debugging messages -// MDNS_DEBUGMSGS is normally set in the project options (or makefile) but can also be set here if desired -// (If you edit the file here to turn on MDNS_DEBUGMSGS while you're debugging some code, be careful -// not to accidentally check-in that change by mistake when you check in your other changes.) - -//#undef MDNS_DEBUGMSGS -//#define MDNS_DEBUGMSGS 2 - -// Set MDNS_CHECK_PRINTF_STYLE_FUNCTIONS to 1 to enable extra GCC compiler warnings -// Note: You don't normally want to do this, because it generates a bunch of -// spurious warnings for the following custom extensions implemented by mDNS_vsnprintf: -// warning: `#' flag used with `%s' printf format (for %#s -- pascal string format) -// warning: repeated `#' flag in format (for %##s -- DNS name string format) -// warning: double format, pointer arg (arg 2) (for %.4a, %.16a, %#a -- IP address formats) -#define MDNS_CHECK_PRINTF_STYLE_FUNCTIONS 0 - -typedef enum - { - MDNS_LOG_MSG, - MDNS_LOG_OPERATION, - MDNS_LOG_SPS, - MDNS_LOG_INFO, - MDNS_LOG_DEBUG, - } mDNSLogLevel_t; - -// Set this symbol to 1 to answer remote queries for our Address, reverse mapping PTR, and HINFO records -#define ANSWER_REMOTE_HOSTNAME_QUERIES 0 - -// Set this symbol to 1 to do extra debug checks on malloc() and free() -// Set this symbol to 2 to write a log message for every malloc() and free() -//#define MACOSX_MDNS_MALLOC_DEBUGGING 1 - -//#define ForceAlerts 1 -//#define LogTimeStamps 1 - -// Developer-settings section ends here - -#if MDNS_CHECK_PRINTF_STYLE_FUNCTIONS -#define IS_A_PRINTF_STYLE_FUNCTION(F,A) __attribute__ ((format(printf,F,A))) -#else -#define IS_A_PRINTF_STYLE_FUNCTION(F,A) -#endif - -#ifdef __cplusplus - extern "C" { -#endif - -// Variable argument macro support. Use ANSI C99 __VA_ARGS__ where possible. Otherwise, use the next best thing. - -#if (defined(__GNUC__)) - #if ((__GNUC__ > 3) || ((__GNUC__ == 3) && (__GNUC_MINOR__ >= 2))) - #define MDNS_C99_VA_ARGS 1 - #define MDNS_GNU_VA_ARGS 0 - #else - #define MDNS_C99_VA_ARGS 0 - #define MDNS_GNU_VA_ARGS 1 - #endif - #define MDNS_HAS_VA_ARG_MACROS 1 -#elif (_MSC_VER >= 1400) // Visual Studio 2005 and later - #define MDNS_C99_VA_ARGS 1 - #define MDNS_GNU_VA_ARGS 0 - #define MDNS_HAS_VA_ARG_MACROS 1 -#elif (defined(__MWERKS__)) - #define MDNS_C99_VA_ARGS 1 - #define MDNS_GNU_VA_ARGS 0 - #define MDNS_HAS_VA_ARG_MACROS 1 -#else - #define MDNS_C99_VA_ARGS 0 - #define MDNS_GNU_VA_ARGS 0 - #define MDNS_HAS_VA_ARG_MACROS 0 -#endif - -#if (MDNS_HAS_VA_ARG_MACROS) - #if (MDNS_C99_VA_ARGS) - #define debug_noop( ... ) ((void)0) - #define LogMsg( ... ) LogMsgWithLevel(MDNS_LOG_MSG, __VA_ARGS__) - #define LogOperation( ... ) do { if (mDNS_LoggingEnabled) LogMsgWithLevel(MDNS_LOG_OPERATION, __VA_ARGS__); } while (0) - #define LogSPS( ... ) do { if (mDNS_LoggingEnabled) LogMsgWithLevel(MDNS_LOG_SPS, __VA_ARGS__); } while (0) - #define LogInfo( ... ) do { if (mDNS_LoggingEnabled) LogMsgWithLevel(MDNS_LOG_INFO, __VA_ARGS__); } while (0) - #elif (MDNS_GNU_VA_ARGS) - #define debug_noop( ARGS... ) ((void)0) - #define LogMsg( ARGS... ) LogMsgWithLevel(MDNS_LOG_MSG, ARGS) - #define LogOperation( ARGS... ) do { if (mDNS_LoggingEnabled) LogMsgWithLevel(MDNS_LOG_OPERATION, ARGS); } while (0) - #define LogSPS( ARGS... ) do { if (mDNS_LoggingEnabled) LogMsgWithLevel(MDNS_LOG_SPS, ARGS); } while (0) - #define LogInfo( ARGS... ) do { if (mDNS_LoggingEnabled) LogMsgWithLevel(MDNS_LOG_INFO, ARGS); } while (0) - #else - #error Unknown variadic macros - #endif -#else - // If your platform does not support variadic macros, you need to define the following variadic functions. - // See mDNSShared/mDNSDebug.c for sample implementation - #define debug_noop 1 ? (void)0 : (void) - #define LogMsg LogMsg_ - #define LogOperation (mDNS_LoggingEnabled == 0) ? ((void)0) : LogOperation_ - #define LogSPS (mDNS_LoggingEnabled == 0) ? ((void)0) : LogSPS_ - #define LogInfo (mDNS_LoggingEnabled == 0) ? ((void)0) : LogInfo_ - extern void LogMsg_(const char *format, ...) IS_A_PRINTF_STYLE_FUNCTION(1,2); - extern void LogOperation_(const char *format, ...) IS_A_PRINTF_STYLE_FUNCTION(1,2); - extern void LogSPS_(const char *format, ...) IS_A_PRINTF_STYLE_FUNCTION(1,2); - extern void LogInfo_(const char *format, ...) IS_A_PRINTF_STYLE_FUNCTION(1,2); -#endif - -#if MDNS_DEBUGMSGS -#define debugf debugf_ -extern void debugf_(const char *format, ...) IS_A_PRINTF_STYLE_FUNCTION(1,2); -#else -#define debugf debug_noop -#endif - -#if MDNS_DEBUGMSGS > 1 -#define verbosedebugf verbosedebugf_ -extern void verbosedebugf_(const char *format, ...) IS_A_PRINTF_STYLE_FUNCTION(1,2); -#else -#define verbosedebugf debug_noop -#endif - -extern int mDNS_LoggingEnabled; -extern int mDNS_PacketLoggingEnabled; -extern int mDNS_DebugMode; // If non-zero, LogMsg() writes to stderr instead of syslog -extern const char ProgramName[]; - -extern void LogMsgWithLevel(mDNSLogLevel_t logLevel, const char *format, ...) IS_A_PRINTF_STYLE_FUNCTION(2,3); -// LogMsgNoIdent needs to be fixed so that it logs without the ident prefix like it used to -// (or completely overhauled to use the new "log to a separate file" facility) -#define LogMsgNoIdent LogMsg - -#if APPLE_OSX_mDNSResponder && MACOSX_MDNS_MALLOC_DEBUGGING >= 1 -extern void *mallocL(char *msg, unsigned int size); -extern void freeL(char *msg, void *x); -extern void LogMemCorruption(const char *format, ...); -extern void uds_validatelists(void); -extern void udns_validatelists(void *const v); -#else -#define mallocL(X,Y) malloc(Y) -#define freeL(X,Y) free(Y) -#endif - -#ifdef __cplusplus - } -#endif - -#endif diff -Nru qtcreator-2.5.0/src/tools/mdnssd/mDNSEmbeddedAPI.h qtcreator-2.5.2/src/tools/mdnssd/mDNSEmbeddedAPI.h --- qtcreator-2.5.0/src/tools/mdnssd/mDNSEmbeddedAPI.h 2012-05-09 12:13:19.000000000 +0000 +++ qtcreator-2.5.2/src/tools/mdnssd/mDNSEmbeddedAPI.h 1970-01-01 00:00:00.000000000 +0000 @@ -1,2964 +0,0 @@ -/* -*- Mode: C; tab-width: 4 -*- - * - * Copyright (c) 2002-2003 Apple Computer, Inc. All rights reserved. - * - * Licensed under the Apache License, Version 2.0 (the "License"); - * you may not use this file except in compliance with the License. - * You may obtain a copy of the License at - * - * http://www.apache.org/licenses/LICENSE-2.0 - * - * Unless required by applicable law or agreed to in writing, software - * distributed under the License is distributed on an "AS IS" BASIS, - * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. - * See the License for the specific language governing permissions and - * limitations under the License. - - NOTE: - If you're building an application that uses DNS Service Discovery - this is probably NOT the header file you're looking for. - In most cases you will want to use /usr/include/dns_sd.h instead. - - This header file defines the lowest level raw interface to mDNSCore, - which is appropriate *only* on tiny embedded systems where everything - runs in a single address space and memory is extremely constrained. - All the APIs here are malloc-free, which means that the caller is - responsible for passing in a pointer to the relevant storage that - will be used in the execution of that call, and (when called with - correct parameters) all the calls are guaranteed to succeed. There - is never a case where a call can suffer intermittent failures because - the implementation calls malloc() and sometimes malloc() returns NULL - because memory is so limited that no more is available. - This is primarily for devices that need to have precisely known fixed - memory requirements, with absolutely no uncertainty or run-time variation, - but that certainty comes at a cost of more difficult programming. - - For applications running on general-purpose desktop operating systems - (Mac OS, Linux, Solaris, Windows, etc.) the API you should use is - /usr/include/dns_sd.h, which defines the API by which multiple - independent client processes communicate their DNS Service Discovery - requests to a single "mdnsd" daemon running in the background. - - Even on platforms that don't run multiple independent processes in - multiple independent address spaces, you can still use the preferred - dns_sd.h APIs by linking in "dnssd_clientshim.c", which implements - the standard "dns_sd.h" API calls, allocates any required storage - using malloc(), and then calls through to the low-level malloc-free - mDNSCore routines defined here. This has the benefit that even though - you're running on a small embedded system with a single address space, - you can still use the exact same client C code as you'd use on a - general-purpose desktop system. - - */ - -#ifndef __mDNSClientAPI_h -#define __mDNSClientAPI_h - -#if defined(EFI32) || defined(EFI64) || defined(EFIX64) -// EFI doesn't have stdarg.h unless it's building with GCC. -#include "Tiano.h" -#if !defined(__GNUC__) -#define va_list VA_LIST -#define va_start(a, b) VA_START(a, b) -#define va_end(a) VA_END(a) -#define va_arg(a, b) VA_ARG(a, b) -#endif -#else -#include // stdarg.h is required for for va_list support for the mDNS_vsnprintf declaration -#endif - -#include "mDNSDebug.h" -#if APPLE_OSX_mDNSResponder -#include -#endif - -#ifdef __cplusplus - extern "C" { -#endif - -// *************************************************************************** -// Function scope indicators - -// If you see "mDNSlocal" before a function name in a C file, it means the function is not callable outside this file -#ifndef mDNSlocal -#define mDNSlocal static -#endif -// If you see "mDNSexport" before a symbol in a C file, it means the symbol is exported for use by clients -// For every "mDNSexport" in a C file, there needs to be a corresponding "extern" declaration in some header file -// (When a C file #includes a header file, the "extern" declarations tell the compiler: -// "This symbol exists -- but not necessarily in this C file.") -#ifndef mDNSexport -#define mDNSexport -#endif - -// Explanation: These local/export markers are a little habit of mine for signaling the programmers' intentions. -// When "mDNSlocal" is just a synonym for "static", and "mDNSexport" is a complete no-op, you could be -// forgiven for asking what purpose they serve. The idea is that if you see "mDNSexport" in front of a -// function definition it means the programmer intended it to be exported and callable from other files -// in the project. If you see "mDNSlocal" in front of a function definition it means the programmer -// intended it to be private to that file. If you see neither in front of a function definition it -// means the programmer forgot (so you should work out which it is supposed to be, and fix it). -// Using "mDNSlocal" instead of "static" makes it easier to do a textual searches for one or the other. -// For example you can do a search for "static" to find if any functions declare any local variables as "static" -// (generally a bad idea unless it's also "const", because static storage usually risks being non-thread-safe) -// without the results being cluttered with hundreds of matches for functions declared static. -// - Stuart Cheshire - -// *************************************************************************** -// Structure packing macro - -// If we're not using GNUC, it's not fatal. -// Most compilers naturally pack the on-the-wire structures correctly anyway, so a plain "struct" is usually fine. -// In the event that structures are not packed correctly, mDNS_Init() will detect this and report an error, so the -// developer will know what's wrong, and can investigate what needs to be done on that compiler to provide proper packing. -#ifndef packedstruct - #if ((__GNUC__ > 2) || ((__GNUC__ == 2) && (__GNUC_MINOR__ >= 9))) - #define packedstruct struct __attribute__((__packed__)) - #define packedunion union __attribute__((__packed__)) - #else - #define packedstruct struct - #define packedunion union - #endif -#endif - -// *************************************************************************** -#if 0 -#pragma mark - DNS Resource Record class and type constants -#endif - -typedef enum // From RFC 1035 - { - kDNSClass_IN = 1, // Internet - kDNSClass_CS = 2, // CSNET - kDNSClass_CH = 3, // CHAOS - kDNSClass_HS = 4, // Hesiod - kDNSClass_NONE = 254, // Used in DNS UPDATE [RFC 2136] - - kDNSClass_Mask = 0x7FFF,// Multicast DNS uses the bottom 15 bits to identify the record class... - kDNSClass_UniqueRRSet = 0x8000,// ... and the top bit indicates that all other cached records are now invalid - - kDNSQClass_ANY = 255, // Not a DNS class, but a DNS query class, meaning "all classes" - kDNSQClass_UnicastResponse = 0x8000 // Top bit set in a question means "unicast response acceptable" - } DNS_ClassValues; - -typedef enum // From RFC 1035 - { - kDNSType_A = 1, // 1 Address - kDNSType_NS, // 2 Name Server - kDNSType_MD, // 3 Mail Destination - kDNSType_MF, // 4 Mail Forwarder - kDNSType_CNAME, // 5 Canonical Name - kDNSType_SOA, // 6 Start of Authority - kDNSType_MB, // 7 Mailbox - kDNSType_MG, // 8 Mail Group - kDNSType_MR, // 9 Mail Rename - kDNSType_NULL, // 10 NULL RR - kDNSType_WKS, // 11 Well-known-service - kDNSType_PTR, // 12 Domain name pointer - kDNSType_HINFO, // 13 Host information - kDNSType_MINFO, // 14 Mailbox information - kDNSType_MX, // 15 Mail Exchanger - kDNSType_TXT, // 16 Arbitrary text string - kDNSType_RP, // 17 Responsible person - kDNSType_AFSDB, // 18 AFS cell database - kDNSType_X25, // 19 X_25 calling address - kDNSType_ISDN, // 20 ISDN calling address - kDNSType_RT, // 21 Router - kDNSType_NSAP, // 22 NSAP address - kDNSType_NSAP_PTR, // 23 Reverse NSAP lookup (deprecated) - kDNSType_SIG, // 24 Security signature - kDNSType_KEY, // 25 Security key - kDNSType_PX, // 26 X.400 mail mapping - kDNSType_GPOS, // 27 Geographical position (withdrawn) - kDNSType_AAAA, // 28 IPv6 Address - kDNSType_LOC, // 29 Location Information - kDNSType_NXT, // 30 Next domain (security) - kDNSType_EID, // 31 Endpoint identifier - kDNSType_NIMLOC, // 32 Nimrod Locator - kDNSType_SRV, // 33 Service record - kDNSType_ATMA, // 34 ATM Address - kDNSType_NAPTR, // 35 Naming Authority PoinTeR - kDNSType_KX, // 36 Key Exchange - kDNSType_CERT, // 37 Certification record - kDNSType_A6, // 38 IPv6 Address (deprecated) - kDNSType_DNAME, // 39 Non-terminal DNAME (for IPv6) - kDNSType_SINK, // 40 Kitchen sink (experimental) - kDNSType_OPT, // 41 EDNS0 option (meta-RR) - kDNSType_APL, // 42 Address Prefix List - kDNSType_DS, // 43 Delegation Signer - kDNSType_SSHFP, // 44 SSH Key Fingerprint - kDNSType_IPSECKEY, // 45 IPSECKEY - kDNSType_RRSIG, // 46 RRSIG - kDNSType_NSEC, // 47 Denial of Existence - kDNSType_DNSKEY, // 48 DNSKEY - kDNSType_DHCID, // 49 DHCP Client Identifier - kDNSType_NSEC3, // 50 Hashed Authenticated Denial of Existence - kDNSType_NSEC3PARAM, // 51 Hashed Authenticated Denial of Existence - - kDNSType_HIP = 55, // 55 Host Identity Protocol - - kDNSType_SPF = 99, // 99 Sender Policy Framework for E-Mail - kDNSType_UINFO, // 100 IANA-Reserved - kDNSType_UID, // 101 IANA-Reserved - kDNSType_GID, // 102 IANA-Reserved - kDNSType_UNSPEC, // 103 IANA-Reserved - - kDNSType_TKEY = 249, // 249 Transaction key - kDNSType_TSIG, // 250 Transaction signature - kDNSType_IXFR, // 251 Incremental zone transfer - kDNSType_AXFR, // 252 Transfer zone of authority - kDNSType_MAILB, // 253 Transfer mailbox records - kDNSType_MAILA, // 254 Transfer mail agent records - kDNSQType_ANY // Not a DNS type, but a DNS query type, meaning "all types" - } DNS_TypeValues; - -// *************************************************************************** -#if 0 -#pragma mark - -#pragma mark - Simple types -#endif - -// mDNS defines its own names for these common types to simplify portability across -// multiple platforms that may each have their own (different) names for these types. -typedef int mDNSBool; -typedef signed char mDNSs8; -typedef unsigned char mDNSu8; -typedef signed short mDNSs16; -typedef unsigned short mDNSu16; - -// says -// __LP64__ _LP64 -// These macros are defined, with value 1, if (and only if) the compilation is -// for a target where long int and pointer both use 64-bits and int uses 32-bit. -// says -// Macro Name __LP64__ Value 1 -// A quick Google search for "defined(__LP64__)" OR "#ifdef __LP64__" gives 2590 hits and -// a search for "#if __LP64__" gives only 12, so I think we'll go with the majority and use defined() -#if defined(_ILP64) || defined(__ILP64__) -typedef signed int32 mDNSs32; -typedef unsigned int32 mDNSu32; -#elif defined(_LP64) || defined(__LP64__) -typedef signed int mDNSs32; -typedef unsigned int mDNSu32; -#else -typedef signed long mDNSs32; -typedef unsigned long mDNSu32; -//typedef signed int mDNSs32; -//typedef unsigned int mDNSu32; -#endif - -// To enforce useful type checking, we make mDNSInterfaceID be a pointer to a dummy struct -// This way, mDNSInterfaceIDs can be assigned, and compared with each other, but not with other types -// Declaring the type to be the typical generic "void *" would lack this type checking -typedef struct mDNSInterfaceID_dummystruct { void *dummy; } *mDNSInterfaceID; - -// These types are for opaque two- and four-byte identifiers. -// The "NotAnInteger" fields of the unions allow the value to be conveniently passed around in a -// register for the sake of efficiency, and compared for equality or inequality, but don't forget -- -// just because it is in a register doesn't mean it is an integer. Operations like greater than, -// less than, add, multiply, increment, decrement, etc., are undefined for opaque identifiers, -// and if you make the mistake of trying to do those using the NotAnInteger field, then you'll -// find you get code that doesn't work consistently on big-endian and little-endian machines. -#if defined(_WIN32) - #pragma pack(push,2) -#endif -typedef union { mDNSu8 b[ 2]; mDNSu16 NotAnInteger; } mDNSOpaque16; -typedef union { mDNSu8 b[ 4]; mDNSu32 NotAnInteger; } mDNSOpaque32; -typedef packedunion { mDNSu8 b[ 6]; mDNSu16 w[3]; mDNSu32 l[1]; } mDNSOpaque48; -typedef union { mDNSu8 b[ 8]; mDNSu16 w[4]; mDNSu32 l[2]; } mDNSOpaque64; -typedef union { mDNSu8 b[16]; mDNSu16 w[8]; mDNSu32 l[4]; } mDNSOpaque128; -#if defined(_WIN32) - #pragma pack(pop) -#endif - -typedef mDNSOpaque16 mDNSIPPort; // An IP port is a two-byte opaque identifier (not an integer) -typedef mDNSOpaque32 mDNSv4Addr; // An IP address is a four-byte opaque identifier (not an integer) -typedef mDNSOpaque128 mDNSv6Addr; // An IPv6 address is a 16-byte opaque identifier (not an integer) -typedef mDNSOpaque48 mDNSEthAddr; // An Ethernet address is a six-byte opaque identifier (not an integer) - -// Bit operations for opaque 64 bit quantity. Uses the 32 bit quantity(l[2]) to set and clear bits -#define mDNSNBBY 8 -#define bit_set_opaque64(op64, index) (op64.l[((index))/(sizeof(mDNSu32) * mDNSNBBY)] |= (1 << ((index) % (sizeof(mDNSu32) * mDNSNBBY)))) -#define bit_clr_opaque64(op64, index) (op64.l[((index))/(sizeof(mDNSu32) * mDNSNBBY)] &= ~(1 << ((index) % (sizeof(mDNSu32) * mDNSNBBY)))) -#define bit_get_opaque64(op64, index) (op64.l[((index))/(sizeof(mDNSu32) * mDNSNBBY)] & (1 << ((index) % (sizeof(mDNSu32) * mDNSNBBY)))) - -enum - { - mDNSAddrType_None = 0, - mDNSAddrType_IPv4 = 4, - mDNSAddrType_IPv6 = 6, - mDNSAddrType_Unknown = ~0 // Special marker value used in known answer list recording - }; - -enum - { - mDNSTransport_None = 0, - mDNSTransport_UDP = 1, - mDNSTransport_TCP = 2 - }; - -typedef struct - { - mDNSs32 type; - union { mDNSv6Addr v6; mDNSv4Addr v4; } ip; - } mDNSAddr; - -enum { mDNSfalse = 0, mDNStrue = 1 }; - -#define mDNSNULL 0L - -enum - { - mStatus_Waiting = 1, - mStatus_NoError = 0, - - // mDNS return values are in the range FFFE FF00 (-65792) to FFFE FFFF (-65537) - // The top end of the range (FFFE FFFF) is used for error codes; - // the bottom end of the range (FFFE FF00) is used for non-error values; - - // Error codes: - mStatus_UnknownErr = -65537, // First value: 0xFFFE FFFF - mStatus_NoSuchNameErr = -65538, - mStatus_NoMemoryErr = -65539, - mStatus_BadParamErr = -65540, - mStatus_BadReferenceErr = -65541, - mStatus_BadStateErr = -65542, - mStatus_BadFlagsErr = -65543, - mStatus_UnsupportedErr = -65544, - mStatus_NotInitializedErr = -65545, - mStatus_NoCache = -65546, - mStatus_AlreadyRegistered = -65547, - mStatus_NameConflict = -65548, - mStatus_Invalid = -65549, - mStatus_Firewall = -65550, - mStatus_Incompatible = -65551, - mStatus_BadInterfaceErr = -65552, - mStatus_Refused = -65553, - mStatus_NoSuchRecord = -65554, - mStatus_NoAuth = -65555, - mStatus_NoSuchKey = -65556, - mStatus_NATTraversal = -65557, - mStatus_DoubleNAT = -65558, - mStatus_BadTime = -65559, - mStatus_BadSig = -65560, // while we define this per RFC 2845, BIND 9 returns Refused for bad/missing signatures - mStatus_BadKey = -65561, - mStatus_TransientErr = -65562, // transient failures, e.g. sending packets shortly after a network transition or wake from sleep - mStatus_ServiceNotRunning = -65563, // Background daemon not running - mStatus_NATPortMappingUnsupported = -65564, // NAT doesn't support NAT-PMP or UPnP - mStatus_NATPortMappingDisabled = -65565, // NAT supports NAT-PMP or UPnP but it's disabled by the administrator - mStatus_NoRouter = -65566, - mStatus_PollingMode = -65567, - mStatus_Timeout = -65568, - // -65568 to -65786 currently unused; available for allocation - - // tcp connection status - mStatus_ConnPending = -65787, - mStatus_ConnFailed = -65788, - mStatus_ConnEstablished = -65789, - - // Non-error values: - mStatus_GrowCache = -65790, - mStatus_ConfigChanged = -65791, - mStatus_MemFree = -65792 // Last value: 0xFFFE FF00 - // mStatus_MemFree is the last legal mDNS error code, at the end of the range allocated for mDNS - }; - -typedef mDNSs32 mStatus; - -// RFC 1034/1035 specify that a domain label consists of a length byte plus up to 63 characters -#define MAX_DOMAIN_LABEL 63 -typedef struct { mDNSu8 c[ 64]; } domainlabel; // One label: length byte and up to 63 characters - -// RFC 1034/1035/2181 specify that a domain name (length bytes and data bytes) may be up to 255 bytes long, -// plus the terminating zero at the end makes 256 bytes total in the on-the-wire format. -#define MAX_DOMAIN_NAME 256 -typedef struct { mDNSu8 c[256]; } domainname; // Up to 256 bytes of length-prefixed domainlabels - -typedef struct { mDNSu8 c[256]; } UTF8str255; // Null-terminated C string - -// The longest legal textual form of a DNS name is 1009 bytes, including the C-string terminating NULL at the end. -// Explanation: -// When a native domainname object is converted to printable textual form using ConvertDomainNameToCString(), -// non-printing characters are represented in the conventional DNS way, as '\ddd', where ddd is a three-digit decimal number. -// The longest legal domain name is 256 bytes, in the form of four labels as shown below: -// Length byte, 63 data bytes, length byte, 63 data bytes, length byte, 63 data bytes, length byte, 62 data bytes, zero byte. -// Each label is encoded textually as characters followed by a trailing dot. -// If every character has to be represented as a four-byte escape sequence, then this makes the maximum textual form four labels -// plus the C-string terminating NULL as shown below: -// 63*4+1 + 63*4+1 + 63*4+1 + 62*4+1 + 1 = 1009. -// Note that MAX_ESCAPED_DOMAIN_LABEL is not normally used: If you're only decoding a single label, escaping is usually not required. -// It is for domain names, where dots are used as label separators, that proper escaping is vital. -#define MAX_ESCAPED_DOMAIN_LABEL 254 -#define MAX_ESCAPED_DOMAIN_NAME 1009 - -// MAX_REVERSE_MAPPING_NAME -// For IPv4: "123.123.123.123.in-addr.arpa." 30 bytes including terminating NUL -// For IPv6: "x.x.x.x.x.x.x.x.x.x.x.x.x.x.x.x.x.x.x.x.x.x.x.x.x.x.x.x.x.x.x.x.ip6.arpa." 74 bytes including terminating NUL - -#define MAX_REVERSE_MAPPING_NAME_V4 30 -#define MAX_REVERSE_MAPPING_NAME_V6 74 -#define MAX_REVERSE_MAPPING_NAME 74 - -// Most records have a TTL of 75 minutes, so that their 80% cache-renewal query occurs once per hour. -// For records containing a hostname (in the name on the left, or in the rdata on the right), -// like A, AAAA, reverse-mapping PTR, and SRV, we use a two-minute TTL by default, because we don't want -// them to hang around for too long in the cache if the host in question crashes or otherwise goes away. - -#define kStandardTTL (3600UL * 100 / 80) -#define kHostNameTTL 120UL - -// Some applications want to register their SRV records with a lower ttl so that in case the server -// using a dynamic port number restarts, the clients will not have stale information for more than -// 10 seconds - -#define kHostNameSmallTTL 10UL - - -// Multicast DNS uses announcements (gratuitous responses) to update peer caches. -// This means it is feasible to use relatively larger TTL values than we might otherwise -// use, because we have a cache coherency protocol to keep the peer caches up to date. -// With Unicast DNS, once an authoritative server gives a record with a certain TTL value to a client -// or caching server, that client or caching server is entitled to hold onto the record until its TTL -// expires, and has no obligation to contact the authoritative server again until that time arrives. -// This means that whereas Multicast DNS can use announcements to pre-emptively update stale data -// before it would otherwise have expired, standard Unicast DNS (not using LLQs) has no equivalent -// mechanism, and TTL expiry is the *only* mechanism by which stale data gets deleted. Because of this, -// we currently limit the TTL to ten seconds in such cases where no dynamic cache updating is possible. -#define kStaticCacheTTL 10 - -#define DefaultTTLforRRType(X) (((X) == kDNSType_A || (X) == kDNSType_AAAA || (X) == kDNSType_SRV) ? kHostNameTTL : kStandardTTL) - -typedef struct AuthRecord_struct AuthRecord; -typedef struct ServiceRecordSet_struct ServiceRecordSet; -typedef struct CacheRecord_struct CacheRecord; -typedef struct CacheGroup_struct CacheGroup; -typedef struct AuthGroup_struct AuthGroup; -typedef struct DNSQuestion_struct DNSQuestion; -typedef struct ZoneData_struct ZoneData; -typedef struct mDNS_struct mDNS; -typedef struct mDNS_PlatformSupport_struct mDNS_PlatformSupport; -typedef struct NATTraversalInfo_struct NATTraversalInfo; - -// Structure to abstract away the differences between TCP/SSL sockets, and one for UDP sockets -// The actual definition of these structures appear in the appropriate platform support code -typedef struct TCPSocket_struct TCPSocket; -typedef struct UDPSocket_struct UDPSocket; - -// *************************************************************************** -#if 0 -#pragma mark - -#pragma mark - DNS Message structures -#endif - -#define mDNS_numZones numQuestions -#define mDNS_numPrereqs numAnswers -#define mDNS_numUpdates numAuthorities - -typedef packedstruct - { - mDNSOpaque16 id; - mDNSOpaque16 flags; - mDNSu16 numQuestions; - mDNSu16 numAnswers; - mDNSu16 numAuthorities; - mDNSu16 numAdditionals; - } DNSMessageHeader; - -// We can send and receive packets up to 9000 bytes (Ethernet Jumbo Frame size, if that ever becomes widely used) -// However, in the normal case we try to limit packets to 1500 bytes so that we don't get IP fragmentation on standard Ethernet -// 40 (IPv6 header) + 8 (UDP header) + 12 (DNS message header) + 1440 (DNS message body) = 1500 total -#define AbsoluteMaxDNSMessageData 8940 -#define NormalMaxDNSMessageData 1440 -typedef packedstruct - { - DNSMessageHeader h; // Note: Size 12 bytes - mDNSu8 data[AbsoluteMaxDNSMessageData]; // 40 (IPv6) + 8 (UDP) + 12 (DNS header) + 8940 (data) = 9000 - } DNSMessage; - -typedef struct tcpInfo_t - { - mDNS *m; - TCPSocket *sock; - DNSMessage request; - int requestLen; - DNSQuestion *question; // For queries - AuthRecord *rr; // For record updates - mDNSAddr Addr; - mDNSIPPort Port; - mDNSIPPort SrcPort; - DNSMessage *reply; - mDNSu16 replylen; - unsigned long nread; - int numReplies; - } tcpInfo_t; - -// *************************************************************************** -#if 0 -#pragma mark - -#pragma mark - Other Packet Format Structures -#endif - -typedef packedstruct - { - mDNSEthAddr dst; - mDNSEthAddr src; - mDNSOpaque16 ethertype; - } EthernetHeader; // 14 bytes - -typedef packedstruct - { - mDNSOpaque16 hrd; - mDNSOpaque16 pro; - mDNSu8 hln; - mDNSu8 pln; - mDNSOpaque16 op; - mDNSEthAddr sha; - mDNSv4Addr spa; - mDNSEthAddr tha; - mDNSv4Addr tpa; - } ARP_EthIP; // 28 bytes - -typedef packedstruct - { - mDNSu8 vlen; - mDNSu8 tos; - mDNSu16 totlen; - mDNSOpaque16 id; - mDNSOpaque16 flagsfrags; - mDNSu8 ttl; - mDNSu8 protocol; // Payload type: 0x06 = TCP, 0x11 = UDP - mDNSu16 checksum; - mDNSv4Addr src; - mDNSv4Addr dst; - } IPv4Header; // 20 bytes - -typedef packedstruct - { - mDNSu32 vcf; // Version, Traffic Class, Flow Label - mDNSu16 len; // Payload Length - mDNSu8 pro; // Type of next header: 0x06 = TCP, 0x11 = UDP, 0x3A = ICMPv6 - mDNSu8 ttl; // Hop Limit - mDNSv6Addr src; - mDNSv6Addr dst; - } IPv6Header; // 40 bytes - -typedef packedstruct - { - mDNSv6Addr src; - mDNSv6Addr dst; - mDNSOpaque32 len; - mDNSOpaque32 pro; - } IPv6PseudoHeader; // 40 bytes - -typedef union - { - mDNSu8 bytes[20]; - ARP_EthIP arp; - IPv4Header v4; - IPv6Header v6; - } NetworkLayerPacket; - -typedef packedstruct - { - mDNSIPPort src; - mDNSIPPort dst; - mDNSu32 seq; - mDNSu32 ack; - mDNSu8 offset; - mDNSu8 flags; - mDNSu16 window; - mDNSu16 checksum; - mDNSu16 urgent; - } TCPHeader; // 20 bytes; IP protocol type 0x06 - -typedef packedstruct - { - mDNSIPPort src; - mDNSIPPort dst; - mDNSu16 len; // Length including UDP header (i.e. minimum value is 8 bytes) - mDNSu16 checksum; - } UDPHeader; // 8 bytes; IP protocol type 0x11 - -typedef packedstruct - { - mDNSu8 type; // 0x87 == Neighbor Solicitation, 0x88 == Neighbor Advertisement - mDNSu8 code; - mDNSu16 checksum; - mDNSu32 flags_res; // R/S/O flags and reserved bits - mDNSv6Addr target; - // Typically 8 bytes of options are also present - } IPv6NDP; // 24 bytes or more; IP protocol type 0x3A - -#define NDP_Sol 0x87 -#define NDP_Adv 0x88 - -#define NDP_Router 0x80 -#define NDP_Solicited 0x40 -#define NDP_Override 0x20 - -#define NDP_SrcLL 1 -#define NDP_TgtLL 2 - -typedef union - { - mDNSu8 bytes[20]; - TCPHeader tcp; - UDPHeader udp; - IPv6NDP ndp; - } TransportLayerPacket; - -typedef packedstruct - { - mDNSOpaque64 InitiatorCookie; - mDNSOpaque64 ResponderCookie; - mDNSu8 NextPayload; - mDNSu8 Version; - mDNSu8 ExchangeType; - mDNSu8 Flags; - mDNSOpaque32 MessageID; - mDNSu32 Length; - } IKEHeader; // 28 bytes - -// *************************************************************************** -#if 0 -#pragma mark - -#pragma mark - Resource Record structures -#endif - -// Authoritative Resource Records: -// There are four basic types: Shared, Advisory, Unique, Known Unique - -// * Shared Resource Records do not have to be unique -// -- Shared Resource Records are used for DNS-SD service PTRs -// -- It is okay for several hosts to have RRs with the same name but different RDATA -// -- We use a random delay on responses to reduce collisions when all the hosts respond to the same query -// -- These RRs typically have moderately high TTLs (e.g. one hour) -// -- These records are announced on startup and topology changes for the benefit of passive listeners -// -- These records send a goodbye packet when deregistering -// -// * Advisory Resource Records are like Shared Resource Records, except they don't send a goodbye packet -// -// * Unique Resource Records should be unique among hosts within any given mDNS scope -// -- The majority of Resource Records are of this type -// -- If two entities on the network have RRs with the same name but different RDATA, this is a conflict -// -- Responses may be sent immediately, because only one host should be responding to any particular query -// -- These RRs typically have low TTLs (e.g. a few minutes) -// -- On startup and after topology changes, a host issues queries to verify uniqueness - -// * Known Unique Resource Records are treated like Unique Resource Records, except that mDNS does -// not have to verify their uniqueness because this is already known by other means (e.g. the RR name -// is derived from the host's IP or Ethernet address, which is already known to be a unique identifier). - -// Summary of properties of different record types: -// Probe? Does this record type send probes before announcing? -// Conflict? Does this record type react if we observe an apparent conflict? -// Goodbye? Does this record type send a goodbye packet on departure? -// -// Probe? Conflict? Goodbye? Notes -// Unregistered Should not appear in any list (sanity check value) -// Shared No No Yes e.g. Service PTR record -// Deregistering No No Yes Shared record about to announce its departure and leave the list -// Advisory No No No -// Unique Yes Yes No Record intended to be unique -- will probe to verify -// Verified Yes Yes No Record has completed probing, and is verified unique -// KnownUnique No Yes No Record is assumed by other means to be unique - -// Valid lifecycle of a record: -// Unregistered -> Shared -> Deregistering -(goodbye)-> Unregistered -// Unregistered -> Advisory -> Unregistered -// Unregistered -> Unique -(probe)-> Verified -> Unregistered -// Unregistered -> KnownUnique -> Unregistered - -// Each Authoritative kDNSRecordType has only one bit set. This makes it easy to quickly see if a record -// is one of a particular set of types simply by performing the appropriate bitwise masking operation. - -// Cache Resource Records (received from the network): -// There are four basic types: Answer, Unique Answer, Additional, Unique Additional -// Bit 7 (the top bit) of kDNSRecordType is always set for Cache Resource Records; always clear for Authoritative Resource Records -// Bit 6 (value 0x40) is set for answer records; clear for authority/additional records -// Bit 5 (value 0x20) is set for records received with the kDNSClass_UniqueRRSet - -enum - { - kDNSRecordTypeUnregistered = 0x00, // Not currently in any list - kDNSRecordTypeDeregistering = 0x01, // Shared record about to announce its departure and leave the list - - kDNSRecordTypeUnique = 0x02, // Will become a kDNSRecordTypeVerified when probing is complete - - kDNSRecordTypeAdvisory = 0x04, // Like Shared, but no goodbye packet - kDNSRecordTypeShared = 0x08, // Shared means record name does not have to be unique -- use random delay on responses - - kDNSRecordTypeVerified = 0x10, // Unique means mDNS should check that name is unique (and then send immediate responses) - kDNSRecordTypeKnownUnique = 0x20, // Known Unique means mDNS can assume name is unique without checking - // For Dynamic Update records, Known Unique means the record must already exist on the server. - kDNSRecordTypeUniqueMask = (kDNSRecordTypeUnique | kDNSRecordTypeVerified | kDNSRecordTypeKnownUnique), - kDNSRecordTypeActiveSharedMask = (kDNSRecordTypeAdvisory | kDNSRecordTypeShared), - kDNSRecordTypeActiveUniqueMask = (kDNSRecordTypeVerified | kDNSRecordTypeKnownUnique), - kDNSRecordTypeActiveMask = (kDNSRecordTypeActiveSharedMask | kDNSRecordTypeActiveUniqueMask), - - kDNSRecordTypePacketAdd = 0x80, // Received in the Additional Section of a DNS Response - kDNSRecordTypePacketAddUnique = 0x90, // Received in the Additional Section of a DNS Response with kDNSClass_UniqueRRSet set - kDNSRecordTypePacketAuth = 0xA0, // Received in the Authorities Section of a DNS Response - kDNSRecordTypePacketAuthUnique = 0xB0, // Received in the Authorities Section of a DNS Response with kDNSClass_UniqueRRSet set - kDNSRecordTypePacketAns = 0xC0, // Received in the Answer Section of a DNS Response - kDNSRecordTypePacketAnsUnique = 0xD0, // Received in the Answer Section of a DNS Response with kDNSClass_UniqueRRSet set - - kDNSRecordTypePacketNegative = 0xF0, // Pseudo-RR generated to cache non-existence results like NXDomain - - kDNSRecordTypePacketUniqueMask = 0x10 // True for PacketAddUnique, PacketAnsUnique, PacketAuthUnique, kDNSRecordTypePacketNegative - }; - -typedef packedstruct { mDNSu16 priority; mDNSu16 weight; mDNSIPPort port; domainname target; } rdataSRV; -typedef packedstruct { mDNSu16 preference; domainname exchange; } rdataMX; -typedef packedstruct { domainname mbox; domainname txt; } rdataRP; -typedef packedstruct { mDNSu16 preference; domainname map822; domainname mapx400; } rdataPX; - -typedef packedstruct - { - domainname mname; - domainname rname; - mDNSs32 serial; // Modular counter; increases when zone changes - mDNSu32 refresh; // Time in seconds that a slave waits after successful replication of the database before it attempts replication again - mDNSu32 retry; // Time in seconds that a slave waits after an unsuccessful replication attempt before it attempts replication again - mDNSu32 expire; // Time in seconds that a slave holds on to old data while replication attempts remain unsuccessful - mDNSu32 min; // Nominally the minimum record TTL for this zone, in seconds; also used for negative caching. - } rdataSOA; - -// EDNS Option Code registrations are recorded in the "DNS EDNS0 Options" section of -// - -#define kDNSOpt_LLQ 1 -#define kDNSOpt_Lease 2 -#define kDNSOpt_NSID 3 -#define kDNSOpt_Owner 4 - -typedef struct - { - mDNSu16 vers; - mDNSu16 llqOp; - mDNSu16 err; // Or UDP reply port, in setup request - // Note: In the in-memory form, there's typically a two-byte space here, so that the following 64-bit id is word-aligned - mDNSOpaque64 id; - mDNSu32 llqlease; - } LLQOptData; - -typedef struct - { - mDNSu8 vers; // Version number of this Owner OPT record - mDNSs8 seq; // Sleep/wake epoch - mDNSEthAddr HMAC; // Host's primary identifier (e.g. MAC of on-board Ethernet) - mDNSEthAddr IMAC; // Interface's MAC address (if different to primary MAC) - mDNSOpaque48 password; // Optional password - } OwnerOptData; - -// Note: rdataOPT format may be repeated an arbitrary number of times in a single resource record -typedef packedstruct - { - mDNSu16 opt; - mDNSu16 optlen; - union { LLQOptData llq; mDNSu32 updatelease; OwnerOptData owner; } u; - } rdataOPT; - -// Space needed to put OPT records into a packet: -// Header 11 bytes (name 1, type 2, class 2, TTL 4, length 2) -// LLQ rdata 18 bytes (opt 2, len 2, vers 2, op 2, err 2, id 8, lease 4) -// Lease rdata 8 bytes (opt 2, len 2, lease 4) -// Owner rdata 12-24 (opt 2, len 2, owner 8-20) - -#define DNSOpt_Header_Space 11 -#define DNSOpt_LLQData_Space (4 + 2 + 2 + 2 + 8 + 4) -#define DNSOpt_LeaseData_Space (4 + 4) -#define DNSOpt_OwnerData_ID_Space (4 + 2 + 6) -#define DNSOpt_OwnerData_ID_Wake_Space (4 + 2 + 6 + 6) -#define DNSOpt_OwnerData_ID_Wake_PW4_Space (4 + 2 + 6 + 6 + 4) -#define DNSOpt_OwnerData_ID_Wake_PW6_Space (4 + 2 + 6 + 6 + 6) - -#define ValidOwnerLength(X) ( (X) == DNSOpt_OwnerData_ID_Space - 4 || \ - (X) == DNSOpt_OwnerData_ID_Wake_Space - 4 || \ - (X) == DNSOpt_OwnerData_ID_Wake_PW4_Space - 4 || \ - (X) == DNSOpt_OwnerData_ID_Wake_PW6_Space - 4 ) - -#define DNSOpt_Owner_Space(A,B) (mDNSSameEthAddress((A),(B)) ? DNSOpt_OwnerData_ID_Space : DNSOpt_OwnerData_ID_Wake_Space) - -#define DNSOpt_Data_Space(O) ( \ - (O)->opt == kDNSOpt_LLQ ? DNSOpt_LLQData_Space : \ - (O)->opt == kDNSOpt_Lease ? DNSOpt_LeaseData_Space : \ - (O)->opt == kDNSOpt_Owner ? DNSOpt_Owner_Space(&(O)->u.owner.HMAC, &(O)->u.owner.IMAC) : 0x10000) - -// A maximal NSEC record is: -// 256 bytes domainname 'nextname' -// + 256 * 34 = 8704 bytes of bitmap data -// = 8960 bytes total -// For now we only support NSEC records encoding DNS types 0-255 and ignore the nextname (we always set it to be the same as the rrname), -// which gives us a fixed in-memory size of 32 bytes (256 bits) -typedef struct - { - mDNSu8 bitmap[32]; - } rdataNSEC; - -// StandardAuthRDSize is 264 (256+8), which is large enough to hold a maximum-sized SRV record (6 + 256 bytes) -// MaximumRDSize is 8K the absolute maximum we support (at least for now) -#define StandardAuthRDSize 264 -#define MaximumRDSize 8192 - -// InlineCacheRDSize is 68 -// Records received from the network with rdata this size or less have their rdata stored right in the CacheRecord object -// Records received from the network with rdata larger than this have additional storage allocated for the rdata -// A quick unscientific sample from a busy network at Apple with lots of machines revealed this: -// 1461 records in cache -// 292 were one-byte TXT records -// 136 were four-byte A records -// 184 were sixteen-byte AAAA records -// 780 were various PTR, TXT and SRV records from 12-64 bytes -// Only 69 records had rdata bigger than 64 bytes -// Note that since CacheRecord object and a CacheGroup object are allocated out of the same pool, it's sensible to -// have them both be the same size. Making one smaller without making the other smaller won't actually save any memory. -#define InlineCacheRDSize 68 - -// On 64-bit, the pointers in a CacheRecord are bigger, and that creates 8 bytes more space for the name in a CacheGroup -#if ENABLE_MULTI_PACKET_QUERY_SNOOPING - #if defined(_ILP64) || defined(__ILP64__) || defined(_LP64) || defined(__LP64__) || defined(_WIN64) - #define InlineCacheGroupNameSize 160 - #else - #define InlineCacheGroupNameSize 148 - #endif -#else - #if defined(_ILP64) || defined(__ILP64__) || defined(_LP64) || defined(__LP64__) || defined(_WIN64) - #define InlineCacheGroupNameSize 144 - #else - #define InlineCacheGroupNameSize 132 - #endif -#endif - -// The RDataBody union defines the common rdata types that fit into our 264-byte limit -typedef union - { - mDNSu8 data[StandardAuthRDSize]; - mDNSv4Addr ipv4; // For 'A' record - domainname name; // For PTR, NS, CNAME, DNAME - UTF8str255 txt; - rdataMX mx; - mDNSv6Addr ipv6; // For 'AAAA' record - rdataSRV srv; - rdataOPT opt[2]; // For EDNS0 OPT record; RDataBody may contain multiple variable-length rdataOPT objects packed together - rdataNSEC nsec; - } RDataBody; - -// The RDataBody2 union is the same as above, except it includes fields for the larger types like soa, rp, px -typedef union - { - mDNSu8 data[StandardAuthRDSize]; - mDNSv4Addr ipv4; // For 'A' record - domainname name; // For PTR, NS, CNAME, DNAME - rdataSOA soa; // This is large; not included in the normal RDataBody definition - UTF8str255 txt; - rdataMX mx; - rdataRP rp; // This is large; not included in the normal RDataBody definition - rdataPX px; // This is large; not included in the normal RDataBody definition - mDNSv6Addr ipv6; // For 'AAAA' record - rdataSRV srv; - rdataOPT opt[2]; // For EDNS0 OPT record; RDataBody may contain multiple variable-length rdataOPT objects packed together - rdataNSEC nsec; - } RDataBody2; - -typedef struct - { - mDNSu16 MaxRDLength; // Amount of storage allocated for rdata (usually sizeof(RDataBody)) - mDNSu16 padding; // So that RDataBody is aligned on 32-bit boundary - RDataBody u; - } RData; - -// sizeofRDataHeader should be 4 bytes -#define sizeofRDataHeader (sizeof(RData) - sizeof(RDataBody)) - -// RData_small is a smaller version of the RData object, used for inline data storage embedded in a CacheRecord_struct -typedef struct - { - mDNSu16 MaxRDLength; // Storage allocated for data (may be greater than InlineCacheRDSize if additional storage follows this object) - mDNSu16 padding; // So that data is aligned on 32-bit boundary - mDNSu8 data[InlineCacheRDSize]; - } RData_small; - -// Note: Within an mDNSRecordCallback mDNS all API calls are legal except mDNS_Init(), mDNS_Exit(), mDNS_Execute() -typedef void mDNSRecordCallback(mDNS *const m, AuthRecord *const rr, mStatus result); - -// Note: -// Restrictions: An mDNSRecordUpdateCallback may not make any mDNS API calls. -// The intent of this callback is to allow the client to free memory, if necessary. -// The internal data structures of the mDNS code may not be in a state where mDNS API calls may be made safely. -typedef void mDNSRecordUpdateCallback(mDNS *const m, AuthRecord *const rr, RData *OldRData, mDNSu16 OldRDLen); - -// *************************************************************************** -#if 0 -#pragma mark - -#pragma mark - NAT Traversal structures and constants -#endif - -#define NATMAP_MAX_RETRY_INTERVAL ((mDNSPlatformOneSecond * 60) * 15) // Max retry interval is 15 minutes -#define NATMAP_MIN_RETRY_INTERVAL (mDNSPlatformOneSecond * 2) // Min retry interval is 2 seconds -#define NATMAP_INIT_RETRY (mDNSPlatformOneSecond / 4) // start at 250ms w/ exponential decay -#define NATMAP_DEFAULT_LEASE (60 * 60 * 2) // 2 hour lease life in seconds -#define NATMAP_VERS 0 - -typedef enum - { - NATOp_AddrRequest = 0, - NATOp_MapUDP = 1, - NATOp_MapTCP = 2, - - NATOp_AddrResponse = 0x80 | 0, - NATOp_MapUDPResponse = 0x80 | 1, - NATOp_MapTCPResponse = 0x80 | 2, - } NATOp_t; - -enum - { - NATErr_None = 0, - NATErr_Vers = 1, - NATErr_Refused = 2, - NATErr_NetFail = 3, - NATErr_Res = 4, - NATErr_Opcode = 5 - }; - -typedef mDNSu16 NATErr_t; - -typedef packedstruct - { - mDNSu8 vers; - mDNSu8 opcode; - } NATAddrRequest; - -typedef packedstruct - { - mDNSu8 vers; - mDNSu8 opcode; - mDNSu16 err; - mDNSu32 upseconds; // Time since last NAT engine reboot, in seconds - mDNSv4Addr ExtAddr; - } NATAddrReply; - -typedef packedstruct - { - mDNSu8 vers; - mDNSu8 opcode; - mDNSOpaque16 unused; - mDNSIPPort intport; - mDNSIPPort extport; - mDNSu32 NATReq_lease; - } NATPortMapRequest; - -typedef packedstruct - { - mDNSu8 vers; - mDNSu8 opcode; - mDNSu16 err; - mDNSu32 upseconds; // Time since last NAT engine reboot, in seconds - mDNSIPPort intport; - mDNSIPPort extport; - mDNSu32 NATRep_lease; - } NATPortMapReply; - -typedef enum - { - LNTDiscoveryOp = 1, - LNTExternalAddrOp = 2, - LNTPortMapOp = 3, - LNTPortMapDeleteOp = 4 - } LNTOp_t; - -#define LNT_MAXBUFSIZE 8192 -typedef struct tcpLNTInfo_struct tcpLNTInfo; -struct tcpLNTInfo_struct - { - tcpLNTInfo *next; - mDNS *m; - NATTraversalInfo *parentNATInfo; // pointer back to the parent NATTraversalInfo - TCPSocket *sock; - LNTOp_t op; // operation performed using this connection - mDNSAddr Address; // router address - mDNSIPPort Port; // router port - mDNSu8 *Request; // xml request to router - int requestLen; - mDNSu8 *Reply; // xml reply from router - int replyLen; - unsigned long nread; // number of bytes read so far - int retries; // number of times we've tried to do this port mapping - }; - -typedef void (*NATTraversalClientCallback)(mDNS *m, NATTraversalInfo *n); - -// if m->timenow < ExpiryTime then we have an active mapping, and we'll renew halfway to expiry -// if m->timenow >= ExpiryTime then our mapping has expired, and we're trying to create one - -struct NATTraversalInfo_struct - { - // Internal state fields. These are used internally by mDNSCore; the client layer needn't be concerned with them. - NATTraversalInfo *next; - - mDNSs32 ExpiryTime; // Time this mapping expires, or zero if no mapping - mDNSs32 retryInterval; // Current interval, between last packet we sent and the next one - mDNSs32 retryPortMap; // If Protocol is nonzero, time to send our next mapping packet - mStatus NewResult; // New error code; will be copied to Result just prior to invoking callback - -#ifdef _LEGACY_NAT_TRAVERSAL_ - tcpLNTInfo tcpInfo; // Legacy NAT traversal (UPnP) TCP connection -#endif - - // Result fields: When the callback is invoked these fields contain the answers the client is looking for - // When the callback is invoked ExternalPort is *usually* set to be the same the same as RequestedPort, except: - // (a) When we're behind a NAT gateway with port mapping disabled, ExternalPort is reported as zero to - // indicate that we don't currently have a working mapping (but RequestedPort retains the external port - // we'd like to get, the next time we meet an accomodating NAT gateway willing to give us one). - // (b) When we have a routable non-RFC1918 address, we don't *need* a port mapping, so ExternalPort - // is reported as the same as our InternalPort, since that is effectively our externally-visible port too. - // Again, RequestedPort retains the external port we'd like to get the next time we find ourself behind a NAT gateway. - // To improve stability of port mappings, RequestedPort is updated any time we get a successful - // mapping response from the NAT-PMP or UPnP gateway. For example, if we ask for port 80, and - // get assigned port 81, then thereafter we'll contine asking for port 81. - mDNSInterfaceID InterfaceID; - mDNSv4Addr ExternalAddress; // Initially set to onesIPv4Addr, until first callback - mDNSIPPort ExternalPort; - mDNSu32 Lifetime; - mStatus Result; - - // Client API fields: The client must set up these fields *before* making any NAT traversal API calls - mDNSu8 Protocol; // NATOp_MapUDP or NATOp_MapTCP, or zero if just requesting the external IP address - mDNSIPPort IntPort; // Client's internal port number (doesn't change) - mDNSIPPort RequestedPort; // Requested external port; may be updated with actual value assigned by gateway - mDNSu32 NATLease; // Requested lifetime in seconds (doesn't change) - NATTraversalClientCallback clientCallback; - void *clientContext; - }; - -enum - { - DNSServer_Untested = 0, - DNSServer_Passed = 1, - DNSServer_Failed = 2, - DNSServer_Disabled = 3 - }; - -enum - { - DNSServer_FlagDelete = 1, - DNSServer_FlagNew = 2 - }; - -enum - { - McastResolver_FlagDelete = 1, - McastResolver_FlagNew = 2 - }; - -typedef struct McastResolver - { - struct McastResolver *next; - mDNSInterfaceID interface; - mDNSu32 flags; // Set when we're planning to delete this from the list - domainname domain; - mDNSu32 timeout; // timeout value for questions - } McastResolver; - -typedef struct DNSServer - { - struct DNSServer *next; - mDNSInterfaceID interface; // For specialized uses; we can have DNS servers reachable over specific interfaces - mDNSAddr addr; - mDNSIPPort port; - mDNSOpaque16 testid; - mDNSu32 flags; // Set when we're planning to delete this from the list - mDNSu32 teststate; // Have we sent bug-detection query to this server? - mDNSs32 lasttest; // Time we sent last bug-detection query to this server - domainname domain; // name->server matching for "split dns" - mDNSs32 penaltyTime; // amount of time this server is penalized - mDNSBool scoped; // interface should be matched against question only - // if scoped is set - mDNSu32 timeout; // timeout value for questions - mDNSBool cellIntf; // Resolver from Cellular Interface ? - } DNSServer; - -typedef struct // Size is 36 bytes when compiling for 32-bit; 48 when compiling for 64-bit - { - mDNSu8 RecordType; // See enum above - mDNSu16 rrtype; - mDNSu16 rrclass; - mDNSu32 rroriginalttl; // In seconds - mDNSu16 rdlength; // Size of the raw rdata, in bytes, in the on-the-wire format - // (In-memory storage may be larger, for structures containing 'holes', like SOA, - // or smaller, for NSEC where we don't bother storing the nextname field) - mDNSu16 rdestimate; // Upper bound on on-the-wire size of rdata after name compression - mDNSu32 namehash; // Name-based (i.e. case-insensitive) hash of name - mDNSu32 rdatahash; // For rdata containing domain name (e.g. PTR, SRV, CNAME etc.), case-insensitive name hash - // else, for all other rdata, 32-bit hash of the raw rdata - // Note: This requirement is important. Various routines like AddAdditionalsToResponseList(), - // ReconfirmAntecedents(), etc., use rdatahash as a pre-flight check to see - // whether it's worth doing a full SameDomainName() call. If the rdatahash - // is not a correct case-insensitive name hash, they'll get false negatives. - - // Grouping pointers together at the end of the structure improves the memory layout efficiency - mDNSInterfaceID InterfaceID; // Set if this RR is specific to one interface - // For records received off the wire, InterfaceID is *always* set to the receiving interface - // For our authoritative records, InterfaceID is usually zero, except for those few records - // that are interface-specific (e.g. address records, especially linklocal addresses) - const domainname *name; - RData *rdata; // Pointer to storage for this rdata - DNSServer *rDNSServer; // Unicast DNS server authoritative for this entry;null for multicast - } ResourceRecord; - -// Unless otherwise noted, states may apply to either independent record registrations or service registrations -typedef enum - { - regState_Zero = 0, - regState_Pending = 1, // update sent, reply not received - regState_Registered = 2, // update sent, reply received - regState_DeregPending = 3, // dereg sent, reply not received - regState_Unregistered = 4, // not in any list - regState_Refresh = 5, // outstanding refresh (or target change) message - regState_NATMap = 6, // establishing NAT port mapping - regState_UpdatePending = 7, // update in flight as result of mDNS_Update call - regState_NoTarget = 8, // SRV Record registration pending registration of hostname - regState_NATError = 9 // unable to complete NAT traversal - } regState_t; - -enum - { - Target_Manual = 0, - Target_AutoHost = 1, - Target_AutoHostAndNATMAP = 2 - }; - -typedef enum - { - mergeState_Zero = 0, - mergeState_DontMerge = 1 // Set on fatal error conditions to disable merging - } mergeState_t; - -struct AuthGroup_struct // Header object for a list of AuthRecords with the same name - { - AuthGroup *next; // Next AuthGroup object in this hash table bucket - mDNSu32 namehash; // Name-based (i.e. case insensitive) hash of name - AuthRecord *members; // List of CacheRecords with this same name - AuthRecord **rrauth_tail; // Tail end of that list - domainname *name; // Common name for all AuthRecords in this list - AuthRecord *NewLocalOnlyRecords; - // Size to here is 20 bytes when compiling 32-bit; 40 bytes when compiling 64-bit - mDNSu8 namestorage[InlineCacheGroupNameSize]; - }; - -#define AUTH_HASH_SLOTS 499 -#define FORALL_AUTHRECORDS(SLOT,AG,AR) \ - for ((SLOT) = 0; (SLOT) < AUTH_HASH_SLOTS; (SLOT)++) \ - for ((AG)=m->rrauth.rrauth_hash[(SLOT)]; (AG); (AG)=(AG)->next) \ - for ((AR) = (AG)->members; (AR); (AR)=(AR)->next) - -typedef union AuthEntity_union AuthEntity; -union AuthEntity_union { AuthEntity *next; AuthGroup ag; }; -typedef struct { - mDNSu32 rrauth_size; // Total number of available auth entries - mDNSu32 rrauth_totalused; // Number of auth entries currently occupied - mDNSu32 rrauth_report; - mDNSu8 rrauth_lock; // For debugging: Set at times when these lists may not be modified - AuthEntity *rrauth_free; - AuthGroup *rrauth_hash[AUTH_HASH_SLOTS]; -}AuthHash; - -// AuthRecordAny includes mDNSInterface_Any and interface specific auth records (anything -// other than P2P or LocalOnly) -typedef enum - { - AuthRecordAny, // registered for *Any, NOT including P2P interfaces - AuthRecordAnyIncludeP2P, // registered for *Any, including P2P interfaces - AuthRecordLocalOnly, - AuthRecordP2P // discovered over D2D/P2P framework - } AuthRecType; - -struct AuthRecord_struct - { - // For examples of how to set up this structure for use in mDNS_Register(), - // see mDNS_AdvertiseInterface() or mDNS_RegisterService(). - // Basically, resrec and persistent metadata need to be set up before calling mDNS_Register(). - // mDNS_SetupResourceRecord() is avaliable as a helper routine to set up most fields to sensible default values for you - - AuthRecord *next; // Next in list; first element of structure for efficiency reasons - // Field Group 1: Common ResourceRecord fields - ResourceRecord resrec; // 36 bytes when compiling for 32-bit; 48 when compiling for 64-bit - - // Field Group 2: Persistent metadata for Authoritative Records - AuthRecord *Additional1; // Recommended additional record to include in response (e.g. SRV for PTR record) - AuthRecord *Additional2; // Another additional (e.g. TXT for PTR record) - AuthRecord *DependentOn; // This record depends on another for its uniqueness checking - AuthRecord *RRSet; // This unique record is part of an RRSet - mDNSRecordCallback *RecordCallback; // Callback function to call for state changes, and to free memory asynchronously on deregistration - void *RecordContext; // Context parameter for the callback function - mDNSu8 AutoTarget; // Set if the target of this record (PTR, CNAME, SRV, etc.) is our host name - mDNSu8 AllowRemoteQuery; // Set if we allow hosts not on the local link to query this record - mDNSu8 ForceMCast; // Set by client to advertise solely via multicast, even for apparently unicast names - - OwnerOptData WakeUp; // WakeUp.HMAC.l[0] nonzero indicates that this is a Sleep Proxy record - mDNSAddr AddressProxy; // For reverse-mapping Sleep Proxy PTR records, address in question - mDNSs32 TimeRcvd; // In platform time units - mDNSs32 TimeExpire; // In platform time units - AuthRecType ARType; // LocalOnly, P2P or Normal ? - - // Field Group 3: Transient state for Authoritative Records - mDNSu8 Acknowledged; // Set if we've given the success callback to the client - mDNSu8 ProbeCount; // Number of probes remaining before this record is valid (kDNSRecordTypeUnique) - mDNSu8 AnnounceCount; // Number of announcements remaining (kDNSRecordTypeShared) - mDNSu8 RequireGoodbye; // Set if this RR has been announced on the wire and will require a goodbye packet - mDNSu8 AnsweredLocalQ; // Set if this AuthRecord has been delivered to any local question (LocalOnly or mDNSInterface_Any) - mDNSu8 IncludeInProbe; // Set if this RR is being put into a probe right now - mDNSu8 ImmedUnicast; // Set if we may send our response directly via unicast to the requester - mDNSInterfaceID SendNSECNow; // Set if we need to generate associated NSEC data for this rrname - mDNSInterfaceID ImmedAnswer; // Someone on this interface issued a query we need to answer (all-ones for all interfaces) -#if MDNS_LOG_ANSWER_SUPPRESSION_TIMES - mDNSs32 ImmedAnswerMarkTime; -#endif - mDNSInterfaceID ImmedAdditional; // Hint that we might want to also send this record, just to be helpful - mDNSInterfaceID SendRNow; // The interface this query is being sent on right now - mDNSv4Addr v4Requester; // Recent v4 query for this record, or all-ones if more than one recent query - mDNSv6Addr v6Requester; // Recent v6 query for this record, or all-ones if more than one recent query - AuthRecord *NextResponse; // Link to the next element in the chain of responses to generate - const mDNSu8 *NR_AnswerTo; // Set if this record was selected by virtue of being a direct answer to a question - AuthRecord *NR_AdditionalTo; // Set if this record was selected by virtue of being additional to another - mDNSs32 ThisAPInterval; // In platform time units: Current interval for announce/probe - mDNSs32 LastAPTime; // In platform time units: Last time we sent announcement/probe - mDNSs32 LastMCTime; // Last time we multicast this record (used to guard against packet-storm attacks) - mDNSInterfaceID LastMCInterface; // Interface this record was multicast on at the time LastMCTime was recorded - RData *NewRData; // Set if we are updating this record with new rdata - mDNSu16 newrdlength; // ... and the length of the new RData - mDNSRecordUpdateCallback *UpdateCallback; - mDNSu32 UpdateCredits; // Token-bucket rate limiting of excessive updates - mDNSs32 NextUpdateCredit; // Time next token is added to bucket - mDNSs32 UpdateBlocked; // Set if update delaying is in effect - - // Field Group 4: Transient uDNS state for Authoritative Records - regState_t state; // Maybe combine this with resrec.RecordType state? Right now it's ambiguous and confusing. - // e.g. rr->resrec.RecordType can be kDNSRecordTypeUnregistered, - // and rr->state can be regState_Unregistered - // What if we find one of those statements is true and the other false? What does that mean? - mDNSBool uselease; // dynamic update contains (should contain) lease option - mDNSs32 expire; // In platform time units: expiration of lease (-1 for static) - mDNSBool Private; // If zone is private, DNS updates may have to be encrypted to prevent eavesdropping - mDNSOpaque16 updateid; // Identifier to match update request and response -- also used when transferring records to Sleep Proxy - const domainname *zone; // the zone that is updated - ZoneData *nta; - struct tcpInfo_t *tcp; - NATTraversalInfo NATinfo; - mDNSBool SRVChanged; // temporarily deregistered service because its SRV target or port changed - mergeState_t mState; // Unicast Record Registrations merge state - mDNSu8 refreshCount; // Number of refreshes to the server - mStatus updateError; // Record update resulted in Error ? - - // uDNS_UpdateRecord support fields - // Do we really need all these in *addition* to NewRData and newrdlength above? - void *UpdateContext; // Context parameter for the update callback function - mDNSu16 OrigRDLen; // previously registered, being deleted - mDNSu16 InFlightRDLen; // currently being registered - mDNSu16 QueuedRDLen; // pending operation (re-transmitting if necessary) THEN register the queued update - RData *OrigRData; - RData *InFlightRData; - RData *QueuedRData; - - // Field Group 5: Large data objects go at the end - domainname namestorage; - RData rdatastorage; // Normally the storage is right here, except for oversized records - // rdatastorage MUST be the last thing in the structure -- when using oversized AuthRecords, extra bytes - // are appended after the end of the AuthRecord, logically augmenting the size of the rdatastorage - // DO NOT ADD ANY MORE FIELDS HERE - }; - -// IsLocalDomain alone is not sufficient to determine that a record is mDNS or uDNS. By default domain names within -// the "local" pseudo-TLD (and within the IPv4 and IPv6 link-local reverse mapping domains) are automatically treated -// as mDNS records, but it is also possible to force any record (even those not within one of the inherently local -// domains) to be handled as an mDNS record by setting the ForceMCast flag, or by setting a non-zero InterfaceID. -// For example, the reverse-mapping PTR record created in AdvertiseInterface sets the ForceMCast flag, since it points to -// a dot-local hostname, and therefore it would make no sense to register this record with a wide-area Unicast DNS server. -// The same applies to Sleep Proxy records, which we will answer for when queried via mDNS, but we never want to try -// to register them with a wide-area Unicast DNS server -- and we probably don't have the required credentials anyway. -// Currently we have no concept of a wide-area uDNS record scoped to a particular interface, so if the InterfaceID is -// nonzero we treat this the same as ForceMCast. -// Note: Question_uDNS(Q) is used in *only* one place -- on entry to mDNS_StartQuery_internal, to decide whether to set TargetQID. -// Everywhere else in the code, the determination of whether a question is unicast is made by checking to see if TargetQID is nonzero. -#define AuthRecord_uDNS(R) ((R)->resrec.InterfaceID == mDNSInterface_Any && !(R)->ForceMCast && !IsLocalDomain((R)->resrec.name)) -#define Question_uDNS(Q) ((Q)->InterfaceID == mDNSInterface_Unicast || \ - ((Q)->InterfaceID != mDNSInterface_LocalOnly && (Q)->InterfaceID != mDNSInterface_P2P && !(Q)->ForceMCast && !IsLocalDomain(&(Q)->qname))) - -#define RRLocalOnly(rr) ((rr)->ARType == AuthRecordLocalOnly || (rr)->ARType == AuthRecordP2P) - -#define RRAny(rr) ((rr)->ARType == AuthRecordAny || (rr)->ARType == AuthRecordAnyIncludeP2P) - -// Question (A or AAAA) that is suppressed currently because IPv4 or IPv6 address -// is not available locally for A or AAAA question respectively -#define QuerySuppressed(Q) ((Q)->SuppressUnusable && (Q)->SuppressQuery) - -#define PrivateQuery(Q) ((Q)->AuthInfo && (Q)->AuthInfo->AutoTunnel) - -// Normally we always lookup the cache and /etc/hosts before sending the query on the wire. For single label -// queries (A and AAAA) that are unqualified (indicated by AppendSearchDomains), we want to append search -// domains before we try them as such -#define ApplySearchDomainsFirst(q) ((q)->AppendSearchDomains && (CountLabels(&((q)->qname))) == 1) - -// Wrapper struct for Auth Records for higher-level code that cannot use the AuthRecord's ->next pointer field -typedef struct ARListElem - { - struct ARListElem *next; - AuthRecord ar; // Note: Must be last element of structure, to accomodate oversized AuthRecords - } ARListElem; - -struct CacheGroup_struct // Header object for a list of CacheRecords with the same name - { - CacheGroup *next; // Next CacheGroup object in this hash table bucket - mDNSu32 namehash; // Name-based (i.e. case insensitive) hash of name - CacheRecord *members; // List of CacheRecords with this same name - CacheRecord **rrcache_tail; // Tail end of that list - domainname *name; // Common name for all CacheRecords in this list - // Size to here is 20 bytes when compiling 32-bit; 40 bytes when compiling 64-bit - mDNSu8 namestorage[InlineCacheGroupNameSize]; - }; - - -struct CacheRecord_struct - { - CacheRecord *next; // Next in list; first element of structure for efficiency reasons - ResourceRecord resrec; // 36 bytes when compiling for 32-bit; 48 when compiling for 64-bit - - // Transient state for Cache Records - CacheRecord *NextInKAList; // Link to the next element in the chain of known answers to send - mDNSs32 TimeRcvd; // In platform time units - mDNSs32 DelayDelivery; // Set if we want to defer delivery of this answer to local clients - mDNSs32 NextRequiredQuery; // In platform time units - mDNSs32 LastUsed; // In platform time units - DNSQuestion *CRActiveQuestion; // Points to an active question referencing this answer. Can never point to a NewQuestion. - mDNSu32 UnansweredQueries; // Number of times we've issued a query for this record without getting an answer - mDNSs32 LastUnansweredTime; // In platform time units; last time we incremented UnansweredQueries -#if ENABLE_MULTI_PACKET_QUERY_SNOOPING - mDNSu32 MPUnansweredQ; // Multi-packet query handling: Number of times we've seen a query for this record - mDNSs32 MPLastUnansweredQT; // Multi-packet query handling: Last time we incremented MPUnansweredQ - mDNSu32 MPUnansweredKA; // Multi-packet query handling: Number of times we've seen this record in a KA list - mDNSBool MPExpectingKA; // Multi-packet query handling: Set when we increment MPUnansweredQ; allows one KA -#endif - CacheRecord *NextInCFList; // Set if this is in the list of records we just received with the cache flush bit set - // Size to here is 76 bytes when compiling 32-bit; 104 bytes when compiling 64-bit - RData_small smallrdatastorage; // Storage for small records is right here (4 bytes header + 68 bytes data = 72 bytes) - }; - -// Storage sufficient to hold either a CacheGroup header or a CacheRecord -// -- for best efficiency (to avoid wasted unused storage) they should be the same size -typedef union CacheEntity_union CacheEntity; -union CacheEntity_union { CacheEntity *next; CacheGroup cg; CacheRecord cr; }; - -typedef struct - { - CacheRecord r; - mDNSu8 _extradata[MaximumRDSize-InlineCacheRDSize]; // Glue on the necessary number of extra bytes - domainname namestorage; // Needs to go *after* the extra rdata bytes - } LargeCacheRecord; - -typedef struct HostnameInfo - { - struct HostnameInfo *next; - NATTraversalInfo natinfo; - domainname fqdn; - AuthRecord arv4; // registered IPv4 address record - AuthRecord arv6; // registered IPv6 address record - mDNSRecordCallback *StatusCallback; // callback to deliver success or error code to client layer - const void *StatusContext; // Client Context - } HostnameInfo; - -typedef struct ExtraResourceRecord_struct ExtraResourceRecord; -struct ExtraResourceRecord_struct - { - ExtraResourceRecord *next; - mDNSu32 ClientID; // Opaque ID field to be used by client to map an AddRecord call to a set of Extra records - AuthRecord r; - // Note: Add any additional fields *before* the AuthRecord in this structure, not at the end. - // In some cases clients can allocate larger chunks of memory and set r->rdata->MaxRDLength to indicate - // that this extra memory is available, which would result in any fields after the AuthRecord getting smashed - }; - -// Note: Within an mDNSServiceCallback mDNS all API calls are legal except mDNS_Init(), mDNS_Exit(), mDNS_Execute() -typedef void mDNSServiceCallback(mDNS *const m, ServiceRecordSet *const sr, mStatus result); - -// A ServiceRecordSet has no special meaning to the core code of the Multicast DNS protocol engine; -// it is just a convenience structure to group together the records that make up a standard service -// registration so that they can be allocted and deallocted together as a single memory object. -// It contains its own ServiceCallback+ServiceContext to report aggregate results up to the next layer of software above. -// It also contains: -// * the basic PTR/SRV/TXT triplet used to represent any DNS-SD service -// * the "_services" PTR record for service enumeration -// * the optional list of SubType PTR records -// * the optional list of additional records attached to the service set (e.g. iChat pictures) - -struct ServiceRecordSet_struct - { - // These internal state fields are used internally by mDNSCore; the client layer needn't be concerned with them. - // No fields need to be set up by the client prior to calling mDNS_RegisterService(); - // all required data is passed as parameters to that function. - mDNSServiceCallback *ServiceCallback; - void *ServiceContext; - mDNSBool Conflict; // Set if this record set was forcibly deregistered because of a conflict - - ExtraResourceRecord *Extras; // Optional list of extra AuthRecords attached to this service registration - mDNSu32 NumSubTypes; - AuthRecord *SubTypes; - AuthRecord RR_ADV; // e.g. _services._dns-sd._udp.local. PTR _printer._tcp.local. - AuthRecord RR_PTR; // e.g. _printer._tcp.local. PTR Name._printer._tcp.local. - AuthRecord RR_SRV; // e.g. Name._printer._tcp.local. SRV 0 0 port target - AuthRecord RR_TXT; // e.g. Name._printer._tcp.local. TXT PrintQueueName - // Don't add any fields after AuthRecord RR_TXT. - // This is where the implicit extra space goes if we allocate a ServiceRecordSet containing an oversized RR_TXT record - }; - -// *************************************************************************** -#if 0 -#pragma mark - -#pragma mark - Question structures -#endif - -// We record the last eight instances of each duplicate query -// This gives us v4/v6 on each of Ethernet, AirPort and Firewire, and two free slots "for future expansion" -// If the host has more active interfaces that this it is not fatal -- duplicate question suppression will degrade gracefully. -// Since we will still remember the last eight, the busiest interfaces will still get the effective duplicate question suppression. -#define DupSuppressInfoSize 8 - -typedef struct - { - mDNSs32 Time; - mDNSInterfaceID InterfaceID; - mDNSs32 Type; // v4 or v6? - } DupSuppressInfo; - -typedef enum - { - LLQ_InitialRequest = 1, - LLQ_SecondaryRequest = 2, - LLQ_Established = 3, - LLQ_Poll = 4 - } LLQ_State; - -// LLQ constants -#define kLLQ_Vers 1 -#define kLLQ_DefLease 7200 // 2 hours -#define kLLQ_MAX_TRIES 3 // retry an operation 3 times max -#define kLLQ_INIT_RESEND 2 // resend an un-ack'd packet after 2 seconds, then double for each additional -// LLQ Operation Codes -#define kLLQOp_Setup 1 -#define kLLQOp_Refresh 2 -#define kLLQOp_Event 3 - -// LLQ Errror Codes -enum - { - LLQErr_NoError = 0, - LLQErr_ServFull = 1, - LLQErr_Static = 2, - LLQErr_FormErr = 3, - LLQErr_NoSuchLLQ = 4, - LLQErr_BadVers = 5, - LLQErr_UnknownErr = 6 - }; - -enum { NoAnswer_Normal = 0, NoAnswer_Suspended = 1, NoAnswer_Fail = 2 }; - -#define HMAC_LEN 64 -#define HMAC_IPAD 0x36 -#define HMAC_OPAD 0x5c -#define MD5_LEN 16 - -#define AutoTunnelUnregistered(X) ( \ - (X)->AutoTunnelHostRecord.resrec.RecordType == kDNSRecordTypeUnregistered && \ - (X)->AutoTunnelDeviceInfo.resrec.RecordType == kDNSRecordTypeUnregistered && \ - (X)->AutoTunnelService. resrec.RecordType == kDNSRecordTypeUnregistered && \ - (X)->AutoTunnel6Record. resrec.RecordType == kDNSRecordTypeUnregistered ) - -// Internal data structure to maintain authentication information -typedef struct DomainAuthInfo - { - struct DomainAuthInfo *next; - mDNSs32 deltime; // If we're planning to delete this DomainAuthInfo, the time we want it deleted - const char* AutoTunnel; // If NULL, this is not an AutoTunnel DAI. Otherwise, this is prepended to the IPSec identifier - AuthRecord AutoTunnelHostRecord; // User-visible hostname; used as SRV target for AutoTunnel services - AuthRecord AutoTunnelTarget; // Opaque hostname of tunnel endpoint; used as SRV target for AutoTunnelService record - AuthRecord AutoTunnelDeviceInfo; // Device info of tunnel endpoint - AuthRecord AutoTunnelService; // Service record (possibly NAT-Mapped) of IKE daemon implementing tunnel endpoint - AuthRecord AutoTunnel6Record; // AutoTunnel AAAA Record obtained from Connectivityd - NATTraversalInfo AutoTunnelNAT; - domainname domain; - domainname keyname; - domainname hostname; - mDNSIPPort port; - char b64keydata[32]; - mDNSu8 keydata_ipad[HMAC_LEN]; // padded key for inner hash rounds - mDNSu8 keydata_opad[HMAC_LEN]; // padded key for outer hash rounds - } DomainAuthInfo; - -// Note: Within an mDNSQuestionCallback mDNS all API calls are legal except mDNS_Init(), mDNS_Exit(), mDNS_Execute() -typedef enum { QC_rmv = 0, QC_add = 1, QC_addnocache = 2 } QC_result; -typedef void mDNSQuestionCallback(mDNS *const m, DNSQuestion *question, const ResourceRecord *const answer, QC_result AddRecord); - -#define NextQSendTime(Q) ((Q)->LastQTime + (Q)->ThisQInterval) -#define ActiveQuestion(Q) ((Q)->ThisQInterval > 0 && !(Q)->DuplicateOf) -#define TimeToSendThisQuestion(Q,time) (ActiveQuestion(Q) && (time) - NextQSendTime(Q) >= 0) - -struct DNSQuestion_struct - { - // Internal state fields. These are used internally by mDNSCore; the client layer needn't be concerned with them. - DNSQuestion *next; - mDNSu32 qnamehash; - mDNSs32 DelayAnswering; // Set if we want to defer answering this question until the cache settles - mDNSs32 LastQTime; // Last scheduled transmission of this Q on *all* applicable interfaces - mDNSs32 ThisQInterval; // LastQTime + ThisQInterval is the next scheduled transmission of this Q - // ThisQInterval > 0 for an active question; - // ThisQInterval = 0 for a suspended question that's still in the list - // ThisQInterval = -1 for a cancelled question (should not still be in list) - mDNSs32 ExpectUnicastResp;// Set when we send a query with the kDNSQClass_UnicastResponse bit set - mDNSs32 LastAnswerPktNum; // The sequence number of the last response packet containing an answer to this Q - mDNSu32 RecentAnswerPkts; // Number of answers since the last time we sent this query - mDNSu32 CurrentAnswers; // Number of records currently in the cache that answer this question - mDNSu32 LargeAnswers; // Number of answers with rdata > 1024 bytes - mDNSu32 UniqueAnswers; // Number of answers received with kDNSClass_UniqueRRSet bit set - mDNSInterfaceID FlappingInterface1;// Set when an interface goes away, to flag if remove events are delivered for this Q - mDNSInterfaceID FlappingInterface2;// Set when an interface goes away, to flag if remove events are delivered for this Q - DomainAuthInfo *AuthInfo; // Non-NULL if query is currently being done using Private DNS - DNSQuestion *DuplicateOf; - DNSQuestion *NextInDQList; - DupSuppressInfo DupSuppress[DupSuppressInfoSize]; - mDNSInterfaceID SendQNow; // The interface this query is being sent on right now - mDNSBool SendOnAll; // Set if we're sending this question on all active interfaces - mDNSu32 RequestUnicast; // Non-zero if we want to send query with kDNSQClass_UnicastResponse bit set - mDNSs32 LastQTxTime; // Last time this Q was sent on one (but not necessarily all) interfaces - mDNSu32 CNAMEReferrals; // Count of how many CNAME redirections we've done - mDNSBool SuppressQuery; // This query should be suppressed and not sent on the wire - mDNSu8 LOAddressAnswers; // Number of answers from the local only auth records that are - // answering A, AAAA and CNAME (/etc/hosts) - mDNSu8 WakeOnResolveCount; // Number of wakes that should be sent on resolve - mDNSs32 StopTime; // Time this question should be stopped by giving them a negative answer - - // Wide Area fields. These are used internally by the uDNS core - UDPSocket *LocalSocket; - mDNSBool deliverAddEvents; // Change in DNSSserver requiring to deliver ADD events - DNSServer *qDNSServer; // Caching server for this query (in the absence of an SRV saying otherwise) - mDNSOpaque64 validDNSServers; // Valid DNSServers for this question - mDNSu16 noServerResponse; // At least one server did not respond. - mDNSu16 triedAllServersOnce; // Tried all DNS servers once - mDNSu8 unansweredQueries;// The number of unanswered queries to this server - - ZoneData *nta; // Used for getting zone data for private or LLQ query - mDNSAddr servAddr; // Address and port learned from _dns-llq, _dns-llq-tls or _dns-query-tls SRV query - mDNSIPPort servPort; - struct tcpInfo_t *tcp; - mDNSIPPort tcpSrcPort; // Local Port TCP packet received on;need this as tcp struct is disposed - // by tcpCallback before calling into mDNSCoreReceive - mDNSu8 NoAnswer; // Set if we want to suppress answers until tunnel setup has completed - - // LLQ-specific fields. These fields are only meaningful when LongLived flag is set - LLQ_State state; - mDNSu32 ReqLease; // seconds (relative) - mDNSs32 expire; // ticks (absolute) - mDNSs16 ntries; // for UDP: the number of packets sent for this LLQ state - // for TCP: there is some ambiguity in the use of this variable, but in general, it is - // the number of TCP/TLS connection attempts for this LLQ state, or - // the number of packets sent for this TCP/TLS connection - mDNSOpaque64 id; - - // Client API fields: The client must set up these fields *before* calling mDNS_StartQuery() - mDNSInterfaceID InterfaceID; // Non-zero if you want to issue queries only on a single specific IP interface - mDNSAddr Target; // Non-zero if you want to direct queries to a specific unicast target address - mDNSIPPort TargetPort; // Must be set if Target is set - mDNSOpaque16 TargetQID; // Must be set if Target is set - domainname qname; - mDNSu16 qtype; - mDNSu16 qclass; - mDNSBool LongLived; // Set by client for calls to mDNS_StartQuery to indicate LLQs to unicast layer. - mDNSBool ExpectUnique; // Set by client if it's expecting unique RR(s) for this question, not shared RRs - mDNSBool ForceMCast; // Set by client to force mDNS query, even for apparently uDNS names - mDNSBool ReturnIntermed; // Set by client to request callbacks for intermediate CNAME/NXDOMAIN results - mDNSBool SuppressUnusable; // Set by client to suppress unusable queries to be sent on the wire - mDNSBool RetryWithSearchDomains; // Retry with search domains if there is no entry in the cache or AuthRecords - mDNSu8 TimeoutQuestion; // Timeout this question if there is no reply in configured time - mDNSu8 WakeOnResolve; // Send wakeup on resolve - mDNSs8 SearchListIndex; // Index into SearchList; Used by the client layer but not touched by core - mDNSs8 AppendSearchDomains; // Search domains can be appended for this query - mDNSs8 AppendLocalSearchDomains; // Search domains ending in .local can be appended for this query - domainname *qnameOrig; // Copy of the original question name if it is not fully qualified - mDNSQuestionCallback *QuestionCallback; - void *QuestionContext; - }; - -typedef struct - { - // Client API fields: The client must set up name and InterfaceID *before* calling mDNS_StartResolveService() - // When the callback is invoked, ip, port, TXTlen and TXTinfo will have been filled in with the results learned from the network. - domainname name; - mDNSInterfaceID InterfaceID; // ID of the interface the response was received on - mDNSAddr ip; // Remote (destination) IP address where this service can be accessed - mDNSIPPort port; // Port where this service can be accessed - mDNSu16 TXTlen; - mDNSu8 TXTinfo[2048]; // Additional demultiplexing information (e.g. LPR queue name) - } ServiceInfo; - -// Note: Within an mDNSServiceInfoQueryCallback mDNS all API calls are legal except mDNS_Init(), mDNS_Exit(), mDNS_Execute() -typedef struct ServiceInfoQuery_struct ServiceInfoQuery; -typedef void mDNSServiceInfoQueryCallback(mDNS *const m, ServiceInfoQuery *query); -struct ServiceInfoQuery_struct - { - // Internal state fields. These are used internally by mDNSCore; the client layer needn't be concerned with them. - // No fields need to be set up by the client prior to calling mDNS_StartResolveService(); - // all required data is passed as parameters to that function. - // The ServiceInfoQuery structure memory is working storage for mDNSCore to discover the requested information - // and place it in the ServiceInfo structure. After the client has called mDNS_StopResolveService(), it may - // dispose of the ServiceInfoQuery structure while retaining the results in the ServiceInfo structure. - DNSQuestion qSRV; - DNSQuestion qTXT; - DNSQuestion qAv4; - DNSQuestion qAv6; - mDNSu8 GotSRV; - mDNSu8 GotTXT; - mDNSu8 GotADD; - mDNSu32 Answers; - ServiceInfo *info; - mDNSServiceInfoQueryCallback *ServiceInfoQueryCallback; - void *ServiceInfoQueryContext; - }; - -typedef enum { ZoneServiceUpdate, ZoneServiceQuery, ZoneServiceLLQ } ZoneService; - -typedef void ZoneDataCallback(mDNS *const m, mStatus err, const ZoneData *result); - -struct ZoneData_struct - { - domainname ChildName; // Name for which we're trying to find the responsible server - ZoneService ZoneService; // Which service we're seeking for this zone (update, query, or LLQ) - domainname *CurrentSOA; // Points to somewhere within ChildName - domainname ZoneName; // Discovered result: Left-hand-side of SOA record - mDNSu16 ZoneClass; // Discovered result: DNS Class from SOA record - domainname Host; // Discovered result: Target host from SRV record - mDNSIPPort Port; // Discovered result: Update port, query port, or LLQ port from SRV record - mDNSAddr Addr; // Discovered result: Address of Target host from SRV record - mDNSBool ZonePrivate; // Discovered result: Does zone require encrypted queries? - ZoneDataCallback *ZoneDataCallback; // Caller-specified function to be called upon completion - void *ZoneDataContext; - DNSQuestion question; // Storage for any active question - }; - -extern ZoneData *StartGetZoneData(mDNS *const m, const domainname *const name, const ZoneService target, ZoneDataCallback callback, void *callbackInfo); -extern void CancelGetZoneData(mDNS *const m, ZoneData *nta); -extern mDNSBool IsGetZoneDataQuestion(DNSQuestion *q); - -typedef struct DNameListElem - { - struct DNameListElem *next; - mDNSu32 uid; - domainname name; - } DNameListElem; - -#if APPLE_OSX_mDNSResponder -// Different states that we go through locating the peer -#define TC_STATE_AAAA_PEER 0x000000001 /* Peer's BTMM IPv6 address */ -#define TC_STATE_AAAA_PEER_RELAY 0x000000002 /* Peer's IPv6 Relay address */ -#define TC_STATE_SRV_PEER 0x000000003 /* Peer's SRV Record corresponding to IPv4 address */ -#define TC_STATE_ADDR_PEER 0x000000004 /* Peer's IPv4 address */ - -typedef struct ClientTunnel - { - struct ClientTunnel *next; - const char *prefix; - domainname dstname; - mDNSBool MarkedForDeletion; - mDNSv6Addr loc_inner; - mDNSv4Addr loc_outer; - mDNSv6Addr loc_outer6; - mDNSv6Addr rmt_inner; - mDNSv4Addr rmt_outer; - mDNSv6Addr rmt_outer6; - mDNSIPPort rmt_outer_port; - mDNSu16 tc_state; - DNSQuestion q; - } ClientTunnel; -#endif - -// *************************************************************************** -#if 0 -#pragma mark - -#pragma mark - NetworkInterfaceInfo_struct -#endif - -typedef struct NetworkInterfaceInfo_struct NetworkInterfaceInfo; - -// A NetworkInterfaceInfo_struct serves two purposes: -// 1. It holds the address, PTR and HINFO records to advertise a given IP address on a given physical interface -// 2. It tells mDNSCore which physical interfaces are available; each physical interface has its own unique InterfaceID. -// Since there may be multiple IP addresses on a single physical interface, -// there may be multiple NetworkInterfaceInfo_structs with the same InterfaceID. -// In this case, to avoid sending the same packet n times, when there's more than one -// struct with the same InterfaceID, mDNSCore picks one member of the set to be the -// active representative of the set; all others have the 'InterfaceActive' flag unset. - -struct NetworkInterfaceInfo_struct - { - // Internal state fields. These are used internally by mDNSCore; the client layer needn't be concerned with them. - NetworkInterfaceInfo *next; - - mDNSu8 InterfaceActive; // Set if interface is sending & receiving packets (see comment above) - mDNSu8 IPv4Available; // If InterfaceActive, set if v4 available on this InterfaceID - mDNSu8 IPv6Available; // If InterfaceActive, set if v6 available on this InterfaceID - - DNSQuestion NetWakeBrowse; - DNSQuestion NetWakeResolve[3]; // For fault-tolerance, we try up to three Sleep Proxies - mDNSAddr SPSAddr[3]; - mDNSIPPort SPSPort[3]; - mDNSs32 NextSPSAttempt; // -1 if we're not currently attempting to register with any Sleep Proxy - mDNSs32 NextSPSAttemptTime; - - // Standard AuthRecords that every Responder host should have (one per active IP address) - AuthRecord RR_A; // 'A' or 'AAAA' (address) record for our ".local" name - AuthRecord RR_PTR; // PTR (reverse lookup) record - AuthRecord RR_HINFO; - - // Client API fields: The client must set up these fields *before* calling mDNS_RegisterInterface() - mDNSInterfaceID InterfaceID; // Identifies physical interface; MUST NOT be 0, -1, or -2 - mDNSAddr ip; // The IPv4 or IPv6 address to advertise - mDNSAddr mask; - mDNSEthAddr MAC; - char ifname[64]; // Windows uses a GUID string for the interface name, which doesn't fit in 16 bytes - mDNSu8 Advertise; // False if you are only searching on this interface - mDNSu8 McastTxRx; // Send/Receive multicast on this { InterfaceID, address family } ? - mDNSu8 NetWake; // Set if Wake-On-Magic-Packet is enabled on this interface - mDNSu8 Loopback; // Set if this is the loopback interface - }; - -#define SLE_DELETE 0x00000001 -#define SLE_WAB_QUERY_STARTED 0x00000002 - -typedef struct SearchListElem - { - struct SearchListElem *next; - domainname domain; - int flag; - mDNSInterfaceID InterfaceID; - DNSQuestion BrowseQ; - DNSQuestion DefBrowseQ; - DNSQuestion AutomaticBrowseQ; - DNSQuestion RegisterQ; - DNSQuestion DefRegisterQ; - int numCfAnswers; - ARListElem *AuthRecs; - } SearchListElem; - -// For domain enumeration and automatic browsing -// This is the user's DNS search list. -// In each of these domains we search for our special pointer records (lb._dns-sd._udp., etc.) -// to discover recommended domains for domain enumeration (browse, default browse, registration, -// default registration) and possibly one or more recommended automatic browsing domains. -extern SearchListElem *SearchList; // This really ought to be part of mDNS_struct -- SC - -// *************************************************************************** -#if 0 -#pragma mark - -#pragma mark - Main mDNS object, used to hold all the mDNS state -#endif - -typedef void mDNSCallback(mDNS *const m, mStatus result); - -#define CACHE_HASH_SLOTS 499 - -enum // Bit flags -- i.e. values should be 1, 2, 4, 8, etc. - { - mDNS_KnownBug_LimitedIPv6 = 1, - mDNS_KnownBug_LossySyslog = 2 // - }; - -enum - { - SleepState_Awake = 0, - SleepState_Transferring = 1, - SleepState_Sleeping = 2 - }; - -struct mDNS_struct - { - // Internal state fields. These hold the main internal state of mDNSCore; - // the client layer needn't be concerned with them. - // No fields need to be set up by the client prior to calling mDNS_Init(); - // all required data is passed as parameters to that function. - - mDNS_PlatformSupport *p; // Pointer to platform-specific data of indeterminite size - mDNSu32 KnownBugs; - mDNSBool CanReceiveUnicastOn5353; - mDNSBool AdvertiseLocalAddresses; - mDNSBool DivertMulticastAdvertisements; // from interfaces that do not advertise local addresses to local-only - mStatus mDNSPlatformStatus; - mDNSIPPort UnicastPort4; - mDNSIPPort UnicastPort6; - mDNSEthAddr PrimaryMAC; // Used as unique host ID - mDNSCallback *MainCallback; - void *MainContext; - - // For debugging: To catch and report locking failures - mDNSu32 mDNS_busy; // Incremented between mDNS_Lock/mDNS_Unlock section - mDNSu32 mDNS_reentrancy; // Incremented when calling a client callback - mDNSu8 lock_rrcache; // For debugging: Set at times when these lists may not be modified - mDNSu8 lock_Questions; - mDNSu8 lock_Records; -#ifndef MaxMsg - #define MaxMsg 160 -#endif - char MsgBuffer[MaxMsg]; // Temp storage used while building error log messages - - // Task Scheduling variables - mDNSs32 timenow_adjust; // Correction applied if we ever discover time went backwards - mDNSs32 timenow; // The time that this particular activation of the mDNS code started - mDNSs32 timenow_last; // The time the last time we ran - mDNSs32 NextScheduledEvent; // Derived from values below - mDNSs32 ShutdownTime; // Set when we're shutting down; allows us to skip some unnecessary steps - mDNSs32 SuppressSending; // Don't send local-link mDNS packets during this time - mDNSs32 NextCacheCheck; // Next time to refresh cache record before it expires - mDNSs32 NextScheduledQuery; // Next time to send query in its exponential backoff sequence - mDNSs32 NextScheduledProbe; // Next time to probe for new authoritative record - mDNSs32 NextScheduledResponse; // Next time to send authoritative record(s) in responses - mDNSs32 NextScheduledNATOp; // Next time to send NAT-traversal packets - mDNSs32 NextScheduledSPS; // Next time to purge expiring Sleep Proxy records - mDNSs32 RandomQueryDelay; // For de-synchronization of query packets on the wire - mDNSu32 RandomReconfirmDelay; // For de-synchronization of reconfirmation queries on the wire - mDNSs32 PktNum; // Unique sequence number assigned to each received packet - mDNSu8 LocalRemoveEvents; // Set if we may need to deliver remove events for local-only questions and/or local-only records - mDNSu8 SleepState; // Set if we're sleeping - mDNSu8 SleepSeqNum; // "Epoch number" of our current period of wakefulness - mDNSu8 SystemWakeOnLANEnabled; // Set if we want to register with a Sleep Proxy before going to sleep - mDNSu8 SentSleepProxyRegistration;// Set if we registered (or tried to register) with a Sleep Proxy - mDNSu8 SystemSleepOnlyIfWakeOnLAN;// Set if we may only sleep if we managed to register with a Sleep Proxy - mDNSs32 AnnounceOwner; // After waking from sleep, include OWNER option in packets until this time - mDNSs32 DelaySleep; // To inhibit re-sleeping too quickly right after wake - mDNSs32 SleepLimit; // Time window to allow deregistrations, etc., - // during which underying platform layer should inhibit system sleep - mDNSs32 NextScheduledSPRetry; // Time next sleep proxy registration action is required. - // Only valid if SleepLimit is nonzero and DelaySleep is zero. - - mDNSs32 NextScheduledStopTime; // Next time to stop a question - - // These fields only required for mDNS Searcher... - DNSQuestion *Questions; // List of all registered questions, active and inactive - DNSQuestion *NewQuestions; // Fresh questions not yet answered from cache - DNSQuestion *CurrentQuestion; // Next question about to be examined in AnswerLocalQuestions() - DNSQuestion *LocalOnlyQuestions; // Questions with InterfaceID set to mDNSInterface_LocalOnly or mDNSInterface_P2P - DNSQuestion *NewLocalOnlyQuestions; // Fresh local-only or P2P questions not yet answered - DNSQuestion *RestartQuestion; // Questions that are being restarted (stop followed by start) - mDNSu32 rrcache_size; // Total number of available cache entries - mDNSu32 rrcache_totalused; // Number of cache entries currently occupied - mDNSu32 rrcache_active; // Number of cache entries currently occupied by records that answer active questions - mDNSu32 rrcache_report; - CacheEntity *rrcache_free; - CacheGroup *rrcache_hash[CACHE_HASH_SLOTS]; - mDNSs32 rrcache_nextcheck[CACHE_HASH_SLOTS]; - - AuthHash rrauth; - - // Fields below only required for mDNS Responder... - domainlabel nicelabel; // Rich text label encoded using canonically precomposed UTF-8 - domainlabel hostlabel; // Conforms to RFC 1034 "letter-digit-hyphen" ARPANET host name rules - domainname MulticastHostname; // Fully Qualified "dot-local" Host Name, e.g. "Foo.local." - UTF8str255 HIHardware; - UTF8str255 HISoftware; - AuthRecord DeviceInfo; - AuthRecord *ResourceRecords; - AuthRecord *DuplicateRecords; // Records currently 'on hold' because they are duplicates of existing records - AuthRecord *NewLocalRecords; // Fresh AuthRecords (public) not yet delivered to our local-only questions - AuthRecord *CurrentRecord; // Next AuthRecord about to be examined - mDNSBool NewLocalOnlyRecords; // Fresh AuthRecords (local only) not yet delivered to our local questions - NetworkInterfaceInfo *HostInterfaces; - mDNSs32 ProbeFailTime; - mDNSu32 NumFailedProbes; - mDNSs32 SuppressProbes; - - // Unicast-specific data - mDNSs32 NextuDNSEvent; // uDNS next event - mDNSs32 NextSRVUpdate; // Time to perform delayed update - - DNSServer *DNSServers; // list of DNS servers - McastResolver *McastResolvers; // list of Mcast Resolvers - - mDNSAddr Router; - mDNSAddr AdvertisedV4; // IPv4 address pointed to by hostname - mDNSAddr AdvertisedV6; // IPv6 address pointed to by hostname - - DomainAuthInfo *AuthInfoList; // list of domains requiring authentication for updates - - DNSQuestion ReverseMap; // Reverse-map query to find static hostname for service target - DNSQuestion AutomaticBrowseDomainQ; - domainname StaticHostname; // Current answer to reverse-map query - domainname FQDN; - HostnameInfo *Hostnames; // List of registered hostnames + hostname metadata - mDNSv6Addr AutoTunnelHostAddr; // IPv6 address advertised for AutoTunnel services on this machine - mDNSBool AutoTunnelHostAddrActive; - // AutoTunnel Relay address has two distinct uses - // AutoTunnelRelayAddrIn: If non-zero, it means that this host can be reached (inbound connection) through the relay - // AutoTunnelRelayAddrOut: If non-zero, it means that this host can use the relay to reach (outbound connection) the - // other hosts through the relay - mDNSv6Addr AutoTunnelRelayAddrIn; - mDNSv6Addr AutoTunnelRelayAddrOut; - domainlabel AutoTunnelLabel; // Used to construct hostname for *IPv4* address of tunnel endpoints - - mDNSBool StartWABQueries; // Start WAB queries for the purpose of domain enumeration - mDNSBool RegisterAutoTunnel6; - - // NAT-Traversal fields - NATTraversalInfo LLQNAT; // Single shared NAT Traversal to receive inbound LLQ notifications - NATTraversalInfo *NATTraversals; - NATTraversalInfo *CurrentNATTraversal; - mDNSs32 retryIntervalGetAddr; // delta between time sent and retry - mDNSs32 retryGetAddr; // absolute time when we retry - mDNSv4Addr ExternalAddress; - - UDPSocket *NATMcastRecvskt; // For receiving NAT-PMP AddrReply multicasts from router on port 5350 - mDNSu32 LastNATupseconds; // NAT engine uptime in seconds, from most recent NAT packet - mDNSs32 LastNATReplyLocalTime; // Local time in ticks when most recent NAT packet was received - mDNSu16 LastNATMapResultCode; // Most recent error code for mappings - - tcpLNTInfo tcpAddrInfo; // legacy NAT traversal TCP connection info for external address - tcpLNTInfo tcpDeviceInfo; // legacy NAT traversal TCP connection info for device info - tcpLNTInfo *tcpInfoUnmapList; // list of pending unmap requests - mDNSInterfaceID UPnPInterfaceID; - UDPSocket *SSDPSocket; // For SSDP request/response - mDNSBool SSDPWANPPPConnection; // whether we should send the SSDP query for WANIPConnection or WANPPPConnection - mDNSIPPort UPnPRouterPort; // port we send discovery messages to - mDNSIPPort UPnPSOAPPort; // port we send SOAP messages to - mDNSu8 *UPnPRouterURL; // router's URL string - mDNSBool UPnPWANPPPConnection; // whether we're using WANIPConnection or WANPPPConnection - mDNSu8 *UPnPSOAPURL; // router's SOAP control URL string - mDNSu8 *UPnPRouterAddressString; // holds both the router's address and port - mDNSu8 *UPnPSOAPAddressString; // holds both address and port for SOAP messages - - // Sleep Proxy Server fields - mDNSu8 SPSType; // 0 = off, 10-99 encodes desirability metric - mDNSu8 SPSPortability; // 10-99 - mDNSu8 SPSMarginalPower; // 10-99 - mDNSu8 SPSTotalPower; // 10-99 - mDNSu8 SPSState; // 0 = off, 1 = running, 2 = shutting down, 3 = suspended during sleep - mDNSInterfaceID SPSProxyListChanged; - UDPSocket *SPSSocket; - ServiceRecordSet SPSRecords; - mDNSQuestionCallback *SPSBrowseCallback; // So the platform layer can do something useful with SPS browse results - int ProxyRecords; // Total number of records we're holding as proxy - #define MAX_PROXY_RECORDS 10000 /* DOS protection: 400 machines at 25 records each */ - -#if APPLE_OSX_mDNSResponder - ClientTunnel *TunnelClients; - uuid_t asl_uuid; // uuid for ASL logging - void *WCF; -#endif - - // Fixed storage, to avoid creating large objects on the stack - // The imsg is declared as a union with a pointer type to enforce CPU-appropriate alignment - union { DNSMessage m; void *p; } imsg; // Incoming message received from wire - DNSMessage omsg; // Outgoing message we're building - LargeCacheRecord rec; // Resource Record extracted from received message - }; - -#define FORALL_CACHERECORDS(SLOT,CG,CR) \ - for ((SLOT) = 0; (SLOT) < CACHE_HASH_SLOTS; (SLOT)++) \ - for ((CG)=m->rrcache_hash[(SLOT)]; (CG); (CG)=(CG)->next) \ - for ((CR) = (CG)->members; (CR); (CR)=(CR)->next) - -// *************************************************************************** -#if 0 -#pragma mark - -#pragma mark - Useful Static Constants -#endif - -extern const mDNSInterfaceID mDNSInterface_Any; // Zero -extern const mDNSInterfaceID mDNSInterface_LocalOnly; // Special value -extern const mDNSInterfaceID mDNSInterface_Unicast; // Special value -extern const mDNSInterfaceID mDNSInterfaceMark; // Special value -extern const mDNSInterfaceID mDNSInterface_P2P; // Special value - -extern const mDNSIPPort DiscardPort; -extern const mDNSIPPort SSHPort; -extern const mDNSIPPort UnicastDNSPort; -extern const mDNSIPPort SSDPPort; -extern const mDNSIPPort IPSECPort; -extern const mDNSIPPort NSIPCPort; -extern const mDNSIPPort NATPMPAnnouncementPort; -extern const mDNSIPPort NATPMPPort; -extern const mDNSIPPort DNSEXTPort; -extern const mDNSIPPort MulticastDNSPort; -extern const mDNSIPPort LoopbackIPCPort; -extern const mDNSIPPort PrivateDNSPort; - -extern const OwnerOptData zeroOwner; - -extern const mDNSIPPort zeroIPPort; -extern const mDNSv4Addr zerov4Addr; -extern const mDNSv6Addr zerov6Addr; -extern const mDNSEthAddr zeroEthAddr; -extern const mDNSv4Addr onesIPv4Addr; -extern const mDNSv6Addr onesIPv6Addr; -extern const mDNSEthAddr onesEthAddr; -extern const mDNSAddr zeroAddr; - -extern const mDNSv4Addr AllDNSAdminGroup; -extern const mDNSv4Addr AllHosts_v4; -extern const mDNSv6Addr AllHosts_v6; -extern const mDNSv6Addr NDP_prefix; -extern const mDNSEthAddr AllHosts_v6_Eth; -extern const mDNSAddr AllDNSLinkGroup_v4; -extern const mDNSAddr AllDNSLinkGroup_v6; - -extern const mDNSOpaque16 zeroID; -extern const mDNSOpaque16 onesID; -extern const mDNSOpaque16 QueryFlags; -extern const mDNSOpaque16 uQueryFlags; -extern const mDNSOpaque16 ResponseFlags; -extern const mDNSOpaque16 UpdateReqFlags; -extern const mDNSOpaque16 UpdateRespFlags; - -extern const mDNSOpaque64 zeroOpaque64; - -extern mDNSBool StrictUnicastOrdering; -extern mDNSu8 NumUnicastDNSServers; - -#define localdomain (*(const domainname *)"\x5" "local") -#define DeviceInfoName (*(const domainname *)"\xC" "_device-info" "\x4" "_tcp") -#define SleepProxyServiceType (*(const domainname *)"\xC" "_sleep-proxy" "\x4" "_udp") - -// *************************************************************************** -#if 0 -#pragma mark - -#pragma mark - Inline functions -#endif - -#if (defined(_MSC_VER)) - #define mDNSinline static __inline -#elif ((__GNUC__ > 2) || ((__GNUC__ == 2) && (__GNUC_MINOR__ >= 9))) - #define mDNSinline static inline -#endif - -// If we're not doing inline functions, then this header needs to have the extern declarations -#if !defined(mDNSinline) -extern mDNSs32 NonZeroTime(mDNSs32 t); -extern mDNSu16 mDNSVal16(mDNSOpaque16 x); -extern mDNSOpaque16 mDNSOpaque16fromIntVal(mDNSu16 v); -#endif - -// If we're compiling the particular C file that instantiates our inlines, then we -// define "mDNSinline" (to empty string) so that we generate code in the following section -#if (!defined(mDNSinline) && mDNS_InstantiateInlines) -#define mDNSinline -#endif - -#ifdef mDNSinline - -mDNSinline mDNSs32 NonZeroTime(mDNSs32 t) { if (t) return(t); else return(1); } - -mDNSinline mDNSu16 mDNSVal16(mDNSOpaque16 x) { return((mDNSu16)((mDNSu16)x.b[0] << 8 | (mDNSu16)x.b[1])); } - -mDNSinline mDNSOpaque16 mDNSOpaque16fromIntVal(mDNSu16 v) - { - mDNSOpaque16 x; - x.b[0] = (mDNSu8)(v >> 8); - x.b[1] = (mDNSu8)(v & 0xFF); - return(x); - } - -#endif - -// *************************************************************************** -#if 0 -#pragma mark - -#pragma mark - Main Client Functions -#endif - -// Every client should call mDNS_Init, passing in storage for the mDNS object and the mDNS_PlatformSupport object. -// -// Clients that are only advertising services should use mDNS_Init_NoCache and mDNS_Init_ZeroCacheSize. -// Clients that plan to perform queries (mDNS_StartQuery, mDNS_StartBrowse, mDNS_StartResolveService, etc.) -// need to provide storage for the resource record cache, or the query calls will return 'mStatus_NoCache'. -// The rrcachestorage parameter is the address of memory for the resource record cache, and -// the rrcachesize parameter is the number of entries in the CacheRecord array passed in. -// (i.e. the size of the cache memory needs to be sizeof(CacheRecord) * rrcachesize). -// OS X 10.3 Panther uses an initial cache size of 64 entries, and then mDNSCore sends an -// mStatus_GrowCache message if it needs more. -// -// Most clients should use mDNS_Init_AdvertiseLocalAddresses. This causes mDNSCore to automatically -// create the correct address records for all the hosts interfaces. If you plan to advertise -// services being offered by the local machine, this is almost always what you want. -// There are two cases where you might use mDNS_Init_DontAdvertiseLocalAddresses: -// 1. A client-only device, that browses for services but doesn't advertise any of its own. -// 2. A proxy-registration service, that advertises services being offered by other machines, and takes -// the appropriate steps to manually create the correct address records for those other machines. -// In principle, a proxy-like registration service could manually create address records for its own machine too, -// but this would be pointless extra effort when using mDNS_Init_AdvertiseLocalAddresses does that for you. -// -// Note that a client-only device that wishes to prohibit multicast advertisements (e.g. from -// higher-layer API calls) must also set DivertMulticastAdvertisements in the mDNS structure and -// advertise local address(es) on a loopback interface. -// -// When mDNS has finished setting up the client's callback is called -// A client can also spin and poll the mDNSPlatformStatus field to see when it changes from mStatus_Waiting to mStatus_NoError -// -// Call mDNS_StartExit to tidy up before exiting -// Because exiting may be an asynchronous process (e.g. if unicast records need to be deregistered) -// client layer may choose to wait until mDNS_ExitNow() returns true before calling mDNS_FinalExit(). -// -// Call mDNS_Register with a completed AuthRecord object to register a resource record -// If the resource record type is kDNSRecordTypeUnique (or kDNSknownunique) then if a conflicting resource record is discovered, -// the resource record's mDNSRecordCallback will be called with error code mStatus_NameConflict. The callback should deregister -// the record, and may then try registering the record again after picking a new name (e.g. by automatically appending a number). -// Following deregistration, the RecordCallback will be called with result mStatus_MemFree to signal that it is safe to deallocate -// the record's storage (memory must be freed asynchronously to allow for goodbye packets and dynamic update deregistration). -// -// Call mDNS_StartQuery to initiate a query. mDNS will proceed to issue Multicast DNS query packets, and any time a response -// is received containing a record which matches the question, the DNSQuestion's mDNSAnswerCallback function will be called -// Call mDNS_StopQuery when no more answers are required -// -// Care should be taken on multi-threaded or interrupt-driven environments. -// The main mDNS routines call mDNSPlatformLock() on entry and mDNSPlatformUnlock() on exit; -// each platform layer needs to implement these appropriately for its respective platform. -// For example, if the support code on a particular platform implements timer callbacks at interrupt time, then -// mDNSPlatformLock/Unlock need to disable interrupts or do similar concurrency control to ensure that the mDNS -// code is not entered by an interrupt-time timer callback while in the middle of processing a client call. - -extern mStatus mDNS_Init (mDNS *const m, mDNS_PlatformSupport *const p, - CacheEntity *rrcachestorage, mDNSu32 rrcachesize, - mDNSBool AdvertiseLocalAddresses, - mDNSCallback *Callback, void *Context); -// See notes above on use of NoCache/ZeroCacheSize -#define mDNS_Init_NoCache mDNSNULL -#define mDNS_Init_ZeroCacheSize 0 -// See notes above on use of Advertise/DontAdvertiseLocalAddresses -#define mDNS_Init_AdvertiseLocalAddresses mDNStrue -#define mDNS_Init_DontAdvertiseLocalAddresses mDNSfalse -#define mDNS_Init_NoInitCallback mDNSNULL -#define mDNS_Init_NoInitCallbackContext mDNSNULL - -extern void mDNS_ConfigChanged(mDNS *const m); -extern void mDNS_GrowCache (mDNS *const m, CacheEntity *storage, mDNSu32 numrecords); -extern void mDNS_GrowAuth (mDNS *const m, AuthEntity *storage, mDNSu32 numrecords); -extern void mDNS_StartExit (mDNS *const m); -extern void mDNS_FinalExit (mDNS *const m); -#define mDNS_Close(m) do { mDNS_StartExit(m); mDNS_FinalExit(m); } while(0) -#define mDNS_ExitNow(m, now) ((now) - (m)->ShutdownTime >= 0 || (!(m)->ResourceRecords)) - -extern mDNSs32 mDNS_Execute (mDNS *const m); - -extern mStatus mDNS_Register (mDNS *const m, AuthRecord *const rr); -extern mStatus mDNS_Update (mDNS *const m, AuthRecord *const rr, mDNSu32 newttl, - const mDNSu16 newrdlength, RData *const newrdata, mDNSRecordUpdateCallback *Callback); -extern mStatus mDNS_Deregister(mDNS *const m, AuthRecord *const rr); - -extern mStatus mDNS_StartQuery(mDNS *const m, DNSQuestion *const question); -extern mStatus mDNS_StopQuery (mDNS *const m, DNSQuestion *const question); -extern mStatus mDNS_StopQueryWithRemoves(mDNS *const m, DNSQuestion *const question); -extern mStatus mDNS_Reconfirm (mDNS *const m, CacheRecord *const cacherr); -extern mStatus mDNS_ReconfirmByValue(mDNS *const m, ResourceRecord *const rr); -extern void mDNS_PurgeCacheResourceRecord(mDNS *const m, CacheRecord *rr); -extern mDNSs32 mDNS_TimeNow(const mDNS *const m); - -extern mStatus mDNS_StartNATOperation(mDNS *const m, NATTraversalInfo *traversal); -extern mStatus mDNS_StopNATOperation(mDNS *const m, NATTraversalInfo *traversal); -extern mStatus mDNS_StopNATOperation_internal(mDNS *m, NATTraversalInfo *traversal); - -extern DomainAuthInfo *GetAuthInfoForName(mDNS *m, const domainname *const name); - -extern void mDNS_UpdateAllowSleep(mDNS *const m); - -// *************************************************************************** -#if 0 -#pragma mark - -#pragma mark - Platform support functions that are accessible to the client layer too -#endif - -extern mDNSs32 mDNSPlatformOneSecond; - -// *************************************************************************** -#if 0 -#pragma mark - -#pragma mark - General utility and helper functions -#endif - -// mDNS_Dereg_normal is used for most calls to mDNS_Deregister_internal -// mDNS_Dereg_rapid is used to send one goodbye instead of three, when we want the memory available for reuse sooner -// mDNS_Dereg_conflict is used to indicate that this record is being forcibly deregistered because of a conflict -// mDNS_Dereg_repeat is used when cleaning up, for records that may have already been forcibly deregistered -typedef enum { mDNS_Dereg_normal, mDNS_Dereg_rapid, mDNS_Dereg_conflict, mDNS_Dereg_repeat } mDNS_Dereg_type; - -// mDNS_RegisterService is a single call to register the set of resource records associated with a given named service. -// -// mDNS_StartResolveService is single call which is equivalent to multiple calls to mDNS_StartQuery, -// to find the IP address, port number, and demultiplexing information for a given named service. -// As with mDNS_StartQuery, it executes asynchronously, and calls the ServiceInfoQueryCallback when the answer is -// found. After the service is resolved, the client should call mDNS_StopResolveService to complete the transaction. -// The client can also call mDNS_StopResolveService at any time to abort the transaction. -// -// mDNS_AddRecordToService adds an additional record to a Service Record Set. This record may be deregistered -// via mDNS_RemoveRecordFromService, or by deregistering the service. mDNS_RemoveRecordFromService is passed a -// callback to free the memory associated with the extra RR when it is safe to do so. The ExtraResourceRecord -// object can be found in the record's context pointer. - -// mDNS_GetBrowseDomains is a special case of the mDNS_StartQuery call, where the resulting answers -// are a list of PTR records indicating (in the rdata) domains that are recommended for browsing. -// After getting the list of domains to browse, call mDNS_StopQuery to end the search. -// mDNS_GetDefaultBrowseDomain returns the name of the domain that should be highlighted by default. -// -// mDNS_GetRegistrationDomains and mDNS_GetDefaultRegistrationDomain are the equivalent calls to get the list -// of one or more domains that should be offered to the user as choices for where they may register their service, -// and the default domain in which to register in the case where the user has made no selection. - -extern void mDNS_SetupResourceRecord(AuthRecord *rr, RData *RDataStorage, mDNSInterfaceID InterfaceID, - mDNSu16 rrtype, mDNSu32 ttl, mDNSu8 RecordType, AuthRecType artype, mDNSRecordCallback Callback, void *Context); - -// mDNS_RegisterService() flags parameter bit definitions -enum - { - regFlagIncludeP2P = 0x1, // include P2P interfaces when using mDNSInterface_Any - regFlagKnownUnique = 0x2 // client guarantees that SRV and TXT record names are unique - }; - -extern mStatus mDNS_RegisterService (mDNS *const m, ServiceRecordSet *sr, - const domainlabel *const name, const domainname *const type, const domainname *const domain, - const domainname *const host, mDNSIPPort port, const mDNSu8 txtinfo[], mDNSu16 txtlen, - AuthRecord *SubTypes, mDNSu32 NumSubTypes, - mDNSInterfaceID InterfaceID, mDNSServiceCallback Callback, void *Context, mDNSu32 flags); -extern mStatus mDNS_AddRecordToService(mDNS *const m, ServiceRecordSet *sr, ExtraResourceRecord *extra, RData *rdata, mDNSu32 ttl, mDNSu32 includeP2P); -extern mStatus mDNS_RemoveRecordFromService(mDNS *const m, ServiceRecordSet *sr, ExtraResourceRecord *extra, mDNSRecordCallback MemFreeCallback, void *Context); -extern mStatus mDNS_RenameAndReregisterService(mDNS *const m, ServiceRecordSet *const sr, const domainlabel *newname); -extern mStatus mDNS_DeregisterService_drt(mDNS *const m, ServiceRecordSet *sr, mDNS_Dereg_type drt); -#define mDNS_DeregisterService(M,S) mDNS_DeregisterService_drt((M), (S), mDNS_Dereg_normal) - -extern mStatus mDNS_RegisterNoSuchService(mDNS *const m, AuthRecord *const rr, - const domainlabel *const name, const domainname *const type, const domainname *const domain, - const domainname *const host, - const mDNSInterfaceID InterfaceID, mDNSRecordCallback Callback, void *Context, mDNSBool includeP2P); -#define mDNS_DeregisterNoSuchService mDNS_Deregister - -extern void mDNS_SetupQuestion(DNSQuestion *const q, const mDNSInterfaceID InterfaceID, const domainname *const name, - const mDNSu16 qtype, mDNSQuestionCallback *const callback, void *const context); - -extern mStatus mDNS_StartBrowse(mDNS *const m, DNSQuestion *const question, - const domainname *const srv, const domainname *const domain, - const mDNSInterfaceID InterfaceID, mDNSBool ForceMCast, mDNSQuestionCallback *Callback, void *Context); -#define mDNS_StopBrowse mDNS_StopQuery - -extern mStatus mDNS_StartResolveService(mDNS *const m, ServiceInfoQuery *query, ServiceInfo *info, mDNSServiceInfoQueryCallback *Callback, void *Context); -extern void mDNS_StopResolveService (mDNS *const m, ServiceInfoQuery *query); - -typedef enum - { - mDNS_DomainTypeBrowse = 0, - mDNS_DomainTypeBrowseDefault = 1, - mDNS_DomainTypeBrowseAutomatic = 2, - mDNS_DomainTypeRegistration = 3, - mDNS_DomainTypeRegistrationDefault = 4, - - mDNS_DomainTypeMax = 4 - } mDNS_DomainType; - -extern const char *const mDNS_DomainTypeNames[]; - -extern mStatus mDNS_GetDomains(mDNS *const m, DNSQuestion *const question, mDNS_DomainType DomainType, const domainname *dom, - const mDNSInterfaceID InterfaceID, mDNSQuestionCallback *Callback, void *Context); -#define mDNS_StopGetDomains mDNS_StopQuery -extern mStatus mDNS_AdvertiseDomains(mDNS *const m, AuthRecord *rr, mDNS_DomainType DomainType, const mDNSInterfaceID InterfaceID, char *domname); -#define mDNS_StopAdvertiseDomains mDNS_Deregister - -extern mDNSOpaque16 mDNS_NewMessageID(mDNS *const m); -extern mDNSBool mDNS_AddressIsLocalSubnet(mDNS *const m, const mDNSInterfaceID InterfaceID, const mDNSAddr *addr); - -extern DNSServer *GetServerForName(mDNS *m, const domainname *name, mDNSInterfaceID InterfaceID); -extern DNSServer *GetServerForQuestion(mDNS *m, DNSQuestion *question); -extern mDNSu32 SetValidDNSServers(mDNS *m, DNSQuestion *question); - -// *************************************************************************** -#if 0 -#pragma mark - -#pragma mark - DNS name utility functions -#endif - -// In order to expose the full capabilities of the DNS protocol (which allows any arbitrary eight-bit values -// in domain name labels, including unlikely characters like ascii nulls and even dots) all the mDNS APIs -// work with DNS's native length-prefixed strings. For convenience in C, the following utility functions -// are provided for converting between C's null-terminated strings and DNS's length-prefixed strings. - -// Assignment -// A simple C structure assignment of a domainname can cause a protection fault by accessing unmapped memory, -// because that object is defined to be 256 bytes long, but not all domainname objects are truly the full size. -// This macro uses mDNSPlatformMemCopy() to make sure it only touches the actual bytes that are valid. -#define AssignDomainName(DST, SRC) do { mDNSu16 len__ = DomainNameLength((SRC)); \ - if (len__ <= MAX_DOMAIN_NAME) mDNSPlatformMemCopy((DST)->c, (SRC)->c, len__); else (DST)->c[0] = 0; } while(0) - -// Comparison functions -#define SameDomainLabelCS(A,B) ((A)[0] == (B)[0] && mDNSPlatformMemSame((A)+1, (B)+1, (A)[0])) -extern mDNSBool SameDomainLabel(const mDNSu8 *a, const mDNSu8 *b); -extern mDNSBool SameDomainName(const domainname *const d1, const domainname *const d2); -extern mDNSBool SameDomainNameCS(const domainname *const d1, const domainname *const d2); -typedef mDNSBool DomainNameComparisonFn(const domainname *const d1, const domainname *const d2); -extern mDNSBool IsLocalDomain(const domainname *d); // returns true for domains that by default should be looked up using link-local multicast - -#define StripFirstLabel(X) ((const domainname *)&(X)->c[(X)->c[0] ? 1 + (X)->c[0] : 0]) - -#define FirstLabel(X) ((const domainlabel *)(X)) -#define SecondLabel(X) ((const domainlabel *)StripFirstLabel(X)) -#define ThirdLabel(X) ((const domainlabel *)StripFirstLabel(StripFirstLabel(X))) - -extern const mDNSu8 *LastLabel(const domainname *d); - -// Get total length of domain name, in native DNS format, including terminal root label -// (e.g. length of "com." is 5 (length byte, three data bytes, final zero) -extern mDNSu16 DomainNameLengthLimit(const domainname *const name, const mDNSu8 *limit); -#define DomainNameLength(name) DomainNameLengthLimit((name), (name)->c + MAX_DOMAIN_NAME) - -// Append functions to append one or more labels to an existing native format domain name: -// AppendLiteralLabelString adds a single label from a literal C string, with no escape character interpretation. -// AppendDNSNameString adds zero or more labels from a C string using conventional DNS dots-and-escaping interpretation -// AppendDomainLabel adds a single label from a native format domainlabel -// AppendDomainName adds zero or more labels from a native format domainname -extern mDNSu8 *AppendLiteralLabelString(domainname *const name, const char *cstr); -extern mDNSu8 *AppendDNSNameString (domainname *const name, const char *cstr); -extern mDNSu8 *AppendDomainLabel (domainname *const name, const domainlabel *const label); -extern mDNSu8 *AppendDomainName (domainname *const name, const domainname *const append); - -// Convert from null-terminated string to native DNS format: -// The DomainLabel form makes a single label from a literal C string, with no escape character interpretation. -// The DomainName form makes native format domain name from a C string using conventional DNS interpretation: -// dots separate labels, and within each label, '\.' represents a literal dot, '\\' represents a literal -// backslash and backslash with three decimal digits (e.g. \000) represents an arbitrary byte value. -extern mDNSBool MakeDomainLabelFromLiteralString(domainlabel *const label, const char *cstr); -extern mDNSu8 *MakeDomainNameFromDNSNameString (domainname *const name, const char *cstr); - -// Convert native format domainlabel or domainname back to C string format -// IMPORTANT: -// When using ConvertDomainLabelToCString, the target buffer must be MAX_ESCAPED_DOMAIN_LABEL (254) bytes long -// to guarantee there will be no buffer overrun. It is only safe to use a buffer shorter than this in rare cases -// where the label is known to be constrained somehow (for example, if the label is known to be either "_tcp" or "_udp"). -// Similarly, when using ConvertDomainNameToCString, the target buffer must be MAX_ESCAPED_DOMAIN_NAME (1009) bytes long. -// See definitions of MAX_ESCAPED_DOMAIN_LABEL and MAX_ESCAPED_DOMAIN_NAME for more detailed explanation. -extern char *ConvertDomainLabelToCString_withescape(const domainlabel *const name, char *cstr, char esc); -#define ConvertDomainLabelToCString_unescaped(D,C) ConvertDomainLabelToCString_withescape((D), (C), 0) -#define ConvertDomainLabelToCString(D,C) ConvertDomainLabelToCString_withescape((D), (C), '\\') -extern char *ConvertDomainNameToCString_withescape(const domainname *const name, char *cstr, char esc); -#define ConvertDomainNameToCString_unescaped(D,C) ConvertDomainNameToCString_withescape((D), (C), 0) -#define ConvertDomainNameToCString(D,C) ConvertDomainNameToCString_withescape((D), (C), '\\') - -extern void ConvertUTF8PstringToRFC1034HostLabel(const mDNSu8 UTF8Name[], domainlabel *const hostlabel); - -extern mDNSu8 *ConstructServiceName(domainname *const fqdn, const domainlabel *name, const domainname *type, const domainname *const domain); -extern mDNSBool DeconstructServiceName(const domainname *const fqdn, domainlabel *const name, domainname *const type, domainname *const domain); - -// Note: Some old functions have been replaced by more sensibly-named versions. -// You can uncomment the hash-defines below if you don't want to have to change your source code right away. -// When updating your code, note that (unlike the old versions) *all* the new routines take the target object -// as their first parameter. -//#define ConvertCStringToDomainName(SRC,DST) MakeDomainNameFromDNSNameString((DST),(SRC)) -//#define ConvertCStringToDomainLabel(SRC,DST) MakeDomainLabelFromLiteralString((DST),(SRC)) -//#define AppendStringLabelToName(DST,SRC) AppendLiteralLabelString((DST),(SRC)) -//#define AppendStringNameToName(DST,SRC) AppendDNSNameString((DST),(SRC)) -//#define AppendDomainLabelToName(DST,SRC) AppendDomainLabel((DST),(SRC)) -//#define AppendDomainNameToName(DST,SRC) AppendDomainName((DST),(SRC)) - -// *************************************************************************** -#if 0 -#pragma mark - -#pragma mark - Other utility functions and macros -#endif - -// mDNS_vsnprintf/snprintf return the number of characters written, excluding the final terminating null. -// The output is always null-terminated: for example, if the output turns out to be exactly buflen long, -// then the output will be truncated by one character to allow space for the terminating null. -// Unlike standard C vsnprintf/snprintf, they return the number of characters *actually* written, -// not the number of characters that *would* have been printed were buflen unlimited. -extern mDNSu32 mDNS_vsnprintf(char *sbuffer, mDNSu32 buflen, const char *fmt, va_list arg); -extern mDNSu32 mDNS_snprintf(char *sbuffer, mDNSu32 buflen, const char *fmt, ...) IS_A_PRINTF_STYLE_FUNCTION(3,4); -extern mDNSu32 NumCacheRecordsForInterfaceID(const mDNS *const m, mDNSInterfaceID id); -extern char *DNSTypeName(mDNSu16 rrtype); -extern char *GetRRDisplayString_rdb(const ResourceRecord *const rr, const RDataBody *const rd1, char *const buffer); -#define RRDisplayString(m, rr) GetRRDisplayString_rdb(rr, &(rr)->rdata->u, (m)->MsgBuffer) -#define ARDisplayString(m, rr) GetRRDisplayString_rdb(&(rr)->resrec, &(rr)->resrec.rdata->u, (m)->MsgBuffer) -#define CRDisplayString(m, rr) GetRRDisplayString_rdb(&(rr)->resrec, &(rr)->resrec.rdata->u, (m)->MsgBuffer) -extern mDNSBool mDNSSameAddress(const mDNSAddr *ip1, const mDNSAddr *ip2); -extern void IncrementLabelSuffix(domainlabel *name, mDNSBool RichText); -extern mDNSBool mDNSv4AddrIsRFC1918(mDNSv4Addr *addr); // returns true for RFC1918 private addresses -#define mDNSAddrIsRFC1918(X) ((X)->type == mDNSAddrType_IPv4 && mDNSv4AddrIsRFC1918(&(X)->ip.v4)) - -#define mDNSSameIPPort(A,B) ((A).NotAnInteger == (B).NotAnInteger) -#define mDNSSameOpaque16(A,B) ((A).NotAnInteger == (B).NotAnInteger) -#define mDNSSameOpaque32(A,B) ((A).NotAnInteger == (B).NotAnInteger) -#define mDNSSameOpaque64(A,B) ((A)->l[0] == (B)->l[0] && (A)->l[1] == (B)->l[1]) - -#define mDNSSameIPv4Address(A,B) ((A).NotAnInteger == (B).NotAnInteger) -#define mDNSSameIPv6Address(A,B) ((A).l[0] == (B).l[0] && (A).l[1] == (B).l[1] && (A).l[2] == (B).l[2] && (A).l[3] == (B).l[3]) -#define mDNSSameEthAddress(A,B) ((A)->w[0] == (B)->w[0] && (A)->w[1] == (B)->w[1] && (A)->w[2] == (B)->w[2]) - -#define mDNSIPPortIsZero(A) ((A).NotAnInteger == 0) -#define mDNSOpaque16IsZero(A) ((A).NotAnInteger == 0) -#define mDNSOpaque64IsZero(A) (((A)->l[0] | (A)->l[1] ) == 0) -#define mDNSIPv4AddressIsZero(A) ((A).NotAnInteger == 0) -#define mDNSIPv6AddressIsZero(A) (((A).l[0] | (A).l[1] | (A).l[2] | (A).l[3]) == 0) -#define mDNSEthAddressIsZero(A) (((A).w[0] | (A).w[1] | (A).w[2] ) == 0) - -#define mDNSIPv4AddressIsOnes(A) ((A).NotAnInteger == 0xFFFFFFFF) -#define mDNSIPv6AddressIsOnes(A) (((A).l[0] & (A).l[1] & (A).l[2] & (A).l[3]) == 0xFFFFFFFF) - -#define mDNSAddressIsAllDNSLinkGroup(X) ( \ - ((X)->type == mDNSAddrType_IPv4 && mDNSSameIPv4Address((X)->ip.v4, AllDNSLinkGroup_v4.ip.v4)) || \ - ((X)->type == mDNSAddrType_IPv6 && mDNSSameIPv6Address((X)->ip.v6, AllDNSLinkGroup_v6.ip.v6)) ) - -#define mDNSAddressIsZero(X) ( \ - ((X)->type == mDNSAddrType_IPv4 && mDNSIPv4AddressIsZero((X)->ip.v4)) || \ - ((X)->type == mDNSAddrType_IPv6 && mDNSIPv6AddressIsZero((X)->ip.v6)) ) - -#define mDNSAddressIsValidNonZero(X) ( \ - ((X)->type == mDNSAddrType_IPv4 && !mDNSIPv4AddressIsZero((X)->ip.v4)) || \ - ((X)->type == mDNSAddrType_IPv6 && !mDNSIPv6AddressIsZero((X)->ip.v6)) ) - -#define mDNSAddressIsOnes(X) ( \ - ((X)->type == mDNSAddrType_IPv4 && mDNSIPv4AddressIsOnes((X)->ip.v4)) || \ - ((X)->type == mDNSAddrType_IPv6 && mDNSIPv6AddressIsOnes((X)->ip.v6)) ) - -#define mDNSAddressIsValid(X) ( \ - ((X)->type == mDNSAddrType_IPv4) ? !(mDNSIPv4AddressIsZero((X)->ip.v4) || mDNSIPv4AddressIsOnes((X)->ip.v4)) : \ - ((X)->type == mDNSAddrType_IPv6) ? !(mDNSIPv6AddressIsZero((X)->ip.v6) || mDNSIPv6AddressIsOnes((X)->ip.v6)) : mDNSfalse) - -#define mDNSv4AddressIsLinkLocal(X) ((X)->b[0] == 169 && (X)->b[1] == 254) -#define mDNSv6AddressIsLinkLocal(X) ((X)->b[0] == 0xFE && ((X)->b[1] & 0xC0) == 0x80) - -#define mDNSAddressIsLinkLocal(X) ( \ - ((X)->type == mDNSAddrType_IPv4) ? mDNSv4AddressIsLinkLocal(&(X)->ip.v4) : \ - ((X)->type == mDNSAddrType_IPv6) ? mDNSv6AddressIsLinkLocal(&(X)->ip.v6) : mDNSfalse) - -#define mDNSv4AddressIsLoopback(X) ((X)->b[0] == 127 && (X)->b[1] == 0 && (X)->b[2] == 0 && (X)->b[3] == 1) -#define mDNSv6AddressIsLoopback(X) ((((X)->l[0] | (X)->l[1] | (X)->l[2]) == 0) && ((X)->b[12] == 0 && (X)->b[13] == 0 && (X)->b[14] == 0 && (X)->b[15] == 1)) - -// *************************************************************************** -#if 0 -#pragma mark - -#pragma mark - Authentication Support -#endif - -// Unicast DNS and Dynamic Update specific Client Calls -// -// mDNS_SetSecretForDomain tells the core to authenticate (via TSIG with an HMAC_MD5 hash of the shared secret) -// when dynamically updating a given zone (and its subdomains). The key used in authentication must be in -// domain name format. The shared secret must be a null-terminated base64 encoded string. A minimum size of -// 16 bytes (128 bits) is recommended for an MD5 hash as per RFC 2485. -// Calling this routine multiple times for a zone replaces previously entered values. Call with a NULL key -// to disable authentication for the zone. A non-NULL autoTunnelPrefix means this is an AutoTunnel domain, -// and the value is prepended to the IPSec identifier (used for key lookup) - -extern mStatus mDNS_SetSecretForDomain(mDNS *m, DomainAuthInfo *info, - const domainname *domain, const domainname *keyname, const char *b64keydata, const domainname *hostname, mDNSIPPort *port, const char *autoTunnelPrefix); - -extern void RecreateNATMappings(mDNS *const m); - -// Hostname/Unicast Interface Configuration - -// All hostnames advertised point to one IPv4 address and/or one IPv6 address, set via SetPrimaryInterfaceInfo. Invoking this routine -// updates all existing hostnames to point to the new address. - -// A hostname is added via AddDynDNSHostName, which points to the primary interface's v4 and/or v6 addresss - -// The status callback is invoked to convey success or failure codes - the callback should not modify the AuthRecord or free memory. -// Added hostnames may be removed (deregistered) via mDNS_RemoveDynDNSHostName. - -// Host domains added prior to specification of the primary interface address and computer name will be deferred until -// these values are initialized. - -// DNS servers used to resolve unicast queries are specified by mDNS_AddDNSServer. -// For "split" DNS configurations, in which queries for different domains are sent to different servers (e.g. VPN and external), -// a domain may be associated with a DNS server. For standard configurations, specify the root label (".") or NULL. - -extern void mDNS_AddDynDNSHostName(mDNS *m, const domainname *fqdn, mDNSRecordCallback *StatusCallback, const void *StatusContext); -extern void mDNS_RemoveDynDNSHostName(mDNS *m, const domainname *fqdn); -extern void mDNS_SetPrimaryInterfaceInfo(mDNS *m, const mDNSAddr *v4addr, const mDNSAddr *v6addr, const mDNSAddr *router); -extern DNSServer *mDNS_AddDNSServer(mDNS *const m, const domainname *d, const mDNSInterfaceID interface, const mDNSAddr *addr, const mDNSIPPort port, mDNSBool scoped, mDNSu32 timeout, mDNSBool cellIntf); -extern void PenalizeDNSServer(mDNS *const m, DNSQuestion *q); -extern void mDNS_AddSearchDomain(const domainname *const domain, mDNSInterfaceID InterfaceID); - -extern McastResolver *mDNS_AddMcastResolver(mDNS *const m, const domainname *d, const mDNSInterfaceID interface, mDNSu32 timeout); - -// We use ((void *)0) here instead of mDNSNULL to avoid compile warnings on gcc 4.2 -#define mDNS_AddSearchDomain_CString(X, I) \ - do { domainname d__; if (((X) != (void*)0) && MakeDomainNameFromDNSNameString(&d__, (X)) && d__.c[0]) mDNS_AddSearchDomain(&d__, I); } while(0) - -// Routines called by the core, exported by DNSDigest.c - -// Convert an arbitrary base64 encoded key key into an HMAC key (stored in AuthInfo struct) -extern mDNSs32 DNSDigest_ConstructHMACKeyfromBase64(DomainAuthInfo *info, const char *b64key); - -// sign a DNS message. The message must be complete, with all values in network byte order. end points to the end -// of the message, and is modified by this routine. numAdditionals is a pointer to the number of additional -// records in HOST byte order, which is incremented upon successful completion of this routine. The function returns -// the new end pointer on success, and NULL on failure. -extern void DNSDigest_SignMessage(DNSMessage *msg, mDNSu8 **end, DomainAuthInfo *info, mDNSu16 tcode); - -#define SwapDNSHeaderBytes(M) do { \ - (M)->h.numQuestions = (mDNSu16)((mDNSu8 *)&(M)->h.numQuestions )[0] << 8 | ((mDNSu8 *)&(M)->h.numQuestions )[1]; \ - (M)->h.numAnswers = (mDNSu16)((mDNSu8 *)&(M)->h.numAnswers )[0] << 8 | ((mDNSu8 *)&(M)->h.numAnswers )[1]; \ - (M)->h.numAuthorities = (mDNSu16)((mDNSu8 *)&(M)->h.numAuthorities)[0] << 8 | ((mDNSu8 *)&(M)->h.numAuthorities)[1]; \ - (M)->h.numAdditionals = (mDNSu16)((mDNSu8 *)&(M)->h.numAdditionals)[0] << 8 | ((mDNSu8 *)&(M)->h.numAdditionals)[1]; \ - } while (0) - -#define DNSDigest_SignMessageHostByteOrder(M,E,INFO) \ - do { SwapDNSHeaderBytes(M); DNSDigest_SignMessage((M), (E), (INFO), 0); SwapDNSHeaderBytes(M); } while (0) - -// verify a DNS message. The message must be complete, with all values in network byte order. end points to the -// end of the record. tsig is a pointer to the resource record that contains the TSIG OPT record. info is -// the matching key to use for verifying the message. This function expects that the additionals member -// of the DNS message header has already had one subtracted from it. -extern mDNSBool DNSDigest_VerifyMessage(DNSMessage *msg, mDNSu8 *end, LargeCacheRecord *tsig, DomainAuthInfo *info, mDNSu16 *rcode, mDNSu16 *tcode); - -// *************************************************************************** -#if 0 -#pragma mark - -#pragma mark - PlatformSupport interface -#endif - -// This section defines the interface to the Platform Support layer. -// Normal client code should not use any of types defined here, or directly call any of the functions defined here. -// The definitions are placed here because sometimes clients do use these calls indirectly, via other supported client operations. -// For example, AssignDomainName is a macro defined using mDNSPlatformMemCopy() - -// Every platform support module must provide the following functions. -// mDNSPlatformInit() typically opens a communication endpoint, and starts listening for mDNS packets. -// When Setup is complete, the platform support layer calls mDNSCoreInitComplete(). -// mDNSPlatformSendUDP() sends one UDP packet -// When a packet is received, the PlatformSupport code calls mDNSCoreReceive() -// mDNSPlatformClose() tidies up on exit -// -// Note: mDNSPlatformMemAllocate/mDNSPlatformMemFree are only required for handling oversized resource records and unicast DNS. -// If your target platform has a well-defined specialized application, and you know that all the records it uses -// are InlineCacheRDSize or less, then you can just make a simple mDNSPlatformMemAllocate() stub that always returns -// NULL. InlineCacheRDSize is a compile-time constant, which is set by default to 68. If you need to handle records -// a little larger than this and you don't want to have to implement run-time allocation and freeing, then you -// can raise the value of this constant to a suitable value (at the expense of increased memory usage). -// -// USE CAUTION WHEN CALLING mDNSPlatformRawTime: The m->timenow_adjust correction factor needs to be added -// Generally speaking: -// Code that's protected by the main mDNS lock should just use the m->timenow value -// Code outside the main mDNS lock should use mDNS_TimeNow(m) to get properly adjusted time -// In certain cases there may be reasons why it's necessary to get the time without taking the lock first -// (e.g. inside the routines that are doing the locking and unlocking, where a call to get the lock would result in a -// recursive loop); in these cases use mDNS_TimeNow_NoLock(m) to get mDNSPlatformRawTime with the proper correction factor added. -// -// mDNSPlatformUTC returns the time, in seconds, since Jan 1st 1970 UTC and is required for generating TSIG records - -extern mStatus mDNSPlatformInit (mDNS *const m); -extern void mDNSPlatformClose (mDNS *const m); -extern mStatus mDNSPlatformSendUDP(const mDNS *const m, const void *const msg, const mDNSu8 *const end, -mDNSInterfaceID InterfaceID, UDPSocket *src, const mDNSAddr *dst, mDNSIPPort dstport); - -extern void mDNSPlatformLock (const mDNS *const m); -extern void mDNSPlatformUnlock (const mDNS *const m); - -extern void mDNSPlatformStrCopy ( void *dst, const void *src); -extern mDNSu32 mDNSPlatformStrLen ( const void *src); -extern void mDNSPlatformMemCopy ( void *dst, const void *src, mDNSu32 len); -extern mDNSBool mDNSPlatformMemSame (const void *dst, const void *src, mDNSu32 len); -extern void mDNSPlatformMemZero ( void *dst, mDNSu32 len); -#if APPLE_OSX_mDNSResponder && MACOSX_MDNS_MALLOC_DEBUGGING -#define mDNSPlatformMemAllocate(X) mallocL(#X, X) -#else -extern void * mDNSPlatformMemAllocate (mDNSu32 len); -#endif -extern void mDNSPlatformMemFree (void *mem); - -// If the platform doesn't have a strong PRNG, we define a naive multiply-and-add based on a seed -// from the platform layer. Long-term, we should embed an arc4 implementation, but the strength -// will still depend on the randomness of the seed. -#if !defined(_PLATFORM_HAS_STRONG_PRNG_) && (_BUILDING_XCODE_PROJECT_ || defined(_WIN32)) -#define _PLATFORM_HAS_STRONG_PRNG_ 1 -#endif -#if _PLATFORM_HAS_STRONG_PRNG_ -extern mDNSu32 mDNSPlatformRandomNumber(void); -#else -extern mDNSu32 mDNSPlatformRandomSeed (void); -#endif // _PLATFORM_HAS_STRONG_PRNG_ - -extern mStatus mDNSPlatformTimeInit (void); -extern mDNSs32 mDNSPlatformRawTime (void); -extern mDNSs32 mDNSPlatformUTC (void); -#define mDNS_TimeNow_NoLock(m) (mDNSPlatformRawTime() + (m)->timenow_adjust) - -#if MDNS_DEBUGMSGS -extern void mDNSPlatformWriteDebugMsg(const char *msg); -#endif -extern void mDNSPlatformWriteLogMsg(const char *ident, const char *msg, mDNSLogLevel_t loglevel); - -#if APPLE_OSX_mDNSResponder -// Utility function for ASL logging -mDNSexport void mDNSASLLog(uuid_t *uuid, const char *subdomain, const char *result, const char *signature, const char *fmt, ...); -#endif - -// Platform support modules should provide the following functions to map between opaque interface IDs -// and interface indexes in order to support the DNS-SD API. If your target platform does not support -// multiple interfaces and/or does not support the DNS-SD API, these functions can be empty. -extern mDNSInterfaceID mDNSPlatformInterfaceIDfromInterfaceIndex(mDNS *const m, mDNSu32 ifindex); -extern mDNSu32 mDNSPlatformInterfaceIndexfromInterfaceID(mDNS *const m, mDNSInterfaceID id, mDNSBool suppressNetworkChange); - -// Every platform support module must provide the following functions if it is to support unicast DNS -// and Dynamic Update. -// All TCP socket operations implemented by the platform layer MUST NOT BLOCK. -// mDNSPlatformTCPConnect initiates a TCP connection with a peer, adding the socket descriptor to the -// main event loop. The return value indicates whether the connection succeeded, failed, or is pending -// (i.e. the call would block.) On return, the descriptor parameter is set to point to the connected socket. -// The TCPConnectionCallback is subsequently invoked when the connection -// completes (in which case the ConnectionEstablished parameter is true), or data is available for -// reading on the socket (indicated by the ConnectionEstablished parameter being false.) If the connection -// asynchronously fails, the TCPConnectionCallback should be invoked as usual, with the error being -// returned in subsequent calls to PlatformReadTCP or PlatformWriteTCP. (This allows for platforms -// with limited asynchronous error detection capabilities.) PlatformReadTCP and PlatformWriteTCP must -// return the number of bytes read/written, 0 if the call would block, and -1 if an error. PlatformReadTCP -// should set the closed argument if the socket has been closed. -// PlatformTCPCloseConnection must close the connection to the peer and remove the descriptor from the -// event loop. CloseConnectin may be called at any time, including in a ConnectionCallback. - -typedef enum - { - kTCPSocketFlags_Zero = 0, - kTCPSocketFlags_UseTLS = (1 << 0) - } TCPSocketFlags; - -typedef void (*TCPConnectionCallback)(TCPSocket *sock, void *context, mDNSBool ConnectionEstablished, mStatus err); -extern TCPSocket *mDNSPlatformTCPSocket(mDNS *const m, TCPSocketFlags flags, mDNSIPPort *port); // creates a TCP socket -extern TCPSocket *mDNSPlatformTCPAccept(TCPSocketFlags flags, int sd); -extern int mDNSPlatformTCPGetFD(TCPSocket *sock); -extern mStatus mDNSPlatformTCPConnect(TCPSocket *sock, const mDNSAddr *dst, mDNSOpaque16 dstport, domainname *hostname, - mDNSInterfaceID InterfaceID, TCPConnectionCallback callback, void *context); -extern void mDNSPlatformTCPCloseConnection(TCPSocket *sock); -extern long mDNSPlatformReadTCP(TCPSocket *sock, void *buf, unsigned long buflen, mDNSBool *closed); -extern long mDNSPlatformWriteTCP(TCPSocket *sock, const char *msg, unsigned long len); -extern UDPSocket *mDNSPlatformUDPSocket(mDNS *const m, const mDNSIPPort requestedport); -extern void mDNSPlatformUDPClose(UDPSocket *sock); -extern void mDNSPlatformReceiveBPF_fd(mDNS *const m, int fd); -extern void mDNSPlatformUpdateProxyList(mDNS *const m, const mDNSInterfaceID InterfaceID); -extern void mDNSPlatformSendRawPacket(const void *const msg, const mDNSu8 *const end, mDNSInterfaceID InterfaceID); -extern void mDNSPlatformSetLocalAddressCacheEntry(mDNS *const m, const mDNSAddr *const tpa, const mDNSEthAddr *const tha, mDNSInterfaceID InterfaceID); -extern void mDNSPlatformSourceAddrForDest(mDNSAddr *const src, const mDNSAddr *const dst); - -// mDNSPlatformTLSSetupCerts/mDNSPlatformTLSTearDownCerts used by dnsextd -extern mStatus mDNSPlatformTLSSetupCerts(void); -extern void mDNSPlatformTLSTearDownCerts(void); - -// Platforms that support unicast browsing and dynamic update registration for clients who do not specify a domain -// in browse/registration calls must implement these routines to get the "default" browse/registration list. - -extern void mDNSPlatformSetDNSConfig(mDNS *const m, mDNSBool setservers, mDNSBool setsearch, domainname *const fqdn, DNameListElem **RegDomains, DNameListElem **BrowseDomains); -extern mStatus mDNSPlatformGetPrimaryInterface(mDNS *const m, mDNSAddr *v4, mDNSAddr *v6, mDNSAddr *router); -extern void mDNSPlatformDynDNSHostNameStatusChanged(const domainname *const dname, const mStatus status); - -extern void mDNSPlatformSetAllowSleep(mDNS *const m, mDNSBool allowSleep, const char *reason); -extern void mDNSPlatformSendWakeupPacket(mDNS *const m, mDNSInterfaceID InterfaceID, char *EthAddr, char *IPAddr, int iteration); -extern mDNSBool mDNSPlatformValidRecordForInterface(AuthRecord *rr, const NetworkInterfaceInfo *intf); - -#ifdef _LEGACY_NAT_TRAVERSAL_ -// Support for legacy NAT traversal protocols, implemented by the platform layer and callable by the core. -extern void LNT_SendDiscoveryMsg(mDNS *m); -extern void LNT_ConfigureRouterInfo(mDNS *m, const mDNSInterfaceID InterfaceID, const mDNSu8 *const data, const mDNSu16 len); -extern mStatus LNT_GetExternalAddress(mDNS *m); -extern mStatus LNT_MapPort(mDNS *m, NATTraversalInfo *const n); -extern mStatus LNT_UnmapPort(mDNS *m, NATTraversalInfo *const n); -extern void LNT_ClearState(mDNS *const m); -#endif // _LEGACY_NAT_TRAVERSAL_ - -// The core mDNS code provides these functions, for the platform support code to call at appropriate times -// -// mDNS_SetFQDN() is called once on startup (typically from mDNSPlatformInit()) -// and then again on each subsequent change of the host name. -// -// mDNS_RegisterInterface() is used by the platform support layer to inform mDNSCore of what -// physical and/or logical interfaces are available for sending and receiving packets. -// Typically it is called on startup for each available interface, but register/deregister may be -// called again later, on multiple occasions, to inform the core of interface configuration changes. -// If set->Advertise is set non-zero, then mDNS_RegisterInterface() also registers the standard -// resource records that should be associated with every publicised IP address/interface: -// -- Name-to-address records (A/AAAA) -// -- Address-to-name records (PTR) -// -- Host information (HINFO) -// IMPORTANT: The specified mDNSInterfaceID MUST NOT be 0, -1, or -2; these values have special meaning -// mDNS_RegisterInterface does not result in the registration of global hostnames via dynamic update - -// see mDNS_SetPrimaryInterfaceInfo, mDNS_AddDynDNSHostName, etc. for this purpose. -// Note that the set may be deallocated immediately after it is deregistered via mDNS_DeegisterInterface. -// -// mDNS_RegisterDNS() is used by the platform support layer to provide the core with the addresses of -// available domain name servers for unicast queries/updates. RegisterDNS() should be called once for -// each name server, typically at startup, or when a new name server becomes available. DeregiterDNS() -// must be called whenever a registered name server becomes unavailable. DeregisterDNSList deregisters -// all registered servers. mDNS_DNSRegistered() returns true if one or more servers are registered in the core. -// -// mDNSCoreInitComplete() is called when the platform support layer is finished. -// Typically this is at the end of mDNSPlatformInit(), but may be later -// (on platforms like OT that allow asynchronous initialization of the networking stack). -// -// mDNSCoreReceive() is called when a UDP packet is received -// -// mDNSCoreMachineSleep() is called when the machine sleeps or wakes -// (This refers to heavyweight laptop-style sleep/wake that disables network access, -// not lightweight second-by-second CPU power management modes.) - -extern void mDNS_SetFQDN(mDNS *const m); -extern void mDNS_ActivateNetWake_internal (mDNS *const m, NetworkInterfaceInfo *set); -extern void mDNS_DeactivateNetWake_internal(mDNS *const m, NetworkInterfaceInfo *set); -extern mStatus mDNS_RegisterInterface (mDNS *const m, NetworkInterfaceInfo *set, mDNSBool flapping); -extern void mDNS_DeregisterInterface(mDNS *const m, NetworkInterfaceInfo *set, mDNSBool flapping); -extern void mDNSCoreInitComplete(mDNS *const m, mStatus result); -extern void mDNSCoreReceive(mDNS *const m, void *const msg, const mDNSu8 *const end, - const mDNSAddr *const srcaddr, const mDNSIPPort srcport, - const mDNSAddr *dstaddr, const mDNSIPPort dstport, const mDNSInterfaceID InterfaceID); -extern void mDNSCoreRestartQueries(mDNS *const m); -typedef void (*FlushCache)(mDNS *const m); -typedef void (*CallbackBeforeStartQuery)(mDNS *const m, void *context); -extern void mDNSCoreRestartAddressQueries(mDNS *const m, mDNSBool SearchDomainsChanged, FlushCache flushCacheRecords, - CallbackBeforeStartQuery beforeQueryStart, void *context); -extern mDNSBool mDNSCoreHaveAdvertisedMulticastServices(mDNS *const m); -extern void mDNSCoreMachineSleep(mDNS *const m, mDNSBool wake); -extern mDNSBool mDNSCoreReadyForSleep(mDNS *m, mDNSs32 now); -extern mDNSs32 mDNSCoreIntervalToNextWake(mDNS *const m, mDNSs32 now); - -extern void mDNSCoreReceiveRawPacket (mDNS *const m, const mDNSu8 *const p, const mDNSu8 *const end, const mDNSInterfaceID InterfaceID); - -extern mDNSBool mDNSAddrIsDNSMulticast(const mDNSAddr *ip); - -extern CacheRecord *CreateNewCacheEntry(mDNS *const m, const mDNSu32 slot, CacheGroup *cg, mDNSs32 delay); -extern void ScheduleNextCacheCheckTime(mDNS *const m, const mDNSu32 slot, const mDNSs32 event); -extern void GrantCacheExtensions(mDNS *const m, DNSQuestion *q, mDNSu32 lease); -extern void MakeNegativeCacheRecord(mDNS *const m, CacheRecord *const cr, - const domainname *const name, const mDNSu32 namehash, const mDNSu16 rrtype, const mDNSu16 rrclass, mDNSu32 ttl_seconds, - mDNSInterfaceID InterfaceID, DNSServer *dnsserver); -extern void CompleteDeregistration(mDNS *const m, AuthRecord *rr); -extern void AnswerCurrentQuestionWithResourceRecord(mDNS *const m, CacheRecord *const rr, const QC_result AddRecord); -extern char *InterfaceNameForID(mDNS *const m, const mDNSInterfaceID InterfaceID); -extern void DNSServerChangeForQuestion(mDNS *const m, DNSQuestion *q, DNSServer *newServer); -extern void ActivateUnicastRegistration(mDNS *const m, AuthRecord *const rr); -extern void CheckSuppressUnusableQuestions(mDNS *const m); -extern void RetrySearchDomainQuestions(mDNS *const m); -extern mDNSBool DomainEnumQuery(const domainname *qname); - -// Used only in logging to restrict the number of /etc/hosts entries printed -extern void FreeEtcHosts(mDNS *const m, AuthRecord *const rr, mStatus result); -// exported for using the hash for /etc/hosts AuthRecords -extern AuthGroup *AuthGroupForName(AuthHash *r, const mDNSu32 slot, const mDNSu32 namehash, const domainname *const name); -extern AuthGroup *AuthGroupForRecord(AuthHash *r, const mDNSu32 slot, const ResourceRecord *const rr); -extern AuthGroup *InsertAuthRecord(mDNS *const m, AuthHash *r, AuthRecord *rr); -extern AuthGroup *RemoveAuthRecord(mDNS *const m, AuthHash *r, AuthRecord *rr); - -// For now this AutoTunnel stuff is specific to Mac OS X. -// In the future, if there's demand, we may see if we can abstract it out cleanly into the platform layer -#if APPLE_OSX_mDNSResponder -extern void AutoTunnelCallback(mDNS *const m, DNSQuestion *question, const ResourceRecord *const answer, QC_result AddRecord); -extern void AddNewClientTunnel(mDNS *const m, DNSQuestion *const q); -extern void SetupLocalAutoTunnelInterface_internal(mDNS *const m, mDNSBool servicesStarting); -extern void UpdateAutoTunnelDomainStatuses(const mDNS *const m); -extern mStatus ActivateLocalProxy(mDNS *const m, char *ifname); -extern void RemoveAutoTunnel6Record(mDNS *const m); -extern mDNSBool RecordReadyForSleep(mDNS *const m, AuthRecord *rr); -#endif - -// *************************************************************************** -#if 0 -#pragma mark - -#pragma mark - Sleep Proxy -#endif - -// Sleep Proxy Server Property Encoding -// -// Sleep Proxy Servers are advertised using a structured service name, consisting of four -// metrics followed by a human-readable name. The metrics assist clients in deciding which -// Sleep Proxy Server(s) to use when multiple are available on the network. Each metric -// is a two-digit decimal number in the range 10-99. Lower metrics are generally better. -// -// AA-BB-CC-DD Name -// -// Metrics: -// -// AA = Intent -// BB = Portability -// CC = Marginal Power -// DD = Total Power -// -// -// ** Intent Metric ** -// -// 20 = Dedicated Sleep Proxy Server -- a device, permanently powered on, -// installed for the express purpose of providing Sleep Proxy Service. -// -// 30 = Primary Network Infrastructure Hardware -- a router, DHCP server, NAT gateway, -// or similar permanently installed device which is permanently powered on. -// This is hardware designed for the express purpose of being network -// infrastructure, and for most home users is typically a single point -// of failure for the local network -- e.g. most home users only have -// a single NAT gateway / DHCP server. Even though in principle the -// hardware might technically be capable of running different software, -// a typical user is unlikely to do that. e.g. AirPort base station. -// -// 40 = Primary Network Infrastructure Software -- a general-purpose computer -// (e.g. Mac, Windows, Linux, etc.) which is currently running DHCP server -// or NAT gateway software, but the user could choose to turn that off -// fairly easily. e.g. iMac running Internet Sharing -// -// 50 = Secondary Network Infrastructure Hardware -- like primary infrastructure -// hardware, except not a single point of failure for the entire local network. -// For example, an AirPort base station in bridge mode. This may have clients -// associated with it, and if it goes away those clients will be inconvenienced, -// but unlike the NAT gateway / DHCP server, the entire local network is not -// dependent on it. -// -// 60 = Secondary Network Infrastructure Software -- like 50, but in a general- -// purpose CPU. -// -// 70 = Incidentally Available Hardware -- a device which has no power switch -// and is generally left powered on all the time. Even though it is not a -// part of what we conventionally consider network infrastructure (router, -// DHCP, NAT, DNS, etc.), and the rest of the network can operate fine -// without it, since it's available and unlikely to be turned off, it is a -// reasonable candidate for providing Sleep Proxy Service e.g. Apple TV, -// or an AirPort base station in client mode, associated with an existing -// wireless network (e.g. AirPort Express connected to a music system, or -// being used to share a USB printer). -// -// 80 = Incidentally Available Software -- a general-purpose computer which -// happens at this time to be set to "never sleep", and as such could be -// useful as a Sleep Proxy Server, but has not been intentionally provided -// for this purpose. Of all the Intent Metric categories this is the -// one most likely to be shut down or put to sleep without warning. -// However, if nothing else is availalable, it may be better than nothing. -// e.g. Office computer in the workplace which has been set to "never sleep" -// -// -// ** Portability Metric ** -// -// Inversely related to mass of device, on the basis that, all other things -// being equal, heavier devices are less likely to be moved than lighter devices. -// E.g. A MacBook running Internet Sharing is probably more likely to be -// put to sleep and taken away than a Mac Pro running Internet Sharing. -// The Portability Metric is a logarithmic decibel scale, computed by taking the -// (approximate) mass of the device in milligrammes, taking the base 10 logarithm -// of that, multiplying by 10, and subtracting the result from 100: -// -// Portability Metric = 100 - (log10(mg) * 10) -// -// The Portability Metric is not necessarily computed literally from the actual -// mass of the device; the intent is just that lower numbers indicate more -// permanent devices, and higher numbers indicate devices more likely to be -// removed from the network, e.g., in order of increasing portability: -// -// Mac Pro < iMac < Laptop < iPhone -// -// Example values: -// -// 10 = 1 metric tonne -// 40 = 1kg -// 70 = 1g -// 90 = 10mg -// -// -// ** Marginal Power and Total Power Metrics ** -// -// The Marginal Power Metric is the power difference between sleeping and staying awake -// to be a Sleep Proxy Server. -// -// The Total Power Metric is the total power consumption when being Sleep Proxy Server. -// -// The Power Metrics use a logarithmic decibel scale, computed as ten times the -// base 10 logarithm of the (approximate) power in microwatts: -// -// Power Metric = log10(uW) * 10 -// -// Higher values indicate higher power consumption. Example values: -// -// 10 = 10 uW -// 20 = 100 uW -// 30 = 1 mW -// 60 = 1 W -// 90 = 1 kW - -typedef enum - { - mDNSSleepProxyMetric_Dedicated = 20, - mDNSSleepProxyMetric_PrimaryHardware = 30, - mDNSSleepProxyMetric_PrimarySoftware = 40, - mDNSSleepProxyMetric_SecondaryHardware = 50, - mDNSSleepProxyMetric_SecondarySoftware = 60, - mDNSSleepProxyMetric_IncidentalHardware = 70, - mDNSSleepProxyMetric_IncidentalSoftware = 80 - } mDNSSleepProxyMetric; - -extern void mDNSCoreBeSleepProxyServer_internal(mDNS *const m, mDNSu8 sps, mDNSu8 port, mDNSu8 marginalpower, mDNSu8 totpower); -#define mDNSCoreBeSleepProxyServer(M,S,P,MP,TP) \ - do { mDNS_Lock(m); mDNSCoreBeSleepProxyServer_internal((M),(S),(P),(MP),(TP)); mDNS_Unlock(m); } while(0) - -extern void FindSPSInCache(mDNS *const m, const DNSQuestion *const q, const CacheRecord *sps[3]); -#define PrototypeSPSName(X) ((X)[0] >= 11 && (X)[3] == '-' && (X)[ 4] == '9' && (X)[ 5] == '9' && \ - (X)[6] == '-' && (X)[ 7] == '9' && (X)[ 8] == '9' && \ - (X)[9] == '-' && (X)[10] == '9' && (X)[11] == '9' ) -#define ValidSPSName(X) ((X)[0] >= 5 && mDNSIsDigit((X)[1]) && mDNSIsDigit((X)[2]) && mDNSIsDigit((X)[4]) && mDNSIsDigit((X)[5])) -#define SPSMetric(X) (!ValidSPSName(X) || PrototypeSPSName(X) ? 1000000 : \ - ((X)[1]-'0') * 100000 + ((X)[2]-'0') * 10000 + ((X)[4]-'0') * 1000 + ((X)[5]-'0') * 100 + ((X)[7]-'0') * 10 + ((X)[8]-'0')) - -// *************************************************************************** -#if 0 -#pragma mark - -#pragma mark - Compile-Time assertion checks -#endif - -// Some C compiler cleverness. We can make the compiler check certain things for -// us, and report compile-time errors if anything is wrong. The usual way to do -// this would be to use a run-time "if" statement, but then you don't find out -// what's wrong until you run the software. This way, if the assertion condition -// is false, the array size is negative, and the complier complains immediately. - -struct CompileTimeAssertionChecks_mDNS - { - // Check that the compiler generated our on-the-wire packet format structure definitions - // properly packed, without adding padding bytes to align fields on 32-bit or 64-bit boundaries. - char assert0[(sizeof(rdataSRV) == 262 ) ? 1 : -1]; - char assert1[(sizeof(DNSMessageHeader) == 12 ) ? 1 : -1]; - char assert2[(sizeof(DNSMessage) == 12+AbsoluteMaxDNSMessageData) ? 1 : -1]; - char assert3[(sizeof(mDNSs8) == 1 ) ? 1 : -1]; - char assert4[(sizeof(mDNSu8) == 1 ) ? 1 : -1]; - char assert5[(sizeof(mDNSs16) == 2 ) ? 1 : -1]; - char assert6[(sizeof(mDNSu16) == 2 ) ? 1 : -1]; - char assert7[(sizeof(mDNSs32) == 4 ) ? 1 : -1]; - char assert8[(sizeof(mDNSu32) == 4 ) ? 1 : -1]; - char assert9[(sizeof(mDNSOpaque16) == 2 ) ? 1 : -1]; - char assertA[(sizeof(mDNSOpaque32) == 4 ) ? 1 : -1]; - char assertB[(sizeof(mDNSOpaque128) == 16 ) ? 1 : -1]; - char assertC[(sizeof(CacheRecord ) == sizeof(CacheGroup) ) ? 1 : -1]; - char assertD[(sizeof(int) >= 4 ) ? 1 : -1]; - char assertE[(StandardAuthRDSize >= 256 ) ? 1 : -1]; - char assertF[(sizeof(EthernetHeader) == 14 ) ? 1 : -1]; - char assertG[(sizeof(ARP_EthIP ) == 28 ) ? 1 : -1]; - char assertH[(sizeof(IPv4Header ) == 20 ) ? 1 : -1]; - char assertI[(sizeof(IPv6Header ) == 40 ) ? 1 : -1]; - char assertJ[(sizeof(IPv6NDP ) == 24 ) ? 1 : -1]; - char assertK[(sizeof(UDPHeader ) == 8 ) ? 1 : -1]; - char assertL[(sizeof(IKEHeader ) == 28 ) ? 1 : -1]; - char assertM[(sizeof(TCPHeader ) == 20 ) ? 1 : -1]; - - // Check our structures are reasonable sizes. Including overly-large buffers, or embedding - // other overly-large structures instead of having a pointer to them, can inadvertently - // cause structure sizes (and therefore memory usage) to balloon unreasonably. - char sizecheck_RDataBody [(sizeof(RDataBody) == 264) ? 1 : -1]; - char sizecheck_ResourceRecord [(sizeof(ResourceRecord) <= 64) ? 1 : -1]; - char sizecheck_AuthRecord [(sizeof(AuthRecord) <= 1208) ? 1 : -1]; - char sizecheck_CacheRecord [(sizeof(CacheRecord) <= 184) ? 1 : -1]; - char sizecheck_CacheGroup [(sizeof(CacheGroup) <= 184) ? 1 : -1]; - char sizecheck_DNSQuestion [(sizeof(DNSQuestion) <= 786) ? 1 : -1]; - char sizecheck_ZoneData [(sizeof(ZoneData) <= 1624) ? 1 : -1]; - char sizecheck_NATTraversalInfo [(sizeof(NATTraversalInfo) <= 192) ? 1 : -1]; - char sizecheck_HostnameInfo [(sizeof(HostnameInfo) <= 3050) ? 1 : -1]; - char sizecheck_DNSServer [(sizeof(DNSServer) <= 328) ? 1 : -1]; - char sizecheck_NetworkInterfaceInfo[(sizeof(NetworkInterfaceInfo) <= 6850) ? 1 : -1]; - char sizecheck_ServiceRecordSet [(sizeof(ServiceRecordSet) <= 5500) ? 1 : -1]; - char sizecheck_DomainAuthInfo [(sizeof(DomainAuthInfo) <= 7808) ? 1 : -1]; - char sizecheck_ServiceInfoQuery [(sizeof(ServiceInfoQuery) <= 3200) ? 1 : -1]; -#if APPLE_OSX_mDNSResponder - char sizecheck_ClientTunnel [(sizeof(ClientTunnel) <= 1148) ? 1 : -1]; -#endif - }; - -// *************************************************************************** - -#ifdef __cplusplus - } -#endif - -#endif diff -Nru qtcreator-2.5.0/src/tools/mdnssd/mDNSPosix.c qtcreator-2.5.2/src/tools/mdnssd/mDNSPosix.c --- qtcreator-2.5.0/src/tools/mdnssd/mDNSPosix.c 2012-05-09 12:13:19.000000000 +0000 +++ qtcreator-2.5.2/src/tools/mdnssd/mDNSPosix.c 1970-01-01 00:00:00.000000000 +0000 @@ -1,1621 +0,0 @@ -/* -*- Mode: C; tab-width: 4 -*- - * - * Copyright (c) 2002-2004 Apple Computer, Inc. All rights reserved. - * - * Licensed under the Apache License, Version 2.0 (the "License"); - * you may not use this file except in compliance with the License. - * You may obtain a copy of the License at - * - * http://www.apache.org/licenses/LICENSE-2.0 - * - * Unless required by applicable law or agreed to in writing, software - * distributed under the License is distributed on an "AS IS" BASIS, - * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. - * See the License for the specific language governing permissions and - * limitations under the License. - * - * Formatting notes: - * This code follows the "Whitesmiths style" C indentation rules. Plenty of discussion - * on C indentation can be found on the web, such as , - * but for the sake of brevity here I will say just this: Curly braces are not syntactially - * part of an "if" statement; they are the beginning and ending markers of a compound statement; - * therefore common sense dictates that if they are part of a compound statement then they - * should be indented to the same level as everything else in that compound statement. - * Indenting curly braces at the same level as the "if" implies that curly braces are - * part of the "if", which is false. (This is as misleading as people who write "char* x,y;" - * thinking that variables x and y are both of type "char*" -- and anyone who doesn't - * understand why variable y is not of type "char*" just proves the point that poor code - * layout leads people to unfortunate misunderstandings about how the C language really works.) - */ - -#include "mDNSEmbeddedAPI.h" // Defines the interface provided to the client layer above -#include "DNSCommon.h" -#include "mDNSPosix.h" // Defines the specific types needed to run mDNS on this platform -#include "dns_sd.h" - -#include -#include -#include -#include -#include -#include -#include -#include -#include -#include -#include -#include -#include -#include -#include -#include -#include // platform support for UTC time - -#if USES_NETLINK -#include -#include -#include -#else // USES_NETLINK -#include -#include -#endif // USES_NETLINK - -#include "mDNSUNP.h" -#include "GenLinkedList.h" - -// *************************************************************************** -// Structures - -// We keep a list of client-supplied event sources in PosixEventSource records -struct PosixEventSource - { - mDNSPosixEventCallback Callback; - void *Context; - int fd; - struct PosixEventSource *Next; - }; -typedef struct PosixEventSource PosixEventSource; - -// Context record for interface change callback -struct IfChangeRec - { - int NotifySD; - mDNS *mDNS; - }; -typedef struct IfChangeRec IfChangeRec; - -// Note that static data is initialized to zero in (modern) C. -static fd_set gEventFDs; -static int gMaxFD; // largest fd in gEventFDs -static GenLinkedList gEventSources; // linked list of PosixEventSource's -static sigset_t gEventSignalSet; // Signals which event loop listens for -static sigset_t gEventSignals; // Signals which were received while inside loop - -// *************************************************************************** -// Globals (for debugging) - -static int num_registered_interfaces = 0; -static int num_pkts_accepted = 0; -static int num_pkts_rejected = 0; - -// *************************************************************************** -// Functions - -int gMDNSPlatformPosixVerboseLevel = 0; - -#define PosixErrorToStatus(errNum) ((errNum) == 0 ? mStatus_NoError : mStatus_UnknownErr) - -mDNSlocal void SockAddrTomDNSAddr(const struct sockaddr *const sa, mDNSAddr *ipAddr, mDNSIPPort *ipPort) - { - switch (sa->sa_family) - { - case AF_INET: - { - struct sockaddr_in *sin = (struct sockaddr_in*)sa; - ipAddr->type = mDNSAddrType_IPv4; - ipAddr->ip.v4.NotAnInteger = sin->sin_addr.s_addr; - if (ipPort) ipPort->NotAnInteger = sin->sin_port; - break; - } - -#if HAVE_IPV6 - case AF_INET6: - { - struct sockaddr_in6 *sin6 = (struct sockaddr_in6*)sa; -#ifndef NOT_HAVE_SA_LEN - assert(sin6->sin6_len == sizeof(*sin6)); -#endif - ipAddr->type = mDNSAddrType_IPv6; - ipAddr->ip.v6 = *(mDNSv6Addr*)&sin6->sin6_addr; - if (ipPort) ipPort->NotAnInteger = sin6->sin6_port; - break; - } -#endif - - default: - verbosedebugf("SockAddrTomDNSAddr: Uknown address family %d\n", sa->sa_family); - ipAddr->type = mDNSAddrType_None; - if (ipPort) ipPort->NotAnInteger = 0; - break; - } - } - -#if COMPILER_LIKES_PRAGMA_MARK -#pragma mark ***** Send and Receive -#endif - -// mDNS core calls this routine when it needs to send a packet. -mDNSexport mStatus mDNSPlatformSendUDP(const mDNS *const m, const void *const msg, const mDNSu8 *const end, - mDNSInterfaceID InterfaceID, UDPSocket *src, const mDNSAddr *dst, mDNSIPPort dstPort) - { - int err = 0; - struct sockaddr_storage to; - PosixNetworkInterface * thisIntf = (PosixNetworkInterface *)(InterfaceID); - int sendingsocket = -1; - - (void)src; // Will need to use this parameter once we implement mDNSPlatformUDPSocket/mDNSPlatformUDPClose - - assert(m != NULL); - assert(msg != NULL); - assert(end != NULL); - assert((((char *) end) - ((char *) msg)) > 0); - - if (dstPort.NotAnInteger == 0) - { - LogMsg("mDNSPlatformSendUDP: Invalid argument -dstPort is set to 0"); - return PosixErrorToStatus(EINVAL); - } - if (dst->type == mDNSAddrType_IPv4) - { - struct sockaddr_in *sin = (struct sockaddr_in*)&to; -#ifndef NOT_HAVE_SA_LEN - sin->sin_len = sizeof(*sin); -#endif - sin->sin_family = AF_INET; - sin->sin_port = dstPort.NotAnInteger; - sin->sin_addr.s_addr = dst->ip.v4.NotAnInteger; - sendingsocket = thisIntf ? thisIntf->multicastSocket4 : m->p->unicastSocket4; - } - -#if HAVE_IPV6 - else if (dst->type == mDNSAddrType_IPv6) - { - struct sockaddr_in6 *sin6 = (struct sockaddr_in6*)&to; - mDNSPlatformMemZero(sin6, sizeof(*sin6)); -#ifndef NOT_HAVE_SA_LEN - sin6->sin6_len = sizeof(*sin6); -#endif - sin6->sin6_family = AF_INET6; - sin6->sin6_port = dstPort.NotAnInteger; - sin6->sin6_addr = *(struct in6_addr*)&dst->ip.v6; - sendingsocket = thisIntf ? thisIntf->multicastSocket6 : m->p->unicastSocket6; - } -#endif - - if (sendingsocket >= 0) - err = sendto(sendingsocket, msg, (char*)end - (char*)msg, 0, (struct sockaddr *)&to, GET_SA_LEN(to)); - - if (err > 0) err = 0; - else if (err < 0) - { - static int MessageCount = 0; - // Don't report EHOSTDOWN (i.e. ARP failure), ENETDOWN, or no route to host for unicast destinations - if (!mDNSAddressIsAllDNSLinkGroup(dst)) - if (errno == EHOSTDOWN || errno == ENETDOWN || errno == EHOSTUNREACH || errno == ENETUNREACH) return(mStatus_TransientErr); - - if (MessageCount < 1000) - { - MessageCount++; - if (thisIntf) - LogMsg("mDNSPlatformSendUDP got error %d (%s) sending packet to %#a on interface %#a/%s/%d", - errno, strerror(errno), dst, &thisIntf->coreIntf.ip, thisIntf->intfName, thisIntf->index); - else - LogMsg("mDNSPlatformSendUDP got error %d (%s) sending packet to %#a", errno, strerror(errno), dst); - } - } - - return PosixErrorToStatus(err); - } - -// This routine is called when the main loop detects that data is available on a socket. -mDNSlocal void SocketDataReady(mDNS *const m, PosixNetworkInterface *intf, int skt) - { - mDNSAddr senderAddr, destAddr; - mDNSIPPort senderPort; - ssize_t packetLen; - DNSMessage packet; - struct my_in_pktinfo packetInfo; - struct sockaddr_storage from; - socklen_t fromLen; - int flags; - mDNSu8 ttl; - mDNSBool reject; - const mDNSInterfaceID InterfaceID = intf ? intf->coreIntf.InterfaceID : NULL; - - assert(m != NULL); - assert(skt >= 0); - - fromLen = sizeof(from); - flags = 0; - packetLen = recvfrom_flags(skt, &packet, sizeof(packet), &flags, (struct sockaddr *) &from, &fromLen, &packetInfo, &ttl); - - if (packetLen >= 0) - { - SockAddrTomDNSAddr((struct sockaddr*)&from, &senderAddr, &senderPort); - SockAddrTomDNSAddr((struct sockaddr*)&packetInfo.ipi_addr, &destAddr, NULL); - - // If we have broken IP_RECVDSTADDR functionality (so far - // I've only seen this on OpenBSD) then apply a hack to - // convince mDNS Core that this isn't a spoof packet. - // Basically what we do is check to see whether the - // packet arrived as a multicast and, if so, set its - // destAddr to the mDNS address. - // - // I must admit that I could just be doing something - // wrong on OpenBSD and hence triggering this problem - // but I'm at a loss as to how. - // - // If this platform doesn't have IP_PKTINFO or IP_RECVDSTADDR, then we have - // no way to tell the destination address or interface this packet arrived on, - // so all we can do is just assume it's a multicast - - #if HAVE_BROKEN_RECVDSTADDR || (!defined(IP_PKTINFO) && !defined(IP_RECVDSTADDR)) - if ((destAddr.NotAnInteger == 0) && (flags & MSG_MCAST)) - { - destAddr.type = senderAddr.type; - if (senderAddr.type == mDNSAddrType_IPv4) destAddr.ip.v4 = AllDNSLinkGroup_v4.ip.v4; - else if (senderAddr.type == mDNSAddrType_IPv6) destAddr.ip.v6 = AllDNSLinkGroup_v6.ip.v6; - } - #endif - - // We only accept the packet if the interface on which it came - // in matches the interface associated with this socket. - // We do this match by name or by index, depending on which - // information is available. recvfrom_flags sets the name - // to "" if the name isn't available, or the index to -1 - // if the index is available. This accomodates the various - // different capabilities of our target platforms. - - reject = mDNSfalse; - if (!intf) - { - // Ignore multicasts accidentally delivered to our unicast receiving socket - if (mDNSAddrIsDNSMulticast(&destAddr)) packetLen = -1; - } - else - { - if (packetInfo.ipi_ifname[0] != 0) reject = (strcmp(packetInfo.ipi_ifname, intf->intfName) != 0); - else if (packetInfo.ipi_ifindex != -1) reject = (packetInfo.ipi_ifindex != intf->index); - - if (reject) - { - verbosedebugf("SocketDataReady ignored a packet from %#a to %#a on interface %s/%d expecting %#a/%s/%d/%d", - &senderAddr, &destAddr, packetInfo.ipi_ifname, packetInfo.ipi_ifindex, - &intf->coreIntf.ip, intf->intfName, intf->index, skt); - packetLen = -1; - num_pkts_rejected++; - if (num_pkts_rejected > (num_pkts_accepted + 1) * (num_registered_interfaces + 1) * 2) - { - fprintf(stderr, - "*** WARNING: Received %d packets; Accepted %d packets; Rejected %d packets because of interface mismatch\n", - num_pkts_accepted + num_pkts_rejected, num_pkts_accepted, num_pkts_rejected); - num_pkts_accepted = 0; - num_pkts_rejected = 0; - } - } - else - { - verbosedebugf("SocketDataReady got a packet from %#a to %#a on interface %#a/%s/%d/%d", - &senderAddr, &destAddr, &intf->coreIntf.ip, intf->intfName, intf->index, skt); - num_pkts_accepted++; - } - } - } - - if (packetLen >= 0) - mDNSCoreReceive(m, &packet, (mDNSu8 *)&packet + packetLen, - &senderAddr, senderPort, &destAddr, MulticastDNSPort, InterfaceID); - } - -mDNSexport TCPSocket *mDNSPlatformTCPSocket(mDNS * const m, TCPSocketFlags flags, mDNSIPPort * port) - { - (void)m; // Unused - (void)flags; // Unused - (void)port; // Unused - return NULL; - } - -mDNSexport TCPSocket *mDNSPlatformTCPAccept(TCPSocketFlags flags, int sd) - { - (void)flags; // Unused - (void)sd; // Unused - return NULL; - } - -mDNSexport int mDNSPlatformTCPGetFD(TCPSocket *sock) - { - (void)sock; // Unused - return -1; - } - -mDNSexport mStatus mDNSPlatformTCPConnect(TCPSocket *sock, const mDNSAddr *dst, mDNSOpaque16 dstport, domainname *hostname, mDNSInterfaceID InterfaceID, - TCPConnectionCallback callback, void *context) - { - (void)sock; // Unused - (void)dst; // Unused - (void)dstport; // Unused - (void)hostname; // Unused - (void)InterfaceID; // Unused - (void)callback; // Unused - (void)context; // Unused - return(mStatus_UnsupportedErr); - } - -mDNSexport void mDNSPlatformTCPCloseConnection(TCPSocket *sock) - { - (void)sock; // Unused - } - -mDNSexport long mDNSPlatformReadTCP(TCPSocket *sock, void *buf, unsigned long buflen, mDNSBool * closed) - { - (void)sock; // Unused - (void)buf; // Unused - (void)buflen; // Unused - (void)closed; // Unused - return 0; - } - -mDNSexport long mDNSPlatformWriteTCP(TCPSocket *sock, const char *msg, unsigned long len) - { - (void)sock; // Unused - (void)msg; // Unused - (void)len; // Unused - return 0; - } - -mDNSexport UDPSocket *mDNSPlatformUDPSocket(mDNS * const m, mDNSIPPort port) - { - (void)m; // Unused - (void)port; // Unused - return NULL; - } - -mDNSexport void mDNSPlatformUDPClose(UDPSocket *sock) - { - (void)sock; // Unused - } - -mDNSexport void mDNSPlatformUpdateProxyList(mDNS *const m, const mDNSInterfaceID InterfaceID) - { - (void)m; // Unused - (void)InterfaceID; // Unused - } - -mDNSexport void mDNSPlatformSendRawPacket(const void *const msg, const mDNSu8 *const end, mDNSInterfaceID InterfaceID) - { - (void)msg; // Unused - (void)end; // Unused - (void)InterfaceID; // Unused - } - -mDNSexport void mDNSPlatformSetLocalAddressCacheEntry(mDNS *const m, const mDNSAddr *const tpa, const mDNSEthAddr *const tha, mDNSInterfaceID InterfaceID) - { - (void)m; // Unused - (void)tpa; // Unused - (void)tha; // Unused - (void)InterfaceID; // Unused - } - -mDNSexport mStatus mDNSPlatformTLSSetupCerts(void) - { - return(mStatus_UnsupportedErr); - } - -mDNSexport void mDNSPlatformTLSTearDownCerts(void) - { - } - -mDNSexport void mDNSPlatformSetAllowSleep(mDNS *const m, mDNSBool allowSleep, const char *reason) - { - (void) m; - (void) allowSleep; - (void) reason; - } - -#if COMPILER_LIKES_PRAGMA_MARK -#pragma mark - -#pragma mark - /etc/hosts support -#endif - -mDNSexport void FreeEtcHosts(mDNS *const m, AuthRecord *const rr, mStatus result) - { - (void)m; // unused - (void)rr; - (void)result; - } - - -#if COMPILER_LIKES_PRAGMA_MARK -#pragma mark ***** DDNS Config Platform Functions -#endif - -mDNSexport void mDNSPlatformSetDNSConfig(mDNS *const m, mDNSBool setservers, mDNSBool setsearch, domainname *const fqdn, DNameListElem **RegDomains, DNameListElem **BrowseDomains) - { - (void) m; - (void) setservers; - (void) fqdn; - (void) setsearch; - (void) RegDomains; - (void) BrowseDomains; - } - -mDNSexport mStatus mDNSPlatformGetPrimaryInterface(mDNS * const m, mDNSAddr * v4, mDNSAddr * v6, mDNSAddr * router) - { - (void) m; - (void) v4; - (void) v6; - (void) router; - - return mStatus_UnsupportedErr; - } - -mDNSexport void mDNSPlatformDynDNSHostNameStatusChanged(const domainname *const dname, const mStatus status) - { - (void) dname; - (void) status; - } - -#if COMPILER_LIKES_PRAGMA_MARK -#pragma mark ***** Init and Term -#endif - -// This gets the current hostname, truncating it at the first dot if necessary -mDNSlocal void GetUserSpecifiedRFC1034ComputerName(domainlabel *const namelabel) - { - int len = 0; - gethostname((char *)(&namelabel->c[1]), MAX_DOMAIN_LABEL); - while (len < MAX_DOMAIN_LABEL && namelabel->c[len+1] && namelabel->c[len+1] != '.') len++; - namelabel->c[0] = len; - } - -// On OS X this gets the text of the field labelled "Computer Name" in the Sharing Prefs Control Panel -// Other platforms can either get the information from the appropriate place, -// or they can alternatively just require all registering services to provide an explicit name -mDNSlocal void GetUserSpecifiedFriendlyComputerName(domainlabel *const namelabel) - { - // On Unix we have no better name than the host name, so we just use that. - GetUserSpecifiedRFC1034ComputerName(namelabel); - } - -mDNSexport int ParseDNSServers(mDNS *m, const char *filePath) - { - char line[256]; - char nameserver[16]; - char keyword[10]; - int numOfServers = 0; - FILE *fp = fopen(filePath, "r"); - if (fp == NULL) return -1; - while (fgets(line,sizeof(line),fp)) - { - struct in_addr ina; - line[255]='\0'; // just to be safe - if (sscanf(line,"%10s %15s", keyword, nameserver) != 2) continue; // it will skip whitespaces - if (strncasecmp(keyword,"nameserver",10)) continue; - if (inet_aton(nameserver, (struct in_addr *)&ina) != 0) - { - mDNSAddr DNSAddr; - DNSAddr.type = mDNSAddrType_IPv4; - DNSAddr.ip.v4.NotAnInteger = ina.s_addr; - mDNS_AddDNSServer(m, NULL, mDNSInterface_Any, &DNSAddr, UnicastDNSPort, mDNSfalse, 0, mDNSfalse); - numOfServers++; - } - } - return (numOfServers > 0) ? 0 : -1; - } - -// Searches the interface list looking for the named interface. -// Returns a pointer to if it found, or NULL otherwise. -mDNSlocal PosixNetworkInterface *SearchForInterfaceByName(mDNS *const m, const char *intfName) - { - PosixNetworkInterface *intf; - - assert(m != NULL); - assert(intfName != NULL); - - intf = (PosixNetworkInterface*)(m->HostInterfaces); - while ((intf != NULL) && (strcmp(intf->intfName, intfName) != 0)) - intf = (PosixNetworkInterface *)(intf->coreIntf.next); - - return intf; - } - -mDNSexport mDNSInterfaceID mDNSPlatformInterfaceIDfromInterfaceIndex(mDNS *const m, mDNSu32 index) - { - PosixNetworkInterface *intf; - - assert(m != NULL); - - if (index == kDNSServiceInterfaceIndexLocalOnly) return(mDNSInterface_LocalOnly); - if (index == kDNSServiceInterfaceIndexP2P ) return(mDNSInterface_P2P); - if (index == kDNSServiceInterfaceIndexAny ) return(mDNSInterface_Any); - - intf = (PosixNetworkInterface*)(m->HostInterfaces); - while ((intf != NULL) && (mDNSu32) intf->index != index) - intf = (PosixNetworkInterface *)(intf->coreIntf.next); - - return (mDNSInterfaceID) intf; - } - -mDNSexport mDNSu32 mDNSPlatformInterfaceIndexfromInterfaceID(mDNS *const m, mDNSInterfaceID id, mDNSBool suppressNetworkChange) - { - PosixNetworkInterface *intf; - (void) suppressNetworkChange; // Unused - - assert(m != NULL); - - if (id == mDNSInterface_LocalOnly) return(kDNSServiceInterfaceIndexLocalOnly); - if (id == mDNSInterface_P2P ) return(kDNSServiceInterfaceIndexP2P); - if (id == mDNSInterface_Any ) return(kDNSServiceInterfaceIndexAny); - - intf = (PosixNetworkInterface*)(m->HostInterfaces); - while ((intf != NULL) && (mDNSInterfaceID) intf != id) - intf = (PosixNetworkInterface *)(intf->coreIntf.next); - - return intf ? intf->index : 0; - } - -// Frees the specified PosixNetworkInterface structure. The underlying -// interface must have already been deregistered with the mDNS core. -mDNSlocal void FreePosixNetworkInterface(PosixNetworkInterface *intf) - { - assert(intf != NULL); - if (intf->intfName != NULL) free((void *)intf->intfName); - if (intf->multicastSocket4 != -1) assert(close(intf->multicastSocket4) == 0); -#if HAVE_IPV6 - if (intf->multicastSocket6 != -1) assert(close(intf->multicastSocket6) == 0); -#endif - free(intf); - } - -// Grab the first interface, deregister it, free it, and repeat until done. -mDNSlocal void ClearInterfaceList(mDNS *const m) - { - assert(m != NULL); - - while (m->HostInterfaces) - { - PosixNetworkInterface *intf = (PosixNetworkInterface*)(m->HostInterfaces); - mDNS_DeregisterInterface(m, &intf->coreIntf, mDNSfalse); - if (gMDNSPlatformPosixVerboseLevel > 0) fprintf(stderr, "Deregistered interface %s\n", intf->intfName); - FreePosixNetworkInterface(intf); - } - num_registered_interfaces = 0; - num_pkts_accepted = 0; - num_pkts_rejected = 0; - } - -// Sets up a send/receive socket. -// If mDNSIPPort port is non-zero, then it's a multicast socket on the specified interface -// If mDNSIPPort port is zero, then it's a randomly assigned port number, used for sending unicast queries -mDNSlocal int SetupSocket(struct sockaddr *intfAddr, mDNSIPPort port, int interfaceIndex, int *sktPtr) - { - int err = 0; - static const int kOn = 1; - static const int kIntTwoFiveFive = 255; - static const unsigned char kByteTwoFiveFive = 255; - const mDNSBool JoinMulticastGroup = (port.NotAnInteger != 0); - - (void) interfaceIndex; // This parameter unused on plaforms that don't have IPv6 - assert(intfAddr != NULL); - assert(sktPtr != NULL); - assert(*sktPtr == -1); - - // Open the socket... - if (intfAddr->sa_family == AF_INET) *sktPtr = socket(PF_INET, SOCK_DGRAM, IPPROTO_UDP); -#if HAVE_IPV6 - else if (intfAddr->sa_family == AF_INET6) *sktPtr = socket(PF_INET6, SOCK_DGRAM, IPPROTO_UDP); -#endif - else return EINVAL; - - if (*sktPtr < 0) { err = errno; perror((intfAddr->sa_family == AF_INET) ? "socket AF_INET" : "socket AF_INET6"); } - - // ... with a shared UDP port, if it's for multicast receiving - if (err == 0 && port.NotAnInteger) - { - #if defined(SO_REUSEPORT) - err = setsockopt(*sktPtr, SOL_SOCKET, SO_REUSEPORT, &kOn, sizeof(kOn)); - #elif defined(SO_REUSEADDR) - err = setsockopt(*sktPtr, SOL_SOCKET, SO_REUSEADDR, &kOn, sizeof(kOn)); - #else - #error This platform has no way to avoid address busy errors on multicast. - #endif - if (err < 0) { err = errno; perror("setsockopt - SO_REUSExxxx"); } - } - - // We want to receive destination addresses and interface identifiers. - if (intfAddr->sa_family == AF_INET) - { - struct ip_mreq imr; - struct sockaddr_in bindAddr; - if (err == 0) - { - #if defined(IP_PKTINFO) // Linux - err = setsockopt(*sktPtr, IPPROTO_IP, IP_PKTINFO, &kOn, sizeof(kOn)); - if (err < 0) { err = errno; perror("setsockopt - IP_PKTINFO"); } - #elif defined(IP_RECVDSTADDR) || defined(IP_RECVIF) // BSD and Solaris - #if defined(IP_RECVDSTADDR) - err = setsockopt(*sktPtr, IPPROTO_IP, IP_RECVDSTADDR, &kOn, sizeof(kOn)); - if (err < 0) { err = errno; perror("setsockopt - IP_RECVDSTADDR"); } - #endif - #if defined(IP_RECVIF) - if (err == 0) - { - err = setsockopt(*sktPtr, IPPROTO_IP, IP_RECVIF, &kOn, sizeof(kOn)); - if (err < 0) { err = errno; perror("setsockopt - IP_RECVIF"); } - } - #endif - #else - #warning This platform has no way to get the destination interface information -- will only work for single-homed hosts - #endif - } - #if defined(IP_RECVTTL) // Linux - if (err == 0) - { - setsockopt(*sktPtr, IPPROTO_IP, IP_RECVTTL, &kOn, sizeof(kOn)); - // We no longer depend on being able to get the received TTL, so don't worry if the option fails - } - #endif - - // Add multicast group membership on this interface - if (err == 0 && JoinMulticastGroup) - { - imr.imr_multiaddr.s_addr = AllDNSLinkGroup_v4.ip.v4.NotAnInteger; - imr.imr_interface = ((struct sockaddr_in*)intfAddr)->sin_addr; - err = setsockopt(*sktPtr, IPPROTO_IP, IP_ADD_MEMBERSHIP, &imr, sizeof(imr)); - if (err < 0) { err = errno; perror("setsockopt - IP_ADD_MEMBERSHIP"); } - } - - // Specify outgoing interface too - if (err == 0 && JoinMulticastGroup) - { - err = setsockopt(*sktPtr, IPPROTO_IP, IP_MULTICAST_IF, &((struct sockaddr_in*)intfAddr)->sin_addr, sizeof(struct in_addr)); - if (err < 0) { err = errno; perror("setsockopt - IP_MULTICAST_IF"); } - } - - // Per the mDNS spec, send unicast packets with TTL 255 - if (err == 0) - { - err = setsockopt(*sktPtr, IPPROTO_IP, IP_TTL, &kIntTwoFiveFive, sizeof(kIntTwoFiveFive)); - if (err < 0) { err = errno; perror("setsockopt - IP_TTL"); } - } - - // and multicast packets with TTL 255 too - // There's some debate as to whether IP_MULTICAST_TTL is an int or a byte so we just try both. - if (err == 0) - { - err = setsockopt(*sktPtr, IPPROTO_IP, IP_MULTICAST_TTL, &kByteTwoFiveFive, sizeof(kByteTwoFiveFive)); - if (err < 0 && errno == EINVAL) - err = setsockopt(*sktPtr, IPPROTO_IP, IP_MULTICAST_TTL, &kIntTwoFiveFive, sizeof(kIntTwoFiveFive)); - if (err < 0) { err = errno; perror("setsockopt - IP_MULTICAST_TTL"); } - } - - // And start listening for packets - if (err == 0) - { - bindAddr.sin_family = AF_INET; - bindAddr.sin_port = port.NotAnInteger; - bindAddr.sin_addr.s_addr = INADDR_ANY; // Want to receive multicasts AND unicasts on this socket - err = bind(*sktPtr, (struct sockaddr *) &bindAddr, sizeof(bindAddr)); - if (err < 0) { err = errno; perror("bind"); fflush(stderr); } - } - } // endif (intfAddr->sa_family == AF_INET) - -#if HAVE_IPV6 - else if (intfAddr->sa_family == AF_INET6) - { - struct ipv6_mreq imr6; - struct sockaddr_in6 bindAddr6; -#if defined(IPV6_PKTINFO) - if (err == 0) - { - err = setsockopt(*sktPtr, IPPROTO_IPV6, IPV6_2292_PKTINFO, &kOn, sizeof(kOn)); - if (err < 0) { err = errno; perror("setsockopt - IPV6_PKTINFO"); } - } - #else - #warning This platform has no way to get the destination interface information for IPv6 -- will only work for single-homed hosts - #endif - #if defined(IPV6_HOPLIMIT) - if (err == 0) - { - err = setsockopt(*sktPtr, IPPROTO_IPV6, IPV6_2292_HOPLIMIT, &kOn, sizeof(kOn)); - if (err < 0) { err = errno; perror("setsockopt - IPV6_HOPLIMIT"); } - } - #endif - - // Add multicast group membership on this interface - if (err == 0 && JoinMulticastGroup) - { - imr6.ipv6mr_multiaddr = *(const struct in6_addr*)&AllDNSLinkGroup_v6.ip.v6; - imr6.ipv6mr_interface = interfaceIndex; - //LogMsg("Joining %.16a on %d", &imr6.ipv6mr_multiaddr, imr6.ipv6mr_interface); - err = setsockopt(*sktPtr, IPPROTO_IPV6, IPV6_JOIN_GROUP, &imr6, sizeof(imr6)); - if (err < 0) - { - err = errno; - verbosedebugf("IPV6_JOIN_GROUP %.16a on %d failed.\n", &imr6.ipv6mr_multiaddr, imr6.ipv6mr_interface); - perror("setsockopt - IPV6_JOIN_GROUP"); - } - } - - // Specify outgoing interface too - if (err == 0 && JoinMulticastGroup) - { - u_int multicast_if = interfaceIndex; - err = setsockopt(*sktPtr, IPPROTO_IPV6, IPV6_MULTICAST_IF, &multicast_if, sizeof(multicast_if)); - if (err < 0) { err = errno; perror("setsockopt - IPV6_MULTICAST_IF"); } - } - - // We want to receive only IPv6 packets on this socket. - // Without this option, we may get IPv4 addresses as mapped addresses. - if (err == 0) - { - err = setsockopt(*sktPtr, IPPROTO_IPV6, IPV6_V6ONLY, &kOn, sizeof(kOn)); - if (err < 0) { err = errno; perror("setsockopt - IPV6_V6ONLY"); } - } - - // Per the mDNS spec, send unicast packets with TTL 255 - if (err == 0) - { - err = setsockopt(*sktPtr, IPPROTO_IPV6, IPV6_UNICAST_HOPS, &kIntTwoFiveFive, sizeof(kIntTwoFiveFive)); - if (err < 0) { err = errno; perror("setsockopt - IPV6_UNICAST_HOPS"); } - } - - // and multicast packets with TTL 255 too - // There's some debate as to whether IPV6_MULTICAST_HOPS is an int or a byte so we just try both. - if (err == 0) - { - err = setsockopt(*sktPtr, IPPROTO_IPV6, IPV6_MULTICAST_HOPS, &kByteTwoFiveFive, sizeof(kByteTwoFiveFive)); - if (err < 0 && errno == EINVAL) - err = setsockopt(*sktPtr, IPPROTO_IPV6, IPV6_MULTICAST_HOPS, &kIntTwoFiveFive, sizeof(kIntTwoFiveFive)); - if (err < 0) { err = errno; perror("setsockopt - IPV6_MULTICAST_HOPS"); } - } - - // And start listening for packets - if (err == 0) - { - mDNSPlatformMemZero(&bindAddr6, sizeof(bindAddr6)); -#ifndef NOT_HAVE_SA_LEN - bindAddr6.sin6_len = sizeof(bindAddr6); -#endif - bindAddr6.sin6_family = AF_INET6; - bindAddr6.sin6_port = port.NotAnInteger; - bindAddr6.sin6_flowinfo = 0; - bindAddr6.sin6_addr = in6addr_any; // Want to receive multicasts AND unicasts on this socket - bindAddr6.sin6_scope_id = 0; - err = bind(*sktPtr, (struct sockaddr *) &bindAddr6, sizeof(bindAddr6)); - if (err < 0) { err = errno; perror("bind"); fflush(stderr); } - } - } // endif (intfAddr->sa_family == AF_INET6) -#endif - - // Set the socket to non-blocking. - if (err == 0) - { - err = fcntl(*sktPtr, F_GETFL, 0); - if (err < 0) err = errno; - else - { - err = fcntl(*sktPtr, F_SETFL, err | O_NONBLOCK); - if (err < 0) err = errno; - } - } - - // Clean up - if (err != 0 && *sktPtr != -1) { assert(close(*sktPtr) == 0); *sktPtr = -1; } - assert((err == 0) == (*sktPtr != -1)); - return err; - } - -// Creates a PosixNetworkInterface for the interface whose IP address is -// intfAddr and whose name is intfName and registers it with mDNS core. -mDNSlocal int SetupOneInterface(mDNS *const m, struct sockaddr *intfAddr, struct sockaddr *intfMask, const char *intfName, int intfIndex) - { - int err = 0; - PosixNetworkInterface *intf; - PosixNetworkInterface *alias = NULL; - - assert(m != NULL); - assert(intfAddr != NULL); - assert(intfName != NULL); - assert(intfMask != NULL); - - // Allocate the interface structure itself. - intf = (PosixNetworkInterface*)malloc(sizeof(*intf)); - if (intf == NULL) { assert(0); err = ENOMEM; } - - // And make a copy of the intfName. - if (err == 0) - { - intf->intfName = strdup(intfName); - if (intf->intfName == NULL) { assert(0); err = ENOMEM; } - } - - if (err == 0) - { - // Set up the fields required by the mDNS core. - SockAddrTomDNSAddr(intfAddr, &intf->coreIntf.ip, NULL); - SockAddrTomDNSAddr(intfMask, &intf->coreIntf.mask, NULL); - - //LogMsg("SetupOneInterface: %#a %#a", &intf->coreIntf.ip, &intf->coreIntf.mask); - strncpy(intf->coreIntf.ifname, intfName, sizeof(intf->coreIntf.ifname)); - intf->coreIntf.ifname[sizeof(intf->coreIntf.ifname)-1] = 0; - intf->coreIntf.Advertise = m->AdvertiseLocalAddresses; - intf->coreIntf.McastTxRx = mDNStrue; - - // Set up the extra fields in PosixNetworkInterface. - assert(intf->intfName != NULL); // intf->intfName already set up above - intf->index = intfIndex; - intf->multicastSocket4 = -1; -#if HAVE_IPV6 - intf->multicastSocket6 = -1; -#endif - alias = SearchForInterfaceByName(m, intf->intfName); - if (alias == NULL) alias = intf; - intf->coreIntf.InterfaceID = (mDNSInterfaceID)alias; - - if (alias != intf) - debugf("SetupOneInterface: %s %#a is an alias of %#a", intfName, &intf->coreIntf.ip, &alias->coreIntf.ip); - } - - // Set up the multicast socket - if (err == 0) - { - if (alias->multicastSocket4 == -1 && intfAddr->sa_family == AF_INET) - err = SetupSocket(intfAddr, MulticastDNSPort, intf->index, &alias->multicastSocket4); -#if HAVE_IPV6 - else if (alias->multicastSocket6 == -1 && intfAddr->sa_family == AF_INET6) - err = SetupSocket(intfAddr, MulticastDNSPort, intf->index, &alias->multicastSocket6); -#endif - } - - // The interface is all ready to go, let's register it with the mDNS core. - if (err == 0) - err = mDNS_RegisterInterface(m, &intf->coreIntf, mDNSfalse); - - // Clean up. - if (err == 0) - { - num_registered_interfaces++; - debugf("SetupOneInterface: %s %#a Registered", intf->intfName, &intf->coreIntf.ip); - if (gMDNSPlatformPosixVerboseLevel > 0) - fprintf(stderr, "Registered interface %s\n", intf->intfName); - } - else - { - // Use intfName instead of intf->intfName in the next line to avoid dereferencing NULL. - debugf("SetupOneInterface: %s %#a failed to register %d", intfName, &intf->coreIntf.ip, err); - if (intf) { FreePosixNetworkInterface(intf); intf = NULL; } - } - - assert((err == 0) == (intf != NULL)); - - return err; - } - -// Call get_ifi_info() to obtain a list of active interfaces and call SetupOneInterface() on each one. -mDNSlocal int SetupInterfaceList(mDNS *const m) - { - mDNSBool foundav4 = mDNSfalse; - int err = 0; - struct ifi_info *intfList = get_ifi_info(AF_INET, mDNStrue); - struct ifi_info *firstLoopback = NULL; - - assert(m != NULL); - debugf("SetupInterfaceList"); - - if (intfList == NULL) err = ENOENT; - -#if HAVE_IPV6 - if (err == 0) /* Link the IPv6 list to the end of the IPv4 list */ - { - struct ifi_info **p = &intfList; - while (*p) p = &(*p)->ifi_next; - *p = get_ifi_info(AF_INET6, mDNStrue); - } -#endif - - if (err == 0) - { - struct ifi_info *i = intfList; - while (i) - { - if ( ((i->ifi_addr->sa_family == AF_INET) -#if HAVE_IPV6 - || (i->ifi_addr->sa_family == AF_INET6) -#endif - ) && (i->ifi_flags & IFF_UP) && !(i->ifi_flags & IFF_POINTOPOINT)) - { - if (i->ifi_flags & IFF_LOOPBACK) - { - if (firstLoopback == NULL) - firstLoopback = i; - } - else - { - if (SetupOneInterface(m, i->ifi_addr, i->ifi_netmask, i->ifi_name, i->ifi_index) == 0) - if (i->ifi_addr->sa_family == AF_INET) - foundav4 = mDNStrue; - } - } - i = i->ifi_next; - } - - // If we found no normal interfaces but we did find a loopback interface, register the - // loopback interface. This allows self-discovery if no interfaces are configured. - // Temporary workaround: Multicast loopback on IPv6 interfaces appears not to work. - // In the interim, we skip loopback interface only if we found at least one v4 interface to use - // if ((m->HostInterfaces == NULL) && (firstLoopback != NULL)) - if (!foundav4 && firstLoopback) - (void) SetupOneInterface(m, firstLoopback->ifi_addr, firstLoopback->ifi_netmask, firstLoopback->ifi_name, firstLoopback->ifi_index); - } - - // Clean up. - if (intfList != NULL) free_ifi_info(intfList); - return err; - } - -#if USES_NETLINK - -// See for a description of NetLink - -// Open a socket that will receive interface change notifications -mDNSlocal mStatus OpenIfNotifySocket(int *pFD) - { - mStatus err = mStatus_NoError; - struct sockaddr_nl snl; - int sock; - int ret; - - sock = socket(AF_NETLINK, SOCK_RAW, NETLINK_ROUTE); - if (sock < 0) - return errno; - - // Configure read to be non-blocking because inbound msg size is not known in advance - (void) fcntl(sock, F_SETFL, O_NONBLOCK); - - /* Subscribe the socket to Link & IP addr notifications. */ - mDNSPlatformMemZero(&snl, sizeof snl); - snl.nl_family = AF_NETLINK; - snl.nl_groups = RTMGRP_LINK | RTMGRP_IPV4_IFADDR; - ret = bind(sock, (struct sockaddr *) &snl, sizeof snl); - if (0 == ret) - *pFD = sock; - else - err = errno; - - return err; - } - -#if MDNS_DEBUGMSGS -mDNSlocal void PrintNetLinkMsg(const struct nlmsghdr *pNLMsg) - { - const char *kNLMsgTypes[] = { "", "NLMSG_NOOP", "NLMSG_ERROR", "NLMSG_DONE", "NLMSG_OVERRUN" }; - const char *kNLRtMsgTypes[] = { "RTM_NEWLINK", "RTM_DELLINK", "RTM_GETLINK", "RTM_NEWADDR", "RTM_DELADDR", "RTM_GETADDR" }; - - printf("nlmsghdr len=%d, type=%s, flags=0x%x\n", pNLMsg->nlmsg_len, - pNLMsg->nlmsg_type < RTM_BASE ? kNLMsgTypes[pNLMsg->nlmsg_type] : kNLRtMsgTypes[pNLMsg->nlmsg_type - RTM_BASE], - pNLMsg->nlmsg_flags); - - if (RTM_NEWLINK <= pNLMsg->nlmsg_type && pNLMsg->nlmsg_type <= RTM_GETLINK) - { - struct ifinfomsg *pIfInfo = (struct ifinfomsg*) NLMSG_DATA(pNLMsg); - printf("ifinfomsg family=%d, type=%d, index=%d, flags=0x%x, change=0x%x\n", pIfInfo->ifi_family, - pIfInfo->ifi_type, pIfInfo->ifi_index, pIfInfo->ifi_flags, pIfInfo->ifi_change); - - } - else if (RTM_NEWADDR <= pNLMsg->nlmsg_type && pNLMsg->nlmsg_type <= RTM_GETADDR) - { - struct ifaddrmsg *pIfAddr = (struct ifaddrmsg*) NLMSG_DATA(pNLMsg); - printf("ifaddrmsg family=%d, index=%d, flags=0x%x\n", pIfAddr->ifa_family, - pIfAddr->ifa_index, pIfAddr->ifa_flags); - } - printf("\n"); - } -#endif - -mDNSlocal mDNSu32 ProcessRoutingNotification(int sd) -// Read through the messages on sd and if any indicate that any interface records should -// be torn down and rebuilt, return affected indices as a bitmask. Otherwise return 0. - { - ssize_t readCount; - char buff[4096]; - struct nlmsghdr *pNLMsg = (struct nlmsghdr*) buff; - mDNSu32 result = 0; - - // The structure here is more complex than it really ought to be because, - // unfortunately, there's no good way to size a buffer in advance large - // enough to hold all pending data and so avoid message fragmentation. - // (Note that FIONREAD is not supported on AF_NETLINK.) - - readCount = read(sd, buff, sizeof buff); - while (1) - { - // Make sure we've got an entire nlmsghdr in the buffer, and payload, too. - // If not, discard already-processed messages in buffer and read more data. - if (((char*) &pNLMsg[1] > (buff + readCount)) || // i.e. *pNLMsg extends off end of buffer - ((char*) pNLMsg + pNLMsg->nlmsg_len > (buff + readCount))) - { - if (buff < (char*) pNLMsg) // we have space to shuffle - { - // discard processed data - readCount -= ((char*) pNLMsg - buff); - memmove(buff, pNLMsg, readCount); - pNLMsg = (struct nlmsghdr*) buff; - - // read more data - readCount += read(sd, buff + readCount, sizeof buff - readCount); - continue; // spin around and revalidate with new readCount - } - else - break; // Otherwise message does not fit in buffer - } - -#if MDNS_DEBUGMSGS - PrintNetLinkMsg(pNLMsg); -#endif - - // Process the NetLink message - if (pNLMsg->nlmsg_type == RTM_GETLINK || pNLMsg->nlmsg_type == RTM_NEWLINK) - result |= 1 << ((struct ifinfomsg*) NLMSG_DATA(pNLMsg))->ifi_index; - else if (pNLMsg->nlmsg_type == RTM_DELADDR || pNLMsg->nlmsg_type == RTM_NEWADDR) - result |= 1 << ((struct ifaddrmsg*) NLMSG_DATA(pNLMsg))->ifa_index; - - // Advance pNLMsg to the next message in the buffer - if ((pNLMsg->nlmsg_flags & NLM_F_MULTI) != 0 && pNLMsg->nlmsg_type != NLMSG_DONE) - { - ssize_t len = readCount - ((char*)pNLMsg - buff); - pNLMsg = NLMSG_NEXT(pNLMsg, len); - } - else - break; // all done! - } - - return result; - } - -#else // USES_NETLINK - -// Open a socket that will receive interface change notifications -mDNSlocal mStatus OpenIfNotifySocket(int *pFD) - { - *pFD = socket(AF_ROUTE, SOCK_RAW, 0); - - if (*pFD < 0) - return mStatus_UnknownErr; - - // Configure read to be non-blocking because inbound msg size is not known in advance - (void) fcntl(*pFD, F_SETFL, O_NONBLOCK); - - return mStatus_NoError; - } - -#if MDNS_DEBUGMSGS -mDNSlocal void PrintRoutingSocketMsg(const struct ifa_msghdr *pRSMsg) - { - const char *kRSMsgTypes[] = { "", "RTM_ADD", "RTM_DELETE", "RTM_CHANGE", "RTM_GET", "RTM_LOSING", - "RTM_REDIRECT", "RTM_MISS", "RTM_LOCK", "RTM_OLDADD", "RTM_OLDDEL", "RTM_RESOLVE", - "RTM_NEWADDR", "RTM_DELADDR", "RTM_IFINFO", "RTM_NEWMADDR", "RTM_DELMADDR" }; - - int index = pRSMsg->ifam_type == RTM_IFINFO ? ((struct if_msghdr*) pRSMsg)->ifm_index : pRSMsg->ifam_index; - - printf("ifa_msghdr len=%d, type=%s, index=%d\n", pRSMsg->ifam_msglen, kRSMsgTypes[pRSMsg->ifam_type], index); - } -#endif - -mDNSlocal mDNSu32 ProcessRoutingNotification(int sd) -// Read through the messages on sd and if any indicate that any interface records should -// be torn down and rebuilt, return affected indices as a bitmask. Otherwise return 0. - { - ssize_t readCount; - char buff[4096]; - struct ifa_msghdr *pRSMsg = (struct ifa_msghdr*) buff; - mDNSu32 result = 0; - - readCount = read(sd, buff, sizeof buff); - if (readCount < (ssize_t) sizeof(struct ifa_msghdr)) - return mStatus_UnsupportedErr; // cannot decipher message - -#if MDNS_DEBUGMSGS - PrintRoutingSocketMsg(pRSMsg); -#endif - - // Process the message - if (pRSMsg->ifam_type == RTM_NEWADDR || pRSMsg->ifam_type == RTM_DELADDR || - pRSMsg->ifam_type == RTM_IFINFO) - { - if (pRSMsg->ifam_type == RTM_IFINFO) - result |= 1 << ((struct if_msghdr*) pRSMsg)->ifm_index; - else - result |= 1 << pRSMsg->ifam_index; - } - - return result; - } - -#endif // USES_NETLINK - -// Called when data appears on interface change notification socket -mDNSlocal void InterfaceChangeCallback(int fd, short filter, void *context) - { - IfChangeRec *pChgRec = (IfChangeRec*) context; - fd_set readFDs; - mDNSu32 changedInterfaces = 0; - struct timeval zeroTimeout = { 0, 0 }; - - (void)fd; // Unused - (void)filter; // Unused - - FD_ZERO(&readFDs); - FD_SET(pChgRec->NotifySD, &readFDs); - - do - { - changedInterfaces |= ProcessRoutingNotification(pChgRec->NotifySD); - } - while (0 < select(pChgRec->NotifySD + 1, &readFDs, (fd_set*) NULL, (fd_set*) NULL, &zeroTimeout)); - - // Currently we rebuild the entire interface list whenever any interface change is - // detected. If this ever proves to be a performance issue in a multi-homed - // configuration, more care should be paid to changedInterfaces. - if (changedInterfaces) - mDNSPlatformPosixRefreshInterfaceList(pChgRec->mDNS); - } - -// Register with either a Routing Socket or RtNetLink to listen for interface changes. -mDNSlocal mStatus WatchForInterfaceChange(mDNS *const m) - { - mStatus err; - IfChangeRec *pChgRec; - - pChgRec = (IfChangeRec*) mDNSPlatformMemAllocate(sizeof *pChgRec); - if (pChgRec == NULL) - return mStatus_NoMemoryErr; - - pChgRec->mDNS = m; - err = OpenIfNotifySocket(&pChgRec->NotifySD); - if (err == 0) - err = mDNSPosixAddFDToEventLoop(pChgRec->NotifySD, InterfaceChangeCallback, pChgRec); - - return err; - } - -// Test to see if we're the first client running on UDP port 5353, by trying to bind to 5353 without using SO_REUSEPORT. -// If we fail, someone else got here first. That's not a big problem; we can share the port for multicast responses -- -// we just need to be aware that we shouldn't expect to successfully receive unicast UDP responses. -mDNSlocal mDNSBool mDNSPlatformInit_CanReceiveUnicast(void) - { - int err; - int s = socket(AF_INET, SOCK_DGRAM, IPPROTO_UDP); - struct sockaddr_in s5353; - s5353.sin_family = AF_INET; - s5353.sin_port = MulticastDNSPort.NotAnInteger; - s5353.sin_addr.s_addr = 0; - err = bind(s, (struct sockaddr *)&s5353, sizeof(s5353)); - close(s); - if (err) debugf("No unicast UDP responses"); - else debugf("Unicast UDP responses okay"); - return(err == 0); - } - -// mDNS core calls this routine to initialise the platform-specific data. -mDNSexport mStatus mDNSPlatformInit(mDNS *const m) - { - int err = 0; - struct sockaddr sa; - assert(m != NULL); - - if (mDNSPlatformInit_CanReceiveUnicast()) m->CanReceiveUnicastOn5353 = mDNStrue; - - // Tell mDNS core the names of this machine. - - // Set up the nice label - m->nicelabel.c[0] = 0; - GetUserSpecifiedFriendlyComputerName(&m->nicelabel); - if (m->nicelabel.c[0] == 0) MakeDomainLabelFromLiteralString(&m->nicelabel, "Computer"); - - // Set up the RFC 1034-compliant label - m->hostlabel.c[0] = 0; - GetUserSpecifiedRFC1034ComputerName(&m->hostlabel); - if (m->hostlabel.c[0] == 0) MakeDomainLabelFromLiteralString(&m->hostlabel, "Computer"); - - mDNS_SetFQDN(m); - - sa.sa_family = AF_INET; - m->p->unicastSocket4 = -1; - if (err == mStatus_NoError) err = SetupSocket(&sa, zeroIPPort, 0, &m->p->unicastSocket4); -#if HAVE_IPV6 - sa.sa_family = AF_INET6; - m->p->unicastSocket6 = -1; - if (err == mStatus_NoError) err = SetupSocket(&sa, zeroIPPort, 0, &m->p->unicastSocket6); -#endif - - // Tell mDNS core about the network interfaces on this machine. - if (err == mStatus_NoError) err = SetupInterfaceList(m); - - // Tell mDNS core about DNS Servers - mDNS_Lock(m); - if (err == mStatus_NoError) ParseDNSServers(m, uDNS_SERVERS_FILE); - mDNS_Unlock(m); - - if (err == mStatus_NoError) - { - err = WatchForInterfaceChange(m); - // Failure to observe interface changes is non-fatal. - if (err != mStatus_NoError) - { - fprintf(stderr, "mDNS(%d) WARNING: Unable to detect interface changes (%d).\n", getpid(), err); - err = mStatus_NoError; - } - } - - // We don't do asynchronous initialization on the Posix platform, so by the time - // we get here the setup will already have succeeded or failed. If it succeeded, - // we should just call mDNSCoreInitComplete() immediately. - if (err == mStatus_NoError) - mDNSCoreInitComplete(m, mStatus_NoError); - - return PosixErrorToStatus(err); - } - -// mDNS core calls this routine to clean up the platform-specific data. -// In our case all we need to do is to tear down every network interface. -mDNSexport void mDNSPlatformClose(mDNS *const m) - { - assert(m != NULL); - ClearInterfaceList(m); - if (m->p->unicastSocket4 != -1) assert(close(m->p->unicastSocket4) == 0); -#if HAVE_IPV6 - if (m->p->unicastSocket6 != -1) assert(close(m->p->unicastSocket6) == 0); -#endif - } - -mDNSexport mStatus mDNSPlatformPosixRefreshInterfaceList(mDNS *const m) - { - int err; - ClearInterfaceList(m); - err = SetupInterfaceList(m); - return PosixErrorToStatus(err); - } - -#if COMPILER_LIKES_PRAGMA_MARK -#pragma mark ***** Locking -#endif - -// On the Posix platform, locking is a no-op because we only ever enter -// mDNS core on the main thread. - -// mDNS core calls this routine when it wants to prevent -// the platform from reentering mDNS core code. -mDNSexport void mDNSPlatformLock (const mDNS *const m) - { - (void) m; // Unused - } - -// mDNS core calls this routine when it release the lock taken by -// mDNSPlatformLock and allow the platform to reenter mDNS core code. -mDNSexport void mDNSPlatformUnlock (const mDNS *const m) - { - (void) m; // Unused - } - -#if COMPILER_LIKES_PRAGMA_MARK -#pragma mark ***** Strings -#endif - -// mDNS core calls this routine to copy C strings. -// On the Posix platform this maps directly to the ANSI C strcpy. -mDNSexport void mDNSPlatformStrCopy(void *dst, const void *src) - { - strcpy((char *)dst, (char *)src); - } - -// mDNS core calls this routine to get the length of a C string. -// On the Posix platform this maps directly to the ANSI C strlen. -mDNSexport mDNSu32 mDNSPlatformStrLen (const void *src) - { - return strlen((char*)src); - } - -// mDNS core calls this routine to copy memory. -// On the Posix platform this maps directly to the ANSI C memcpy. -mDNSexport void mDNSPlatformMemCopy(void *dst, const void *src, mDNSu32 len) - { - memcpy(dst, src, len); - } - -// mDNS core calls this routine to test whether blocks of memory are byte-for-byte -// identical. On the Posix platform this is a simple wrapper around ANSI C memcmp. -mDNSexport mDNSBool mDNSPlatformMemSame(const void *dst, const void *src, mDNSu32 len) - { - return memcmp(dst, src, len) == 0; - } - -// mDNS core calls this routine to clear blocks of memory. -// On the Posix platform this is a simple wrapper around ANSI C memset. -mDNSexport void mDNSPlatformMemZero(void *dst, mDNSu32 len) - { - memset(dst, 0, len); - } - -mDNSexport void * mDNSPlatformMemAllocate(mDNSu32 len) { return(malloc(len)); } -mDNSexport void mDNSPlatformMemFree (void *mem) { free(mem); } - -mDNSexport mDNSu32 mDNSPlatformRandomSeed(void) - { - struct timeval tv; - gettimeofday(&tv, NULL); - return(tv.tv_usec); - } - -mDNSexport mDNSs32 mDNSPlatformOneSecond = 1024; - -mDNSexport mStatus mDNSPlatformTimeInit(void) - { - // No special setup is required on Posix -- we just use gettimeofday(); - // This is not really safe, because gettimeofday can go backwards if the user manually changes the date or time - // We should find a better way to do this - return(mStatus_NoError); - } - -mDNSexport mDNSs32 mDNSPlatformRawTime() - { - struct timeval tv; - gettimeofday(&tv, NULL); - // tv.tv_sec is seconds since 1st January 1970 (GMT, with no adjustment for daylight savings time) - // tv.tv_usec is microseconds since the start of this second (i.e. values 0 to 999999) - // We use the lower 22 bits of tv.tv_sec for the top 22 bits of our result - // and we multiply tv.tv_usec by 16 / 15625 to get a value in the range 0-1023 to go in the bottom 10 bits. - // This gives us a proper modular (cyclic) counter that has a resolution of roughly 1ms (actually 1/1024 second) - // and correctly cycles every 2^22 seconds (4194304 seconds = approx 48 days). - return((tv.tv_sec << 10) | (tv.tv_usec * 16 / 15625)); - } - -mDNSexport mDNSs32 mDNSPlatformUTC(void) - { - return time(NULL); - } - -mDNSexport void mDNSPlatformSendWakeupPacket(mDNS *const m, mDNSInterfaceID InterfaceID, char *EthAddr, char *IPAddr, int iteration) - { - (void) m; - (void) InterfaceID; - (void) EthAddr; - (void) IPAddr; - (void) iteration; - } - -mDNSexport mDNSBool mDNSPlatformValidRecordForInterface(AuthRecord *rr, const NetworkInterfaceInfo *intf) - { - (void) rr; - (void) intf; - - return 1; - } - -mDNSlocal void mDNSPosixAddToFDSet(int *nfds, fd_set *readfds, int s) - { - if (*nfds < s + 1) *nfds = s + 1; - FD_SET(s, readfds); - } - -mDNSexport void mDNSPosixGetFDSet(mDNS *m, int *nfds, fd_set *readfds, struct timeval *timeout) - { - mDNSs32 ticks; - struct timeval interval; - - // 1. Call mDNS_Execute() to let mDNSCore do what it needs to do - mDNSs32 nextevent = mDNS_Execute(m); - - // 2. Build our list of active file descriptors - PosixNetworkInterface *info = (PosixNetworkInterface *)(m->HostInterfaces); - if (m->p->unicastSocket4 != -1) mDNSPosixAddToFDSet(nfds, readfds, m->p->unicastSocket4); -#if HAVE_IPV6 - if (m->p->unicastSocket6 != -1) mDNSPosixAddToFDSet(nfds, readfds, m->p->unicastSocket6); -#endif - while (info) - { - if (info->multicastSocket4 != -1) mDNSPosixAddToFDSet(nfds, readfds, info->multicastSocket4); -#if HAVE_IPV6 - if (info->multicastSocket6 != -1) mDNSPosixAddToFDSet(nfds, readfds, info->multicastSocket6); -#endif - info = (PosixNetworkInterface *)(info->coreIntf.next); - } - - // 3. Calculate the time remaining to the next scheduled event (in struct timeval format) - ticks = nextevent - mDNS_TimeNow(m); - if (ticks < 1) ticks = 1; - interval.tv_sec = ticks >> 10; // The high 22 bits are seconds - interval.tv_usec = ((ticks & 0x3FF) * 15625) / 16; // The low 10 bits are 1024ths - - // 4. If client's proposed timeout is more than what we want, then reduce it - if (timeout->tv_sec > interval.tv_sec || - (timeout->tv_sec == interval.tv_sec && timeout->tv_usec > interval.tv_usec)) - *timeout = interval; - } - -mDNSexport void mDNSPosixProcessFDSet(mDNS *const m, fd_set *readfds) - { - PosixNetworkInterface *info; - assert(m != NULL); - assert(readfds != NULL); - info = (PosixNetworkInterface *)(m->HostInterfaces); - - if (m->p->unicastSocket4 != -1 && FD_ISSET(m->p->unicastSocket4, readfds)) - { - FD_CLR(m->p->unicastSocket4, readfds); - SocketDataReady(m, NULL, m->p->unicastSocket4); - } -#if HAVE_IPV6 - if (m->p->unicastSocket6 != -1 && FD_ISSET(m->p->unicastSocket6, readfds)) - { - FD_CLR(m->p->unicastSocket6, readfds); - SocketDataReady(m, NULL, m->p->unicastSocket6); - } -#endif - - while (info) - { - if (info->multicastSocket4 != -1 && FD_ISSET(info->multicastSocket4, readfds)) - { - FD_CLR(info->multicastSocket4, readfds); - SocketDataReady(m, info, info->multicastSocket4); - } -#if HAVE_IPV6 - if (info->multicastSocket6 != -1 && FD_ISSET(info->multicastSocket6, readfds)) - { - FD_CLR(info->multicastSocket6, readfds); - SocketDataReady(m, info, info->multicastSocket6); - } -#endif - info = (PosixNetworkInterface *)(info->coreIntf.next); - } - } - -// update gMaxFD -mDNSlocal void DetermineMaxEventFD(void) - { - PosixEventSource *iSource; - - gMaxFD = 0; - for (iSource=(PosixEventSource*)gEventSources.Head; iSource; iSource = iSource->Next) - if (gMaxFD < iSource->fd) - gMaxFD = iSource->fd; - } - -// Add a file descriptor to the set that mDNSPosixRunEventLoopOnce() listens to. -mStatus mDNSPosixAddFDToEventLoop(int fd, mDNSPosixEventCallback callback, void *context) - { - PosixEventSource *newSource; - - if (gEventSources.LinkOffset == 0) - InitLinkedList(&gEventSources, offsetof(PosixEventSource, Next)); - - if (fd >= (int) FD_SETSIZE || fd < 0) - return mStatus_UnsupportedErr; - if (callback == NULL) - return mStatus_BadParamErr; - - newSource = (PosixEventSource*) malloc(sizeof *newSource); - if (NULL == newSource) - return mStatus_NoMemoryErr; - - newSource->Callback = callback; - newSource->Context = context; - newSource->fd = fd; - - AddToTail(&gEventSources, newSource); - FD_SET(fd, &gEventFDs); - - DetermineMaxEventFD(); - - return mStatus_NoError; - } - -// Remove a file descriptor from the set that mDNSPosixRunEventLoopOnce() listens to. -mStatus mDNSPosixRemoveFDFromEventLoop(int fd) - { - PosixEventSource *iSource; - - for (iSource=(PosixEventSource*)gEventSources.Head; iSource; iSource = iSource->Next) - { - if (fd == iSource->fd) - { - FD_CLR(fd, &gEventFDs); - RemoveFromList(&gEventSources, iSource); - free(iSource); - DetermineMaxEventFD(); - return mStatus_NoError; - } - } - return mStatus_NoSuchNameErr; - } - -// Simply note the received signal in gEventSignals. -mDNSlocal void NoteSignal(int signum) - { - sigaddset(&gEventSignals, signum); - } - -// Tell the event package to listen for signal and report it in mDNSPosixRunEventLoopOnce(). -mStatus mDNSPosixListenForSignalInEventLoop(int signum) - { - struct sigaction action; - mStatus err; - - mDNSPlatformMemZero(&action, sizeof action); // more portable than member-wise assignment - action.sa_handler = NoteSignal; - err = sigaction(signum, &action, (struct sigaction*) NULL); - - sigaddset(&gEventSignalSet, signum); - - return err; - } - -// Tell the event package to stop listening for signal in mDNSPosixRunEventLoopOnce(). -mStatus mDNSPosixIgnoreSignalInEventLoop(int signum) - { - struct sigaction action; - mStatus err; - - mDNSPlatformMemZero(&action, sizeof action); // more portable than member-wise assignment - action.sa_handler = SIG_DFL; - err = sigaction(signum, &action, (struct sigaction*) NULL); - - sigdelset(&gEventSignalSet, signum); - - return err; - } - -// Do a single pass through the attendent event sources and dispatch any found to their callbacks. -// Return as soon as internal timeout expires, or a signal we're listening for is received. -mStatus mDNSPosixRunEventLoopOnce(mDNS *m, const struct timeval *pTimeout, - sigset_t *pSignalsReceived, mDNSBool *pDataDispatched) - { - fd_set listenFDs = gEventFDs; - int fdMax = 0, numReady; - struct timeval timeout = *pTimeout; - - // Include the sockets that are listening to the wire in our select() set - mDNSPosixGetFDSet(m, &fdMax, &listenFDs, &timeout); // timeout may get modified - if (fdMax < gMaxFD) - fdMax = gMaxFD; - - numReady = select(fdMax + 1, &listenFDs, (fd_set*) NULL, (fd_set*) NULL, &timeout); - - // If any data appeared, invoke its callback - if (numReady > 0) - { - PosixEventSource *iSource; - - (void) mDNSPosixProcessFDSet(m, &listenFDs); // call this first to process wire data for clients - - for (iSource=(PosixEventSource*)gEventSources.Head; iSource; iSource = iSource->Next) - { - if (FD_ISSET(iSource->fd, &listenFDs)) - { - iSource->Callback(iSource->fd, 0, iSource->Context); - break; // in case callback removed elements from gEventSources - } - } - *pDataDispatched = mDNStrue; - } - else - *pDataDispatched = mDNSfalse; - - (void) sigprocmask(SIG_BLOCK, &gEventSignalSet, (sigset_t*) NULL); - *pSignalsReceived = gEventSignals; - sigemptyset(&gEventSignals); - (void) sigprocmask(SIG_UNBLOCK, &gEventSignalSet, (sigset_t*) NULL); - - return mStatus_NoError; - } diff -Nru qtcreator-2.5.0/src/tools/mdnssd/mDNSPosix.h qtcreator-2.5.2/src/tools/mdnssd/mDNSPosix.h --- qtcreator-2.5.0/src/tools/mdnssd/mDNSPosix.h 2012-05-09 12:13:19.000000000 +0000 +++ qtcreator-2.5.2/src/tools/mdnssd/mDNSPosix.h 1970-01-01 00:00:00.000000000 +0000 @@ -1,85 +0,0 @@ -/* -*- Mode: C; tab-width: 4 -*- - * - * Copyright (c) 2002-2004 Apple Computer, Inc. All rights reserved. - * - * Licensed under the Apache License, Version 2.0 (the "License"); - * you may not use this file except in compliance with the License. - * You may obtain a copy of the License at - * - * http://www.apache.org/licenses/LICENSE-2.0 - * - * Unless required by applicable law or agreed to in writing, software - * distributed under the License is distributed on an "AS IS" BASIS, - * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. - * See the License for the specific language governing permissions and - * limitations under the License. - */ - -#ifndef __mDNSPlatformPosix_h -#define __mDNSPlatformPosix_h - -#include -#include - -#ifdef __cplusplus - extern "C" { -#endif - -// PosixNetworkInterface is a record extension of the core NetworkInterfaceInfo -// type that supports extra fields needed by the Posix platform. -// -// IMPORTANT: coreIntf must be the first field in the structure because -// we cast between pointers to the two different types regularly. - -typedef struct PosixNetworkInterface PosixNetworkInterface; - -struct PosixNetworkInterface - { - NetworkInterfaceInfo coreIntf; - const char * intfName; - PosixNetworkInterface * aliasIntf; - int index; - int multicastSocket4; -#if HAVE_IPV6 - int multicastSocket6; -#endif - }; - -// This is a global because debugf_() needs to be able to check its value -extern int gMDNSPlatformPosixVerboseLevel; - -struct mDNS_PlatformSupport_struct - { - int unicastSocket4; -#if HAVE_IPV6 - int unicastSocket6; -#endif - }; - -#define uDNS_SERVERS_FILE "/etc/resolv.conf" -extern int ParseDNSServers(mDNS *m, const char *filePath); -extern mStatus mDNSPlatformPosixRefreshInterfaceList(mDNS *const m); - // See comment in implementation. - -// Call mDNSPosixGetFDSet before calling select(), to update the parameters -// as may be necessary to meet the needs of the mDNSCore code. -// The timeout pointer MUST NOT be NULL. -// Set timeout->tv_sec to 0x3FFFFFFF if you want to have effectively no timeout -// After calling mDNSPosixGetFDSet(), call select(nfds, &readfds, NULL, NULL, &timeout); as usual -// After select() returns, call mDNSPosixProcessFDSet() to let mDNSCore do its work -extern void mDNSPosixGetFDSet(mDNS *m, int *nfds, fd_set *readfds, struct timeval *timeout); -extern void mDNSPosixProcessFDSet(mDNS *const m, fd_set *readfds); - -typedef void (*mDNSPosixEventCallback)(int fd, short filter, void *context); - -extern mStatus mDNSPosixAddFDToEventLoop( int fd, mDNSPosixEventCallback callback, void *context); -extern mStatus mDNSPosixRemoveFDFromEventLoop( int fd); -extern mStatus mDNSPosixListenForSignalInEventLoop( int signum); -extern mStatus mDNSPosixIgnoreSignalInEventLoop( int signum); -extern mStatus mDNSPosixRunEventLoopOnce( mDNS *m, const struct timeval *pTimeout, sigset_t *pSignalsReceived, mDNSBool *pDataDispatched); - -#ifdef __cplusplus - } -#endif - -#endif diff -Nru qtcreator-2.5.0/src/tools/mdnssd/mDNSUNP.c qtcreator-2.5.2/src/tools/mdnssd/mDNSUNP.c --- qtcreator-2.5.0/src/tools/mdnssd/mDNSUNP.c 2012-05-09 12:13:19.000000000 +0000 +++ qtcreator-2.5.2/src/tools/mdnssd/mDNSUNP.c 1970-01-01 00:00:00.000000000 +0000 @@ -1,719 +0,0 @@ -/* -*- Mode: C; tab-width: 4 -*- - * - * Copyright (c) 2002-2004 Apple Computer, Inc. All rights reserved. - * - * Licensed under the Apache License, Version 2.0 (the "License"); - * you may not use this file except in compliance with the License. - * You may obtain a copy of the License at - * - * http://www.apache.org/licenses/LICENSE-2.0 - * - * Unless required by applicable law or agreed to in writing, software - * distributed under the License is distributed on an "AS IS" BASIS, - * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. - * See the License for the specific language governing permissions and - * limitations under the License. - */ - -#include "mDNSUNP.h" - -#include -#include -#include -#include -#include -#include -#include -#include -#include - -/* Some weird platforms derived from 4.4BSD Lite (e.g. EFI) need the ALIGN(P) - macro, usually defined in or someplace like that, to make sure the - CMSG_NXTHDR macro is well-formed. On such platforms, the symbol NEED_ALIGN_MACRO - should be set to the name of the header to include to get the ALIGN(P) macro. -*/ -#ifdef NEED_ALIGN_MACRO -#include NEED_ALIGN_MACRO -#endif - -/* Solaris defined SIOCGIFCONF etc in but - other platforms don't even have that include file. So, - if we haven't yet got a definition, let's try to find - . -*/ - -#ifndef SIOCGIFCONF - #include -#endif - -/* sockaddr_dl is only referenced if we're using IP_RECVIF, - so only include the header in that case. -*/ - -#ifdef IP_RECVIF - #include -#endif - -#if defined(AF_INET6) && HAVE_IPV6 && !HAVE_LINUX -#include -#include -// Note: netinet/in_var.h implicitly includes netinet6/in6_var.h for us -#endif - -#if defined(AF_INET6) && HAVE_IPV6 && HAVE_LINUX -#include -#include - -/* Converts a prefix length to IPv6 network mask */ -void plen_to_mask(int plen, char *addr) { - int i; - int colons=7; /* Number of colons in IPv6 address */ - int bits_in_block=16; /* Bits per IPv6 block */ - for(i=0;i<=colons;i++) { - int block, ones=0xffff, ones_in_block; - if (plen>bits_in_block) ones_in_block=bits_in_block; - else ones_in_block=plen; - block = ones & (ones << (bits_in_block-ones_in_block)); - i==0 ? sprintf(addr, "%x", block) : sprintf(addr, "%s:%x", addr, block); - plen -= ones_in_block; - } - } - -/* Gets IPv6 interface information from the /proc filesystem in linux*/ -struct ifi_info *get_ifi_info_linuxv6(int family, int doaliases) - { - struct ifi_info *ifi, *ifihead, **ifipnext, *ifipold, **ifiptr; - FILE *fp; - char addr[8][5]; - int flags, myflags, index, plen, scope; - char ifname[9], lastname[IFNAMSIZ]; - char addr6[32+7+1]; /* don't forget the seven ':' */ - struct addrinfo hints, *res0; - struct sockaddr_in6 *sin6; - struct in6_addr *addrptr; - int err; - int sockfd = -1; - struct ifreq ifr; - - res0=NULL; - ifihead = NULL; - ifipnext = &ifihead; - lastname[0] = 0; - - if ((fp = fopen(PROC_IFINET6_PATH, "r")) != NULL) { - sockfd = socket(AF_INET6, SOCK_DGRAM, 0); - if (sockfd < 0) { - goto gotError; - } - while (fscanf(fp, - "%4s%4s%4s%4s%4s%4s%4s%4s %02x %02x %02x %02x %8s\n", - addr[0],addr[1],addr[2],addr[3], - addr[4],addr[5],addr[6],addr[7], - &index, &plen, &scope, &flags, ifname) != EOF) { - - myflags = 0; - if (strncmp(lastname, ifname, IFNAMSIZ) == 0) { - if (doaliases == 0) - continue; /* already processed this interface */ - myflags = IFI_ALIAS; - } - memcpy(lastname, ifname, IFNAMSIZ); - ifi = (struct ifi_info*)calloc(1, sizeof(struct ifi_info)); - if (ifi == NULL) { - goto gotError; - } - - ifipold = *ifipnext; /* need this later */ - ifiptr = ifipnext; - *ifipnext = ifi; /* prev points to this new one */ - ifipnext = &ifi->ifi_next; /* pointer to next one goes here */ - - sprintf(addr6, "%s:%s:%s:%s:%s:%s:%s:%s", - addr[0],addr[1],addr[2],addr[3], - addr[4],addr[5],addr[6],addr[7]); - - /* Add address of the interface */ - memset(&hints, 0, sizeof(hints)); - hints.ai_family = AF_INET6; - hints.ai_flags = AI_NUMERICHOST; - err = getaddrinfo(addr6, NULL, &hints, &res0); - if (err) { - goto gotError; - } - ifi->ifi_addr = calloc(1, sizeof(struct sockaddr_in6)); - if (ifi->ifi_addr == NULL) { - goto gotError; - } - memcpy(ifi->ifi_addr, res0->ai_addr, sizeof(struct sockaddr_in6)); - - /* Add netmask of the interface */ - char ipv6addr[INET6_ADDRSTRLEN]; - plen_to_mask(plen, ipv6addr); - ifi->ifi_netmask = calloc(1, sizeof(struct sockaddr_in6)); - if (ifi->ifi_addr == NULL) { - goto gotError; - } - sin6=calloc(1, sizeof(struct sockaddr_in6)); - addrptr=calloc(1, sizeof(struct in6_addr)); - inet_pton(family, ipv6addr, addrptr); - sin6->sin6_family=family; - sin6->sin6_addr=*addrptr; - sin6->sin6_scope_id=scope; - memcpy(ifi->ifi_netmask, sin6, sizeof(struct sockaddr_in6)); - free(sin6); - - - /* Add interface name */ - memcpy(ifi->ifi_name, ifname, IFI_NAME); - - /* Add interface index */ - ifi->ifi_index = index; - - /* Add interface flags*/ - memcpy(ifr.ifr_name, ifname, IFNAMSIZ); - if (ioctl(sockfd, SIOCGIFFLAGS, &ifr) < 0) { - if (errno == EADDRNOTAVAIL) { - /* - * If the main interface is configured with no IP address but - * an alias interface exists with an IP address, you get - * EADDRNOTAVAIL for the main interface - */ - free(ifi->ifi_addr); - free(ifi); - ifipnext = ifiptr; - *ifipnext = ifipold; - continue; - } else { - goto gotError; - } - } - ifi->ifi_flags = ifr.ifr_flags; - freeaddrinfo(res0); - res0=NULL; - } - } - goto done; - - gotError: - if (ifihead != NULL) { - free_ifi_info(ifihead); - ifihead = NULL; - } - if (res0 != NULL) { - freeaddrinfo(res0); - res0=NULL; - } - done: - if (sockfd != -1) { - assert(close(sockfd) == 0); - } - return(ifihead); /* pointer to first structure in linked list */ - } -#endif // defined(AF_INET6) && HAVE_IPV6 && HAVE_LINUX - -struct ifi_info *get_ifi_info(int family, int doaliases) -{ - int junk; - struct ifi_info *ifi, *ifihead, **ifipnext, *ifipold, **ifiptr; - int sockfd, sockf6, len, lastlen, flags, myflags; -#ifdef NOT_HAVE_IF_NAMETOINDEX - int index = 200; -#endif - char *ptr, *buf, lastname[IFNAMSIZ], *cptr; - struct ifconf ifc; - struct ifreq *ifr, ifrcopy; - struct sockaddr_in *sinptr; - -#if defined(AF_INET6) && HAVE_IPV6 - struct sockaddr_in6 *sinptr6; -#endif - -#if defined(AF_INET6) && HAVE_IPV6 && HAVE_LINUX - if (family == AF_INET6) return get_ifi_info_linuxv6(family, doaliases); -#endif - - sockfd = -1; - sockf6 = -1; - buf = NULL; - ifihead = NULL; - - sockfd = socket(AF_INET, SOCK_DGRAM, 0); - if (sockfd < 0) { - goto gotError; - } - - lastlen = 0; - len = 100 * sizeof(struct ifreq); /* initial buffer size guess */ - for ( ; ; ) { - buf = (char*)malloc(len); - if (buf == NULL) { - goto gotError; - } - ifc.ifc_len = len; - ifc.ifc_buf = buf; - if (ioctl(sockfd, SIOCGIFCONF, &ifc) < 0) { - if (errno != EINVAL || lastlen != 0) { - goto gotError; - } - } else { - if (ifc.ifc_len == lastlen) - break; /* success, len has not changed */ - lastlen = ifc.ifc_len; - } - len += 10 * sizeof(struct ifreq); /* increment */ - free(buf); - } - ifihead = NULL; - ifipnext = &ifihead; - lastname[0] = 0; -/* end get_ifi_info1 */ - -/* include get_ifi_info2 */ - for (ptr = buf; ptr < buf + ifc.ifc_len; ) { - ifr = (struct ifreq *) ptr; - - /* Advance to next one in buffer */ - if (sizeof(struct ifreq) > sizeof(ifr->ifr_name) + GET_SA_LEN(ifr->ifr_addr)) - ptr += sizeof(struct ifreq); - else - ptr += sizeof(ifr->ifr_name) + GET_SA_LEN(ifr->ifr_addr); - -// fprintf(stderr, "intf %p name=%s AF=%d\n", index, ifr->ifr_name, ifr->ifr_addr.sa_family); - - if (ifr->ifr_addr.sa_family != family) - continue; /* ignore if not desired address family */ - - myflags = 0; - if ( (cptr = strchr(ifr->ifr_name, ':')) != NULL) - *cptr = 0; /* replace colon will null */ - if (strncmp(lastname, ifr->ifr_name, IFNAMSIZ) == 0) { - if (doaliases == 0) - continue; /* already processed this interface */ - myflags = IFI_ALIAS; - } - memcpy(lastname, ifr->ifr_name, IFNAMSIZ); - - ifrcopy = *ifr; - if (ioctl(sockfd, SIOCGIFFLAGS, &ifrcopy) < 0) { - goto gotError; - } - - flags = ifrcopy.ifr_flags; - if ((flags & IFF_UP) == 0) - continue; /* ignore if interface not up */ - - ifi = (struct ifi_info*)calloc(1, sizeof(struct ifi_info)); - if (ifi == NULL) { - goto gotError; - } - ifipold = *ifipnext; /* need this later */ - ifiptr = ifipnext; - *ifipnext = ifi; /* prev points to this new one */ - ifipnext = &ifi->ifi_next; /* pointer to next one goes here */ - - ifi->ifi_flags = flags; /* IFF_xxx values */ - ifi->ifi_myflags = myflags; /* IFI_xxx values */ -#ifndef NOT_HAVE_IF_NAMETOINDEX - ifi->ifi_index = if_nametoindex(ifr->ifr_name); -#else - ifrcopy = *ifr; -#ifdef SIOCGIFINDEX - if ( 0 >= ioctl(sockfd, SIOCGIFINDEX, &ifrcopy)) - ifi->ifi_index = ifrcopy.ifr_index; - else -#endif - ifi->ifi_index = index++; /* SIOCGIFINDEX is broken on Solaris 2.5ish, so fake it */ -#endif - memcpy(ifi->ifi_name, ifr->ifr_name, IFI_NAME); - ifi->ifi_name[IFI_NAME-1] = '\0'; -/* end get_ifi_info2 */ -/* include get_ifi_info3 */ - switch (ifr->ifr_addr.sa_family) { - case AF_INET: - sinptr = (struct sockaddr_in *) &ifr->ifr_addr; - if (ifi->ifi_addr == NULL) { - ifi->ifi_addr = (struct sockaddr*)calloc(1, sizeof(struct sockaddr_in)); - if (ifi->ifi_addr == NULL) { - goto gotError; - } - memcpy(ifi->ifi_addr, sinptr, sizeof(struct sockaddr_in)); - -#ifdef SIOCGIFNETMASK - if (ioctl(sockfd, SIOCGIFNETMASK, &ifrcopy) < 0) { - if (errno == EADDRNOTAVAIL) { - /* - * If the main interface is configured with no IP address but - * an alias interface exists with an IP address, you get - * EADDRNOTAVAIL for the main interface - */ - free(ifi->ifi_addr); - free(ifi); - ifipnext = ifiptr; - *ifipnext = ifipold; - continue; - } else { - goto gotError; - } - } - - ifi->ifi_netmask = (struct sockaddr*)calloc(1, sizeof(struct sockaddr_in)); - if (ifi->ifi_netmask == NULL) goto gotError; - sinptr = (struct sockaddr_in *) &ifrcopy.ifr_addr; - /* The BSD ioctls (including Mac OS X) stick some weird values in for sin_len and sin_family */ -#ifndef NOT_HAVE_SA_LEN - sinptr->sin_len = sizeof(struct sockaddr_in); -#endif - sinptr->sin_family = AF_INET; - memcpy(ifi->ifi_netmask, sinptr, sizeof(struct sockaddr_in)); -#endif - -#ifdef SIOCGIFBRDADDR - if (flags & IFF_BROADCAST) { - if (ioctl(sockfd, SIOCGIFBRDADDR, &ifrcopy) < 0) { - goto gotError; - } - sinptr = (struct sockaddr_in *) &ifrcopy.ifr_broadaddr; - /* The BSD ioctls (including Mac OS X) stick some weird values in for sin_len and sin_family */ -#ifndef NOT_HAVE_SA_LEN - sinptr->sin_len = sizeof( struct sockaddr_in ); -#endif - sinptr->sin_family = AF_INET; - ifi->ifi_brdaddr = (struct sockaddr*)calloc(1, sizeof(struct sockaddr_in)); - if (ifi->ifi_brdaddr == NULL) { - goto gotError; - } - memcpy(ifi->ifi_brdaddr, sinptr, sizeof(struct sockaddr_in)); - } -#endif - -#ifdef SIOCGIFDSTADDR - if (flags & IFF_POINTOPOINT) { - if (ioctl(sockfd, SIOCGIFDSTADDR, &ifrcopy) < 0) { - goto gotError; - } - sinptr = (struct sockaddr_in *) &ifrcopy.ifr_dstaddr; - /* The BSD ioctls (including Mac OS X) stick some weird values in for sin_len and sin_family */ -#ifndef NOT_HAVE_SA_LEN - sinptr->sin_len = sizeof( struct sockaddr_in ); -#endif - sinptr->sin_family = AF_INET; - ifi->ifi_dstaddr = (struct sockaddr*)calloc(1, sizeof(struct sockaddr_in)); - if (ifi->ifi_dstaddr == NULL) { - goto gotError; - } - memcpy(ifi->ifi_dstaddr, sinptr, sizeof(struct sockaddr_in)); - } -#endif - } - break; - -#if defined(AF_INET6) && HAVE_IPV6 - case AF_INET6: - sinptr6 = (struct sockaddr_in6 *) &ifr->ifr_addr; - if (ifi->ifi_addr == NULL) { - ifi->ifi_addr = calloc(1, sizeof(struct sockaddr_in6)); - if (ifi->ifi_addr == NULL) { - goto gotError; - } - - /* Some platforms (*BSD) inject the prefix in IPv6LL addresses */ - /* We need to strip that out */ - if (IN6_IS_ADDR_LINKLOCAL(&sinptr6->sin6_addr)) - sinptr6->sin6_addr.s6_addr[2] = sinptr6->sin6_addr.s6_addr[3] = 0; - memcpy(ifi->ifi_addr, sinptr6, sizeof(struct sockaddr_in6)); - -#ifdef SIOCGIFNETMASK_IN6 - { - struct in6_ifreq ifr6; - if (sockf6 == -1) - sockf6 = socket(AF_INET6, SOCK_DGRAM, 0); - memset(&ifr6, 0, sizeof(ifr6)); - memcpy(&ifr6.ifr_name, &ifr->ifr_name, sizeof(ifr6.ifr_name )); - memcpy(&ifr6.ifr_ifru.ifru_addr, &ifr->ifr_addr, sizeof(ifr6.ifr_ifru.ifru_addr)); - if (ioctl(sockf6, SIOCGIFNETMASK_IN6, &ifr6) < 0) { - if (errno == EADDRNOTAVAIL) { - /* - * If the main interface is configured with no IP address but - * an alias interface exists with an IP address, you get - * EADDRNOTAVAIL for the main interface - */ - free(ifi->ifi_addr); - free(ifi); - ifipnext = ifiptr; - *ifipnext = ifipold; - continue; - } else { - goto gotError; - } - } - ifi->ifi_netmask = (struct sockaddr*)calloc(1, sizeof(struct sockaddr_in6)); - if (ifi->ifi_netmask == NULL) goto gotError; - sinptr6 = (struct sockaddr_in6 *) &ifr6.ifr_ifru.ifru_addr; - memcpy(ifi->ifi_netmask, sinptr6, sizeof(struct sockaddr_in6)); - } -#endif - } - break; -#endif - - default: - break; - } - } - goto done; - -gotError: - if (ifihead != NULL) { - free_ifi_info(ifihead); - ifihead = NULL; - } - -done: - if (buf != NULL) { - free(buf); - } - if (sockfd != -1) { - junk = close(sockfd); - assert(junk == 0); - } - if (sockf6 != -1) { - junk = close(sockf6); - assert(junk == 0); - } - return(ifihead); /* pointer to first structure in linked list */ -} -/* end get_ifi_info3 */ - -/* include free_ifi_info */ -void -free_ifi_info(struct ifi_info *ifihead) -{ - struct ifi_info *ifi, *ifinext; - - for (ifi = ifihead; ifi != NULL; ifi = ifinext) { - if (ifi->ifi_addr != NULL) - free(ifi->ifi_addr); - if (ifi->ifi_netmask != NULL) - free(ifi->ifi_netmask); - if (ifi->ifi_brdaddr != NULL) - free(ifi->ifi_brdaddr); - if (ifi->ifi_dstaddr != NULL) - free(ifi->ifi_dstaddr); - ifinext = ifi->ifi_next; /* can't fetch ifi_next after free() */ - free(ifi); /* the ifi_info{} itself */ - } -} -/* end free_ifi_info */ - -ssize_t -recvfrom_flags(int fd, void *ptr, size_t nbytes, int *flagsp, - struct sockaddr *sa, socklen_t *salenptr, struct my_in_pktinfo *pktp, u_char *ttl) -{ - struct msghdr msg; - struct iovec iov[1]; - ssize_t n; - -#ifdef CMSG_FIRSTHDR - struct cmsghdr *cmptr; - union { - struct cmsghdr cm; - char control[1024]; - } control_un; - - *ttl = 255; // If kernel fails to provide TTL data then assume the TTL was 255 as it should be - - msg.msg_control = control_un.control; - msg.msg_controllen = sizeof(control_un.control); - msg.msg_flags = 0; -#else - memset(&msg, 0, sizeof(msg)); /* make certain msg_accrightslen = 0 */ -#endif /* CMSG_FIRSTHDR */ - - msg.msg_name = (char *) sa; - msg.msg_namelen = *salenptr; - iov[0].iov_base = (char *)ptr; - iov[0].iov_len = nbytes; - msg.msg_iov = iov; - msg.msg_iovlen = 1; - - if ( (n = recvmsg(fd, &msg, *flagsp)) < 0) - return(n); - - *salenptr = msg.msg_namelen; /* pass back results */ - if (pktp) { - /* 0.0.0.0, i/f = -1 */ - /* We set the interface to -1 so that the caller can - tell whether we returned a meaningful value or - just some default. Previously this code just - set the value to 0, but I'm concerned that 0 - might be a valid interface value. - */ - memset(pktp, 0, sizeof(struct my_in_pktinfo)); - pktp->ipi_ifindex = -1; - } -/* end recvfrom_flags1 */ - -/* include recvfrom_flags2 */ -#ifndef CMSG_FIRSTHDR - #warning CMSG_FIRSTHDR not defined. Will not be able to determine destination address, received interface, etc. - *flagsp = 0; /* pass back results */ - return(n); -#else - - *flagsp = msg.msg_flags; /* pass back results */ - if (msg.msg_controllen < (socklen_t)sizeof(struct cmsghdr) || - (msg.msg_flags & MSG_CTRUNC) || pktp == NULL) - return(n); - - for (cmptr = CMSG_FIRSTHDR(&msg); cmptr != NULL; - cmptr = CMSG_NXTHDR(&msg, cmptr)) { - -#ifdef IP_PKTINFO -#if in_pktinfo_definition_is_missing -struct in_pktinfo -{ - int ipi_ifindex; - struct in_addr ipi_spec_dst; - struct in_addr ipi_addr; -}; -#endif - if (cmptr->cmsg_level == IPPROTO_IP && - cmptr->cmsg_type == IP_PKTINFO) { - struct in_pktinfo *tmp; - struct sockaddr_in *sin = (struct sockaddr_in*)&pktp->ipi_addr; - - tmp = (struct in_pktinfo *) CMSG_DATA(cmptr); - sin->sin_family = AF_INET; - sin->sin_addr = tmp->ipi_addr; - sin->sin_port = 0; - pktp->ipi_ifindex = tmp->ipi_ifindex; - continue; - } -#endif - -#ifdef IP_RECVDSTADDR - if (cmptr->cmsg_level == IPPROTO_IP && - cmptr->cmsg_type == IP_RECVDSTADDR) { - struct sockaddr_in *sin = (struct sockaddr_in*)&pktp->ipi_addr; - - sin->sin_family = AF_INET; - sin->sin_addr = *(struct in_addr*)CMSG_DATA(cmptr); - sin->sin_port = 0; - continue; - } -#endif - -#ifdef IP_RECVIF - if (cmptr->cmsg_level == IPPROTO_IP && - cmptr->cmsg_type == IP_RECVIF) { - struct sockaddr_dl *sdl = (struct sockaddr_dl *) CMSG_DATA(cmptr); -#ifndef HAVE_BROKEN_RECVIF_NAME - int nameLen = (sdl->sdl_nlen < IFI_NAME - 1) ? sdl->sdl_nlen : (IFI_NAME - 1); - strncpy(pktp->ipi_ifname, sdl->sdl_data, nameLen); -#endif - pktp->ipi_ifindex = sdl->sdl_index; -#ifdef HAVE_BROKEN_RECVIF_NAME - if (sdl->sdl_index == 0) { - pktp->ipi_ifindex = *(uint_t*)sdl; - } -#endif - assert(pktp->ipi_ifname[IFI_NAME - 1] == 0); - // null terminated because of memset above - continue; - } -#endif - -#ifdef IP_RECVTTL - if (cmptr->cmsg_level == IPPROTO_IP && - cmptr->cmsg_type == IP_RECVTTL) { - *ttl = *(u_char*)CMSG_DATA(cmptr); - continue; - } - else if (cmptr->cmsg_level == IPPROTO_IP && - cmptr->cmsg_type == IP_TTL) { // some implementations seem to send IP_TTL instead of IP_RECVTTL - *ttl = *(int*)CMSG_DATA(cmptr); - continue; - } -#endif - -#if defined(IPV6_PKTINFO) && HAVE_IPV6 - if (cmptr->cmsg_level == IPPROTO_IPV6 && - cmptr->cmsg_type == IPV6_2292_PKTINFO) { - struct sockaddr_in6 *sin6 = (struct sockaddr_in6*)&pktp->ipi_addr; - struct in6_pktinfo *ip6_info = (struct in6_pktinfo*)CMSG_DATA(cmptr); - - sin6->sin6_family = AF_INET6; -#ifndef NOT_HAVE_SA_LEN - sin6->sin6_len = sizeof(*sin6); -#endif - sin6->sin6_addr = ip6_info->ipi6_addr; - sin6->sin6_flowinfo = 0; - sin6->sin6_scope_id = 0; - sin6->sin6_port = 0; - pktp->ipi_ifindex = ip6_info->ipi6_ifindex; - continue; - } -#endif - -#if defined(IPV6_HOPLIMIT) && HAVE_IPV6 - if (cmptr->cmsg_level == IPPROTO_IPV6 && - cmptr->cmsg_type == IPV6_2292_HOPLIMIT) { - *ttl = *(int*)CMSG_DATA(cmptr); - continue; - } -#endif - assert(0); // unknown ancillary data - } - return(n); -#endif /* CMSG_FIRSTHDR */ -} - -// ********************************************************************************************** - -// daemonize the process. Adapted from "Unix Network Programming" vol 1 by Stevens, section 12.4. -// Returns 0 on success, -1 on failure. - -#ifdef NOT_HAVE_DAEMON -#include -#include -#include - -int daemon(int nochdir, int noclose) - { - switch (fork()) - { - case -1: return (-1); // Fork failed - case 0: break; // Child -- continue - default: _exit(0); // Parent -- exit - } - - if (setsid() == -1) return(-1); - - signal(SIGHUP, SIG_IGN); - - switch (fork()) // Fork again, primarily for reasons of Unix trivia - { - case -1: return (-1); // Fork failed - case 0: break; // Child -- continue - default: _exit(0); // Parent -- exit - } - - if (!nochdir) (void)chdir("/"); - umask(0); - - if (!noclose) - { - int fd = open("/dev/null", O_RDWR, 0); - if (fd != -1) - { - // Avoid unnecessarily duplicating a file descriptor to itself - if (fd != STDIN_FILENO) (void)dup2(fd, STDIN_FILENO); - if (fd != STDOUT_FILENO) (void)dup2(fd, STDOUT_FILENO); - if (fd != STDERR_FILENO) (void)dup2(fd, STDERR_FILENO); - if (fd != STDIN_FILENO && fd != STDOUT_FILENO && fd != STDERR_FILENO) - (void)close (fd); - } - } - return (0); - } -#endif /* NOT_HAVE_DAEMON */ diff -Nru qtcreator-2.5.0/src/tools/mdnssd/mDNSUNP.h qtcreator-2.5.2/src/tools/mdnssd/mDNSUNP.h --- qtcreator-2.5.0/src/tools/mdnssd/mDNSUNP.h 2012-05-09 12:13:19.000000000 +0000 +++ qtcreator-2.5.2/src/tools/mdnssd/mDNSUNP.h 1970-01-01 00:00:00.000000000 +0000 @@ -1,130 +0,0 @@ -/* -*- Mode: C; tab-width: 4 -*- - * - * Copyright (c) 2002-2004 Apple Computer, Inc. All rights reserved. - * - * Licensed under the Apache License, Version 2.0 (the "License"); - * you may not use this file except in compliance with the License. - * You may obtain a copy of the License at - * - * http://www.apache.org/licenses/LICENSE-2.0 - * - * Unless required by applicable law or agreed to in writing, software - * distributed under the License is distributed on an "AS IS" BASIS, - * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. - * See the License for the specific language governing permissions and - * limitations under the License. - */ - -#ifndef __mDNSUNP_h -#define __mDNSUNP_h - -#include -#include -#include -#include - -#ifdef HAVE_LINUX -#include -#define IPV6_2292_PKTINFO IPV6_2292PKTINFO -#define IPV6_2292_HOPLIMIT IPV6_2292HOPLIMIT -#else -// The following are the supported non-linux posix OSes - -// netbsd, freebsd and openbsd. -#if HAVE_IPV6 -#define IPV6_2292_PKTINFO 19 -#define IPV6_2292_HOPLIMIT 20 -#endif -#endif - -#ifdef __cplusplus - extern "C" { -#endif - -#ifdef NOT_HAVE_SOCKLEN_T - typedef unsigned int socklen_t; -#endif - -#if !defined(_SS_MAXSIZE) -#if HAVE_IPV6 -#define sockaddr_storage sockaddr_in6 -#else -#define sockaddr_storage sockaddr -#endif // HAVE_IPV6 -#endif // !defined(_SS_MAXSIZE) - -#ifndef NOT_HAVE_SA_LEN -#define GET_SA_LEN(X) (sizeof(struct sockaddr) > ((struct sockaddr*)&(X))->sa_len ? \ - sizeof(struct sockaddr) : ((struct sockaddr*)&(X))->sa_len ) -#elif HAVE_IPV6 -#define GET_SA_LEN(X) (((struct sockaddr*)&(X))->sa_family == AF_INET ? sizeof(struct sockaddr_in) : \ - ((struct sockaddr*)&(X))->sa_family == AF_INET6 ? sizeof(struct sockaddr_in6) : sizeof(struct sockaddr)) -#else -#define GET_SA_LEN(X) (((struct sockaddr*)&(X))->sa_family == AF_INET ? sizeof(struct sockaddr_in) : sizeof(struct sockaddr)) -#endif - -#define IFI_NAME 16 /* same as IFNAMSIZ in */ -#define IFI_HADDR 8 /* allow for 64-bit EUI-64 in future */ - -// Renamed from my_in_pktinfo because in_pktinfo is used by Linux. - -struct my_in_pktinfo { - struct sockaddr_storage ipi_addr; - int ipi_ifindex; /* received interface index */ - char ipi_ifname[IFI_NAME]; /* received interface name */ -}; - -/* From the text (Stevens, section 20.2): */ -/* 'As an example of recvmsg we will write a function named recvfrom_flags that */ -/* is similar to recvfrom but also returns: */ -/* 1. the returned msg_flags value, */ -/* 2. the destination addres of the received datagram (from the IP_RECVDSTADDR socket option, and */ -/* 3. the index of the interface on which the datagram was received (the IP_RECVIF socket option).' */ -extern ssize_t recvfrom_flags(int fd, void *ptr, size_t nbytes, int *flagsp, - struct sockaddr *sa, socklen_t *salenptr, struct my_in_pktinfo *pktp, u_char *ttl); - -struct ifi_info { - char ifi_name[IFI_NAME]; /* interface name, null terminated */ - u_char ifi_haddr[IFI_HADDR]; /* hardware address */ - u_short ifi_hlen; /* #bytes in hardware address: 0, 6, 8 */ - short ifi_flags; /* IFF_xxx constants from */ - short ifi_myflags; /* our own IFI_xxx flags */ - int ifi_index; /* interface index */ - struct sockaddr *ifi_addr; /* primary address */ - struct sockaddr *ifi_netmask; - struct sockaddr *ifi_brdaddr;/* broadcast address */ - struct sockaddr *ifi_dstaddr;/* destination address */ - struct ifi_info *ifi_next; /* next of these structures */ -}; - -#if defined(AF_INET6) && HAVE_IPV6 && HAVE_LINUX -#define PROC_IFINET6_PATH "/proc/net/if_inet6" -extern struct ifi_info *get_ifi_info_linuxv6(int family, int doaliases); -#endif - -#if defined(AF_INET6) && HAVE_IPV6 -#define INET6_ADDRSTRLEN 46 /*Maximum length of IPv6 address */ -#endif - - - -#define IFI_ALIAS 1 /* ifi_addr is an alias */ - -/* From the text (Stevens, section 16.6): */ -/* 'Since many programs need to know all the interfaces on a system, we will develop a */ -/* function of our own named get_ifi_info that returns a linked list of structures, one */ -/* for each interface that is currently "up."' */ -extern struct ifi_info *get_ifi_info(int family, int doaliases); - -/* 'The free_ifi_info function, which takes a pointer that was */ -/* returned by get_ifi_info and frees all the dynamic memory.' */ -extern void free_ifi_info(struct ifi_info *); - -#ifdef NOT_HAVE_DAEMON -extern int daemon(int nochdir, int noclose); -#endif - -#ifdef __cplusplus - } -#endif - -#endif diff -Nru qtcreator-2.5.0/src/tools/mdnssd/mdnssd.pro qtcreator-2.5.2/src/tools/mdnssd/mdnssd.pro --- qtcreator-2.5.0/src/tools/mdnssd/mdnssd.pro 2012-05-09 12:13:19.000000000 +0000 +++ qtcreator-2.5.2/src/tools/mdnssd/mdnssd.pro 1970-01-01 00:00:00.000000000 +0000 @@ -1,59 +0,0 @@ -#------------------------------------------------- -# -# Project created by QtCreator 2011-10-14T10:22:27 -# -#------------------------------------------------- - -QT -= gui -QT += core - -TARGET = mdnssd -CONFIG += console -CONFIG -= app_bundle - -TEMPLATE = app - -include(../../../qtcreator.pri) -DESTDIR = $$IDE_BIN_PATH - -DEFINES += PID_FILE=\\\"/var/run/mdnsd.pid\\\" MDNS_UDS_SERVERPATH=\\\"/var/run/mdnsd\\\" MDNS_DEBUGMSGS=0 - -SOURCES += \ - uds_daemon.c \ - uDNS.c \ - PosixDaemon.c \ - PlatformCommon.c \ - mDNSUNP.c \ - mDNSPosix.c \ - mDNSDebug.c \ - mDNS.c \ - GenLinkedList.c \ - dnssd_ipc.c \ - DNSDigest.c \ - DNSCommon.c - -HEADERS += \ - uds_daemon.h \ - uDNS.h \ - PlatformCommon.h \ - mDNSUNP.h \ - mDNSPosix.h \ - mDNSEmbeddedAPI.h \ - mDNSDebug.h \ - GenLinkedList.h \ - dnssd_ipc.h \ - DNSCommon.h \ - DebugServices.h \ - dns_sd.h - -*-g++ { - QMAKE_CFLAGS += -Wno-unused-but-set-variable - QMAKE_CXXFLAGS += -Wno-unused-but-set-variable -} -linux-* { -DEFINES += _GNU_SOURCE HAVE_IPV6 NOT_HAVE_SA_LEN USES_NETLINK HAVE_LINUX TARGET_OS_LINUX -} -macx { -DEFINES += HAVE_IPV6 __MAC_OS_X_VERSION_MIN_REQUIRED=__MAC_OS_X_VERSION_10_4 __APPLE_USE_RFC_2292 -} - diff -Nru qtcreator-2.5.0/src/tools/mdnssd/uDNS.c qtcreator-2.5.2/src/tools/mdnssd/uDNS.c --- qtcreator-2.5.0/src/tools/mdnssd/uDNS.c 2012-05-09 12:13:19.000000000 +0000 +++ qtcreator-2.5.2/src/tools/mdnssd/uDNS.c 1970-01-01 00:00:00.000000000 +0000 @@ -1,4927 +0,0 @@ -/* -*- Mode: C; tab-width: 4 -*- - * - * Copyright (c) 2002-2006 Apple Computer, Inc. All rights reserved. - * - * Licensed under the Apache License, Version 2.0 (the "License"); - * you may not use this file except in compliance with the License. - * You may obtain a copy of the License at - * - * http://www.apache.org/licenses/LICENSE-2.0 - * - * Unless required by applicable law or agreed to in writing, software - * distributed under the License is distributed on an "AS IS" BASIS, - * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. - * See the License for the specific language governing permissions and - * limitations under the License. - - * To Do: - * Elimate all mDNSPlatformMemAllocate/mDNSPlatformMemFree from this code -- the core code - * is supposed to be malloc-free so that it runs in constant memory determined at compile-time. - * Any dynamic run-time requirements should be handled by the platform layer below or client layer above - */ - -#if APPLE_OSX_mDNSResponder -#include -#endif -#include "uDNS.h" - -#if(defined(_MSC_VER)) - // Disable "assignment within conditional expression". - // Other compilers understand the convention that if you place the assignment expression within an extra pair - // of parentheses, this signals to the compiler that you really intended an assignment and no warning is necessary. - // The Microsoft compiler doesn't understand this convention, so in the absense of any other way to signal - // to the compiler that the assignment is intentional, we have to just turn this warning off completely. - #pragma warning(disable:4706) -#endif - -// For domain enumeration and automatic browsing -// This is the user's DNS search list. -// In each of these domains we search for our special pointer records (lb._dns-sd._udp., etc.) -// to discover recommended domains for domain enumeration (browse, default browse, registration, -// default registration) and possibly one or more recommended automatic browsing domains. -mDNSexport SearchListElem *SearchList = mDNSNULL; - -// The value can be set to true by the Platform code e.g., MacOSX uses the plist mechanism -mDNSBool StrictUnicastOrdering = mDNSfalse; - -// We keep track of the number of unicast DNS servers and log a message when we exceed 64. -// Currently the unicast queries maintain a 64 bit map to track the valid DNS servers for that -// question. Bit position is the index into the DNS server list. This is done so to try all -// the servers exactly once before giving up. If we could allocate memory in the core, then -// arbitrary limitation of 64 DNSServers can be removed. -mDNSu8 NumUnicastDNSServers = 0; -#define MAX_UNICAST_DNS_SERVERS 64 - -// *************************************************************************** -#if COMPILER_LIKES_PRAGMA_MARK -#pragma mark - General Utility Functions -#endif - -// set retry timestamp for record with exponential backoff -mDNSlocal void SetRecordRetry(mDNS *const m, AuthRecord *rr, mDNSu32 random) - { - rr->LastAPTime = m->timenow; - - if (rr->expire && rr->refreshCount < MAX_UPDATE_REFRESH_COUNT) - { - mDNSs32 remaining = rr->expire - m->timenow; - rr->refreshCount++; - if (remaining > MIN_UPDATE_REFRESH_TIME) - { - // Refresh at 70% + random (currently it is 0 to 10%) - rr->ThisAPInterval = 7 * (remaining/10) + (random ? random : mDNSRandom(remaining/10)); - // Don't update more often than 5 minutes - if (rr->ThisAPInterval < MIN_UPDATE_REFRESH_TIME) - rr->ThisAPInterval = MIN_UPDATE_REFRESH_TIME; - LogInfo("SetRecordRetry refresh in %d of %d for %s", - rr->ThisAPInterval/mDNSPlatformOneSecond, (rr->expire - m->timenow)/mDNSPlatformOneSecond, ARDisplayString(m, rr)); - } - else - { - rr->ThisAPInterval = MIN_UPDATE_REFRESH_TIME; - LogInfo("SetRecordRetry clamping to min refresh in %d of %d for %s", - rr->ThisAPInterval/mDNSPlatformOneSecond, (rr->expire - m->timenow)/mDNSPlatformOneSecond, ARDisplayString(m, rr)); - } - return; - } - - rr->expire = 0; - - rr->ThisAPInterval = rr->ThisAPInterval * QuestionIntervalStep; // Same Retry logic as Unicast Queries - if (rr->ThisAPInterval < INIT_RECORD_REG_INTERVAL) - rr->ThisAPInterval = INIT_RECORD_REG_INTERVAL; - if (rr->ThisAPInterval > MAX_RECORD_REG_INTERVAL) - rr->ThisAPInterval = MAX_RECORD_REG_INTERVAL; - - LogInfo("SetRecordRetry retry in %d ms for %s", rr->ThisAPInterval, ARDisplayString(m, rr)); - } - -// *************************************************************************** -#if COMPILER_LIKES_PRAGMA_MARK -#pragma mark - Name Server List Management -#endif - -mDNSexport DNSServer *mDNS_AddDNSServer(mDNS *const m, const domainname *d, const mDNSInterfaceID interface, const mDNSAddr *addr, const mDNSIPPort port, mDNSBool scoped, mDNSu32 timeout, mDNSBool cellIntf) - { - DNSServer **p = &m->DNSServers; - DNSServer *tmp = mDNSNULL; - - if ((NumUnicastDNSServers + 1) > MAX_UNICAST_DNS_SERVERS) - { - LogMsg("mDNS_AddDNSServer: DNS server limit of %d reached, not adding this server", MAX_UNICAST_DNS_SERVERS); - return mDNSNULL; - } - - if (!d) d = (const domainname *)""; - - LogInfo("mDNS_AddDNSServer: Adding %#a for %##s, InterfaceID %p, scoped %d", addr, d->c, interface, scoped); - if (m->mDNS_busy != m->mDNS_reentrancy+1) - LogMsg("mDNS_AddDNSServer: Lock not held! mDNS_busy (%ld) mDNS_reentrancy (%ld)", m->mDNS_busy, m->mDNS_reentrancy); - - while (*p) // Check if we already have this {interface,address,port,domain} tuple registered - { - if ((*p)->scoped == scoped && (*p)->interface == interface && (*p)->teststate != DNSServer_Disabled && - mDNSSameAddress(&(*p)->addr, addr) && mDNSSameIPPort((*p)->port, port) && SameDomainName(&(*p)->domain, d)) - { - if (!((*p)->flags & DNSServer_FlagDelete)) debugf("Note: DNS Server %#a:%d for domain %##s (%p) registered more than once", addr, mDNSVal16(port), d->c, interface); - (*p)->flags &= ~DNSServer_FlagDelete; - tmp = *p; - *p = tmp->next; - tmp->next = mDNSNULL; - } - else - p=&(*p)->next; - } - - if (tmp) *p = tmp; // move to end of list, to ensure ordering from platform layer - else - { - // allocate, add to list - *p = mDNSPlatformMemAllocate(sizeof(**p)); - if (!*p) LogMsg("Error: mDNS_AddDNSServer - malloc"); - else - { - NumUnicastDNSServers++; - (*p)->scoped = scoped; - (*p)->interface = interface; - (*p)->addr = *addr; - (*p)->port = port; - (*p)->flags = DNSServer_FlagNew; - (*p)->teststate = /* DNSServer_Untested */ DNSServer_Passed; - (*p)->lasttest = m->timenow - INIT_UCAST_POLL_INTERVAL; - (*p)->timeout = timeout; - (*p)->cellIntf = cellIntf; - AssignDomainName(&(*p)->domain, d); - (*p)->next = mDNSNULL; - } - } - (*p)->penaltyTime = 0; - return(*p); - } - -// PenalizeDNSServer is called when the number of queries to the unicast -// DNS server exceeds MAX_UCAST_UNANSWERED_QUERIES or when we receive an -// error e.g., SERV_FAIL from DNS server. -mDNSexport void PenalizeDNSServer(mDNS *const m, DNSQuestion *q) - { - DNSServer *new; - DNSServer *orig = q->qDNSServer; - - if (m->mDNS_busy != m->mDNS_reentrancy+1) - LogMsg("PenalizeDNSServer: Lock not held! mDNS_busy (%ld) mDNS_reentrancy (%ld)", m->mDNS_busy, m->mDNS_reentrancy); - - // This should never happen. Whenever we change DNS server, we change the ID on the question and hence - // we should never accept a response after we penalize a DNS server e.g., send two queries, no response, - // penalize DNS server and no new servers to pick for the question and hence qDNSServer is NULL. If we - // receive a response now, the DNS server can be NULL. But we won't because the ID already has been - // changed. - if (!q->qDNSServer) - { - LogMsg("PenalizeDNSServer: ERROR!! Null DNS server for %##s (%s) %d", q->qname.c, DNSTypeName(q->qtype), q->unansweredQueries); - goto end; - } - - LogInfo("PenalizeDNSServer: Penalizing DNS server %#a:%d question (%##s) for question %p %##s (%s) SuppressUnusable %d", - &q->qDNSServer->addr, mDNSVal16(q->qDNSServer->port), q->qDNSServer->domain.c, q, q->qname.c, DNSTypeName(q->qtype), - q->SuppressUnusable); - - // If strict ordering of unicast servers needs to be preserved, we just lookup - // the next best match server below - // - // If strict ordering is not required which is the default behavior, we penalize the server - // for DNSSERVER_PENALTY_TIME. We may also use additional logic e.g., don't penalize for PTR - // in the future. - - if (!StrictUnicastOrdering) - { - LogInfo("PenalizeDNSServer: Strict Unicast Ordering is FALSE"); - // We penalize the server so that new queries don't pick this server for DNSSERVER_PENALTY_TIME - // XXX Include other logic here to see if this server should really be penalized - // - if (q->qtype == kDNSType_PTR) - { - LogInfo("PenalizeDNSServer: Not Penalizing PTR question"); - } - else - { - LogInfo("PenalizeDNSServer: Penalizing question type %d", q->qtype); - q->qDNSServer->penaltyTime = NonZeroTime(m->timenow + DNSSERVER_PENALTY_TIME); - } - } - else - { - LogInfo("PenalizeDNSServer: Strict Unicast Ordering is TRUE"); - } - -end: - new = GetServerForQuestion(m, q); - - - if (new == orig) - { - if (new) - LogMsg("PenalizeDNSServer: ERROR!! GetServerForQuestion returned the same server %#a:%d", &new->addr, - mDNSVal16(new->port)); - else - LogMsg("PenalizeDNSServer: ERROR!! GetServerForQuestion returned the same server NULL"); - q->ThisQInterval = 0; // Inactivate this question so that we dont bombard the network - } - else - { - // The new DNSServer is set in DNSServerChangeForQuestion - DNSServerChangeForQuestion(m, q, new); - - if (new) - { - LogInfo("PenalizeDNSServer: Server for %##s (%s) changed to %#a:%d (%##s)", - q->qname.c, DNSTypeName(q->qtype), &q->qDNSServer->addr, mDNSVal16(q->qDNSServer->port), q->qDNSServer->domain.c); - // We want to try the next server immediately. As the question may already have backed off, reset - // the interval. We do this only the first time when we try all the DNS servers. Once we reached the end of - // list and retrying all the servers again e.g., at least one server failed to respond in the previous try, we - // use the normal backoff which is done in uDNS_CheckCurrentQuestion when we send the packet out. - if (!q->triedAllServersOnce) - { - q->ThisQInterval = InitialQuestionInterval; - q->LastQTime = m->timenow - q->ThisQInterval; - SetNextQueryTime(m, q); - } - } - else - { - // We don't have any more DNS servers for this question. If some server in the list did not return - // any response, we need to keep retrying till we get a response. uDNS_CheckCurrentQuestion handles - // this case. - // - // If all servers responded with a negative response, We need to do two things. First, generate a - // negative response so that applications get a reply. We also need to reinitialize the DNS servers - // so that when the cache expires, we can restart the query. - // - // Negative response may be generated in two ways. - // - // 1. AnswerQuestionForDNSServerChanges (called from DNSServerChangedForQuestion) might find some - // cache entries and answer this question. - // 2. uDNS_CheckCurrentQuestion will create a new cache entry and answer this question - // - // For (1), it might be okay to reinitialize the DNS servers here. But for (2), we can't do it here - // because uDNS_CheckCurrentQuestion will try resending the queries. Hence, to be consistent, we - // defer reintializing the DNS servers up until generating a negative cache response. - // - // Be careful not to touch the ThisQInterval here. For a normal question, when we answer the question - // in AnswerCurrentQuestionWithResourceRecord will set ThisQInterval to MaxQuestionInterval and hence - // the next query will not happen until cache expiry. If it is a long lived question, - // AnswerCurrentQuestionWithResourceRecord will not set it to MaxQuestionInterval. In that case, - // we want the normal backoff to work. - LogInfo("PenalizeDNSServer: Server for %p, %##s (%s) changed to NULL, Interval %d", q, q->qname.c, DNSTypeName(q->qtype), q->ThisQInterval); - } - q->unansweredQueries = 0; - - } - } - -// *************************************************************************** -#if COMPILER_LIKES_PRAGMA_MARK -#pragma mark - authorization management -#endif - -mDNSlocal DomainAuthInfo *GetAuthInfoForName_direct(mDNS *m, const domainname *const name) - { - const domainname *n = name; - while (n->c[0]) - { - DomainAuthInfo *ptr; - for (ptr = m->AuthInfoList; ptr; ptr = ptr->next) - if (SameDomainName(&ptr->domain, n)) - { - debugf("GetAuthInfoForName %##s Matched %##s Key name %##s", name->c, ptr->domain.c, ptr->keyname.c); - return(ptr); - } - n = (const domainname *)(n->c + 1 + n->c[0]); - } - //LogInfo("GetAuthInfoForName none found for %##s", name->c); - return mDNSNULL; - } - -// MUST be called with lock held -mDNSexport DomainAuthInfo *GetAuthInfoForName_internal(mDNS *m, const domainname *const name) - { - DomainAuthInfo **p = &m->AuthInfoList; - - if (m->mDNS_busy != m->mDNS_reentrancy+1) - LogMsg("GetAuthInfoForName_internal: Lock not held! mDNS_busy (%ld) mDNS_reentrancy (%ld)", m->mDNS_busy, m->mDNS_reentrancy); - - // First purge any dead keys from the list - while (*p) - { - if ((*p)->deltime && m->timenow - (*p)->deltime >= 0 && AutoTunnelUnregistered(*p)) - { - DNSQuestion *q; - DomainAuthInfo *info = *p; - LogInfo("GetAuthInfoForName_internal deleting expired key %##s %##s", info->domain.c, info->keyname.c); - *p = info->next; // Cut DomainAuthInfo from list *before* scanning our question list updating AuthInfo pointers - for (q = m->Questions; q; q=q->next) - if (q->AuthInfo == info) - { - q->AuthInfo = GetAuthInfoForName_direct(m, &q->qname); - debugf("GetAuthInfoForName_internal updated q->AuthInfo from %##s to %##s for %##s (%s)", - info->domain.c, q->AuthInfo ? q->AuthInfo->domain.c : mDNSNULL, q->qname.c, DNSTypeName(q->qtype)); - } - - // Probably not essential, but just to be safe, zero out the secret key data - // so we don't leave it hanging around in memory - // (where it could potentially get exposed via some other bug) - mDNSPlatformMemZero(info, sizeof(*info)); - mDNSPlatformMemFree(info); - } - else - p = &(*p)->next; - } - - return(GetAuthInfoForName_direct(m, name)); - } - -mDNSexport DomainAuthInfo *GetAuthInfoForName(mDNS *m, const domainname *const name) - { - DomainAuthInfo *d; - mDNS_Lock(m); - d = GetAuthInfoForName_internal(m, name); - mDNS_Unlock(m); - return(d); - } - -// MUST be called with the lock held -mDNSexport mStatus mDNS_SetSecretForDomain(mDNS *m, DomainAuthInfo *info, - const domainname *domain, const domainname *keyname, const char *b64keydata, const domainname *hostname, mDNSIPPort *port, const char *autoTunnelPrefix) - { - DNSQuestion *q; - DomainAuthInfo **p = &m->AuthInfoList; - if (!info || !b64keydata) { LogMsg("mDNS_SetSecretForDomain: ERROR: info %p b64keydata %p", info, b64keydata); return(mStatus_BadParamErr); } - - LogInfo("mDNS_SetSecretForDomain: domain %##s key %##s%s%s", domain->c, keyname->c, autoTunnelPrefix ? " prefix " : "", autoTunnelPrefix ? autoTunnelPrefix : ""); - - info->AutoTunnel = autoTunnelPrefix; - AssignDomainName(&info->domain, domain); - AssignDomainName(&info->keyname, keyname); - if (hostname) - AssignDomainName(&info->hostname, hostname); - else - info->hostname.c[0] = 0; - if (port) - info->port = *port; - else - info->port = zeroIPPort; - mDNS_snprintf(info->b64keydata, sizeof(info->b64keydata), "%s", b64keydata); - - if (DNSDigest_ConstructHMACKeyfromBase64(info, b64keydata) < 0) - { - LogMsg("mDNS_SetSecretForDomain: ERROR: Could not convert shared secret from base64: domain %##s key %##s %s", domain->c, keyname->c, mDNS_LoggingEnabled ? b64keydata : ""); - return(mStatus_BadParamErr); - } - - // Don't clear deltime until after we've ascertained that b64keydata is valid - info->deltime = 0; - - while (*p && (*p) != info) p=&(*p)->next; - if (*p) {LogInfo("mDNS_SetSecretForDomain: Domain %##s Already in list", (*p)->domain.c); return(mStatus_AlreadyRegistered);} - - // Caution: Only zero AutoTunnelHostRecord.namestorage and AutoTunnelNAT.clientContext AFTER we've determined that this is a NEW DomainAuthInfo - // being added to the list. Otherwise we risk smashing our AutoTunnel host records and NATOperation that are already active and in use. - info->AutoTunnelHostRecord.resrec.RecordType = kDNSRecordTypeUnregistered; - info->AutoTunnelHostRecord.namestorage.c[0] = 0; - info->AutoTunnelTarget .resrec.RecordType = kDNSRecordTypeUnregistered; - info->AutoTunnelDeviceInfo.resrec.RecordType = kDNSRecordTypeUnregistered; - info->AutoTunnelService .resrec.RecordType = kDNSRecordTypeUnregistered; - info->AutoTunnel6Record .resrec.RecordType = kDNSRecordTypeUnregistered; - info->AutoTunnelNAT.clientContext = mDNSNULL; - info->next = mDNSNULL; - *p = info; - - // Check to see if adding this new DomainAuthInfo has changed the credentials for any of our questions - for (q = m->Questions; q; q=q->next) - { - DomainAuthInfo *newinfo = GetAuthInfoForQuestion(m, q); - if (q->AuthInfo != newinfo) - { - debugf("mDNS_SetSecretForDomain updating q->AuthInfo from %##s to %##s for %##s (%s)", - q->AuthInfo ? q->AuthInfo->domain.c : mDNSNULL, - newinfo ? newinfo ->domain.c : mDNSNULL, q->qname.c, DNSTypeName(q->qtype)); - q->AuthInfo = newinfo; - } - } - - return(mStatus_NoError); - } - -// *************************************************************************** -#if COMPILER_LIKES_PRAGMA_MARK -#pragma mark - -#pragma mark - NAT Traversal -#endif - -mDNSlocal mStatus uDNS_SendNATMsg(mDNS *m, NATTraversalInfo *info) - { - mStatus err = mStatus_NoError; - - // send msg if we have a router and it is a private address - if (!mDNSIPv4AddressIsZero(m->Router.ip.v4) && mDNSv4AddrIsRFC1918(&m->Router.ip.v4)) - { - union { NATAddrRequest NATAddrReq; NATPortMapRequest NATPortReq; } u = { { NATMAP_VERS, NATOp_AddrRequest } } ; - const mDNSu8 *end = (mDNSu8 *)&u + sizeof(NATAddrRequest); - - if (info) // For NATOp_MapUDP and NATOp_MapTCP, fill in additional fields - { - mDNSu8 *p = (mDNSu8 *)&u.NATPortReq.NATReq_lease; - u.NATPortReq.opcode = info->Protocol; - u.NATPortReq.unused = zeroID; - u.NATPortReq.intport = info->IntPort; - u.NATPortReq.extport = info->RequestedPort; - p[0] = (mDNSu8)((info->NATLease >> 24) & 0xFF); - p[1] = (mDNSu8)((info->NATLease >> 16) & 0xFF); - p[2] = (mDNSu8)((info->NATLease >> 8) & 0xFF); - p[3] = (mDNSu8)( info->NATLease & 0xFF); - end = (mDNSu8 *)&u + sizeof(NATPortMapRequest); - } - - err = mDNSPlatformSendUDP(m, (mDNSu8 *)&u, end, 0, mDNSNULL, &m->Router, NATPMPPort); - -#ifdef _LEGACY_NAT_TRAVERSAL_ - if (mDNSIPPortIsZero(m->UPnPRouterPort) || mDNSIPPortIsZero(m->UPnPSOAPPort)) LNT_SendDiscoveryMsg(m); - else if (info) err = LNT_MapPort(m, info); - else err = LNT_GetExternalAddress(m); -#endif // _LEGACY_NAT_TRAVERSAL_ - } - return(err); - } - -mDNSexport void RecreateNATMappings(mDNS *const m) - { - NATTraversalInfo *n; - for (n = m->NATTraversals; n; n=n->next) - { - n->ExpiryTime = 0; // Mark this mapping as expired - n->retryInterval = NATMAP_INIT_RETRY; - n->retryPortMap = m->timenow; -#ifdef _LEGACY_NAT_TRAVERSAL_ - if (n->tcpInfo.sock) { mDNSPlatformTCPCloseConnection(n->tcpInfo.sock); n->tcpInfo.sock = mDNSNULL; } -#endif // _LEGACY_NAT_TRAVERSAL_ - } - - m->NextScheduledNATOp = m->timenow; // Need to send packets immediately - } - -mDNSexport void natTraversalHandleAddressReply(mDNS *const m, mDNSu16 err, mDNSv4Addr ExtAddr) - { - static mDNSu16 last_err = 0; - - if (err) - { - if (err != last_err) LogMsg("Error getting external address %d", err); - ExtAddr = zerov4Addr; - } - else - { - LogInfo("Received external IP address %.4a from NAT", &ExtAddr); - if (mDNSv4AddrIsRFC1918(&ExtAddr)) - LogMsg("Double NAT (external NAT gateway address %.4a is also a private RFC 1918 address)", &ExtAddr); - if (mDNSIPv4AddressIsZero(ExtAddr)) - err = NATErr_NetFail; // fake error to handle routers that pathologically report success with the zero address - } - - if (!mDNSSameIPv4Address(m->ExternalAddress, ExtAddr)) - { - m->ExternalAddress = ExtAddr; - RecreateNATMappings(m); // Also sets NextScheduledNATOp for us - } - - if (!err) // Success, back-off to maximum interval - m->retryIntervalGetAddr = NATMAP_MAX_RETRY_INTERVAL; - else if (!last_err) // Failure after success, retry quickly (then back-off exponentially) - m->retryIntervalGetAddr = NATMAP_INIT_RETRY; - // else back-off normally in case of pathological failures - - m->retryGetAddr = m->timenow + m->retryIntervalGetAddr; - if (m->NextScheduledNATOp - m->retryIntervalGetAddr > 0) - m->NextScheduledNATOp = m->retryIntervalGetAddr; - - last_err = err; - } - -// Both places that call NATSetNextRenewalTime() update m->NextScheduledNATOp correctly afterwards -mDNSlocal void NATSetNextRenewalTime(mDNS *const m, NATTraversalInfo *n) - { - n->retryInterval = (n->ExpiryTime - m->timenow)/2; - if (n->retryInterval < NATMAP_MIN_RETRY_INTERVAL) // Min retry interval is 2 seconds - n->retryInterval = NATMAP_MIN_RETRY_INTERVAL; - n->retryPortMap = m->timenow + n->retryInterval; - } - -// Note: When called from handleLNTPortMappingResponse() only pkt->err, pkt->extport and pkt->NATRep_lease fields are filled in -mDNSexport void natTraversalHandlePortMapReply(mDNS *const m, NATTraversalInfo *n, const mDNSInterfaceID InterfaceID, mDNSu16 err, mDNSIPPort extport, mDNSu32 lease) - { - const char *prot = n->Protocol == NATOp_MapUDP ? "UDP" : n->Protocol == NATOp_MapTCP ? "TCP" : "?"; - (void)prot; - n->NewResult = err; - if (err || lease == 0 || mDNSIPPortIsZero(extport)) - { - LogInfo("natTraversalHandlePortMapReply: %p Response %s Port %5d External Port %5d lease %d error %d", - n, prot, mDNSVal16(n->IntPort), mDNSVal16(extport), lease, err); - n->retryInterval = NATMAP_MAX_RETRY_INTERVAL; - n->retryPortMap = m->timenow + NATMAP_MAX_RETRY_INTERVAL; - // No need to set m->NextScheduledNATOp here, since we're only ever extending the m->retryPortMap time - if (err == NATErr_Refused) n->NewResult = mStatus_NATPortMappingDisabled; - else if (err > NATErr_None && err <= NATErr_Opcode) n->NewResult = mStatus_NATPortMappingUnsupported; - } - else - { - if (lease > 999999999UL / mDNSPlatformOneSecond) - lease = 999999999UL / mDNSPlatformOneSecond; - n->ExpiryTime = NonZeroTime(m->timenow + lease * mDNSPlatformOneSecond); - - if (!mDNSSameIPPort(n->RequestedPort, extport)) - LogInfo("natTraversalHandlePortMapReply: %p Response %s Port %5d External Port %5d changed to %5d", - n, prot, mDNSVal16(n->IntPort), mDNSVal16(n->RequestedPort), mDNSVal16(extport)); - - n->InterfaceID = InterfaceID; - n->RequestedPort = extport; - - LogInfo("natTraversalHandlePortMapReply: %p Response %s Port %5d External Port %5d lease %d", - n, prot, mDNSVal16(n->IntPort), mDNSVal16(extport), lease); - - NATSetNextRenewalTime(m, n); // Got our port mapping; now set timer to renew it at halfway point - m->NextScheduledNATOp = m->timenow; // May need to invoke client callback immediately - } - } - -// Must be called with the mDNS_Lock held -mDNSexport mStatus mDNS_StartNATOperation_internal(mDNS *const m, NATTraversalInfo *traversal) - { - NATTraversalInfo **n; - - LogInfo("mDNS_StartNATOperation_internal %p Protocol %d IntPort %d RequestedPort %d NATLease %d", traversal, - traversal->Protocol, mDNSVal16(traversal->IntPort), mDNSVal16(traversal->RequestedPort), traversal->NATLease); - - // Note: It important that new traversal requests are appended at the *end* of the list, not prepended at the start - for (n = &m->NATTraversals; *n; n=&(*n)->next) - { - if (traversal == *n) - { - LogMsg("Error! Tried to add a NAT traversal that's already in the active list: request %p Prot %d Int %d TTL %d", - traversal, traversal->Protocol, mDNSVal16(traversal->IntPort), traversal->NATLease); - #if ForceAlerts - *(long*)0 = 0; - #endif - return(mStatus_AlreadyRegistered); - } - if (traversal->Protocol && traversal->Protocol == (*n)->Protocol && mDNSSameIPPort(traversal->IntPort, (*n)->IntPort) && - !mDNSSameIPPort(traversal->IntPort, SSHPort)) - LogMsg("Warning: Created port mapping request %p Prot %d Int %d TTL %d " - "duplicates existing port mapping request %p Prot %d Int %d TTL %d", - traversal, traversal->Protocol, mDNSVal16(traversal->IntPort), traversal->NATLease, - *n, (*n) ->Protocol, mDNSVal16((*n) ->IntPort), (*n) ->NATLease); - } - - // Initialize necessary fields - traversal->next = mDNSNULL; - traversal->ExpiryTime = 0; - traversal->retryInterval = NATMAP_INIT_RETRY; - traversal->retryPortMap = m->timenow; - traversal->NewResult = mStatus_NoError; - traversal->ExternalAddress = onesIPv4Addr; - traversal->ExternalPort = zeroIPPort; - traversal->Lifetime = 0; - traversal->Result = mStatus_NoError; - - // set default lease if necessary - if (!traversal->NATLease) traversal->NATLease = NATMAP_DEFAULT_LEASE; - -#ifdef _LEGACY_NAT_TRAVERSAL_ - mDNSPlatformMemZero(&traversal->tcpInfo, sizeof(traversal->tcpInfo)); -#endif // _LEGACY_NAT_TRAVERSAL_ - - if (!m->NATTraversals) // If this is our first NAT request, kick off an address request too - { - m->retryGetAddr = m->timenow; - m->retryIntervalGetAddr = NATMAP_INIT_RETRY; - } - - m->NextScheduledNATOp = m->timenow; // This will always trigger sending the packet ASAP, and generate client callback if necessary - - *n = traversal; // Append new NATTraversalInfo to the end of our list - - return(mStatus_NoError); - } - -// Must be called with the mDNS_Lock held -mDNSexport mStatus mDNS_StopNATOperation_internal(mDNS *m, NATTraversalInfo *traversal) - { - mDNSBool unmap = mDNStrue; - NATTraversalInfo *p; - NATTraversalInfo **ptr = &m->NATTraversals; - - while (*ptr && *ptr != traversal) ptr=&(*ptr)->next; - if (*ptr) *ptr = (*ptr)->next; // If we found it, cut this NATTraversalInfo struct from our list - else - { - LogMsg("mDNS_StopNATOperation_internal: NATTraversalInfo %p not found in list", traversal); - return(mStatus_BadReferenceErr); - } - - LogInfo("mDNS_StopNATOperation_internal %p %d %d %d %d", traversal, - traversal->Protocol, mDNSVal16(traversal->IntPort), mDNSVal16(traversal->RequestedPort), traversal->NATLease); - - if (m->CurrentNATTraversal == traversal) - m->CurrentNATTraversal = m->CurrentNATTraversal->next; - - if (traversal->Protocol) - for (p = m->NATTraversals; p; p=p->next) - if (traversal->Protocol == p->Protocol && mDNSSameIPPort(traversal->IntPort, p->IntPort)) - { - if (!mDNSSameIPPort(traversal->IntPort, SSHPort)) - LogMsg("Warning: Removed port mapping request %p Prot %d Int %d TTL %d " - "duplicates existing port mapping request %p Prot %d Int %d TTL %d", - traversal, traversal->Protocol, mDNSVal16(traversal->IntPort), traversal->NATLease, - p, p ->Protocol, mDNSVal16(p ->IntPort), p ->NATLease); - unmap = mDNSfalse; - } - - if (traversal->ExpiryTime && unmap) - { - traversal->NATLease = 0; - traversal->retryInterval = 0; - uDNS_SendNATMsg(m, traversal); - } - - // Even if we DIDN'T make a successful UPnP mapping yet, we might still have a partially-open TCP connection we need to clean up - #ifdef _LEGACY_NAT_TRAVERSAL_ - { - mStatus err = LNT_UnmapPort(m, traversal); - if (err) LogMsg("Legacy NAT Traversal - unmap request failed with error %d", err); - } - #endif // _LEGACY_NAT_TRAVERSAL_ - - return(mStatus_NoError); - } - -mDNSexport mStatus mDNS_StartNATOperation(mDNS *const m, NATTraversalInfo *traversal) - { - mStatus status; - mDNS_Lock(m); - status = mDNS_StartNATOperation_internal(m, traversal); - mDNS_Unlock(m); - return(status); - } - -mDNSexport mStatus mDNS_StopNATOperation(mDNS *const m, NATTraversalInfo *traversal) - { - mStatus status; - mDNS_Lock(m); - status = mDNS_StopNATOperation_internal(m, traversal); - mDNS_Unlock(m); - return(status); - } - -// *************************************************************************** -#if COMPILER_LIKES_PRAGMA_MARK -#pragma mark - -#pragma mark - Long-Lived Queries -#endif - -// Lock must be held -- otherwise m->timenow is undefined -mDNSlocal void StartLLQPolling(mDNS *const m, DNSQuestion *q) - { - debugf("StartLLQPolling: %##s", q->qname.c); - q->state = LLQ_Poll; - q->ThisQInterval = INIT_UCAST_POLL_INTERVAL; - // We want to send our poll query ASAP, but the "+ 1" is because if we set the time to now, - // we risk causing spurious "SendQueries didn't send all its queries" log messages - q->LastQTime = m->timenow - q->ThisQInterval + 1; - SetNextQueryTime(m, q); -#if APPLE_OSX_mDNSResponder - UpdateAutoTunnelDomainStatuses(m); -#endif - } - -mDNSlocal mDNSu8 *putLLQ(DNSMessage *const msg, mDNSu8 *ptr, const DNSQuestion *const question, const LLQOptData *const data) - { - AuthRecord rr; - ResourceRecord *opt = &rr.resrec; - rdataOPT *optRD; - - //!!!KRS when we implement multiple llqs per message, we'll need to memmove anything past the question section - ptr = putQuestion(msg, ptr, msg->data + AbsoluteMaxDNSMessageData, &question->qname, question->qtype, question->qclass); - if (!ptr) { LogMsg("ERROR: putLLQ - putQuestion"); return mDNSNULL; } - - // locate OptRR if it exists, set pointer to end - // !!!KRS implement me - - // format opt rr (fields not specified are zero-valued) - mDNS_SetupResourceRecord(&rr, mDNSNULL, mDNSInterface_Any, kDNSType_OPT, kStandardTTL, kDNSRecordTypeKnownUnique, AuthRecordAny, mDNSNULL, mDNSNULL); - opt->rrclass = NormalMaxDNSMessageData; - opt->rdlength = sizeof(rdataOPT); // One option in this OPT record - opt->rdestimate = sizeof(rdataOPT); - - optRD = &rr.resrec.rdata->u.opt[0]; - optRD->opt = kDNSOpt_LLQ; - optRD->u.llq = *data; - ptr = PutResourceRecordTTLJumbo(msg, ptr, &msg->h.numAdditionals, opt, 0); - if (!ptr) { LogMsg("ERROR: putLLQ - PutResourceRecordTTLJumbo"); return mDNSNULL; } - - return ptr; - } - -// Normally we'd just request event packets be sent directly to m->LLQNAT.ExternalPort, except... -// with LLQs over TLS/TCP we're doing a weird thing where instead of requesting packets be sent to ExternalAddress:ExternalPort -// we're requesting that packets be sent to ExternalPort, but at the source address of our outgoing TCP connection. -// Normally, after going through the NAT gateway, the source address of our outgoing TCP connection is the same as ExternalAddress, -// so this is fine, except when the TCP connection ends up going over a VPN tunnel instead. -// To work around this, if we find that the source address for our TCP connection is not a private address, we tell the Dot Mac -// LLQ server to send events to us directly at port 5353 on that address, instead of at our mapped external NAT port. - -mDNSlocal mDNSu16 GetLLQEventPort(const mDNS *const m, const mDNSAddr *const dst) - { - mDNSAddr src; - mDNSPlatformSourceAddrForDest(&src, dst); - //LogMsg("GetLLQEventPort: src %#a for dst %#a (%d)", &src, dst, mDNSv4AddrIsRFC1918(&src.ip.v4) ? mDNSVal16(m->LLQNAT.ExternalPort) : 0); - return(mDNSv4AddrIsRFC1918(&src.ip.v4) ? mDNSVal16(m->LLQNAT.ExternalPort) : mDNSVal16(MulticastDNSPort)); - } - -// Normally called with llq set. -// May be called with llq NULL, when retransmitting a lost Challenge Response -mDNSlocal void sendChallengeResponse(mDNS *const m, DNSQuestion *const q, const LLQOptData *llq) - { - mDNSu8 *responsePtr = m->omsg.data; - LLQOptData llqBuf; - - if (q->tcp) { LogMsg("sendChallengeResponse: ERROR!!: question %##s (%s) tcp non-NULL", q->qname.c, DNSTypeName(q->qtype)); return; } - - if (PrivateQuery(q)) { LogMsg("sendChallengeResponse: ERROR!!: Private Query %##s (%s)", q->qname.c, DNSTypeName(q->qtype)); return; } - - if (q->ntries++ == kLLQ_MAX_TRIES) - { - LogMsg("sendChallengeResponse: %d failed attempts for LLQ %##s", kLLQ_MAX_TRIES, q->qname.c); - StartLLQPolling(m,q); - return; - } - - if (!llq) // Retransmission: need to make a new LLQOptData - { - llqBuf.vers = kLLQ_Vers; - llqBuf.llqOp = kLLQOp_Setup; - llqBuf.err = LLQErr_NoError; // Don't need to tell server UDP notification port when sending over UDP - llqBuf.id = q->id; - llqBuf.llqlease = q->ReqLease; - llq = &llqBuf; - } - - q->LastQTime = m->timenow; - q->ThisQInterval = q->tcp ? 0 : (kLLQ_INIT_RESEND * q->ntries * mDNSPlatformOneSecond); // If using TCP, don't need to retransmit - SetNextQueryTime(m, q); - - // To simulate loss of challenge response packet, uncomment line below - //if (q->ntries == 1) return; - - InitializeDNSMessage(&m->omsg.h, q->TargetQID, uQueryFlags); - responsePtr = putLLQ(&m->omsg, responsePtr, q, llq); - if (responsePtr) - { - mStatus err = mDNSSendDNSMessage(m, &m->omsg, responsePtr, mDNSInterface_Any, q->LocalSocket, &q->servAddr, q->servPort, mDNSNULL, mDNSNULL); - if (err) { LogMsg("sendChallengeResponse: mDNSSendDNSMessage%s failed: %d", q->tcp ? " (TCP)" : "", err); } - } - else StartLLQPolling(m,q); - } - -mDNSlocal void SetLLQTimer(mDNS *const m, DNSQuestion *const q, const LLQOptData *const llq) - { - mDNSs32 lease = (mDNSs32)llq->llqlease * mDNSPlatformOneSecond; - q->ReqLease = llq->llqlease; - q->LastQTime = m->timenow; - q->expire = m->timenow + lease; - q->ThisQInterval = lease/2 + mDNSRandom(lease/10); - debugf("SetLLQTimer setting %##s (%s) to %d %d", q->qname.c, DNSTypeName(q->qtype), lease/mDNSPlatformOneSecond, q->ThisQInterval/mDNSPlatformOneSecond); - SetNextQueryTime(m, q); - } - -mDNSlocal void recvSetupResponse(mDNS *const m, mDNSu8 rcode, DNSQuestion *const q, const LLQOptData *const llq) - { - if (rcode && rcode != kDNSFlag1_RC_NXDomain) - { LogMsg("ERROR: recvSetupResponse %##s (%s) - rcode && rcode != kDNSFlag1_RC_NXDomain", q->qname.c, DNSTypeName(q->qtype)); return; } - - if (llq->llqOp != kLLQOp_Setup) - { LogMsg("ERROR: recvSetupResponse %##s (%s) - bad op %d", q->qname.c, DNSTypeName(q->qtype), llq->llqOp); return; } - - if (llq->vers != kLLQ_Vers) - { LogMsg("ERROR: recvSetupResponse %##s (%s) - bad vers %d", q->qname.c, DNSTypeName(q->qtype), llq->vers); return; } - - if (q->state == LLQ_InitialRequest) - { - //LogInfo("Got LLQ_InitialRequest"); - - if (llq->err) { LogMsg("recvSetupResponse - received llq->err %d from server", llq->err); StartLLQPolling(m,q); return; } - - if (q->ReqLease != llq->llqlease) - debugf("recvSetupResponse: requested lease %lu, granted lease %lu", q->ReqLease, llq->llqlease); - - // cache expiration in case we go to sleep before finishing setup - q->ReqLease = llq->llqlease; - q->expire = m->timenow + ((mDNSs32)llq->llqlease * mDNSPlatformOneSecond); - - // update state - q->state = LLQ_SecondaryRequest; - q->id = llq->id; - q->ntries = 0; // first attempt to send response - sendChallengeResponse(m, q, llq); - } - else if (q->state == LLQ_SecondaryRequest) - { - //LogInfo("Got LLQ_SecondaryRequest"); - - // Fix this immediately if not sooner. Copy the id from the LLQOptData into our DNSQuestion struct. This is only - // an issue for private LLQs, because we skip parts 2 and 3 of the handshake. This is related to a bigger - // problem of the current implementation of TCP LLQ setup: we're not handling state transitions correctly - // if the server sends back SERVFULL or STATIC. - if (PrivateQuery(q)) - { - LogInfo("Private LLQ_SecondaryRequest; copying id %08X%08X", llq->id.l[0], llq->id.l[1]); - q->id = llq->id; - } - - if (llq->err) { LogMsg("ERROR: recvSetupResponse %##s (%s) code %d from server", q->qname.c, DNSTypeName(q->qtype), llq->err); StartLLQPolling(m,q); return; } - if (!mDNSSameOpaque64(&q->id, &llq->id)) - { LogMsg("recvSetupResponse - ID changed. discarding"); return; } // this can happen rarely (on packet loss + reordering) - q->state = LLQ_Established; - q->ntries = 0; - SetLLQTimer(m, q, llq); -#if APPLE_OSX_mDNSResponder - UpdateAutoTunnelDomainStatuses(m); -#endif - } - } - -mDNSexport uDNS_LLQType uDNS_recvLLQResponse(mDNS *const m, const DNSMessage *const msg, const mDNSu8 *const end, - const mDNSAddr *const srcaddr, const mDNSIPPort srcport, DNSQuestion **matchQuestion) - { - DNSQuestion pktQ, *q; - if (msg->h.numQuestions && getQuestion(msg, msg->data, end, 0, &pktQ)) - { - const rdataOPT *opt = GetLLQOptData(m, msg, end); - - for (q = m->Questions; q; q = q->next) - { - if (!mDNSOpaque16IsZero(q->TargetQID) && q->LongLived && q->qtype == pktQ.qtype && q->qnamehash == pktQ.qnamehash && SameDomainName(&q->qname, &pktQ.qname)) - { - debugf("uDNS_recvLLQResponse found %##s (%s) %d %#a %#a %X %X %X %X %d", - q->qname.c, DNSTypeName(q->qtype), q->state, srcaddr, &q->servAddr, - opt ? opt->u.llq.id.l[0] : 0, opt ? opt->u.llq.id.l[1] : 0, q->id.l[0], q->id.l[1], opt ? opt->u.llq.llqOp : 0); - if (q->state == LLQ_Poll) debugf("uDNS_LLQ_Events: q->state == LLQ_Poll msg->h.id %d q->TargetQID %d", mDNSVal16(msg->h.id), mDNSVal16(q->TargetQID)); - if (q->state == LLQ_Poll && mDNSSameOpaque16(msg->h.id, q->TargetQID)) - { - m->rec.r.resrec.RecordType = 0; // Clear RecordType to show we're not still using it - - // Don't reset the state to IntialRequest as we may write that to the dynamic store - // and PrefPane might wrongly think that we are "Starting" instead of "Polling". If - // we are in polling state because of NAT-PMP disabled or DoubleNAT, next LLQNATCallback - // would kick us back to LLQInitialRequest. So, resetting the state here may not be useful. - // - // If we have a good NAT (neither NAT-PMP disabled nor Double-NAT), then we should not be - // possibly in polling state. To be safe, we want to retry from the start in that case - // as there may not be another LLQNATCallback - // - // NOTE: We can be in polling state if we cannot resolve the SOA record i.e, servAddr is set to - // all ones. In that case, we would set it in LLQ_InitialRequest as it overrides the NAT-PMP or - // Double-NAT state. - if (!mDNSAddressIsOnes(&q->servAddr) && !mDNSIPPortIsZero(m->LLQNAT.ExternalPort) && - !m->LLQNAT.Result) - { - debugf("uDNS_recvLLQResponse got poll response; moving to LLQ_InitialRequest for %##s (%s)", q->qname.c, DNSTypeName(q->qtype)); - q->state = LLQ_InitialRequest; - } - q->servPort = zeroIPPort; // Clear servPort so that startLLQHandshake will retry the GetZoneData processing - q->ThisQInterval = LLQ_POLL_INTERVAL + mDNSRandom(LLQ_POLL_INTERVAL/10); // Retry LLQ setup in approx 15 minutes - q->LastQTime = m->timenow; - SetNextQueryTime(m, q); - *matchQuestion = q; - return uDNS_LLQ_Entire; // uDNS_LLQ_Entire means flush stale records; assume a large effective TTL - } - // Note: In LLQ Event packets, the msg->h.id does not match our q->TargetQID, because in that case the msg->h.id nonce is selected by the server - else if (opt && q->state == LLQ_Established && opt->u.llq.llqOp == kLLQOp_Event && mDNSSameOpaque64(&opt->u.llq.id, &q->id)) - { - mDNSu8 *ackEnd; - //debugf("Sending LLQ ack for %##s (%s)", q->qname.c, DNSTypeName(q->qtype)); - InitializeDNSMessage(&m->omsg.h, msg->h.id, ResponseFlags); - ackEnd = putLLQ(&m->omsg, m->omsg.data, q, &opt->u.llq); - if (ackEnd) mDNSSendDNSMessage(m, &m->omsg, ackEnd, mDNSInterface_Any, q->LocalSocket, srcaddr, srcport, mDNSNULL, mDNSNULL); - m->rec.r.resrec.RecordType = 0; // Clear RecordType to show we're not still using it - debugf("uDNS_LLQ_Events: q->state == LLQ_Established msg->h.id %d q->TargetQID %d", mDNSVal16(msg->h.id), mDNSVal16(q->TargetQID)); - *matchQuestion = q; - return uDNS_LLQ_Events; - } - if (opt && mDNSSameOpaque16(msg->h.id, q->TargetQID)) - { - if (q->state == LLQ_Established && opt->u.llq.llqOp == kLLQOp_Refresh && mDNSSameOpaque64(&opt->u.llq.id, &q->id) && msg->h.numAdditionals && !msg->h.numAnswers) - { - if (opt->u.llq.err != LLQErr_NoError) LogMsg("recvRefreshReply: received error %d from server", opt->u.llq.err); - else - { - //LogInfo("Received refresh confirmation ntries %d for %##s (%s)", q->ntries, q->qname.c, DNSTypeName(q->qtype)); - // If we're waiting to go to sleep, then this LLQ deletion may have been the thing - // we were waiting for, so schedule another check to see if we can sleep now. - if (opt->u.llq.llqlease == 0 && m->SleepLimit) m->NextScheduledSPRetry = m->timenow; - GrantCacheExtensions(m, q, opt->u.llq.llqlease); - SetLLQTimer(m, q, &opt->u.llq); - q->ntries = 0; - } - m->rec.r.resrec.RecordType = 0; // Clear RecordType to show we're not still using it - *matchQuestion = q; - return uDNS_LLQ_Ignore; - } - if (q->state < LLQ_Established && mDNSSameAddress(srcaddr, &q->servAddr)) - { - LLQ_State oldstate = q->state; - recvSetupResponse(m, msg->h.flags.b[1] & kDNSFlag1_RC_Mask, q, &opt->u.llq); - m->rec.r.resrec.RecordType = 0; // Clear RecordType to show we're not still using it - // We have a protocol anomaly here in the LLQ definition. - // Both the challenge packet from the server and the ack+answers packet have opt->u.llq.llqOp == kLLQOp_Setup. - // However, we need to treat them differently: - // The challenge packet has no answers in it, and tells us nothing about whether our cache entries - // are still valid, so this packet should not cause us to do anything that messes with our cache. - // The ack+answers packet gives us the whole truth, so we should handle it by updating our cache - // to match the answers in the packet, and only the answers in the packet. - *matchQuestion = q; - return (oldstate == LLQ_SecondaryRequest ? uDNS_LLQ_Entire : uDNS_LLQ_Ignore); - } - } - } - } - m->rec.r.resrec.RecordType = 0; // Clear RecordType to show we're not still using it - } - *matchQuestion = mDNSNULL; - return uDNS_LLQ_Not; - } - -// Stub definition of TCPSocket_struct so we can access flags field. (Rest of TCPSocket_struct is platform-dependent.) -struct TCPSocket_struct { TCPSocketFlags flags; /* ... */ }; - -// tcpCallback is called to handle events (e.g. connection opening and data reception) on TCP connections for -// Private DNS operations -- private queries, private LLQs, private record updates and private service updates -mDNSlocal void tcpCallback(TCPSocket *sock, void *context, mDNSBool ConnectionEstablished, mStatus err) - { - tcpInfo_t *tcpInfo = (tcpInfo_t *)context; - mDNSBool closed = mDNSfalse; - mDNS *m = tcpInfo->m; - DNSQuestion *const q = tcpInfo->question; - tcpInfo_t **backpointer = - q ? &q ->tcp : - tcpInfo->rr ? &tcpInfo->rr ->tcp : mDNSNULL; - if (backpointer && *backpointer != tcpInfo) - LogMsg("tcpCallback: %d backpointer %p incorrect tcpInfo %p question %p rr %p", - mDNSPlatformTCPGetFD(tcpInfo->sock), *backpointer, tcpInfo, q, tcpInfo->rr); - - if (err) goto exit; - - if (ConnectionEstablished) - { - mDNSu8 *end = ((mDNSu8*) &tcpInfo->request) + tcpInfo->requestLen; - DomainAuthInfo *AuthInfo; - - // Defensive coding for Crash in mDNSResponder at GetAuthInfoForName_internal + 366 - // Don't know yet what's causing this, but at least we can be cautious and try to avoid crashing if we find our pointers in an unexpected state - if (tcpInfo->rr && tcpInfo->rr->resrec.name != &tcpInfo->rr->namestorage) - LogMsg("tcpCallback: ERROR: tcpInfo->rr->resrec.name %p != &tcpInfo->rr->namestorage %p", - tcpInfo->rr->resrec.name, &tcpInfo->rr->namestorage); - if (tcpInfo->rr && tcpInfo->rr-> resrec.name != &tcpInfo->rr-> namestorage) return; - - AuthInfo = tcpInfo->rr ? GetAuthInfoForName(m, tcpInfo->rr->resrec.name) : mDNSNULL; - - // connection is established - send the message - if (q && q->LongLived && q->state == LLQ_Established) - { - // Lease renewal over TCP, resulting from opening a TCP connection in sendLLQRefresh - end = ((mDNSu8*) &tcpInfo->request) + tcpInfo->requestLen; - } - else if (q && q->LongLived && q->state != LLQ_Poll && !mDNSIPPortIsZero(m->LLQNAT.ExternalPort) && !mDNSIPPortIsZero(q->servPort)) - { - // Notes: - // If we have a NAT port mapping, ExternalPort is the external port - // If we have a routable address so we don't need a port mapping, ExternalPort is the same as our own internal port - // If we need a NAT port mapping but can't get one, then ExternalPort is zero - LLQOptData llqData; // set llq rdata - llqData.vers = kLLQ_Vers; - llqData.llqOp = kLLQOp_Setup; - llqData.err = GetLLQEventPort(m, &tcpInfo->Addr); // We're using TCP; tell server what UDP port to send notifications to - LogInfo("tcpCallback: eventPort %d", llqData.err); - llqData.id = zeroOpaque64; - llqData.llqlease = kLLQ_DefLease; - InitializeDNSMessage(&tcpInfo->request.h, q->TargetQID, uQueryFlags); - end = putLLQ(&tcpInfo->request, tcpInfo->request.data, q, &llqData); - if (!end) { LogMsg("ERROR: tcpCallback - putLLQ"); err = mStatus_UnknownErr; goto exit; } - AuthInfo = q->AuthInfo; // Need to add TSIG to this message - q->ntries = 0; // Reset ntries so that tcp/tls connection failures don't affect sendChallengeResponse failures - } - else if (q) - { - // LLQ Polling mode or non-LLQ uDNS over TCP - InitializeDNSMessage(&tcpInfo->request.h, q->TargetQID, uQueryFlags); - end = putQuestion(&tcpInfo->request, tcpInfo->request.data, tcpInfo->request.data + AbsoluteMaxDNSMessageData, &q->qname, q->qtype, q->qclass); - AuthInfo = q->AuthInfo; // Need to add TSIG to this message - } - - err = mDNSSendDNSMessage(m, &tcpInfo->request, end, mDNSInterface_Any, mDNSNULL, &tcpInfo->Addr, tcpInfo->Port, sock, AuthInfo); - if (err) { debugf("ERROR: tcpCallback: mDNSSendDNSMessage - %d", err); err = mStatus_UnknownErr; goto exit; } - - // Record time we sent this question - if (q) - { - mDNS_Lock(m); - q->LastQTime = m->timenow; - if (q->ThisQInterval < (256 * mDNSPlatformOneSecond)) // Now we have a TCP connection open, make sure we wait at least 256 seconds before retrying - q->ThisQInterval = (256 * mDNSPlatformOneSecond); - SetNextQueryTime(m, q); - mDNS_Unlock(m); - } - } - else - { - long n; - if (tcpInfo->nread < 2) // First read the two-byte length preceeding the DNS message - { - mDNSu8 *lenptr = (mDNSu8 *)&tcpInfo->replylen; - n = mDNSPlatformReadTCP(sock, lenptr + tcpInfo->nread, 2 - tcpInfo->nread, &closed); - if (n < 0) - { - LogMsg("ERROR: tcpCallback - attempt to read message length failed (%d)", n); - err = mStatus_ConnFailed; - goto exit; - } - else if (closed) - { - // It's perfectly fine for this socket to close after the first reply. The server might - // be sending gratuitous replies using UDP and doesn't have a need to leave the TCP socket open. - // We'll only log this event if we've never received a reply before. - // BIND 9 appears to close an idle connection after 30 seconds. - if (tcpInfo->numReplies == 0) - { - LogMsg("ERROR: socket closed prematurely tcpInfo->nread = %d", tcpInfo->nread); - err = mStatus_ConnFailed; - goto exit; - } - else - { - // Note that we may not be doing the best thing if an error occurs after we've sent a second request - // over this tcp connection. That is, we only track whether we've received at least one response - // which may have been to a previous request sent over this tcp connection. - if (backpointer) *backpointer = mDNSNULL; // Clear client backpointer FIRST so we don't risk double-disposing our tcpInfo_t - DisposeTCPConn(tcpInfo); - return; - } - } - - tcpInfo->nread += n; - if (tcpInfo->nread < 2) goto exit; - - tcpInfo->replylen = (mDNSu16)((mDNSu16)lenptr[0] << 8 | lenptr[1]); - if (tcpInfo->replylen < sizeof(DNSMessageHeader)) - { LogMsg("ERROR: tcpCallback - length too short (%d bytes)", tcpInfo->replylen); err = mStatus_UnknownErr; goto exit; } - - tcpInfo->reply = mDNSPlatformMemAllocate(tcpInfo->replylen); - if (!tcpInfo->reply) { LogMsg("ERROR: tcpCallback - malloc failed"); err = mStatus_NoMemoryErr; goto exit; } - } - - n = mDNSPlatformReadTCP(sock, ((char *)tcpInfo->reply) + (tcpInfo->nread - 2), tcpInfo->replylen - (tcpInfo->nread - 2), &closed); - - if (n < 0) - { - LogMsg("ERROR: tcpCallback - read returned %d", n); - err = mStatus_ConnFailed; - goto exit; - } - else if (closed) - { - if (tcpInfo->numReplies == 0) - { - LogMsg("ERROR: socket closed prematurely tcpInfo->nread = %d", tcpInfo->nread); - err = mStatus_ConnFailed; - goto exit; - } - else - { - // Note that we may not be doing the best thing if an error occurs after we've sent a second request - // over this tcp connection. That is, we only track whether we've received at least one response - // which may have been to a previous request sent over this tcp connection. - if (backpointer) *backpointer = mDNSNULL; // Clear client backpointer FIRST so we don't risk double-disposing our tcpInfo_t - DisposeTCPConn(tcpInfo); - return; - } - } - - tcpInfo->nread += n; - - if ((tcpInfo->nread - 2) == tcpInfo->replylen) - { - mDNSBool tls; - DNSMessage *reply = tcpInfo->reply; - mDNSu8 *end = (mDNSu8 *)tcpInfo->reply + tcpInfo->replylen; - mDNSAddr Addr = tcpInfo->Addr; - mDNSIPPort Port = tcpInfo->Port; - mDNSIPPort srcPort = zeroIPPort; - tcpInfo->numReplies++; - tcpInfo->reply = mDNSNULL; // Detach reply buffer from tcpInfo_t, to make sure client callback can't cause it to be disposed - tcpInfo->nread = 0; - tcpInfo->replylen = 0; - - // If we're going to dispose this connection, do it FIRST, before calling client callback - // Note: Sleep code depends on us clearing *backpointer here -- it uses the clearing of rr->tcp - // as the signal that the DNS deregistration operation with the server has completed, and the machine may now sleep - // If we clear the tcp pointer in the question, mDNSCoreReceiveResponse cannot find a matching question. Hence - // we store the minimal information i.e., the source port of the connection in the question itself. - // Dereference sock before it is disposed in DisposeTCPConn below. - - if (sock->flags & kTCPSocketFlags_UseTLS) tls = mDNStrue; - else tls = mDNSfalse; - - if (q && q->tcp) {srcPort = q->tcp->SrcPort; q->tcpSrcPort = srcPort;} - - if (backpointer) - if (!q || !q->LongLived || m->SleepState) - { *backpointer = mDNSNULL; DisposeTCPConn(tcpInfo); } - - mDNSCoreReceive(m, reply, end, &Addr, Port, tls ? (mDNSAddr *)1 : mDNSNULL, srcPort, 0); - // USE CAUTION HERE: Invoking mDNSCoreReceive may have caused the environment to change, including canceling this operation itself - - mDNSPlatformMemFree(reply); - return; - } - } - -exit: - - if (err) - { - // Clear client backpointer FIRST -- that way if one of the callbacks cancels its operation - // we won't end up double-disposing our tcpInfo_t - if (backpointer) *backpointer = mDNSNULL; - - mDNS_Lock(m); // Need to grab the lock to get m->timenow - - if (q) - { - if (q->ThisQInterval == 0) - { - // We get here when we fail to establish a new TCP/TLS connection that would have been used for a new LLQ request or an LLQ renewal. - // Note that ThisQInterval is also zero when sendChallengeResponse resends the LLQ request on an extant TCP/TLS connection. - q->LastQTime = m->timenow; - if (q->LongLived) - { - // We didn't get the chance to send our request packet before the TCP/TLS connection failed. - // We want to retry quickly, but want to back off exponentially in case the server is having issues. - // Since ThisQInterval was 0, we can't just multiply by QuestionIntervalStep, we must track the number - // of TCP/TLS connection failures using ntries. - mDNSu32 count = q->ntries + 1; // want to wait at least 1 second before retrying - - q->ThisQInterval = InitialQuestionInterval; - - for (;count;count--) - q->ThisQInterval *= QuestionIntervalStep; - - if (q->ThisQInterval > LLQ_POLL_INTERVAL) - q->ThisQInterval = LLQ_POLL_INTERVAL; - else - q->ntries++; - - LogMsg("tcpCallback: stream connection for LLQ %##s (%s) failed %d times, retrying in %d ms", q->qname.c, DNSTypeName(q->qtype), q->ntries, q->ThisQInterval); - } - else - { - q->ThisQInterval = MAX_UCAST_POLL_INTERVAL; - LogMsg("tcpCallback: stream connection for %##s (%s) failed, retrying in %d ms", q->qname.c, DNSTypeName(q->qtype), q->ThisQInterval); - } - SetNextQueryTime(m, q); - } - else if (NextQSendTime(q) - m->timenow > (q->LongLived ? LLQ_POLL_INTERVAL : MAX_UCAST_POLL_INTERVAL)) - { - // If we get an error and our next scheduled query for this question is more than the max interval from now, - // reset the next query to ensure we wait no longer the maximum interval from now before trying again. - q->LastQTime = m->timenow; - q->ThisQInterval = q->LongLived ? LLQ_POLL_INTERVAL : MAX_UCAST_POLL_INTERVAL; - SetNextQueryTime(m, q); - LogMsg("tcpCallback: stream connection for %##s (%s) failed, retrying in %d ms", q->qname.c, DNSTypeName(q->qtype), q->ThisQInterval); - } - - // We're about to dispose of the TCP connection, so we must reset the state to retry over TCP/TLS - // because sendChallengeResponse will send the query via UDP if we don't have a tcp pointer. - // Resetting to LLQ_InitialRequest will cause uDNS_CheckCurrentQuestion to call startLLQHandshake, which - // will attempt to establish a new tcp connection. - if (q->LongLived && q->state == LLQ_SecondaryRequest) - q->state = LLQ_InitialRequest; - - // ConnFailed may happen if the server sends a TCP reset or TLS fails, in which case we want to retry establishing the LLQ - // quickly rather than switching to polling mode. This case is handled by the above code to set q->ThisQInterval just above. - // If the error isn't ConnFailed, then the LLQ is in bad shape, so we switch to polling mode. - if (err != mStatus_ConnFailed) - { - if (q->LongLived && q->state != LLQ_Poll) StartLLQPolling(m, q); - } - } - - mDNS_Unlock(m); - - DisposeTCPConn(tcpInfo); - } - } - -mDNSlocal tcpInfo_t *MakeTCPConn(mDNS *const m, const DNSMessage *const msg, const mDNSu8 *const end, - TCPSocketFlags flags, const mDNSAddr *const Addr, const mDNSIPPort Port, domainname *hostname, - DNSQuestion *const question, AuthRecord *const rr) - { - mStatus err; - mDNSIPPort srcport = zeroIPPort; - tcpInfo_t *info; - - if ((flags & kTCPSocketFlags_UseTLS) && (!hostname || !hostname->c[0])) - { LogMsg("MakeTCPConn: TLS connection being setup with NULL hostname"); return mDNSNULL; } - - info = (tcpInfo_t *)mDNSPlatformMemAllocate(sizeof(tcpInfo_t)); - if (!info) { LogMsg("ERROR: MakeTCP - memallocate failed"); return(mDNSNULL); } - mDNSPlatformMemZero(info, sizeof(tcpInfo_t)); - - info->m = m; - info->sock = mDNSPlatformTCPSocket(m, flags, &srcport); - info->requestLen = 0; - info->question = question; - info->rr = rr; - info->Addr = *Addr; - info->Port = Port; - info->reply = mDNSNULL; - info->replylen = 0; - info->nread = 0; - info->numReplies = 0; - info->SrcPort = srcport; - - if (msg) - { - info->requestLen = (int) (end - ((mDNSu8*)msg)); - mDNSPlatformMemCopy(&info->request, msg, info->requestLen); - } - - if (!info->sock) { LogMsg("MakeTCPConn: unable to create TCP socket"); mDNSPlatformMemFree(info); return(mDNSNULL); } - err = mDNSPlatformTCPConnect(info->sock, Addr, Port, hostname, (question ? question->InterfaceID : mDNSNULL), tcpCallback, info); - - // Probably suboptimal here. - // Instead of returning mDNSNULL here on failure, we should probably invoke the callback with an error code. - // That way clients can put all the error handling and retry/recovery code in one place, - // instead of having to handle immediate errors in one place and async errors in another. - // Also: "err == mStatus_ConnEstablished" probably never happens. - - // Don't need to log "connection failed" in customer builds -- it happens quite often during sleep, wake, configuration changes, etc. - if (err == mStatus_ConnEstablished) { tcpCallback(info->sock, info, mDNStrue, mStatus_NoError); } - else if (err != mStatus_ConnPending ) { LogInfo("MakeTCPConn: connection failed"); DisposeTCPConn(info); return(mDNSNULL); } - return(info); - } - -mDNSexport void DisposeTCPConn(struct tcpInfo_t *tcp) - { - mDNSPlatformTCPCloseConnection(tcp->sock); - if (tcp->reply) mDNSPlatformMemFree(tcp->reply); - mDNSPlatformMemFree(tcp); - } - -// Lock must be held -mDNSexport void startLLQHandshake(mDNS *m, DNSQuestion *q) - { - if (mDNSIPv4AddressIsOnes(m->LLQNAT.ExternalAddress)) - { - LogInfo("startLLQHandshake: waiting for NAT status for %##s (%s)", q->qname.c, DNSTypeName(q->qtype)); - q->ThisQInterval = LLQ_POLL_INTERVAL + mDNSRandom(LLQ_POLL_INTERVAL/10); // Retry in approx 15 minutes - q->LastQTime = m->timenow; - SetNextQueryTime(m, q); - return; - } - - // Either we don't have NAT-PMP support (ExternalPort is zero) or behind a Double NAT that may or - // may not have NAT-PMP support (NATResult is non-zero) - if (mDNSIPPortIsZero(m->LLQNAT.ExternalPort) || m->LLQNAT.Result) - { - LogInfo("startLLQHandshake: Cannot receive inbound packets; will poll for %##s (%s) External Port %d, NAT Result %d", - q->qname.c, DNSTypeName(q->qtype), mDNSVal16(m->LLQNAT.ExternalPort), m->LLQNAT.Result); - StartLLQPolling(m, q); - return; - } - - if (mDNSIPPortIsZero(q->servPort)) - { - debugf("startLLQHandshake: StartGetZoneData for %##s (%s)", q->qname.c, DNSTypeName(q->qtype)); - q->ThisQInterval = LLQ_POLL_INTERVAL + mDNSRandom(LLQ_POLL_INTERVAL/10); // Retry in approx 15 minutes - q->LastQTime = m->timenow; - SetNextQueryTime(m, q); - q->servAddr = zeroAddr; - // We know q->servPort is zero because of check above - if (q->nta) CancelGetZoneData(m, q->nta); - q->nta = StartGetZoneData(m, &q->qname, ZoneServiceLLQ, LLQGotZoneData, q); - return; - } - - if (PrivateQuery(q)) - { - if (q->tcp) LogInfo("startLLQHandshake: Disposing existing TCP connection for %##s (%s)", q->qname.c, DNSTypeName(q->qtype)); - if (q->tcp) { DisposeTCPConn(q->tcp); q->tcp = mDNSNULL; } - if (!q->nta) - { - // Normally we lookup the zone data and then call this function. And we never free the zone data - // for "PrivateQuery". But sometimes this can happen due to some race conditions. When we - // switch networks, we might end up "Polling" the network e.g., we are behind a Double NAT. - // When we poll, we free the zone information as we send the query to the server (See - // PrivateQueryGotZoneData). The NAT callback (LLQNATCallback) may happen soon after that. If we - // are still behind Double NAT, we would have returned early in this function. But we could - // have switched to a network with no NATs and we should get the zone data again. - LogInfo("startLLQHandshake: nta is NULL for %##s (%s)", q->qname.c, DNSTypeName(q->qtype)); - q->nta = StartGetZoneData(m, &q->qname, ZoneServiceLLQ, LLQGotZoneData, q); - return; - } - else if (!q->nta->Host.c[0]) - { - // This should not happen. If it happens, we print a log and MakeTCPConn will fail if it can't find a hostname - LogMsg("startLLQHandshake: ERROR!!: nta non NULL for %##s (%s) but HostName %d NULL, LongLived %d", q->qname.c, DNSTypeName(q->qtype), q->nta->Host.c[0], q->LongLived); - } - q->tcp = MakeTCPConn(m, mDNSNULL, mDNSNULL, kTCPSocketFlags_UseTLS, &q->servAddr, q->servPort, &q->nta->Host, q, mDNSNULL); - if (!q->tcp) - q->ThisQInterval = mDNSPlatformOneSecond * 5; // If TCP failed (transient networking glitch) try again in five seconds - else - { - q->state = LLQ_SecondaryRequest; // Right now, for private DNS, we skip the four-way LLQ handshake - q->ReqLease = kLLQ_DefLease; - q->ThisQInterval = 0; - } - q->LastQTime = m->timenow; - SetNextQueryTime(m, q); - } - else - { - debugf("startLLQHandshake: m->AdvertisedV4 %#a%s Server %#a:%d%s %##s (%s)", - &m->AdvertisedV4, mDNSv4AddrIsRFC1918(&m->AdvertisedV4.ip.v4) ? " (RFC 1918)" : "", - &q->servAddr, mDNSVal16(q->servPort), mDNSAddrIsRFC1918(&q->servAddr) ? " (RFC 1918)" : "", - q->qname.c, DNSTypeName(q->qtype)); - - if (q->ntries++ >= kLLQ_MAX_TRIES) - { - LogMsg("startLLQHandshake: %d failed attempts for LLQ %##s Polling.", kLLQ_MAX_TRIES, q->qname.c); - StartLLQPolling(m, q); - } - else - { - mDNSu8 *end; - LLQOptData llqData; - - // set llq rdata - llqData.vers = kLLQ_Vers; - llqData.llqOp = kLLQOp_Setup; - llqData.err = LLQErr_NoError; // Don't need to tell server UDP notification port when sending over UDP - llqData.id = zeroOpaque64; - llqData.llqlease = kLLQ_DefLease; - - InitializeDNSMessage(&m->omsg.h, q->TargetQID, uQueryFlags); - end = putLLQ(&m->omsg, m->omsg.data, q, &llqData); - if (!end) { LogMsg("ERROR: startLLQHandshake - putLLQ"); StartLLQPolling(m,q); return; } - - mDNSSendDNSMessage(m, &m->omsg, end, mDNSInterface_Any, q->LocalSocket, &q->servAddr, q->servPort, mDNSNULL, mDNSNULL); - - // update question state - q->state = LLQ_InitialRequest; - q->ReqLease = kLLQ_DefLease; - q->ThisQInterval = (kLLQ_INIT_RESEND * mDNSPlatformOneSecond); - q->LastQTime = m->timenow; - SetNextQueryTime(m, q); - } - } - } - -// forward declaration so GetServiceTarget can do reverse lookup if needed -mDNSlocal void GetStaticHostname(mDNS *m); - -mDNSexport const domainname *GetServiceTarget(mDNS *m, AuthRecord *const rr) - { - debugf("GetServiceTarget %##s", rr->resrec.name->c); - - if (!rr->AutoTarget) // If not automatically tracking this host's current name, just return the existing target - return(&rr->resrec.rdata->u.srv.target); - else - { -#if APPLE_OSX_mDNSResponder - DomainAuthInfo *AuthInfo = GetAuthInfoForName_internal(m, rr->resrec.name); - if (AuthInfo && AuthInfo->AutoTunnel) - { - // If this AutoTunnel is not yet active, start it now (which entails activating its NAT Traversal request, - // which will subsequently advertise the appropriate records when the NAT Traversal returns a result) - if (!AuthInfo->AutoTunnelNAT.clientContext && m->AutoTunnelHostAddr.b[0]) - { - LogInfo("GetServiceTarget: Calling SetupLocalAutoTunnelInterface_internal"); - SetupLocalAutoTunnelInterface_internal(m, mDNStrue); - } - if (AuthInfo->AutoTunnelHostRecord.namestorage.c[0] == 0) return(mDNSNULL); - debugf("GetServiceTarget: Returning %##s", AuthInfo->AutoTunnelHostRecord.namestorage.c); - return(&AuthInfo->AutoTunnelHostRecord.namestorage); - } - else -#endif // APPLE_OSX_mDNSResponder - { - const int srvcount = CountLabels(rr->resrec.name); - HostnameInfo *besthi = mDNSNULL, *hi; - int best = 0; - for (hi = m->Hostnames; hi; hi = hi->next) - if (hi->arv4.state == regState_Registered || hi->arv4.state == regState_Refresh || - hi->arv6.state == regState_Registered || hi->arv6.state == regState_Refresh) - { - int x, hostcount = CountLabels(&hi->fqdn); - for (x = hostcount < srvcount ? hostcount : srvcount; x > 0 && x > best; x--) - if (SameDomainName(SkipLeadingLabels(rr->resrec.name, srvcount - x), SkipLeadingLabels(&hi->fqdn, hostcount - x))) - { best = x; besthi = hi; } - } - - if (besthi) return(&besthi->fqdn); - } - if (m->StaticHostname.c[0]) return(&m->StaticHostname); - else GetStaticHostname(m); // asynchronously do reverse lookup for primary IPv4 address - LogInfo("GetServiceTarget: Returning NULL for %s", ARDisplayString(m, rr)); - return(mDNSNULL); - } - } - -mDNSlocal const domainname *PUBLIC_UPDATE_SERVICE_TYPE = (const domainname*)"\x0B_dns-update" "\x04_udp"; -mDNSlocal const domainname *PUBLIC_LLQ_SERVICE_TYPE = (const domainname*)"\x08_dns-llq" "\x04_udp"; - -mDNSlocal const domainname *PRIVATE_UPDATE_SERVICE_TYPE = (const domainname*)"\x0F_dns-update-tls" "\x04_tcp"; -mDNSlocal const domainname *PRIVATE_QUERY_SERVICE_TYPE = (const domainname*)"\x0E_dns-query-tls" "\x04_tcp"; -mDNSlocal const domainname *PRIVATE_LLQ_SERVICE_TYPE = (const domainname*)"\x0C_dns-llq-tls" "\x04_tcp"; - -#define ZoneDataSRV(X) (\ - (X)->ZoneService == ZoneServiceUpdate ? ((X)->ZonePrivate ? PRIVATE_UPDATE_SERVICE_TYPE : PUBLIC_UPDATE_SERVICE_TYPE) : \ - (X)->ZoneService == ZoneServiceQuery ? ((X)->ZonePrivate ? PRIVATE_QUERY_SERVICE_TYPE : (const domainname*)"" ) : \ - (X)->ZoneService == ZoneServiceLLQ ? ((X)->ZonePrivate ? PRIVATE_LLQ_SERVICE_TYPE : PUBLIC_LLQ_SERVICE_TYPE ) : (const domainname*)"") - -// Forward reference: GetZoneData_StartQuery references GetZoneData_QuestionCallback, and -// GetZoneData_QuestionCallback calls GetZoneData_StartQuery -mDNSlocal mStatus GetZoneData_StartQuery(mDNS *const m, ZoneData *zd, mDNSu16 qtype); - -// GetZoneData_QuestionCallback is called from normal client callback context (core API calls allowed) -mDNSlocal void GetZoneData_QuestionCallback(mDNS *const m, DNSQuestion *question, const ResourceRecord *const answer, QC_result AddRecord) - { - ZoneData *zd = (ZoneData*)question->QuestionContext; - - debugf("GetZoneData_QuestionCallback: %s %s", AddRecord ? "Add" : "Rmv", RRDisplayString(m, answer)); - - if (!AddRecord) return; // Don't care about REMOVE events - if (AddRecord == QC_addnocache && answer->rdlength == 0) return; // Don't care about transient failure indications - if (answer->rrtype != question->qtype) return; // Don't care about CNAMEs - - if (answer->rrtype == kDNSType_SOA) - { - debugf("GetZoneData GOT SOA %s", RRDisplayString(m, answer)); - mDNS_StopQuery(m, question); - if (question->ThisQInterval != -1) - LogMsg("GetZoneData_QuestionCallback: Question %##s (%s) ThisQInterval %d not -1", question->qname.c, DNSTypeName(question->qtype), question->ThisQInterval); - if (answer->rdlength) - { - AssignDomainName(&zd->ZoneName, answer->name); - zd->ZoneClass = answer->rrclass; - AssignDomainName(&zd->question.qname, &zd->ZoneName); - GetZoneData_StartQuery(m, zd, kDNSType_SRV); - } - else if (zd->CurrentSOA->c[0]) - { - DomainAuthInfo *AuthInfo = GetAuthInfoForName(m, zd->CurrentSOA); - if (AuthInfo && AuthInfo->AutoTunnel) - { - // To keep the load on the server down, we don't chop down on - // SOA lookups for AutoTunnels - LogInfo("GetZoneData_QuestionCallback: not chopping labels for %##s", zd->CurrentSOA->c); - zd->ZoneDataCallback(m, mStatus_NoSuchNameErr, zd); - } - else - { - zd->CurrentSOA = (domainname *)(zd->CurrentSOA->c + zd->CurrentSOA->c[0]+1); - AssignDomainName(&zd->question.qname, zd->CurrentSOA); - GetZoneData_StartQuery(m, zd, kDNSType_SOA); - } - } - else - { - LogInfo("GetZoneData recursed to root label of %##s without finding SOA", zd->ChildName.c); - zd->ZoneDataCallback(m, mStatus_NoSuchNameErr, zd); - } - } - else if (answer->rrtype == kDNSType_SRV) - { - debugf("GetZoneData GOT SRV %s", RRDisplayString(m, answer)); - mDNS_StopQuery(m, question); - if (question->ThisQInterval != -1) - LogMsg("GetZoneData_QuestionCallback: Question %##s (%s) ThisQInterval %d not -1", question->qname.c, DNSTypeName(question->qtype), question->ThisQInterval); -// Right now we don't want to fail back to non-encrypted operations -// If the AuthInfo has the AutoTunnel field set, then we want private or nothing -// BTMM: Don't fallback to unencrypted operations when SRV lookup fails -#if 0 - if (!answer->rdlength && zd->ZonePrivate && zd->ZoneService != ZoneServiceQuery) - { - zd->ZonePrivate = mDNSfalse; // Causes ZoneDataSRV() to yield a different SRV name when building the query - GetZoneData_StartQuery(m, zd, kDNSType_SRV); // Try again, non-private this time - } - else -#endif - { - if (answer->rdlength) - { - AssignDomainName(&zd->Host, &answer->rdata->u.srv.target); - zd->Port = answer->rdata->u.srv.port; - AssignDomainName(&zd->question.qname, &zd->Host); - GetZoneData_StartQuery(m, zd, kDNSType_A); - } - else - { - zd->ZonePrivate = mDNSfalse; - zd->Host.c[0] = 0; - zd->Port = zeroIPPort; - zd->Addr = zeroAddr; - zd->ZoneDataCallback(m, mStatus_NoError, zd); - } - } - } - else if (answer->rrtype == kDNSType_A) - { - debugf("GetZoneData GOT A %s", RRDisplayString(m, answer)); - mDNS_StopQuery(m, question); - if (question->ThisQInterval != -1) - LogMsg("GetZoneData_QuestionCallback: Question %##s (%s) ThisQInterval %d not -1", question->qname.c, DNSTypeName(question->qtype), question->ThisQInterval); - zd->Addr.type = mDNSAddrType_IPv4; - zd->Addr.ip.v4 = (answer->rdlength == 4) ? answer->rdata->u.ipv4 : zerov4Addr; - // In order to simulate firewalls blocking our outgoing TCP connections, returning immediate ICMP errors or TCP resets, - // the code below will make us try to connect to loopback, resulting in an immediate "port unreachable" failure. - // This helps us test to make sure we handle this case gracefully - // BTMM: mDNSResponder taking 100 percent CPU after upgrading to 10.5.1 -#if 0 - zd->Addr.ip.v4.b[0] = 127; - zd->Addr.ip.v4.b[1] = 0; - zd->Addr.ip.v4.b[2] = 0; - zd->Addr.ip.v4.b[3] = 1; -#endif - // The caller needs to free the memory when done with zone data - zd->ZoneDataCallback(m, mStatus_NoError, zd); - } - } - -// GetZoneData_StartQuery is called from normal client context (lock not held, or client callback) -mDNSlocal mStatus GetZoneData_StartQuery(mDNS *const m, ZoneData *zd, mDNSu16 qtype) - { - if (qtype == kDNSType_SRV) - { - AssignDomainName(&zd->question.qname, ZoneDataSRV(zd)); - AppendDomainName(&zd->question.qname, &zd->ZoneName); - debugf("lookupDNSPort %##s", zd->question.qname.c); - } - - // CancelGetZoneData can get called at any time. We should stop the question if it has not been - // stopped already. A value of -1 for ThisQInterval indicates that the question is not active - // yet. - zd->question.ThisQInterval = -1; - zd->question.InterfaceID = mDNSInterface_Any; - zd->question.Target = zeroAddr; - //zd->question.qname.c[0] = 0; // Already set - zd->question.qtype = qtype; - zd->question.qclass = kDNSClass_IN; - zd->question.LongLived = mDNSfalse; - zd->question.ExpectUnique = mDNStrue; - zd->question.ForceMCast = mDNSfalse; - zd->question.ReturnIntermed = mDNStrue; - zd->question.SuppressUnusable = mDNSfalse; - zd->question.SearchListIndex = 0; - zd->question.AppendSearchDomains = 0; - zd->question.RetryWithSearchDomains = mDNSfalse; - zd->question.TimeoutQuestion = 0; - zd->question.WakeOnResolve = 0; - zd->question.qnameOrig = mDNSNULL; - zd->question.QuestionCallback = GetZoneData_QuestionCallback; - zd->question.QuestionContext = zd; - - //LogMsg("GetZoneData_StartQuery %##s (%s) %p", zd->question.qname.c, DNSTypeName(zd->question.qtype), zd->question.Private); - return(mDNS_StartQuery(m, &zd->question)); - } - -// StartGetZoneData is an internal routine (i.e. must be called with the lock already held) -mDNSexport ZoneData *StartGetZoneData(mDNS *const m, const domainname *const name, const ZoneService target, ZoneDataCallback callback, void *ZoneDataContext) - { - DomainAuthInfo *AuthInfo = GetAuthInfoForName_internal(m, name); - int initialskip = (AuthInfo && AuthInfo->AutoTunnel) ? DomainNameLength(name) - DomainNameLength(&AuthInfo->domain) : 0; - ZoneData *zd = (ZoneData*)mDNSPlatformMemAllocate(sizeof(ZoneData)); - if (!zd) { LogMsg("ERROR: StartGetZoneData - mDNSPlatformMemAllocate failed"); return mDNSNULL; } - mDNSPlatformMemZero(zd, sizeof(ZoneData)); - AssignDomainName(&zd->ChildName, name); - zd->ZoneService = target; - zd->CurrentSOA = (domainname *)(&zd->ChildName.c[initialskip]); - zd->ZoneName.c[0] = 0; - zd->ZoneClass = 0; - zd->Host.c[0] = 0; - zd->Port = zeroIPPort; - zd->Addr = zeroAddr; - zd->ZonePrivate = AuthInfo && AuthInfo->AutoTunnel ? mDNStrue : mDNSfalse; - zd->ZoneDataCallback = callback; - zd->ZoneDataContext = ZoneDataContext; - - zd->question.QuestionContext = zd; - - mDNS_DropLockBeforeCallback(); // GetZoneData_StartQuery expects to be called from a normal callback, so we emulate that here - if (AuthInfo && AuthInfo->AutoTunnel && !mDNSIPPortIsZero(AuthInfo->port)) - { - LogInfo("StartGetZoneData: Bypassing SOA, SRV query for %##s", AuthInfo->domain.c); - // We bypass SOA and SRV queries if we know the hostname and port already from the configuration. - // Today this is only true for AutoTunnel. As we bypass, we need to infer a few things: - // - // 1. Zone name is the same as the AuthInfo domain - // 2. ZoneClass is kDNSClass_IN which should be a safe assumption - // - // If we want to make this bypass mechanism work for non-AutoTunnels also, (1) has to hold - // good. Otherwise, it has to be configured also. - - AssignDomainName(&zd->ZoneName, &AuthInfo->domain); - zd->ZoneClass = kDNSClass_IN; - AssignDomainName(&zd->Host, &AuthInfo->hostname); - zd->Port = AuthInfo->port; - AssignDomainName(&zd->question.qname, &zd->Host); - GetZoneData_StartQuery(m, zd, kDNSType_A); - } - else - { - if (AuthInfo && AuthInfo->AutoTunnel) LogInfo("StartGetZoneData: Not Bypassing SOA, SRV query for %##s", AuthInfo->domain.c); - AssignDomainName(&zd->question.qname, zd->CurrentSOA); - GetZoneData_StartQuery(m, zd, kDNSType_SOA); - } - mDNS_ReclaimLockAfterCallback(); - - return zd; - } - -// Returns if the question is a GetZoneData question. These questions are special in -// that they are created internally while resolving a private query or LLQs. -mDNSexport mDNSBool IsGetZoneDataQuestion(DNSQuestion *q) - { - if (q->QuestionCallback == GetZoneData_QuestionCallback) return(mDNStrue); - else return(mDNSfalse); - } - -// GetZoneData queries are a special case -- even if we have a key for them, we don't do them privately, -// because that would result in an infinite loop (i.e. to do a private query we first need to get -// the _dns-query-tls SRV record for the zone, and we can't do *that* privately because to do so -// we'd need to already know the _dns-query-tls SRV record. -// Also, as a general rule, we never do SOA queries privately -mDNSexport DomainAuthInfo *GetAuthInfoForQuestion(mDNS *m, const DNSQuestion *const q) // Must be called with lock held - { - if (q->QuestionCallback == GetZoneData_QuestionCallback) return(mDNSNULL); - if (q->qtype == kDNSType_SOA ) return(mDNSNULL); - return(GetAuthInfoForName_internal(m, &q->qname)); - } - -// *************************************************************************** -#if COMPILER_LIKES_PRAGMA_MARK -#pragma mark - host name and interface management -#endif - -mDNSlocal void SendRecordRegistration(mDNS *const m, AuthRecord *rr); -mDNSlocal void SendRecordDeregistration(mDNS *m, AuthRecord *rr); -mDNSlocal mDNSBool IsRecordMergeable(mDNS *const m, AuthRecord *rr, mDNSs32 time); - -// When this function is called, service record is already deregistered. We just -// have to deregister the PTR and TXT records. -mDNSlocal void UpdateAllServiceRecords(mDNS *const m, AuthRecord *rr, mDNSBool reg) - { - AuthRecord *r, *srvRR; - - if (rr->resrec.rrtype != kDNSType_SRV) { LogMsg("UpdateAllServiceRecords:ERROR!! ResourceRecord not a service record %s", ARDisplayString(m, rr)); return; } - - if (reg && rr->state == regState_NoTarget) { LogMsg("UpdateAllServiceRecords:ERROR!! SRV record %s in noTarget state during registration", ARDisplayString(m, rr)); return; } - - LogInfo("UpdateAllServiceRecords: ResourceRecord %s", ARDisplayString(m, rr)); - - for (r = m->ResourceRecords; r; r=r->next) - { - if (!AuthRecord_uDNS(r)) continue; - srvRR = mDNSNULL; - if (r->resrec.rrtype == kDNSType_PTR) - srvRR = r->Additional1; - else if (r->resrec.rrtype == kDNSType_TXT) - srvRR = r->DependentOn; - if (srvRR && srvRR->resrec.rrtype != kDNSType_SRV) - LogMsg("UpdateAllServiceRecords: ERROR!! Resource record %s wrong, expecting SRV type", ARDisplayString(m, srvRR)); - if (srvRR == rr) - { - if (!reg) - { - LogInfo("UpdateAllServiceRecords: deregistering %s", ARDisplayString(m, r)); - r->SRVChanged = mDNStrue; - r->ThisAPInterval = INIT_RECORD_REG_INTERVAL; - r->LastAPTime = m->timenow - INIT_RECORD_REG_INTERVAL; - r->state = regState_DeregPending; - } - else - { - // Clearing SRVchanged is a safety measure. If our pevious dereg never - // came back and we had a target change, we are starting fresh - r->SRVChanged = mDNSfalse; - // if it is already registered or in the process of registering, then don't - // bother re-registering. This happens today for non-BTMM domains where the - // TXT and PTR get registered before SRV records because of the delay in - // getting the port mapping. There is no point in re-registering the TXT - // and PTR records. - if ((r->state == regState_Registered) || - (r->state == regState_Pending && r->nta && !mDNSIPv4AddressIsZero(r->nta->Addr.ip.v4))) - LogInfo("UpdateAllServiceRecords: not registering %s, state %d", ARDisplayString(m, r), r->state); - else - { - LogInfo("UpdateAllServiceRecords: registering %s, state %d", ARDisplayString(m, r), r->state); - ActivateUnicastRegistration(m, r); - } - } - } - } - } - -// Called in normal client context (lock not held) -// Currently only supports SRV records for nat mapping -mDNSlocal void CompleteRecordNatMap(mDNS *m, NATTraversalInfo *n) - { - const domainname *target; - domainname *srvt; - AuthRecord *rr = (AuthRecord *)n->clientContext; - debugf("SRVNatMap complete %.4a IntPort %u ExternalPort %u NATLease %u", &n->ExternalAddress, mDNSVal16(n->IntPort), mDNSVal16(n->ExternalPort), n->NATLease); - - if (!rr) { LogMsg("CompleteRecordNatMap called with unknown AuthRecord object"); return; } - if (!n->NATLease) { LogMsg("CompleteRecordNatMap No NATLease for %s", ARDisplayString(m, rr)); return; } - - if (rr->resrec.rrtype != kDNSType_SRV) {LogMsg("CompleteRecordNatMap: Not a service record %s", ARDisplayString(m, rr)); return; } - - if (rr->resrec.RecordType == kDNSRecordTypeDeregistering) { LogInfo("CompleteRecordNatMap called for %s, Service deregistering", ARDisplayString(m, rr)); return; } - - if (rr->state == regState_DeregPending) { LogInfo("CompleteRecordNatMap called for %s, record in DeregPending", ARDisplayString(m, rr)); return; } - - // As we free the zone info after registering/deregistering with the server (See hndlRecordUpdateReply), - // we need to restart the get zone data and nat mapping request to get the latest mapping result as we can't handle it - // at this moment. Restart from the beginning. - if (!rr->nta || mDNSIPv4AddressIsZero(rr->nta->Addr.ip.v4)) - { - LogInfo("CompleteRecordNatMap called for %s but no zone information!", ARDisplayString(m, rr)); - // We need to clear out the NATinfo state so that it will result in re-acquiring the mapping - // and hence this callback called again. - if (rr->NATinfo.clientContext) - { - mDNS_StopNATOperation_internal(m, &rr->NATinfo); - rr->NATinfo.clientContext = mDNSNULL; - } - rr->state = regState_Pending; - rr->ThisAPInterval = INIT_RECORD_REG_INTERVAL; - rr->LastAPTime = m->timenow - INIT_RECORD_REG_INTERVAL; - return; - } - - mDNS_Lock(m); - // Reevaluate the target always as Target could have changed while - // we were getting the port mapping (See UpdateOneSRVRecord) - target = GetServiceTarget(m, rr); - srvt = GetRRDomainNameTarget(&rr->resrec); - if (!target || target->c[0] == 0 || mDNSIPPortIsZero(n->ExternalPort)) - { - if (target && target->c[0]) - LogInfo("CompleteRecordNatMap - Target %##s for ResourceRecord %##s, ExternalPort %d", target->c, rr->resrec.name->c, mDNSVal16(n->ExternalPort)); - else - LogInfo("CompleteRecordNatMap - no target for %##s, ExternalPort %d", rr->resrec.name->c, mDNSVal16(n->ExternalPort)); - if (srvt) srvt->c[0] = 0; - rr->state = regState_NoTarget; - rr->resrec.rdlength = rr->resrec.rdestimate = 0; - mDNS_Unlock(m); - UpdateAllServiceRecords(m, rr, mDNSfalse); - return; - } - LogInfo("CompleteRecordNatMap - Target %##s for ResourceRecord %##s, ExternalPort %d", target->c, rr->resrec.name->c, mDNSVal16(n->ExternalPort)); - // This function might get called multiple times during a network transition event. Previosuly, we could - // have put the SRV record in NoTarget state above and deregistered all the other records. When this - // function gets called again with a non-zero ExternalPort, we need to set the target and register the - // other records again. - if (srvt && !SameDomainName(srvt, target)) - { - AssignDomainName(srvt, target); - SetNewRData(&rr->resrec, mDNSNULL, 0); // Update rdlength, rdestimate, rdatahash - } - - // SRVChanged is set when when the target of the SRV record changes (See UpdateOneSRVRecord). - // As a result of the target change, we might register just that SRV Record if it was - // previously registered and we have a new target OR deregister SRV (and the associated - // PTR/TXT records) if we don't have a target anymore. When we get a response from the server, - // SRVChanged state tells that we registered/deregistered because of a target change - // and hence handle accordingly e.g., if we deregistered, put the records in NoTarget state OR - // if we registered then put it in Registered state. - // - // Here, we are registering all the records again from the beginning. Treat this as first time - // registration rather than a temporary target change. - rr->SRVChanged = mDNSfalse; - - // We want IsRecordMergeable to check whether it is a record whose update can be - // sent with others. We set the time before we call IsRecordMergeable, so that - // it does not fail this record based on time. We are interested in other checks - // at this time - rr->state = regState_Pending; - rr->ThisAPInterval = INIT_RECORD_REG_INTERVAL; - rr->LastAPTime = m->timenow - INIT_RECORD_REG_INTERVAL; - if (IsRecordMergeable(m, rr, m->timenow + MERGE_DELAY_TIME)) - // Delay the record registration by MERGE_DELAY_TIME so that we can merge them - // into one update - rr->LastAPTime += MERGE_DELAY_TIME; - mDNS_Unlock(m); - // We call this always even though it may not be necessary always e.g., normal registration - // process where TXT and PTR gets registered followed by the SRV record after it gets - // the port mapping. In that case, UpdateAllServiceRecords handles the optimization. The - // update of TXT and PTR record is required if we entered noTargetState before as explained - // above. - UpdateAllServiceRecords(m, rr, mDNStrue); - } - -mDNSlocal void StartRecordNatMap(mDNS *m, AuthRecord *rr) - { - const mDNSu8 *p; - mDNSu8 protocol; - - if (rr->resrec.rrtype != kDNSType_SRV) - { - LogInfo("StartRecordNatMap: Resource Record %##s type %d, not supported", rr->resrec.name->c, rr->resrec.rrtype); - return; - } - p = rr->resrec.name->c; - //Assume ... - // Skip the first two labels to get to the transport protocol - if (p[0]) p += 1 + p[0]; - if (p[0]) p += 1 + p[0]; - if (SameDomainLabel(p, (mDNSu8 *)"\x4" "_tcp")) protocol = NATOp_MapTCP; - else if (SameDomainLabel(p, (mDNSu8 *)"\x4" "_udp")) protocol = NATOp_MapUDP; - else { LogMsg("StartRecordNatMap: could not determine transport protocol of service %##s", rr->resrec.name->c); return; } - - //LogMsg("StartRecordNatMap: clientContext %p IntPort %d srv.port %d %s", - // rr->NATinfo.clientContext, mDNSVal16(rr->NATinfo.IntPort), mDNSVal16(rr->resrec.rdata->u.srv.port), ARDisplayString(m, rr)); - if (rr->NATinfo.clientContext) mDNS_StopNATOperation_internal(m, &rr->NATinfo); - rr->NATinfo.Protocol = protocol; - - // Shouldn't be trying to set IntPort here -- - // BuildUpdateMessage overwrites srs->RR_SRV.resrec.rdata->u.srv.port with external (mapped) port number - rr->NATinfo.IntPort = rr->resrec.rdata->u.srv.port; - rr->NATinfo.RequestedPort = rr->resrec.rdata->u.srv.port; - rr->NATinfo.NATLease = 0; // Request default lease - rr->NATinfo.clientCallback = CompleteRecordNatMap; - rr->NATinfo.clientContext = rr; - mDNS_StartNATOperation_internal(m, &rr->NATinfo); - } - -// Unlink an Auth Record from the m->ResourceRecords list. -// When a resource record enters regState_NoTarget initially, mDNS_Register_internal -// does not initialize completely e.g., it cannot check for duplicates etc. The resource -// record is temporarily left in the ResourceRecords list so that we can initialize later -// when the target is resolvable. Similarly, when host name changes, we enter regState_NoTarget -// and we do the same. - -// This UnlinkResourceRecord routine is very worrying. It bypasses all the normal cleanup performed -// by mDNS_Deregister_internal and just unceremoniously cuts the record from the active list. -// This is why re-regsitering this record was producing syslog messages like this: -// "Error! Tried to add a NAT traversal that's already in the active list" -// Right now UnlinkResourceRecord is fortunately only called by RegisterAllServiceRecords, -// which then immediately calls mDNS_Register_internal to re-register the record, which probably -// masked more serious problems. Any other use of UnlinkResourceRecord is likely to lead to crashes. -// For now we'll workaround that specific problem by explicitly calling mDNS_StopNATOperation_internal, -// but long-term we should either stop cancelling the record registration and then re-registering it, -// or if we really do need to do this for some reason it should be done via the usual -// mDNS_Deregister_internal path instead of just cutting the record from the list. - -mDNSlocal mStatus UnlinkResourceRecord(mDNS *const m, AuthRecord *const rr) - { - AuthRecord **list = &m->ResourceRecords; - while (*list && *list != rr) list = &(*list)->next; - if (*list) - { - *list = rr->next; - rr->next = mDNSNULL; - - // Temporary workaround to cancel any active NAT mapping operation - if (rr->NATinfo.clientContext) - { - mDNS_StopNATOperation_internal(m, &rr->NATinfo); - rr->NATinfo.clientContext = mDNSNULL; - if (rr->resrec.rrtype == kDNSType_SRV) rr->resrec.rdata->u.srv.port = rr->NATinfo.IntPort; - } - - return(mStatus_NoError); - } - LogMsg("UnlinkResourceRecord:ERROR!! - no such active record %##s", rr->resrec.name->c); - return(mStatus_NoSuchRecord); - } - -// We need to go through mDNS_Register again as we did not complete the -// full initialization last time e.g., duplicate checks. -// After we register, we will be in regState_GetZoneData. -mDNSlocal void RegisterAllServiceRecords(mDNS *const m, AuthRecord *rr) - { - LogInfo("RegisterAllServiceRecords: Service Record %##s", rr->resrec.name->c); - // First Register the service record, we do this differently from other records because - // when it entered NoTarget state, it did not go through complete initialization - rr->SRVChanged = mDNSfalse; - UnlinkResourceRecord(m, rr); - mDNS_Register_internal(m, rr); - // Register the other records - UpdateAllServiceRecords(m, rr, mDNStrue); - } - -// Called with lock held -mDNSlocal void UpdateOneSRVRecord(mDNS *m, AuthRecord *rr) - { - // Target change if: - // We have a target and were previously waiting for one, or - // We had a target and no longer do, or - // The target has changed - - domainname *curtarget = &rr->resrec.rdata->u.srv.target; - const domainname *const nt = GetServiceTarget(m, rr); - const domainname *const newtarget = nt ? nt : (domainname*)""; - mDNSBool TargetChanged = (newtarget->c[0] && rr->state == regState_NoTarget) || !SameDomainName(curtarget, newtarget); - mDNSBool HaveZoneData = rr->nta && !mDNSIPv4AddressIsZero(rr->nta->Addr.ip.v4); - - // Nat state change if: - // We were behind a NAT, and now we are behind a new NAT, or - // We're not behind a NAT but our port was previously mapped to a different external port - // We were not behind a NAT and now we are - - mDNSIPPort port = rr->resrec.rdata->u.srv.port; - mDNSBool NowNeedNATMAP = (rr->AutoTarget == Target_AutoHostAndNATMAP && !mDNSIPPortIsZero(port) && mDNSv4AddrIsRFC1918(&m->AdvertisedV4.ip.v4) && rr->nta && !mDNSAddrIsRFC1918(&rr->nta->Addr)); - mDNSBool WereBehindNAT = (rr->NATinfo.clientContext != mDNSNULL); - mDNSBool PortWasMapped = (rr->NATinfo.clientContext && !mDNSSameIPPort(rr->NATinfo.RequestedPort, port)); // I think this is always false -- SC Sept 07 - mDNSBool NATChanged = (!WereBehindNAT && NowNeedNATMAP) || (!NowNeedNATMAP && PortWasMapped); - - (void)HaveZoneData; //unused - - LogInfo("UpdateOneSRVRecord: Resource Record %s TargetChanged %d, NewTarget %##s", ARDisplayString(m, rr), TargetChanged, nt->c); - - debugf("UpdateOneSRVRecord: %##s newtarget %##s TargetChanged %d HaveZoneData %d port %d NowNeedNATMAP %d WereBehindNAT %d PortWasMapped %d NATChanged %d", - rr->resrec.name->c, newtarget, - TargetChanged, HaveZoneData, mDNSVal16(port), NowNeedNATMAP, WereBehindNAT, PortWasMapped, NATChanged); - - if (m->mDNS_busy != m->mDNS_reentrancy+1) - LogMsg("UpdateOneSRVRecord: Lock not held! mDNS_busy (%ld) mDNS_reentrancy (%ld)", m->mDNS_busy, m->mDNS_reentrancy); - - if (!TargetChanged && !NATChanged) return; - - // If we are deregistering the record, then ignore any NAT/Target change. - if (rr->resrec.RecordType == kDNSRecordTypeDeregistering) - { - LogInfo("UpdateOneSRVRecord: Deregistering record, Ignoring TargetChanged %d, NATChanged %d for %##s, state %d", TargetChanged, NATChanged, - rr->resrec.name->c, rr->state); - return; - } - - if (newtarget) - LogInfo("UpdateOneSRVRecord: TargetChanged %d, NATChanged %d for %##s, state %d, newtarget %##s", TargetChanged, NATChanged, rr->resrec.name->c, rr->state, newtarget->c); - else - LogInfo("UpdateOneSRVRecord: TargetChanged %d, NATChanged %d for %##s, state %d, null newtarget", TargetChanged, NATChanged, rr->resrec.name->c, rr->state); - switch(rr->state) - { - case regState_NATMap: - // In these states, the SRV has either not yet been registered (it will get up-to-date information when it is) - // or is in the process of, or has already been, deregistered. This assumes that whenever we transition out - // of this state, we need to look at the target again. - return; - - case regState_UpdatePending: - // We are getting a Target change/NAT change while the SRV record is being updated ? - // let us not do anything for now. - return; - - case regState_NATError: - if (!NATChanged) return; - // if nat changed, register if we have a target (below) - - case regState_NoTarget: - if (!newtarget->c[0]) - { - LogInfo("UpdateOneSRVRecord: No target yet for Resource Record %s", ARDisplayString(m, rr)); - return; - } - RegisterAllServiceRecords(m , rr); - return; - case regState_DeregPending: - // We are in DeregPending either because the service was deregistered from above or we handled - // a NAT/Target change before and sent the deregistration below. There are a few race conditions - // possible - // - // 1. We are handling a second NAT/Target change while the first dereg is in progress. It is possible - // that first dereg never made it through because there was no network connectivity e.g., disconnecting - // from network triggers this function due to a target change and later connecting to the network - // retriggers this function but the deregistration never made it through yet. Just fall through. - // If there is a target register otherwise deregister. - // - // 2. While we sent the dereg during a previous NAT/Target change, uDNS_DeregisterRecord gets - // called as part of service deregistration. When the response comes back, we call - // CompleteDeregistration rather than handle NAT/Target change because the record is in - // kDNSRecordTypeDeregistering state. - // - // 3. If the upper layer deregisters the service, we check for kDNSRecordTypeDeregistering both - // here in this function to avoid handling NAT/Target change and in hndlRecordUpdateReply to call - // CompleteDeregistration instead of handling NAT/Target change. Hence, we are not concerned - // about that case here. - // - // We just handle case (1) by falling through - case regState_Pending: - case regState_Refresh: - case regState_Registered: - // target or nat changed. deregister service. upon completion, we'll look for a new target - rr->SRVChanged = mDNStrue; - rr->ThisAPInterval = INIT_RECORD_REG_INTERVAL; - rr->LastAPTime = m->timenow - INIT_RECORD_REG_INTERVAL; - if (newtarget->c[0]) - { - LogInfo("UpdateOneSRVRecord: SRV record changed for service %##s, registering with new target %##s", - rr->resrec.name->c, newtarget->c); - rr->state = regState_Pending; - } - else - { - LogInfo("UpdateOneSRVRecord: SRV record changed for service %##s de-registering", rr->resrec.name->c); - rr->state = regState_DeregPending; - UpdateAllServiceRecords(m, rr, mDNSfalse); - } - return; - case regState_Unregistered: - default: LogMsg("UpdateOneSRVRecord: Unknown state %d for %##s", rr->state, rr->resrec.name->c); - } - } - -mDNSexport void UpdateAllSRVRecords(mDNS *m) - { - m->NextSRVUpdate = 0; - LogInfo("UpdateAllSRVRecords %d", m->SleepState); - - if (m->CurrentRecord) - LogMsg("UpdateAllSRVRecords ERROR m->CurrentRecord already set %s", ARDisplayString(m, m->CurrentRecord)); - m->CurrentRecord = m->ResourceRecords; - while (m->CurrentRecord) - { - AuthRecord *rptr = m->CurrentRecord; - m->CurrentRecord = m->CurrentRecord->next; - if (AuthRecord_uDNS(rptr) && rptr->resrec.rrtype == kDNSType_SRV) - UpdateOneSRVRecord(m, rptr); - } - } - -// Forward reference: AdvertiseHostname references HostnameCallback, and HostnameCallback calls AdvertiseHostname -mDNSlocal void HostnameCallback(mDNS *const m, AuthRecord *const rr, mStatus result); - -// Called in normal client context (lock not held) -mDNSlocal void hostnameGetPublicAddressCallback(mDNS *m, NATTraversalInfo *n) - { - HostnameInfo *h = (HostnameInfo *)n->clientContext; - - if (!h) { LogMsg("RegisterHostnameRecord: registration cancelled"); return; } - - if (!n->Result) - { - if (mDNSIPv4AddressIsZero(n->ExternalAddress) || mDNSv4AddrIsRFC1918(&n->ExternalAddress)) return; - - if (h->arv4.resrec.RecordType) - { - if (mDNSSameIPv4Address(h->arv4.resrec.rdata->u.ipv4, n->ExternalAddress)) return; // If address unchanged, do nothing - LogInfo("Updating hostname %p %##s IPv4 from %.4a to %.4a (NAT gateway's external address)",n, - h->arv4.resrec.name->c, &h->arv4.resrec.rdata->u.ipv4, &n->ExternalAddress); - mDNS_Deregister(m, &h->arv4); // mStatus_MemFree callback will re-register with new address - } - else - { - LogInfo("Advertising hostname %##s IPv4 %.4a (NAT gateway's external address)", h->arv4.resrec.name->c, &n->ExternalAddress); - h->arv4.resrec.RecordType = kDNSRecordTypeKnownUnique; - h->arv4.resrec.rdata->u.ipv4 = n->ExternalAddress; - mDNS_Register(m, &h->arv4); - } - } - } - -// register record or begin NAT traversal -mDNSlocal void AdvertiseHostname(mDNS *m, HostnameInfo *h) - { - if (!mDNSIPv4AddressIsZero(m->AdvertisedV4.ip.v4) && h->arv4.resrec.RecordType == kDNSRecordTypeUnregistered) - { - mDNS_SetupResourceRecord(&h->arv4, mDNSNULL, mDNSInterface_Any, kDNSType_A, kHostNameTTL, kDNSRecordTypeUnregistered, AuthRecordAny, HostnameCallback, h); - AssignDomainName(&h->arv4.namestorage, &h->fqdn); - h->arv4.resrec.rdata->u.ipv4 = m->AdvertisedV4.ip.v4; - h->arv4.state = regState_Unregistered; - if (mDNSv4AddrIsRFC1918(&m->AdvertisedV4.ip.v4)) - { - // If we already have a NAT query active, stop it and restart it to make sure we get another callback - if (h->natinfo.clientContext) mDNS_StopNATOperation_internal(m, &h->natinfo); - h->natinfo.Protocol = 0; - h->natinfo.IntPort = zeroIPPort; - h->natinfo.RequestedPort = zeroIPPort; - h->natinfo.NATLease = 0; - h->natinfo.clientCallback = hostnameGetPublicAddressCallback; - h->natinfo.clientContext = h; - mDNS_StartNATOperation_internal(m, &h->natinfo); - } - else - { - LogInfo("Advertising hostname %##s IPv4 %.4a", h->arv4.resrec.name->c, &m->AdvertisedV4.ip.v4); - h->arv4.resrec.RecordType = kDNSRecordTypeKnownUnique; - mDNS_Register_internal(m, &h->arv4); - } - } - - if (!mDNSIPv6AddressIsZero(m->AdvertisedV6.ip.v6) && h->arv6.resrec.RecordType == kDNSRecordTypeUnregistered) - { - mDNS_SetupResourceRecord(&h->arv6, mDNSNULL, mDNSInterface_Any, kDNSType_AAAA, kHostNameTTL, kDNSRecordTypeKnownUnique, AuthRecordAny, HostnameCallback, h); - AssignDomainName(&h->arv6.namestorage, &h->fqdn); - h->arv6.resrec.rdata->u.ipv6 = m->AdvertisedV6.ip.v6; - h->arv6.state = regState_Unregistered; - LogInfo("Advertising hostname %##s IPv6 %.16a", h->arv6.resrec.name->c, &m->AdvertisedV6.ip.v6); - mDNS_Register_internal(m, &h->arv6); - } - } - -mDNSlocal void HostnameCallback(mDNS *const m, AuthRecord *const rr, mStatus result) - { - HostnameInfo *hi = (HostnameInfo *)rr->RecordContext; - - if (result == mStatus_MemFree) - { - if (hi) - { - // If we're still in the Hostnames list, update to new address - HostnameInfo *i; - LogInfo("HostnameCallback: Got mStatus_MemFree for %p %p %s", hi, rr, ARDisplayString(m, rr)); - for (i = m->Hostnames; i; i = i->next) - if (rr == &i->arv4 || rr == &i->arv6) - { mDNS_Lock(m); AdvertiseHostname(m, i); mDNS_Unlock(m); return; } - - // Else, we're not still in the Hostnames list, so free the memory - if (hi->arv4.resrec.RecordType == kDNSRecordTypeUnregistered && - hi->arv6.resrec.RecordType == kDNSRecordTypeUnregistered) - { - if (hi->natinfo.clientContext) mDNS_StopNATOperation_internal(m, &hi->natinfo); - hi->natinfo.clientContext = mDNSNULL; - mDNSPlatformMemFree(hi); // free hi when both v4 and v6 AuthRecs deallocated - } - } - return; - } - - if (result) - { - // don't unlink or free - we can retry when we get a new address/router - if (rr->resrec.rrtype == kDNSType_A) - LogMsg("HostnameCallback: Error %d for registration of %##s IP %.4a", result, rr->resrec.name->c, &rr->resrec.rdata->u.ipv4); - else - LogMsg("HostnameCallback: Error %d for registration of %##s IP %.16a", result, rr->resrec.name->c, &rr->resrec.rdata->u.ipv6); - if (!hi) { mDNSPlatformMemFree(rr); return; } - if (rr->state != regState_Unregistered) LogMsg("Error: HostnameCallback invoked with error code for record not in regState_Unregistered!"); - - if (hi->arv4.state == regState_Unregistered && - hi->arv6.state == regState_Unregistered) - { - // only deliver status if both v4 and v6 fail - rr->RecordContext = (void *)hi->StatusContext; - if (hi->StatusCallback) - hi->StatusCallback(m, rr, result); // client may NOT make API calls here - rr->RecordContext = (void *)hi; - } - return; - } - - // register any pending services that require a target - mDNS_Lock(m); - m->NextSRVUpdate = NonZeroTime(m->timenow); - mDNS_Unlock(m); - - // Deliver success to client - if (!hi) { LogMsg("HostnameCallback invoked with orphaned address record"); return; } - if (rr->resrec.rrtype == kDNSType_A) - LogInfo("Registered hostname %##s IP %.4a", rr->resrec.name->c, &rr->resrec.rdata->u.ipv4); - else - LogInfo("Registered hostname %##s IP %.16a", rr->resrec.name->c, &rr->resrec.rdata->u.ipv6); - - rr->RecordContext = (void *)hi->StatusContext; - if (hi->StatusCallback) - hi->StatusCallback(m, rr, result); // client may NOT make API calls here - rr->RecordContext = (void *)hi; - } - -mDNSlocal void FoundStaticHostname(mDNS *const m, DNSQuestion *question, const ResourceRecord *const answer, QC_result AddRecord) - { - const domainname *pktname = &answer->rdata->u.name; - domainname *storedname = &m->StaticHostname; - HostnameInfo *h = m->Hostnames; - - (void)question; - - if (answer->rdlength != 0) - LogInfo("FoundStaticHostname: question %##s -> answer %##s (%s)", question->qname.c, answer->rdata->u.name.c, AddRecord ? "ADD" : "RMV"); - else - LogInfo("FoundStaticHostname: question %##s -> answer NULL (%s)", question->qname.c, AddRecord ? "ADD" : "RMV"); - - if (AddRecord && answer->rdlength != 0 && !SameDomainName(pktname, storedname)) - { - AssignDomainName(storedname, pktname); - while (h) - { - if (h->arv4.state == regState_Pending || h->arv4.state == regState_NATMap || h->arv6.state == regState_Pending) - { - // if we're in the process of registering a dynamic hostname, delay SRV update so we don't have to reregister services if the dynamic name succeeds - m->NextSRVUpdate = NonZeroTime(m->timenow + 5 * mDNSPlatformOneSecond); - debugf("FoundStaticHostname: NextSRVUpdate in %d %d", m->NextSRVUpdate - m->timenow, m->timenow); - return; - } - h = h->next; - } - mDNS_Lock(m); - m->NextSRVUpdate = NonZeroTime(m->timenow); - mDNS_Unlock(m); - } - else if (!AddRecord && SameDomainName(pktname, storedname)) - { - mDNS_Lock(m); - storedname->c[0] = 0; - m->NextSRVUpdate = NonZeroTime(m->timenow); - mDNS_Unlock(m); - } - } - -// Called with lock held -mDNSlocal void GetStaticHostname(mDNS *m) - { - char buf[MAX_REVERSE_MAPPING_NAME_V4]; - DNSQuestion *q = &m->ReverseMap; - mDNSu8 *ip = m->AdvertisedV4.ip.v4.b; - mStatus err; - - if (m->ReverseMap.ThisQInterval != -1) return; // already running - if (mDNSIPv4AddressIsZero(m->AdvertisedV4.ip.v4)) return; - - mDNSPlatformMemZero(q, sizeof(*q)); - // Note: This is reverse order compared to a normal dotted-decimal IP address, so we can't use our customary "%.4a" format code - mDNS_snprintf(buf, sizeof(buf), "%d.%d.%d.%d.in-addr.arpa.", ip[3], ip[2], ip[1], ip[0]); - if (!MakeDomainNameFromDNSNameString(&q->qname, buf)) { LogMsg("Error: GetStaticHostname - bad name %s", buf); return; } - - q->InterfaceID = mDNSInterface_Any; - q->Target = zeroAddr; - q->qtype = kDNSType_PTR; - q->qclass = kDNSClass_IN; - q->LongLived = mDNSfalse; - q->ExpectUnique = mDNSfalse; - q->ForceMCast = mDNSfalse; - q->ReturnIntermed = mDNStrue; - q->SuppressUnusable = mDNSfalse; - q->SearchListIndex = 0; - q->AppendSearchDomains = 0; - q->RetryWithSearchDomains = mDNSfalse; - q->TimeoutQuestion = 0; - q->WakeOnResolve = 0; - q->qnameOrig = mDNSNULL; - q->QuestionCallback = FoundStaticHostname; - q->QuestionContext = mDNSNULL; - - LogInfo("GetStaticHostname: %##s (%s)", q->qname.c, DNSTypeName(q->qtype)); - err = mDNS_StartQuery_internal(m, q); - if (err) LogMsg("Error: GetStaticHostname - StartQuery returned error %d", err); - } - -mDNSexport void mDNS_AddDynDNSHostName(mDNS *m, const domainname *fqdn, mDNSRecordCallback *StatusCallback, const void *StatusContext) - { - HostnameInfo **ptr = &m->Hostnames; - - LogInfo("mDNS_AddDynDNSHostName %##s", fqdn); - - while (*ptr && !SameDomainName(fqdn, &(*ptr)->fqdn)) ptr = &(*ptr)->next; - if (*ptr) { LogMsg("DynDNSHostName %##s already in list", fqdn->c); return; } - - // allocate and format new address record - *ptr = mDNSPlatformMemAllocate(sizeof(**ptr)); - if (!*ptr) { LogMsg("ERROR: mDNS_AddDynDNSHostName - malloc"); return; } - - mDNSPlatformMemZero(*ptr, sizeof(**ptr)); - AssignDomainName(&(*ptr)->fqdn, fqdn); - (*ptr)->arv4.state = regState_Unregistered; - (*ptr)->arv6.state = regState_Unregistered; - (*ptr)->StatusCallback = StatusCallback; - (*ptr)->StatusContext = StatusContext; - - AdvertiseHostname(m, *ptr); - } - -mDNSexport void mDNS_RemoveDynDNSHostName(mDNS *m, const domainname *fqdn) - { - HostnameInfo **ptr = &m->Hostnames; - - LogInfo("mDNS_RemoveDynDNSHostName %##s", fqdn); - - while (*ptr && !SameDomainName(fqdn, &(*ptr)->fqdn)) ptr = &(*ptr)->next; - if (!*ptr) LogMsg("mDNS_RemoveDynDNSHostName: no such domainname %##s", fqdn->c); - else - { - HostnameInfo *hi = *ptr; - // We do it this way because, if we have no active v6 record, the "mDNS_Deregister_internal(m, &hi->arv4);" - // below could free the memory, and we have to make sure we don't touch hi fields after that. - mDNSBool f4 = hi->arv4.resrec.RecordType != kDNSRecordTypeUnregistered && hi->arv4.state != regState_Unregistered; - mDNSBool f6 = hi->arv6.resrec.RecordType != kDNSRecordTypeUnregistered && hi->arv6.state != regState_Unregistered; - if (f4) LogInfo("mDNS_RemoveDynDNSHostName removing v4 %##s", fqdn); - if (f6) LogInfo("mDNS_RemoveDynDNSHostName removing v6 %##s", fqdn); - *ptr = (*ptr)->next; // unlink - if (f4) mDNS_Deregister_internal(m, &hi->arv4, mDNS_Dereg_normal); - if (f6) mDNS_Deregister_internal(m, &hi->arv6, mDNS_Dereg_normal); - // When both deregistrations complete we'll free the memory in the mStatus_MemFree callback - } - if (!m->mDNS_busy) LogMsg("mDNS_RemoveDynDNSHostName: ERROR: Lock not held"); - m->NextSRVUpdate = NonZeroTime(m->timenow); - } - -// Currently called without holding the lock -// Maybe we should change that? -mDNSexport void mDNS_SetPrimaryInterfaceInfo(mDNS *m, const mDNSAddr *v4addr, const mDNSAddr *v6addr, const mDNSAddr *router) - { - mDNSBool v4Changed, v6Changed, RouterChanged; - - if (m->mDNS_busy != m->mDNS_reentrancy) - LogMsg("mDNS_SetPrimaryInterfaceInfo: mDNS_busy (%ld) != mDNS_reentrancy (%ld)", m->mDNS_busy, m->mDNS_reentrancy); - - if (v4addr && v4addr->type != mDNSAddrType_IPv4) { LogMsg("mDNS_SetPrimaryInterfaceInfo v4 address - incorrect type. Discarding. %#a", v4addr); return; } - if (v6addr && v6addr->type != mDNSAddrType_IPv6) { LogMsg("mDNS_SetPrimaryInterfaceInfo v6 address - incorrect type. Discarding. %#a", v6addr); return; } - if (router && router->type != mDNSAddrType_IPv4) { LogMsg("mDNS_SetPrimaryInterfaceInfo passed non-v4 router. Discarding. %#a", router); return; } - - mDNS_Lock(m); - - v4Changed = !mDNSSameIPv4Address(m->AdvertisedV4.ip.v4, v4addr ? v4addr->ip.v4 : zerov4Addr); - v6Changed = !mDNSSameIPv6Address(m->AdvertisedV6.ip.v6, v6addr ? v6addr->ip.v6 : zerov6Addr); - RouterChanged = !mDNSSameIPv4Address(m->Router.ip.v4, router ? router->ip.v4 : zerov4Addr); - - if (v4addr && (v4Changed || RouterChanged)) - debugf("mDNS_SetPrimaryInterfaceInfo: address changed from %#a to %#a", &m->AdvertisedV4, v4addr); - - if (v4addr) m->AdvertisedV4 = *v4addr; else m->AdvertisedV4.ip.v4 = zerov4Addr; - if (v6addr) m->AdvertisedV6 = *v6addr; else m->AdvertisedV6.ip.v6 = zerov6Addr; - if (router) m->Router = *router; else m->Router .ip.v4 = zerov4Addr; - // setting router to zero indicates that nat mappings must be reestablished when router is reset - - if (v4Changed || RouterChanged || v6Changed) - { - HostnameInfo *i; - LogInfo("mDNS_SetPrimaryInterfaceInfo: %s%s%s%#a %#a %#a", - v4Changed ? "v4Changed " : "", - RouterChanged ? "RouterChanged " : "", - v6Changed ? "v6Changed " : "", v4addr, v6addr, router); - - for (i = m->Hostnames; i; i = i->next) - { - LogInfo("mDNS_SetPrimaryInterfaceInfo updating host name registrations for %##s", i->fqdn.c); - - if (i->arv4.resrec.RecordType > kDNSRecordTypeDeregistering && - !mDNSSameIPv4Address(i->arv4.resrec.rdata->u.ipv4, m->AdvertisedV4.ip.v4)) - { - LogInfo("mDNS_SetPrimaryInterfaceInfo deregistering %s", ARDisplayString(m, &i->arv4)); - mDNS_Deregister_internal(m, &i->arv4, mDNS_Dereg_normal); - } - - if (i->arv6.resrec.RecordType > kDNSRecordTypeDeregistering && - !mDNSSameIPv6Address(i->arv6.resrec.rdata->u.ipv6, m->AdvertisedV6.ip.v6)) - { - LogInfo("mDNS_SetPrimaryInterfaceInfo deregistering %s", ARDisplayString(m, &i->arv6)); - mDNS_Deregister_internal(m, &i->arv6, mDNS_Dereg_normal); - } - - // AdvertiseHostname will only register new address records. - // For records still in the process of deregistering it will ignore them, and let the mStatus_MemFree callback handle them. - AdvertiseHostname(m, i); - } - - if (v4Changed || RouterChanged) - { - // If we have a non-zero IPv4 address, we should try immediately to see if we have a NAT gateway - // If we have no IPv4 address, we don't want to be in quite such a hurry to report failures to our clients - // Sleeping server sometimes briefly disappears over Back to My Mac after it wakes up - m->ExternalAddress = zerov4Addr; - m->retryIntervalGetAddr = NATMAP_INIT_RETRY; - m->retryGetAddr = m->timenow + (v4addr ? 0 : mDNSPlatformOneSecond * 5); - m->NextScheduledNATOp = m->timenow; - m->LastNATMapResultCode = NATErr_None; -#ifdef _LEGACY_NAT_TRAVERSAL_ - LNT_ClearState(m); -#endif // _LEGACY_NAT_TRAVERSAL_ - LogInfo("mDNS_SetPrimaryInterfaceInfo:%s%s: retryGetAddr in %d %d", - v4Changed ? " v4Changed" : "", - RouterChanged ? " RouterChanged" : "", - m->retryGetAddr - m->timenow, m->timenow); - } - - if (m->ReverseMap.ThisQInterval != -1) mDNS_StopQuery_internal(m, &m->ReverseMap); - m->StaticHostname.c[0] = 0; - - m->NextSRVUpdate = NonZeroTime(m->timenow); - -#if APPLE_OSX_mDNSResponder - if (RouterChanged) uuid_generate(m->asl_uuid); - UpdateAutoTunnelDomainStatuses(m); -#endif - } - - mDNS_Unlock(m); - } - -// *************************************************************************** -#if COMPILER_LIKES_PRAGMA_MARK -#pragma mark - Incoming Message Processing -#endif - -mDNSlocal mStatus ParseTSIGError(mDNS *const m, const DNSMessage *const msg, const mDNSu8 *const end, const domainname *const displayname) - { - const mDNSu8 *ptr; - mStatus err = mStatus_NoError; - int i; - - ptr = LocateAdditionals(msg, end); - if (!ptr) goto finish; - - for (i = 0; i < msg->h.numAdditionals; i++) - { - ptr = GetLargeResourceRecord(m, msg, ptr, end, 0, kDNSRecordTypePacketAdd, &m->rec); - if (!ptr) goto finish; - if (m->rec.r.resrec.RecordType != kDNSRecordTypePacketNegative && m->rec.r.resrec.rrtype == kDNSType_TSIG) - { - mDNSu32 macsize; - mDNSu8 *rd = m->rec.r.resrec.rdata->u.data; - mDNSu8 *rdend = rd + m->rec.r.resrec.rdlength; - int alglen = DomainNameLengthLimit(&m->rec.r.resrec.rdata->u.name, rdend); - if (alglen > MAX_DOMAIN_NAME) goto finish; - rd += alglen; // algorithm name - if (rd + 6 > rdend) goto finish; - rd += 6; // 48-bit timestamp - if (rd + sizeof(mDNSOpaque16) > rdend) goto finish; - rd += sizeof(mDNSOpaque16); // fudge - if (rd + sizeof(mDNSOpaque16) > rdend) goto finish; - macsize = mDNSVal16(*(mDNSOpaque16 *)rd); - rd += sizeof(mDNSOpaque16); // MAC size - if (rd + macsize > rdend) goto finish; - rd += macsize; - if (rd + sizeof(mDNSOpaque16) > rdend) goto finish; - rd += sizeof(mDNSOpaque16); // orig id - if (rd + sizeof(mDNSOpaque16) > rdend) goto finish; - err = mDNSVal16(*(mDNSOpaque16 *)rd); // error code - - if (err == TSIG_ErrBadSig) { LogMsg("%##s: bad signature", displayname->c); err = mStatus_BadSig; } - else if (err == TSIG_ErrBadKey) { LogMsg("%##s: bad key", displayname->c); err = mStatus_BadKey; } - else if (err == TSIG_ErrBadTime) { LogMsg("%##s: bad time", displayname->c); err = mStatus_BadTime; } - else if (err) { LogMsg("%##s: unknown tsig error %d", displayname->c, err); err = mStatus_UnknownErr; } - goto finish; - } - m->rec.r.resrec.RecordType = 0; // Clear RecordType to show we're not still using it - } - - finish: - m->rec.r.resrec.RecordType = 0; // Clear RecordType to show we're not still using it - return err; - } - -mDNSlocal mStatus checkUpdateResult(mDNS *const m, const domainname *const displayname, const mDNSu8 rcode, const DNSMessage *const msg, const mDNSu8 *const end) - { - (void)msg; // currently unused, needed for TSIG errors - if (!rcode) return mStatus_NoError; - else if (rcode == kDNSFlag1_RC_YXDomain) - { - debugf("name in use: %##s", displayname->c); - return mStatus_NameConflict; - } - else if (rcode == kDNSFlag1_RC_Refused) - { - LogMsg("Update %##s refused", displayname->c); - return mStatus_Refused; - } - else if (rcode == kDNSFlag1_RC_NXRRSet) - { - LogMsg("Reregister refused (NXRRSET): %##s", displayname->c); - return mStatus_NoSuchRecord; - } - else if (rcode == kDNSFlag1_RC_NotAuth) - { - // TSIG errors should come with FormErr as per RFC 2845, but BIND 9 sends them with NotAuth so we look here too - mStatus tsigerr = ParseTSIGError(m, msg, end, displayname); - if (!tsigerr) - { - LogMsg("Permission denied (NOAUTH): %##s", displayname->c); - return mStatus_UnknownErr; - } - else return tsigerr; - } - else if (rcode == kDNSFlag1_RC_FormErr) - { - mStatus tsigerr = ParseTSIGError(m, msg, end, displayname); - if (!tsigerr) - { - LogMsg("Format Error: %##s", displayname->c); - return mStatus_UnknownErr; - } - else return tsigerr; - } - else - { - LogMsg("Update %##s failed with rcode %d", displayname->c, rcode); - return mStatus_UnknownErr; - } - } - -// We add three Additional Records for unicast resource record registrations -// which is a function of AuthInfo and AutoTunnel properties -mDNSlocal mDNSu32 RRAdditionalSize(mDNS *const m, DomainAuthInfo *AuthInfo) - { - mDNSu32 leaseSize, hinfoSize, tsigSize; - mDNSu32 rr_base_size = 10; // type (2) class (2) TTL (4) rdlength (2) - - // OPT RR : Emptyname(.) + base size + rdataOPT - leaseSize = 1 + rr_base_size + sizeof(rdataOPT); - - // HINFO: Resource Record Name + base size + RDATA - // HINFO is added only for autotunnels - hinfoSize = 0; - if (AuthInfo && AuthInfo->AutoTunnel) - hinfoSize = (m->hostlabel.c[0] + 1) + DomainNameLength(&AuthInfo->domain) + - rr_base_size + (2 + m->HIHardware.c[0] + m->HISoftware.c[0]); - - //TSIG: Resource Record Name + base size + RDATA - // RDATA: - // Algorithm name: hmac-md5.sig-alg.reg.int (8+7+3+3 + 5 bytes for length = 26 bytes) - // Time: 6 bytes - // Fudge: 2 bytes - // Mac Size: 2 bytes - // Mac: 16 bytes - // ID: 2 bytes - // Error: 2 bytes - // Len: 2 bytes - // Total: 58 bytes - tsigSize = 0; - if (AuthInfo) tsigSize = DomainNameLength(&AuthInfo->keyname) + rr_base_size + 58; - - return (leaseSize + hinfoSize + tsigSize); - } - -//Note: Make sure that RREstimatedSize is updated accordingly if anything that is done here -//would modify rdlength/rdestimate -mDNSlocal mDNSu8* BuildUpdateMessage(mDNS *const m, mDNSu8 *ptr, AuthRecord *rr, mDNSu8 *limit) - { - //If this record is deregistering, then just send the deletion record - if (rr->state == regState_DeregPending) - { - rr->expire = 0; // Indicate that we have no active registration any more - ptr = putDeletionRecordWithLimit(&m->omsg, ptr, &rr->resrec, limit); - if (!ptr) goto exit; - return ptr; - } - - // This is a common function to both sending an update in a group or individual - // records separately. Hence, we change the state here. - if (rr->state == regState_Registered) rr->state = regState_Refresh; - if (rr->state != regState_Refresh && rr->state != regState_UpdatePending) - rr->state = regState_Pending; - - // For Advisory records like e.g., _services._dns-sd, which is shared, don't send goodbyes as multiple - // host might be registering records and deregistering from one does not make sense - if (rr->resrec.RecordType != kDNSRecordTypeAdvisory) rr->RequireGoodbye = mDNStrue; - - if ((rr->resrec.rrtype == kDNSType_SRV) && (rr->AutoTarget == Target_AutoHostAndNATMAP) && - !mDNSIPPortIsZero(rr->NATinfo.ExternalPort)) - { - rr->resrec.rdata->u.srv.port = rr->NATinfo.ExternalPort; - } - - if (rr->state == regState_UpdatePending) - { - // delete old RData - SetNewRData(&rr->resrec, rr->OrigRData, rr->OrigRDLen); - if (!(ptr = putDeletionRecordWithLimit(&m->omsg, ptr, &rr->resrec, limit))) goto exit; // delete old rdata - - // add new RData - SetNewRData(&rr->resrec, rr->InFlightRData, rr->InFlightRDLen); - if (!(ptr = PutResourceRecordTTLWithLimit(&m->omsg, ptr, &m->omsg.h.mDNS_numUpdates, &rr->resrec, rr->resrec.rroriginalttl, limit))) goto exit; - } - else - { - if (rr->resrec.RecordType == kDNSRecordTypeKnownUnique || rr->resrec.RecordType == kDNSRecordTypeVerified) - { - // KnownUnique : Delete any previous value - // For Unicast registrations, we don't verify that it is unique, but set to verified and hence we want to - // delete any previous value - ptr = putDeleteRRSetWithLimit(&m->omsg, ptr, rr->resrec.name, rr->resrec.rrtype, limit); - if (!ptr) goto exit; - } - else if (rr->resrec.RecordType != kDNSRecordTypeShared) - { - // For now don't do this, until we have the logic for intelligent grouping of individual records into logical service record sets - //ptr = putPrereqNameNotInUse(rr->resrec.name, &m->omsg, ptr, end); - if (!ptr) goto exit; - } - - ptr = PutResourceRecordTTLWithLimit(&m->omsg, ptr, &m->omsg.h.mDNS_numUpdates, &rr->resrec, rr->resrec.rroriginalttl, limit); - if (!ptr) goto exit; - } - - return ptr; -exit: - LogMsg("BuildUpdateMessage: Error formatting message for %s", ARDisplayString(m, rr)); - return mDNSNULL; - } - -// Called with lock held -mDNSlocal void SendRecordRegistration(mDNS *const m, AuthRecord *rr) - { - mDNSu8 *ptr = m->omsg.data; - mStatus err = mStatus_UnknownErr; - mDNSu8 *limit; - DomainAuthInfo *AuthInfo; - - // For the ability to register large TXT records, we limit the single record registrations - // to AbsoluteMaxDNSMessageData - limit = ptr + AbsoluteMaxDNSMessageData; - - AuthInfo = GetAuthInfoForName_internal(m, rr->resrec.name); - limit -= RRAdditionalSize(m, AuthInfo); - - if (m->mDNS_busy != m->mDNS_reentrancy+1) - LogMsg("SendRecordRegistration: Lock not held! mDNS_busy (%ld) mDNS_reentrancy (%ld)", m->mDNS_busy, m->mDNS_reentrancy); - - if (!rr->nta || mDNSIPv4AddressIsZero(rr->nta->Addr.ip.v4)) - { - // We never call this function when there is no zone information . Log a message if it ever happens. - LogMsg("SendRecordRegistration: No Zone information, should not happen %s", ARDisplayString(m, rr)); - return; - } - - rr->updateid = mDNS_NewMessageID(m); - InitializeDNSMessage(&m->omsg.h, rr->updateid, UpdateReqFlags); - - // set zone - ptr = putZone(&m->omsg, ptr, limit, rr->zone, mDNSOpaque16fromIntVal(rr->resrec.rrclass)); - if (!ptr) goto exit; - - if (!(ptr = BuildUpdateMessage(m, ptr, rr, limit))) goto exit; - - if (rr->uselease) - { - ptr = putUpdateLeaseWithLimit(&m->omsg, ptr, DEFAULT_UPDATE_LEASE, limit); - if (!ptr) goto exit; - } - if (rr->Private) - { - LogInfo("SendRecordRegistration TCP %p %s", rr->tcp, ARDisplayString(m, rr)); - if (rr->tcp) LogInfo("SendRecordRegistration: Disposing existing TCP connection for %s", ARDisplayString(m, rr)); - if (rr->tcp) { DisposeTCPConn(rr->tcp); rr->tcp = mDNSNULL; } - if (!rr->nta) { LogMsg("SendRecordRegistration:Private:ERROR!! nta is NULL for %s", ARDisplayString(m, rr)); return; } - rr->tcp = MakeTCPConn(m, &m->omsg, ptr, kTCPSocketFlags_UseTLS, &rr->nta->Addr, rr->nta->Port, &rr->nta->Host, mDNSNULL, rr); - } - else - { - LogInfo("SendRecordRegistration UDP %s", ARDisplayString(m, rr)); - if (!rr->nta) { LogMsg("SendRecordRegistration:ERROR!! nta is NULL for %s", ARDisplayString(m, rr)); return; } - err = mDNSSendDNSMessage(m, &m->omsg, ptr, mDNSInterface_Any, mDNSNULL, &rr->nta->Addr, rr->nta->Port, mDNSNULL, GetAuthInfoForName_internal(m, rr->resrec.name)); - if (err) debugf("ERROR: SendRecordRegistration - mDNSSendDNSMessage - %d", err); - } - - SetRecordRetry(m, rr, 0); - return; -exit: - LogMsg("SendRecordRegistration: Error formatting message for %s, disabling further updates", ARDisplayString(m, rr)); - // Disable this record from future updates - rr->state = regState_NoTarget; - } - -// Is the given record "rr" eligible for merging ? -mDNSlocal mDNSBool IsRecordMergeable(mDNS *const m, AuthRecord *rr, mDNSs32 time) - { - DomainAuthInfo *info; - (void) m; //unused - // A record is eligible for merge, if the following properties are met. - // - // 1. uDNS Resource Record - // 2. It is time to send them now - // 3. It is in proper state - // 4. Update zone has been resolved - // 5. if DomainAuthInfo exists for the zone, it should not be soon deleted - // 6. Zone information is present - // 7. Update server is not zero - // 8. It has a non-null zone - // 9. It uses a lease option - // 10. DontMerge is not set - // - // Following code is implemented as separate "if" statements instead of one "if" statement - // is for better debugging purposes e.g., we know exactly what failed if debugging turned on. - - if (!AuthRecord_uDNS(rr)) return mDNSfalse; - - if (rr->LastAPTime + rr->ThisAPInterval - time > 0) - { debugf("IsRecordMergeable: Time %d not reached for %s", rr->LastAPTime + rr->ThisAPInterval - m->timenow, ARDisplayString(m, rr)); return mDNSfalse; } - - if (!rr->zone) return mDNSfalse; - - info = GetAuthInfoForName_internal(m, rr->zone); - - if (info && info->deltime && m->timenow - info->deltime >= 0) {debugf("IsRecordMergeable: Domain %##s will be deleted soon", info->domain.c); return mDNSfalse;} - - if (rr->state != regState_DeregPending && rr->state != regState_Pending && rr->state != regState_Registered && rr->state != regState_Refresh && rr->state != regState_UpdatePending) - { debugf("IsRecordMergeable: state %d not right %s", rr->state, ARDisplayString(m, rr)); return mDNSfalse; } - - if (!rr->nta || mDNSIPv4AddressIsZero(rr->nta->Addr.ip.v4)) return mDNSfalse; - - if (!rr->uselease) return mDNSfalse; - - if (rr->mState == mergeState_DontMerge) {debugf("IsRecordMergeable Dontmerge true %s", ARDisplayString(m, rr));return mDNSfalse;} - debugf("IsRecordMergeable: Returning true for %s", ARDisplayString(m, rr)); - return mDNStrue; - } - -// Is the resource record "rr" eligible to merge to with "currentRR" ? -mDNSlocal mDNSBool AreRecordsMergeable(mDNS *const m, AuthRecord *currentRR, AuthRecord *rr, mDNSs32 time) - { - // A record is eligible to merge with another record as long it is eligible for merge in itself - // and it has the same zone information as the other record - if (!IsRecordMergeable(m, rr, time)) return mDNSfalse; - - if (!SameDomainName(currentRR->zone, rr->zone)) - { debugf("AreRecordMergeable zone mismatch current rr Zone %##s, rr zone %##s", currentRR->zone->c, rr->zone->c); return mDNSfalse; } - - if (!mDNSSameIPv4Address(currentRR->nta->Addr.ip.v4, rr->nta->Addr.ip.v4)) return mDNSfalse; - - if (!mDNSSameIPPort(currentRR->nta->Port, rr->nta->Port)) return mDNSfalse; - - debugf("AreRecordsMergeable: Returning true for %s", ARDisplayString(m, rr)); - return mDNStrue; - } - -// If we can't build the message successfully because of problems in pre-computing -// the space, we disable merging for all the current records -mDNSlocal void RRMergeFailure(mDNS *const m) - { - AuthRecord *rr; - for (rr = m->ResourceRecords; rr; rr = rr->next) - { - rr->mState = mergeState_DontMerge; - rr->SendRNow = mDNSNULL; - // Restarting the registration is much simpler than saving and restoring - // the exact time - ActivateUnicastRegistration(m, rr); - } - } - -mDNSlocal void SendGroupRRMessage(mDNS *const m, AuthRecord *anchorRR, mDNSu8 *ptr, DomainAuthInfo *info) - { - mDNSu8 *limit; - if (!anchorRR) {debugf("SendGroupRRMessage: Could not merge records"); return;} - - if (info && info->AutoTunnel) limit = m->omsg.data + AbsoluteMaxDNSMessageData; - else limit = m->omsg.data + NormalMaxDNSMessageData; - - // This has to go in the additional section and hence need to be done last - ptr = putUpdateLeaseWithLimit(&m->omsg, ptr, DEFAULT_UPDATE_LEASE, limit); - if (!ptr) - { - LogMsg("SendGroupRRMessage: ERROR: Could not put lease option, failing the group registration"); - // if we can't put the lease, we need to undo the merge - RRMergeFailure(m); - return; - } - if (anchorRR->Private) - { - if (anchorRR->tcp) debugf("SendGroupRRMessage: Disposing existing TCP connection for %s", ARDisplayString(m, anchorRR)); - if (anchorRR->tcp) { DisposeTCPConn(anchorRR->tcp); anchorRR->tcp = mDNSNULL; } - if (!anchorRR->nta) { LogMsg("SendGroupRRMessage:ERROR!! nta is NULL for %s", ARDisplayString(m, anchorRR)); return; } - anchorRR->tcp = MakeTCPConn(m, &m->omsg, ptr, kTCPSocketFlags_UseTLS, &anchorRR->nta->Addr, anchorRR->nta->Port, &anchorRR->nta->Host, mDNSNULL, anchorRR); - if (!anchorRR->tcp) LogInfo("SendGroupRRMessage: Cannot establish TCP connection for %s", ARDisplayString(m, anchorRR)); - else LogInfo("SendGroupRRMessage: Sent a group update ID: %d start %p, end %p, limit %p", mDNSVal16(m->omsg.h.id), m->omsg.data, ptr, limit); - } - else - { - mStatus err = mDNSSendDNSMessage(m, &m->omsg, ptr, mDNSInterface_Any, mDNSNULL, &anchorRR->nta->Addr, anchorRR->nta->Port, mDNSNULL, info); - if (err) LogInfo("SendGroupRRMessage: Cannot send UDP message for %s", ARDisplayString(m, anchorRR)); - else LogInfo("SendGroupRRMessage: Sent a group UDP update ID: %d start %p, end %p, limit %p", mDNSVal16(m->omsg.h.id), m->omsg.data, ptr, limit); - } - return; - } - -// As we always include the zone information and the resource records contain zone name -// at the end, it will get compressed. Hence, we subtract zoneSize and add two bytes for -// the compression pointer -mDNSlocal mDNSu32 RREstimatedSize(AuthRecord *rr, int zoneSize) - { - int rdlength; - - // Note: Estimation of the record size has to mirror the logic in BuildUpdateMessage, otherwise estimation - // would be wrong. Currently BuildUpdateMessage calls SetNewRData in UpdatePending case. Hence, we need - // to account for that here. Otherwise, we might under estimate the size. - if (rr->state == regState_UpdatePending) - // old RData that will be deleted - // new RData that will be added - rdlength = rr->OrigRDLen + rr->InFlightRDLen; - else - rdlength = rr->resrec.rdestimate; - - if (rr->state == regState_DeregPending) - { - debugf("RREstimatedSize: ResourceRecord %##s (%s), DomainNameLength %d, zoneSize %d, rdestimate %d", - rr->resrec.name->c, DNSTypeName(rr->resrec.rrtype), DomainNameLength(rr->resrec.name), zoneSize, rdlength); - return DomainNameLength(rr->resrec.name) - zoneSize + 2 + 10 + rdlength; - } - - // For SRV, TXT, AAAA etc. that are Unique/Verified, we also send a Deletion Record - if (rr->resrec.RecordType == kDNSRecordTypeKnownUnique || rr->resrec.RecordType == kDNSRecordTypeVerified) - { - // Deletion Record: Resource Record Name + Base size (10) + 0 - // Record: Resource Record Name (Compressed = 2) + Base size (10) + rdestimate - - debugf("RREstimatedSize: ResourceRecord %##s (%s), DomainNameLength %d, zoneSize %d, rdestimate %d", - rr->resrec.name->c, DNSTypeName(rr->resrec.rrtype), DomainNameLength(rr->resrec.name), zoneSize, rdlength); - return DomainNameLength(rr->resrec.name) - zoneSize + 2 + 10 + 2 + 10 + rdlength; - } - else - { - return DomainNameLength(rr->resrec.name) - zoneSize + 2 + 10 + rdlength; - } - } - -mDNSlocal AuthRecord *MarkRRForSending(mDNS *const m) - { - AuthRecord *rr; - AuthRecord *firstRR = mDNSNULL; - - // Look for records that needs to be sent in the next two seconds (MERGE_DELAY_TIME is set to 1 second). - // The logic is as follows. - // - // 1. Record 1 finishes getting zone data and its registration gets delayed by 1 second - // 2. Record 2 comes 0.1 second later, finishes getting its zone data and its registration is also delayed by - // 1 second which is now scheduled at 1.1 second - // - // By looking for 1 second into the future (m->timenow + MERGE_DELAY_TIME below does that) we have merged both - // of the above records. Note that we can't look for records too much into the future as this will affect the - // retry logic. The first retry is scheduled at 3 seconds. Hence, we should always look smaller than that. - // Anything more than one second will affect the first retry to happen sooner. - // - // Note: As a side effect of looking one second into the future to facilitate merging, the retries happen - // one second sooner. - for (rr = m->ResourceRecords; rr; rr = rr->next) - { - if (!firstRR) - { - if (!IsRecordMergeable(m, rr, m->timenow + MERGE_DELAY_TIME)) continue; - firstRR = rr; - } - else if (!AreRecordsMergeable(m, firstRR, rr, m->timenow + MERGE_DELAY_TIME)) continue; - - if (rr->SendRNow) LogMsg("MarkRRForSending: Resourcerecord %s already marked for sending", ARDisplayString(m, rr)); - rr->SendRNow = mDNSInterfaceMark; - } - - // We parsed through all records and found something to send. The services/records might - // get registered at different times but we want the refreshes to be all merged and sent - // as one update. Hence, we accelerate some of the records so that they will sync up in - // the future. Look at the records excluding the ones that we have already sent in the - // previous pass. If it half way through its scheduled refresh/retransmit, merge them - // into this packet. - // - // Note that we only look at Registered/Refresh state to keep it simple. As we don't know - // whether the current update will fit into one or more packets, merging a resource record - // (which is in a different state) that has been scheduled for retransmit would trigger - // sending more packets. - if (firstRR) - { - int acc = 0; - for (rr = m->ResourceRecords; rr; rr = rr->next) - { - if ((rr->state != regState_Registered && rr->state != regState_Refresh) || - (rr->SendRNow == mDNSInterfaceMark) || - (!AreRecordsMergeable(m, firstRR, rr, m->timenow + rr->ThisAPInterval/2))) - continue; - rr->SendRNow = mDNSInterfaceMark; - acc++; - } - if (acc) LogInfo("MarkRRForSending: Accelereated %d records", acc); - } - return firstRR; - } - -mDNSlocal mDNSBool SendGroupUpdates(mDNS *const m) - { - mDNSOpaque16 msgid; - mDNSs32 spaceleft = 0; - mDNSs32 zoneSize, rrSize; - mDNSu8 *oldnext; // for debugging - mDNSu8 *next = m->omsg.data; - AuthRecord *rr; - AuthRecord *anchorRR = mDNSNULL; - int nrecords = 0; - AuthRecord *startRR = m->ResourceRecords; - mDNSu8 *limit = mDNSNULL; - DomainAuthInfo *AuthInfo = mDNSNULL; - mDNSBool sentallRecords = mDNStrue; - - - // We try to fit as many ResourceRecords as possible in AbsoluteNormal/MaxDNSMessageData. Before we start - // putting in resource records, we need to reserve space for a few things. Every group/packet should - // have the following. - // - // 1) Needs space for the Zone information (which needs to be at the beginning) - // 2) Additional section MUST have space for lease option, HINFO and TSIG option (which needs to - // to be at the end) - // - // In future we need to reserve space for the pre-requisites which also goes at the beginning. - // To accomodate pre-requisites in the future, first we walk the whole list marking records - // that can be sent in this packet and computing the space needed for these records. - // For TXT and SRV records, we delete the previous record if any by sending the same - // resource record with ANY RDATA and zero rdlen. Hence, we need to have space for both of them. - - while (startRR) - { - AuthInfo = mDNSNULL; - anchorRR = mDNSNULL; - nrecords = 0; - zoneSize = 0; - for (rr = startRR; rr; rr = rr->next) - { - if (rr->SendRNow != mDNSInterfaceMark) continue; - - rr->SendRNow = mDNSNULL; - - if (!anchorRR) - { - AuthInfo = GetAuthInfoForName_internal(m, rr->zone); - - // Though we allow single record registrations for UDP to be AbsoluteMaxDNSMessageData (See - // SendRecordRegistration) to handle large TXT records, to avoid fragmentation we limit UDP - // message to NormalMaxDNSMessageData - if (AuthInfo && AuthInfo->AutoTunnel) spaceleft = AbsoluteMaxDNSMessageData; - else spaceleft = NormalMaxDNSMessageData; - - next = m->omsg.data; - spaceleft -= RRAdditionalSize(m, AuthInfo); - if (spaceleft <= 0) - { - LogMsg("SendGroupUpdates: ERROR!!: spaceleft is zero at the beginning"); - RRMergeFailure(m); - return mDNSfalse; - } - limit = next + spaceleft; - - // Build the initial part of message before putting in the other records - msgid = mDNS_NewMessageID(m); - InitializeDNSMessage(&m->omsg.h, msgid, UpdateReqFlags); - - // We need zone information at the beginning of the packet. Length: ZNAME, ZTYPE(2), ZCLASS(2) - // zone has to be non-NULL for a record to be mergeable, hence it is safe to set/ examine zone - //without checking for NULL. - zoneSize = DomainNameLength(rr->zone) + 4; - spaceleft -= zoneSize; - if (spaceleft <= 0) - { - LogMsg("SendGroupUpdates: ERROR no space for zone information, disabling merge"); - RRMergeFailure(m); - return mDNSfalse; - } - next = putZone(&m->omsg, next, limit, rr->zone, mDNSOpaque16fromIntVal(rr->resrec.rrclass)); - if (!next) - { - LogMsg("SendGroupUpdates: ERROR! Cannot put zone, disabling merge"); - RRMergeFailure(m); - return mDNSfalse; - } - anchorRR = rr; - } - - rrSize = RREstimatedSize(rr, zoneSize - 4); - - if ((spaceleft - rrSize) < 0) - { - // If we can't fit even a single message, skip it, it will be sent separately - // in CheckRecordUpdates - if (!nrecords) - { - LogInfo("SendGroupUpdates: Skipping message %s, spaceleft %d, rrSize %d", ARDisplayString(m, rr), spaceleft, rrSize); - // Mark this as not sent so that the caller knows about it - rr->SendRNow = mDNSInterfaceMark; - // We need to remove the merge delay so that we can send it immediately - rr->ThisAPInterval = INIT_RECORD_REG_INTERVAL; - rr->LastAPTime = m->timenow - INIT_RECORD_REG_INTERVAL; - rr = rr->next; - anchorRR = mDNSNULL; - sentallRecords = mDNSfalse; - } - else - { - LogInfo("SendGroupUpdates:1: Parsed %d records and sending using %s, spaceleft %d, rrSize %d", nrecords, ARDisplayString(m, anchorRR), spaceleft, rrSize); - SendGroupRRMessage(m, anchorRR, next, AuthInfo); - } - break; // breaks out of for loop - } - spaceleft -= rrSize; - oldnext = next; - LogInfo("SendGroupUpdates: Building a message with resource record %s, next %p, state %d, ttl %d", ARDisplayString(m, rr), next, rr->state, rr->resrec.rroriginalttl); - if (!(next = BuildUpdateMessage(m, next, rr, limit))) - { - // We calculated the space and if we can't fit in, we had some bug in the calculation, - // disable merge completely. - LogMsg("SendGroupUpdates: ptr NULL while building message with %s", ARDisplayString(m, rr)); - RRMergeFailure(m); - return mDNSfalse; - } - // If our estimate was higher, adjust to the actual size - if ((next - oldnext) > rrSize) - LogMsg("SendGroupUpdates: ERROR!! Record size estimation is wrong for %s, Estimate %d, Actual %d, state %d", ARDisplayString(m, rr), rrSize, next - oldnext, rr->state); - else { spaceleft += rrSize; spaceleft -= (next - oldnext); } - - nrecords++; - // We could have sent an update earlier with this "rr" as anchorRR for which we never got a response. - // To preserve ordering, we blow away the previous connection before sending this. - if (rr->tcp) { DisposeTCPConn(rr->tcp); rr->tcp = mDNSNULL;} - rr->updateid = msgid; - - // By setting the retry time interval here, we will not be looking at these records - // again when we return to CheckGroupRecordUpdates. - SetRecordRetry(m, rr, 0); - } - // Either we have parsed all the records or stopped at "rr" above due to lack of space - startRR = rr; - } - - if (anchorRR) - { - LogInfo("SendGroupUpdates: Parsed %d records and sending using %s", nrecords, ARDisplayString(m, anchorRR)); - SendGroupRRMessage(m, anchorRR, next, AuthInfo); - } - return sentallRecords; - } - -// Merge the record registrations and send them as a group only if they -// have same DomainAuthInfo and hence the same key to put the TSIG -mDNSlocal void CheckGroupRecordUpdates(mDNS *const m) - { - AuthRecord *rr, *nextRR; - // Keep sending as long as there is at least one record to be sent - while (MarkRRForSending(m)) - { - if (!SendGroupUpdates(m)) - { - // if everything that was marked was not sent, send them out individually - for (rr = m->ResourceRecords; rr; rr = nextRR) - { - // SendRecordRegistrtion might delete the rr from list, hence - // dereference nextRR before calling the function - nextRR = rr->next; - if (rr->SendRNow == mDNSInterfaceMark) - { - // Any records marked for sending should be eligible to be sent out - // immediately. Just being cautious - if (rr->LastAPTime + rr->ThisAPInterval - m->timenow > 0) - { LogMsg("CheckGroupRecordUpdates: ERROR!! Resourcerecord %s not ready", ARDisplayString(m, rr)); continue; } - rr->SendRNow = mDNSNULL; - SendRecordRegistration(m, rr); - } - } - } - } - - debugf("CheckGroupRecordUpdates: No work, returning"); - return; - } - -mDNSlocal void hndlSRVChanged(mDNS *const m, AuthRecord *rr) - { - // Reevaluate the target always as NAT/Target could have changed while - // we were registering/deeregistering - domainname *dt; - const domainname *target = GetServiceTarget(m, rr); - if (!target || target->c[0] == 0) - { - // we don't have a target, if we just derregistered, then we don't have to do anything - if (rr->state == regState_DeregPending) - { - LogInfo("hndlSRVChanged: SRVChanged, No Target, SRV Deregistered for %##s, state %d", rr->resrec.name->c, - rr->state); - rr->SRVChanged = mDNSfalse; - dt = GetRRDomainNameTarget(&rr->resrec); - if (dt) dt->c[0] = 0; - rr->state = regState_NoTarget; // Wait for the next target change - rr->resrec.rdlength = rr->resrec.rdestimate = 0; - return; - } - - // we don't have a target, if we just registered, we need to deregister - if (rr->state == regState_Pending) - { - LogInfo("hndlSRVChanged: SRVChanged, No Target, Deregistering again %##s, state %d", rr->resrec.name->c, rr->state); - rr->ThisAPInterval = INIT_RECORD_REG_INTERVAL; - rr->LastAPTime = m->timenow - INIT_RECORD_REG_INTERVAL; - rr->state = regState_DeregPending; - return; - } - LogInfo("hndlSRVChanged: Not in DeregPending or RegPending state %##s, state %d", rr->resrec.name->c, rr->state); - } - else - { - // If we were in registered state and SRV changed to NULL, we deregister and come back here - // if we have a target, we need to register again. - // - // if we just registered check to see if it is same. If it is different just re-register the - // SRV and its assoicated records - // - // UpdateOneSRVRecord takes care of re-registering all service records - if ((rr->state == regState_DeregPending) || - (rr->state == regState_Pending && !SameDomainName(target, &rr->resrec.rdata->u.srv.target))) - { - dt = GetRRDomainNameTarget(&rr->resrec); - if (dt) dt->c[0] = 0; - rr->state = regState_NoTarget; // NoTarget will allow us to pick up new target OR nat traversal state - rr->resrec.rdlength = rr->resrec.rdestimate = 0; - LogInfo("hndlSRVChanged: SRVChanged, Valid Target %##s, Registering all records for %##s, state %d", - target->c, rr->resrec.name->c, rr->state); - rr->SRVChanged = mDNSfalse; - UpdateOneSRVRecord(m, rr); - return; - } - // Target did not change while this record was registering. Hence, we go to - // Registered state - the state we started from. - if (rr->state == regState_Pending) rr->state = regState_Registered; - } - - rr->SRVChanged = mDNSfalse; - } - -// Called with lock held -mDNSlocal void hndlRecordUpdateReply(mDNS *m, AuthRecord *rr, mStatus err, mDNSu32 random) - { - mDNSBool InvokeCallback = mDNStrue; - mDNSIPPort UpdatePort = zeroIPPort; - - if (m->mDNS_busy != m->mDNS_reentrancy+1) - LogMsg("hndlRecordUpdateReply: Lock not held! mDNS_busy (%ld) mDNS_reentrancy (%ld)", m->mDNS_busy, m->mDNS_reentrancy); - - LogInfo("hndlRecordUpdateReply: err %d ID %d state %d %s(%p)", err, mDNSVal16(rr->updateid), rr->state, ARDisplayString(m, rr), rr); - - rr->updateError = err; -#if APPLE_OSX_mDNSResponder - if (err == mStatus_BadSig) UpdateAutoTunnelDomainStatuses(m); -#endif - - SetRecordRetry(m, rr, random); - - rr->updateid = zeroID; // Make sure that this is not considered as part of a group anymore - // Later when need to send an update, we will get the zone data again. Thus we avoid - // using stale information. - // - // Note: By clearing out the zone info here, it also helps better merging of records - // in some cases. For example, when we get out regState_NoTarget state e.g., move out - // of Double NAT, we want all the records to be in one update. Some BTMM records like - // _autotunnel6 and host records are registered/deregistered when NAT state changes. - // As they are re-registered the zone information is cleared out. To merge with other - // records that might be possibly going out, clearing out the information here helps - // as all of them try to get the zone data. - if (rr->nta) - { - // We always expect the question to be stopped when we get a valid response from the server. - // If the zone info tries to change during this time, updateid would be different and hence - // this response should not have been accepted. - if (rr->nta->question.ThisQInterval != -1) - LogMsg("hndlRecordUpdateReply: ResourceRecord %s, zone info question %##s (%s) interval %d not -1", - ARDisplayString(m, rr), rr->nta->question.qname.c, DNSTypeName(rr->nta->question.qtype), rr->nta->question.ThisQInterval); - UpdatePort = rr->nta->Port; - CancelGetZoneData(m, rr->nta); - rr->nta = mDNSNULL; - } - - // If we are deregistering the record, then complete the deregistration. Ignore any NAT/SRV change - // that could have happened during that time. - if (rr->resrec.RecordType == kDNSRecordTypeDeregistering && rr->state == regState_DeregPending) - { - debugf("hndlRecordUpdateReply: Received reply for deregister record %##s type %d", rr->resrec.name->c, rr->resrec.rrtype); - if (err) LogMsg("ERROR: Deregistration of record %##s type %d failed with error %d", - rr->resrec.name->c, rr->resrec.rrtype, err); - rr->state = regState_Unregistered; - CompleteDeregistration(m, rr); - return; - } - - // We are returning early without updating the state. When we come back from sleep we will re-register after - // re-initializing all the state as though it is a first registration. If the record can't be registered e.g., - // no target, it will be deregistered. Hence, the updating to the right state should not matter when going - // to sleep. - if (m->SleepState) - { - // Need to set it to NoTarget state so that RecordReadyForSleep knows that - // we are done - if (rr->resrec.rrtype == kDNSType_SRV && rr->state == regState_DeregPending) - rr->state = regState_NoTarget; - return; - } - - if (rr->state == regState_UpdatePending) - { - if (err) LogMsg("Update record failed for %##s (err %d)", rr->resrec.name->c, err); - rr->state = regState_Registered; - // deallocate old RData - if (rr->UpdateCallback) rr->UpdateCallback(m, rr, rr->OrigRData, rr->OrigRDLen); - SetNewRData(&rr->resrec, rr->InFlightRData, rr->InFlightRDLen); - rr->OrigRData = mDNSNULL; - rr->InFlightRData = mDNSNULL; - } - - if (rr->SRVChanged) - { - if (rr->resrec.rrtype == kDNSType_SRV) - hndlSRVChanged(m, rr); - else - { - LogInfo("hndlRecordUpdateReply: Deregistered %##s (%s), state %d", rr->resrec.name->c, DNSTypeName(rr->resrec.rrtype), rr->state); - rr->SRVChanged = mDNSfalse; - if (rr->state != regState_DeregPending) LogMsg("hndlRecordUpdateReply: ResourceRecord %s not in DeregPending state %d", ARDisplayString(m, rr), rr->state); - rr->state = regState_NoTarget; // Wait for the next target change - } - return; - } - - if (rr->state == regState_Pending || rr->state == regState_Refresh) - { - if (!err) - { - if (rr->state == regState_Refresh) InvokeCallback = mDNSfalse; - rr->state = regState_Registered; - } - else - { - // Retry without lease only for non-Private domains - LogMsg("hndlRecordUpdateReply: Registration of record %##s type %d failed with error %d", rr->resrec.name->c, rr->resrec.rrtype, err); - if (!rr->Private && rr->uselease && err == mStatus_UnknownErr && mDNSSameIPPort(UpdatePort, UnicastDNSPort)) - { - LogMsg("hndlRecordUpdateReply: Will retry update of record %##s without lease option", rr->resrec.name->c); - rr->uselease = mDNSfalse; - rr->ThisAPInterval = INIT_RECORD_REG_INTERVAL; - rr->LastAPTime = m->timenow - INIT_RECORD_REG_INTERVAL; - return; - } - // Communicate the error to the application in the callback below - } - } - - if (rr->QueuedRData && rr->state == regState_Registered) - { - rr->state = regState_UpdatePending; - rr->InFlightRData = rr->QueuedRData; - rr->InFlightRDLen = rr->QueuedRDLen; - rr->OrigRData = rr->resrec.rdata; - rr->OrigRDLen = rr->resrec.rdlength; - rr->QueuedRData = mDNSNULL; - rr->ThisAPInterval = INIT_RECORD_REG_INTERVAL; - rr->LastAPTime = m->timenow - INIT_RECORD_REG_INTERVAL; - return; - } - - // Don't invoke the callback on error as this may not be useful to the client. - // The client may potentially delete the resource record on error which we normally - // delete during deregistration - if (!err && InvokeCallback && rr->RecordCallback) - { - LogInfo("hndlRecordUpdateReply: Calling record callback on %##s", rr->resrec.name->c); - mDNS_DropLockBeforeCallback(); - rr->RecordCallback(m, rr, err); - mDNS_ReclaimLockAfterCallback(); - } - // CAUTION: MUST NOT do anything more with rr after calling rr->Callback(), because the client's callback function - // is allowed to do anything, including starting/stopping queries, registering/deregistering records, etc. - } - -mDNSexport void uDNS_ReceiveNATPMPPacket(mDNS *m, const mDNSInterfaceID InterfaceID, mDNSu8 *pkt, mDNSu16 len) - { - NATTraversalInfo *ptr; - NATAddrReply *AddrReply = (NATAddrReply *)pkt; - NATPortMapReply *PortMapReply = (NATPortMapReply *)pkt; - mDNSu32 nat_elapsed, our_elapsed; - - // Minimum packet is vers (1) opcode (1) err (2) upseconds (4) = 8 bytes - if (!AddrReply->err && len < 8) { LogMsg("NAT Traversal message too short (%d bytes)", len); return; } - if (AddrReply->vers != NATMAP_VERS) { LogMsg("Received NAT Traversal response with version %d (expected %d)", pkt[0], NATMAP_VERS); return; } - - // Read multi-byte numeric values (fields are identical in a NATPortMapReply) - AddrReply->err = (mDNSu16) ( (mDNSu16)pkt[2] << 8 | pkt[3]); - AddrReply->upseconds = (mDNSs32) ((mDNSs32)pkt[4] << 24 | (mDNSs32)pkt[5] << 16 | (mDNSs32)pkt[6] << 8 | pkt[7]); - - nat_elapsed = AddrReply->upseconds - m->LastNATupseconds; - our_elapsed = (m->timenow - m->LastNATReplyLocalTime) / mDNSPlatformOneSecond; - debugf("uDNS_ReceiveNATPMPPacket %X upseconds %u nat_elapsed %d our_elapsed %d", AddrReply->opcode, AddrReply->upseconds, nat_elapsed, our_elapsed); - - // We compute a conservative estimate of how much the NAT gateways's clock should have advanced - // 1. We subtract 12.5% from our own measured elapsed time, to allow for NAT gateways that have an inacurate clock that runs slowly - // 2. We add a two-second safety margin to allow for rounding errors: e.g. - // -- if NAT gateway sends a packet at t=2.000 seconds, then one at t=7.999, that's approximately 6 real seconds, - // but based on the values in the packet (2,7) the apparent difference according to the packet is only 5 seconds - // -- if we're slow handling packets and/or we have coarse clock granularity, - // we could receive the t=2 packet at our t=1.999 seconds, which we round down to 1 - // and the t=7.999 packet at our t=8.000 seconds, which we record as 8, - // giving an apparent local time difference of 7 seconds - // The two-second safety margin coves this possible calculation discrepancy - if (AddrReply->upseconds < m->LastNATupseconds || nat_elapsed + 2 < our_elapsed - our_elapsed/8) - { LogMsg("NAT gateway %#a rebooted", &m->Router); RecreateNATMappings(m); } - - m->LastNATupseconds = AddrReply->upseconds; - m->LastNATReplyLocalTime = m->timenow; -#ifdef _LEGACY_NAT_TRAVERSAL_ - LNT_ClearState(m); -#endif // _LEGACY_NAT_TRAVERSAL_ - - if (AddrReply->opcode == NATOp_AddrResponse) - { -#if APPLE_OSX_mDNSResponder - static char msgbuf[16]; - mDNS_snprintf(msgbuf, sizeof(msgbuf), "%d", AddrReply->err); - mDNSASLLog((uuid_t *)&m->asl_uuid, "natt.natpmp.AddressRequest", AddrReply->err ? "failure" : "success", msgbuf, ""); -#endif - if (!AddrReply->err && len < sizeof(NATAddrReply)) { LogMsg("NAT Traversal AddrResponse message too short (%d bytes)", len); return; } - natTraversalHandleAddressReply(m, AddrReply->err, AddrReply->ExtAddr); - } - else if (AddrReply->opcode == NATOp_MapUDPResponse || AddrReply->opcode == NATOp_MapTCPResponse) - { - mDNSu8 Protocol = AddrReply->opcode & 0x7F; -#if APPLE_OSX_mDNSResponder - static char msgbuf[16]; - mDNS_snprintf(msgbuf, sizeof(msgbuf), "%s - %d", AddrReply->opcode == NATOp_MapUDPResponse ? "UDP" : "TCP", PortMapReply->err); - mDNSASLLog((uuid_t *)&m->asl_uuid, "natt.natpmp.PortMapRequest", PortMapReply->err ? "failure" : "success", msgbuf, ""); -#endif - if (!PortMapReply->err) - { - if (len < sizeof(NATPortMapReply)) { LogMsg("NAT Traversal PortMapReply message too short (%d bytes)", len); return; } - PortMapReply->NATRep_lease = (mDNSu32) ((mDNSu32)pkt[12] << 24 | (mDNSu32)pkt[13] << 16 | (mDNSu32)pkt[14] << 8 | pkt[15]); - } - - // Since some NAT-PMP server implementations don't return the requested internal port in - // the reply, we can't associate this reply with a particular NATTraversalInfo structure. - // We globally keep track of the most recent error code for mappings. - m->LastNATMapResultCode = PortMapReply->err; - - for (ptr = m->NATTraversals; ptr; ptr=ptr->next) - if (ptr->Protocol == Protocol && mDNSSameIPPort(ptr->IntPort, PortMapReply->intport)) - natTraversalHandlePortMapReply(m, ptr, InterfaceID, PortMapReply->err, PortMapReply->extport, PortMapReply->NATRep_lease); - } - else { LogMsg("Received NAT Traversal response with version unknown opcode 0x%X", AddrReply->opcode); return; } - - // Don't need an SSDP socket if we get a NAT-PMP packet - if (m->SSDPSocket) { debugf("uDNS_ReceiveNATPMPPacket destroying SSDPSocket %p", &m->SSDPSocket); mDNSPlatformUDPClose(m->SSDPSocket); m->SSDPSocket = mDNSNULL; } - } - -// Shorten DNS-SD queries to avoid NAT bugs -// Add check to avoid crashing NAT gateways that have buggy DNS relay code -// -// We know of bugs in home NAT gateways that cause them to crash if they receive certain DNS queries. -// The DNS queries that make them crash are perfectly legal DNS queries, but even if they weren't, -// the gateway shouldn't crash -- in today's world of viruses and network attacks, software has to -// be written assuming that a malicious attacker could send them any packet, properly-formed or not. -// Still, we don't want to be crashing people's home gateways, so we go out of our way to avoid -// the queries that crash them. -// -// Some examples: -// -// 1. Any query where the name ends in ".in-addr.arpa." and the text before this is 32 or more bytes. -// The query type does not need to be PTR -- the gateway will crash for any query type. -// e.g. "ping long-name-crashes-the-buggy-router.in-addr.arpa" will crash one of these. -// -// 2. Any query that results in a large response with the TC bit set. -// -// 3. Any PTR query that doesn't begin with four decimal numbers. -// These gateways appear to assume that the only possible PTR query is a reverse-mapping query -// (e.g. "1.0.168.192.in-addr.arpa") and if they ever get a PTR query where the first four -// labels are not all decimal numbers in the range 0-255, they handle that by crashing. -// These gateways also ignore the remainder of the name following the four decimal numbers -// -- whether or not it actually says in-addr.arpa, they just make up an answer anyway. -// -// The challenge therefore is to craft a query that will discern whether the DNS server -// is one of these buggy ones, without crashing it. Furthermore we don't want our test -// queries making it all the way to the root name servers, putting extra load on those -// name servers and giving Apple a bad reputation. To this end we send this query: -// dig -t ptr 1.0.0.127.dnsbugtest.1.0.0.127.in-addr.arpa. -// -// The text preceding the ".in-addr.arpa." is under 32 bytes, so it won't cause crash (1). -// It will not yield a large response with the TC bit set, so it won't cause crash (2). -// It starts with four decimal numbers, so it won't cause crash (3). -// The name falls within the "1.0.0.127.in-addr.arpa." domain, the reverse-mapping name for the local -// loopback address, and therefore the query will black-hole at the first properly-configured DNS server -// it reaches, making it highly unlikely that this query will make it all the way to the root. -// -// Finally, the correct response to this query is NXDOMAIN or a similar error, but the -// gateways that ignore the remainder of the name following the four decimal numbers -// give themselves away by actually returning a result for this nonsense query. - -mDNSlocal const domainname *DNSRelayTestQuestion = (const domainname*) - "\x1" "1" "\x1" "0" "\x1" "0" "\x3" "127" "\xa" "dnsbugtest" - "\x1" "1" "\x1" "0" "\x1" "0" "\x3" "127" "\x7" "in-addr" "\x4" "arpa"; - -// See comments above for DNSRelayTestQuestion -// If this is the kind of query that has the risk of crashing buggy DNS servers, we do a test question first -mDNSlocal mDNSBool NoTestQuery(DNSQuestion *q) - { - int i; - mDNSu8 *p = q->qname.c; - if (q->AuthInfo) return(mDNStrue); // Don't need a test query for private queries sent directly to authoritative server over TLS/TCP - if (q->qtype != kDNSType_PTR) return(mDNStrue); // Don't need a test query for any non-PTR queries - for (i=0; i<4; i++) // If qname does not begin with num.num.num.num, can't skip the test query - { - if (p[0] < 1 || p[0] > 3) return(mDNSfalse); - if ( p[1] < '0' || p[1] > '9' ) return(mDNSfalse); - if (p[0] >= 2 && (p[2] < '0' || p[2] > '9')) return(mDNSfalse); - if (p[0] >= 3 && (p[3] < '0' || p[3] > '9')) return(mDNSfalse); - p += 1 + p[0]; - } - // If remainder of qname is ".in-addr.arpa.", this is a vanilla reverse-mapping query and - // we can safely do it without needing a test query first, otherwise we need the test query. - return(SameDomainName((domainname*)p, (const domainname*)"\x7" "in-addr" "\x4" "arpa")); - } - -// Returns mDNStrue if response was handled -mDNSlocal mDNSBool uDNS_ReceiveTestQuestionResponse(mDNS *const m, DNSMessage *const msg, const mDNSu8 *const end, - const mDNSAddr *const srcaddr, const mDNSIPPort srcport) - { - const mDNSu8 *ptr = msg->data; - DNSQuestion pktq; - DNSServer *s; - mDNSu32 result = 0; - - // 1. Find out if this is an answer to one of our test questions - if (msg->h.numQuestions != 1) return(mDNSfalse); - ptr = getQuestion(msg, ptr, end, mDNSInterface_Any, &pktq); - if (!ptr) return(mDNSfalse); - if (pktq.qtype != kDNSType_PTR || pktq.qclass != kDNSClass_IN) return(mDNSfalse); - if (!SameDomainName(&pktq.qname, DNSRelayTestQuestion)) return(mDNSfalse); - - // 2. If the DNS relay gave us a positive response, then it's got buggy firmware - // else, if the DNS relay gave us an error or no-answer response, it passed our test - if ((msg->h.flags.b[1] & kDNSFlag1_RC_Mask) == kDNSFlag1_RC_NoErr && msg->h.numAnswers > 0) - result = DNSServer_Failed; - else - result = DNSServer_Passed; - - // 3. Find occurrences of this server in our list, and mark them appropriately - for (s = m->DNSServers; s; s = s->next) - { - mDNSBool matchaddr = (s->teststate != result && mDNSSameAddress(srcaddr, &s->addr) && mDNSSameIPPort(srcport, s->port)); - mDNSBool matchid = (s->teststate == DNSServer_Untested && mDNSSameOpaque16(msg->h.id, s->testid)); - if (matchaddr || matchid) - { - DNSQuestion *q; - s->teststate = result; - if (result == DNSServer_Passed) - { - LogInfo("DNS Server %#a:%d (%#a:%d) %d passed%s", - &s->addr, mDNSVal16(s->port), srcaddr, mDNSVal16(srcport), mDNSVal16(s->testid), - matchaddr ? "" : " NOTE: Reply did not come from address to which query was sent"); - } - else - { - LogMsg("NOTE: Wide-Area Service Discovery disabled to avoid crashing defective DNS relay %#a:%d (%#a:%d) %d%s", - &s->addr, mDNSVal16(s->port), srcaddr, mDNSVal16(srcport), mDNSVal16(s->testid), - matchaddr ? "" : " NOTE: Reply did not come from address to which query was sent"); - } - - // If this server has just changed state from DNSServer_Untested to DNSServer_Passed, then retrigger any waiting questions. - // We use the NoTestQuery() test so that we only retrigger questions that were actually blocked waiting for this test to complete. - if (result == DNSServer_Passed) // Unblock any questions that were waiting for this result - for (q = m->Questions; q; q=q->next) - if (q->qDNSServer == s && !NoTestQuery(q)) - { - q->ThisQInterval = INIT_UCAST_POLL_INTERVAL / QuestionIntervalStep; - q->unansweredQueries = 0; - q->LastQTime = m->timenow - q->ThisQInterval; - m->NextScheduledQuery = m->timenow; - } - } - } - - return(mDNStrue); // Return mDNStrue to tell uDNS_ReceiveMsg it doesn't need to process this packet further - } - -// Called from mDNSCoreReceive with the lock held -mDNSexport void uDNS_ReceiveMsg(mDNS *const m, DNSMessage *const msg, const mDNSu8 *const end, const mDNSAddr *const srcaddr, const mDNSIPPort srcport) - { - DNSQuestion *qptr; - mStatus err = mStatus_NoError; - - mDNSu8 StdR = kDNSFlag0_QR_Response | kDNSFlag0_OP_StdQuery; - mDNSu8 UpdateR = kDNSFlag0_QR_Response | kDNSFlag0_OP_Update; - mDNSu8 QR_OP = (mDNSu8)(msg->h.flags.b[0] & kDNSFlag0_QROP_Mask); - mDNSu8 rcode = (mDNSu8)(msg->h.flags.b[1] & kDNSFlag1_RC_Mask); - - (void)srcport; // Unused - - debugf("uDNS_ReceiveMsg from %#-15a with " - "%2d Question%s %2d Answer%s %2d Authorit%s %2d Additional%s %d bytes", - srcaddr, - msg->h.numQuestions, msg->h.numQuestions == 1 ? ", " : "s,", - msg->h.numAnswers, msg->h.numAnswers == 1 ? ", " : "s,", - msg->h.numAuthorities, msg->h.numAuthorities == 1 ? "y, " : "ies,", - msg->h.numAdditionals, msg->h.numAdditionals == 1 ? "" : "s", end - msg->data); - - if (QR_OP == StdR) - { - //if (srcaddr && recvLLQResponse(m, msg, end, srcaddr, srcport)) return; - if (uDNS_ReceiveTestQuestionResponse(m, msg, end, srcaddr, srcport)) return; - for (qptr = m->Questions; qptr; qptr = qptr->next) - if (msg->h.flags.b[0] & kDNSFlag0_TC && mDNSSameOpaque16(qptr->TargetQID, msg->h.id) && m->timenow - qptr->LastQTime < RESPONSE_WINDOW) - { - if (!srcaddr) LogMsg("uDNS_ReceiveMsg: TCP DNS response had TC bit set: ignoring"); - else - { - // Don't reuse TCP connections. We might have failed over to a different DNS server - // while the first TCP connection is in progress. We need a new TCP connection to the - // new DNS server. So, always try to establish a new connection. - if (qptr->tcp) { DisposeTCPConn(qptr->tcp); qptr->tcp = mDNSNULL; } - qptr->tcp = MakeTCPConn(m, mDNSNULL, mDNSNULL, kTCPSocketFlags_Zero, srcaddr, srcport, mDNSNULL, qptr, mDNSNULL); - } - } - } - - if (QR_OP == UpdateR) - { - mDNSu32 lease = GetPktLease(m, msg, end); - mDNSs32 expire = m->timenow + (mDNSs32)lease * mDNSPlatformOneSecond; - mDNSu32 random = mDNSRandom((mDNSs32)lease * mDNSPlatformOneSecond/10); - - //rcode = kDNSFlag1_RC_ServFail; // Simulate server failure (rcode 2) - - // Walk through all the records that matches the messageID. There could be multiple - // records if we had sent them in a group - if (m->CurrentRecord) - LogMsg("uDNS_ReceiveMsg ERROR m->CurrentRecord already set %s", ARDisplayString(m, m->CurrentRecord)); - m->CurrentRecord = m->ResourceRecords; - while (m->CurrentRecord) - { - AuthRecord *rptr = m->CurrentRecord; - m->CurrentRecord = m->CurrentRecord->next; - if (AuthRecord_uDNS(rptr) && mDNSSameOpaque16(rptr->updateid, msg->h.id)) - { - err = checkUpdateResult(m, rptr->resrec.name, rcode, msg, end); - if (!err && rptr->uselease && lease) - if (rptr->expire - expire >= 0 || rptr->state != regState_UpdatePending) - { - rptr->expire = expire; - rptr->refreshCount = 0; - } - // We pass the random value to make sure that if we update multiple - // records, they all get the same random value - hndlRecordUpdateReply(m, rptr, err, random); - } - } - } - debugf("Received unexpected response: ID %d matches no active records", mDNSVal16(msg->h.id)); - } - -// *************************************************************************** -#if COMPILER_LIKES_PRAGMA_MARK -#pragma mark - Query Routines -#endif - -mDNSexport void sendLLQRefresh(mDNS *m, DNSQuestion *q) - { - mDNSu8 *end; - LLQOptData llq; - mDNSu8 *limit = m->omsg.data + AbsoluteMaxDNSMessageData; - - if (q->ReqLease) - if ((q->state == LLQ_Established && q->ntries >= kLLQ_MAX_TRIES) || q->expire - m->timenow < 0) - { - LogMsg("Unable to refresh LLQ %##s (%s) - will retry in %d seconds", q->qname.c, DNSTypeName(q->qtype), LLQ_POLL_INTERVAL / mDNSPlatformOneSecond); - StartLLQPolling(m,q); - return; - } - - llq.vers = kLLQ_Vers; - llq.llqOp = kLLQOp_Refresh; - llq.err = q->tcp ? GetLLQEventPort(m, &q->servAddr) : LLQErr_NoError; // If using TCP tell server what UDP port to send notifications to - llq.id = q->id; - llq.llqlease = q->ReqLease; - - InitializeDNSMessage(&m->omsg.h, q->TargetQID, uQueryFlags); - end = putLLQ(&m->omsg, m->omsg.data, q, &llq); - if (!end) { LogMsg("sendLLQRefresh: putLLQ failed %##s (%s)", q->qname.c, DNSTypeName(q->qtype)); return; } - - // Note that we (conditionally) add HINFO and TSIG here, since the question might be going away, - // so we may not be able to reference it (most importantly it's AuthInfo) when we actually send the message - end = putHINFO(m, &m->omsg, end, q->AuthInfo, limit); - if (!end) { LogMsg("sendLLQRefresh: putHINFO failed %##s (%s)", q->qname.c, DNSTypeName(q->qtype)); return; } - - if (PrivateQuery(q)) - { - DNSDigest_SignMessageHostByteOrder(&m->omsg, &end, q->AuthInfo); - if (!end) { LogMsg("sendLLQRefresh: DNSDigest_SignMessage failed %##s (%s)", q->qname.c, DNSTypeName(q->qtype)); return; } - } - - if (PrivateQuery(q) && !q->tcp) - { - LogInfo("sendLLQRefresh setting up new TLS session %##s (%s)", q->qname.c, DNSTypeName(q->qtype)); - if (!q->nta) { LogMsg("sendLLQRefresh:ERROR!! q->nta is NULL for %##s (%s)", q->qname.c, DNSTypeName(q->qtype)); return; } - q->tcp = MakeTCPConn(m, &m->omsg, end, kTCPSocketFlags_UseTLS, &q->servAddr, q->servPort, &q->nta->Host, q, mDNSNULL); - } - else - { - mStatus err; - - // if AuthInfo and AuthInfo->AutoTunnel is set, we use the TCP socket but don't need to pass the AuthInfo as - // we already protected the message above. - LogInfo("sendLLQRefresh: using existing %s session %##s (%s)", PrivateQuery(q) ? "TLS" : "UDP", - q->qname.c, DNSTypeName(q->qtype)); - - err = mDNSSendDNSMessage(m, &m->omsg, end, mDNSInterface_Any, q->LocalSocket, &q->servAddr, q->servPort, q->tcp ? q->tcp->sock : mDNSNULL, mDNSNULL); - if (err) - { - LogMsg("sendLLQRefresh: mDNSSendDNSMessage%s failed: %d", q->tcp ? " (TCP)" : "", err); - if (q->tcp) { DisposeTCPConn(q->tcp); q->tcp = mDNSNULL; } - } - } - - q->ntries++; - - debugf("sendLLQRefresh ntries %d %##s (%s)", q->ntries, q->qname.c, DNSTypeName(q->qtype)); - - q->LastQTime = m->timenow; - SetNextQueryTime(m, q); - } - -mDNSexport void LLQGotZoneData(mDNS *const m, mStatus err, const ZoneData *zoneInfo) - { - DNSQuestion *q = (DNSQuestion *)zoneInfo->ZoneDataContext; - - mDNS_Lock(m); - - // If we get here it means that the GetZoneData operation has completed. - // We hold on to the zone data if it is AutoTunnel as we use the hostname - // in zoneInfo during the TLS connection setup. - q->servAddr = zeroAddr; - q->servPort = zeroIPPort; - - if (!err && zoneInfo && !mDNSIPPortIsZero(zoneInfo->Port) && !mDNSAddressIsZero(&zoneInfo->Addr) && zoneInfo->Host.c[0]) - { - q->servAddr = zoneInfo->Addr; - q->servPort = zoneInfo->Port; - if (!PrivateQuery(q)) - { - // We don't need the zone data as we use it only for the Host information which we - // don't need if we are not going to use TLS connections. - if (q->nta) - { - if (q->nta != zoneInfo) LogMsg("LLQGotZoneData: nta (%p) != zoneInfo (%p) %##s (%s)", q->nta, zoneInfo, q->qname.c, DNSTypeName(q->qtype)); - CancelGetZoneData(m, q->nta); - q->nta = mDNSNULL; - } - } - q->ntries = 0; - debugf("LLQGotZoneData %#a:%d", &q->servAddr, mDNSVal16(q->servPort)); - startLLQHandshake(m, q); - } - else - { - if (q->nta) - { - if (q->nta != zoneInfo) LogMsg("LLQGotZoneData: nta (%p) != zoneInfo (%p) %##s (%s)", q->nta, zoneInfo, q->qname.c, DNSTypeName(q->qtype)); - CancelGetZoneData(m, q->nta); - q->nta = mDNSNULL; - } - StartLLQPolling(m,q); - if (err == mStatus_NoSuchNameErr) - { - // this actually failed, so mark it by setting address to all ones - q->servAddr.type = mDNSAddrType_IPv4; - q->servAddr.ip.v4 = onesIPv4Addr; - } - } - - mDNS_Unlock(m); - } - -// Called in normal callback context (i.e. mDNS_busy and mDNS_reentrancy are both 1) -mDNSlocal void PrivateQueryGotZoneData(mDNS *const m, mStatus err, const ZoneData *zoneInfo) - { - DNSQuestion *q = (DNSQuestion *) zoneInfo->ZoneDataContext; - - LogInfo("PrivateQueryGotZoneData %##s (%s) err %d Zone %##s Private %d", q->qname.c, DNSTypeName(q->qtype), err, zoneInfo->ZoneName.c, zoneInfo->ZonePrivate); - - if (q->nta != zoneInfo) LogMsg("PrivateQueryGotZoneData:ERROR!!: nta (%p) != zoneInfo (%p) %##s (%s)", q->nta, zoneInfo, q->qname.c, DNSTypeName(q->qtype)); - - if (err || !zoneInfo || mDNSAddressIsZero(&zoneInfo->Addr) || mDNSIPPortIsZero(zoneInfo->Port) || !zoneInfo->Host.c[0]) - { - LogInfo("PrivateQueryGotZoneData: ERROR!! %##s (%s) invoked with error code %d %p %#a:%d", - q->qname.c, DNSTypeName(q->qtype), err, zoneInfo, - zoneInfo ? &zoneInfo->Addr : mDNSNULL, - zoneInfo ? mDNSVal16(zoneInfo->Port) : 0); - CancelGetZoneData(m, q->nta); - q->nta = mDNSNULL; - return; - } - - if (!zoneInfo->ZonePrivate) - { - debugf("Private port lookup failed -- retrying without TLS -- %##s (%s)", q->qname.c, DNSTypeName(q->qtype)); - q->AuthInfo = mDNSNULL; // Clear AuthInfo so we try again non-private - q->ThisQInterval = InitialQuestionInterval; - q->LastQTime = m->timenow - q->ThisQInterval; - CancelGetZoneData(m, q->nta); - q->nta = mDNSNULL; - mDNS_Lock(m); - SetNextQueryTime(m, q); - mDNS_Unlock(m); - return; - // Next call to uDNS_CheckCurrentQuestion() will do this as a non-private query - } - - if (!PrivateQuery(q)) - { - LogMsg("PrivateQueryGotZoneData: ERROR!! Not a private query %##s (%s) AuthInfo %p", q->qname.c, DNSTypeName(q->qtype), q->AuthInfo); - CancelGetZoneData(m, q->nta); - q->nta = mDNSNULL; - return; - } - - q->TargetQID = mDNS_NewMessageID(m); - if (q->tcp) { DisposeTCPConn(q->tcp); q->tcp = mDNSNULL; } - if (!q->nta) { LogMsg("PrivateQueryGotZoneData:ERROR!! nta is NULL for %##s (%s)", q->qname.c, DNSTypeName(q->qtype)); return; } - q->tcp = MakeTCPConn(m, mDNSNULL, mDNSNULL, kTCPSocketFlags_UseTLS, &zoneInfo->Addr, zoneInfo->Port, &q->nta->Host, q, mDNSNULL); - if (q->nta) { CancelGetZoneData(m, q->nta); q->nta = mDNSNULL; } - } - -// *************************************************************************** -#if COMPILER_LIKES_PRAGMA_MARK -#pragma mark - Dynamic Updates -#endif - -// Called in normal callback context (i.e. mDNS_busy and mDNS_reentrancy are both 1) -mDNSexport void RecordRegistrationGotZoneData(mDNS *const m, mStatus err, const ZoneData *zoneData) - { - AuthRecord *newRR = (AuthRecord*)zoneData->ZoneDataContext; - AuthRecord *ptr; - int c1, c2; - - if (newRR->nta != zoneData) - LogMsg("RecordRegistrationGotZoneData: nta (%p) != zoneData (%p) %##s (%s)", newRR->nta, zoneData, newRR->resrec.name->c, DNSTypeName(newRR->resrec.rrtype)); - - if (m->mDNS_busy != m->mDNS_reentrancy) - LogMsg("RecordRegistrationGotZoneData: mDNS_busy (%ld) != mDNS_reentrancy (%ld)", m->mDNS_busy, m->mDNS_reentrancy); - - // make sure record is still in list (!!!) - for (ptr = m->ResourceRecords; ptr; ptr = ptr->next) if (ptr == newRR) break; - if (!ptr) - { - LogMsg("RecordRegistrationGotZoneData - RR no longer in list. Discarding."); - CancelGetZoneData(m, newRR->nta); - newRR->nta = mDNSNULL; - return; - } - - // check error/result - if (err) - { - if (err != mStatus_NoSuchNameErr) LogMsg("RecordRegistrationGotZoneData: error %d", err); - CancelGetZoneData(m, newRR->nta); - newRR->nta = mDNSNULL; - return; - } - - if (!zoneData) { LogMsg("ERROR: RecordRegistrationGotZoneData invoked with NULL result and no error"); return; } - - if (newRR->resrec.rrclass != zoneData->ZoneClass) - { - LogMsg("ERROR: New resource record's class (%d) does not match zone class (%d)", newRR->resrec.rrclass, zoneData->ZoneClass); - CancelGetZoneData(m, newRR->nta); - newRR->nta = mDNSNULL; - return; - } - - // Don't try to do updates to the root name server. - // We might be tempted also to block updates to any single-label name server (e.g. com, edu, net, etc.) but some - // organizations use their own private pseudo-TLD, like ".home", etc, and we don't want to block that. - if (zoneData->ZoneName.c[0] == 0) - { - LogInfo("RecordRegistrationGotZoneData: No name server found claiming responsibility for \"%##s\"!", newRR->resrec.name->c); - CancelGetZoneData(m, newRR->nta); - newRR->nta = mDNSNULL; - return; - } - - // Store discovered zone data - c1 = CountLabels(newRR->resrec.name); - c2 = CountLabels(&zoneData->ZoneName); - if (c2 > c1) - { - LogMsg("RecordRegistrationGotZoneData: Zone \"%##s\" is longer than \"%##s\"", zoneData->ZoneName.c, newRR->resrec.name->c); - CancelGetZoneData(m, newRR->nta); - newRR->nta = mDNSNULL; - return; - } - newRR->zone = SkipLeadingLabels(newRR->resrec.name, c1-c2); - if (!SameDomainName(newRR->zone, &zoneData->ZoneName)) - { - LogMsg("RecordRegistrationGotZoneData: Zone \"%##s\" does not match \"%##s\" for \"%##s\"", newRR->zone->c, zoneData->ZoneName.c, newRR->resrec.name->c); - CancelGetZoneData(m, newRR->nta); - newRR->nta = mDNSNULL; - return; - } - - if (mDNSIPPortIsZero(zoneData->Port) || mDNSAddressIsZero(&zoneData->Addr) || !zoneData->Host.c[0]) - { - LogInfo("RecordRegistrationGotZoneData: No _dns-update._udp service found for \"%##s\"!", newRR->resrec.name->c); - CancelGetZoneData(m, newRR->nta); - newRR->nta = mDNSNULL; - return; - } - - newRR->Private = zoneData->ZonePrivate; - debugf("RecordRegistrationGotZoneData: Set zone information for %##s %##s to %#a:%d", - newRR->resrec.name->c, zoneData->ZoneName.c, &zoneData->Addr, mDNSVal16(zoneData->Port)); - - // If we are deregistering, uDNS_DeregisterRecord will do that as it has the zone data now. - if (newRR->state == regState_DeregPending) - { - mDNS_Lock(m); - uDNS_DeregisterRecord(m, newRR); - mDNS_Unlock(m); - return; - } - - if (newRR->resrec.rrtype == kDNSType_SRV) - { - const domainname *target; - // Reevaluate the target always as NAT/Target could have changed while - // we were fetching zone data. - mDNS_Lock(m); - target = GetServiceTarget(m, newRR); - mDNS_Unlock(m); - if (!target || target->c[0] == 0) - { - domainname *t = GetRRDomainNameTarget(&newRR->resrec); - LogInfo("RecordRegistrationGotZoneData - no target for %##s", newRR->resrec.name->c); - if (t) t->c[0] = 0; - newRR->resrec.rdlength = newRR->resrec.rdestimate = 0; - newRR->state = regState_NoTarget; - CancelGetZoneData(m, newRR->nta); - newRR->nta = mDNSNULL; - return; - } - } - // If we have non-zero service port (always?) - // and a private address, and update server is non-private - // and this service is AutoTarget - // then initiate a NAT mapping request. On completion it will do SendRecordRegistration() for us - if (newRR->resrec.rrtype == kDNSType_SRV && !mDNSIPPortIsZero(newRR->resrec.rdata->u.srv.port) && - mDNSv4AddrIsRFC1918(&m->AdvertisedV4.ip.v4) && newRR->nta && !mDNSAddrIsRFC1918(&newRR->nta->Addr) && - newRR->AutoTarget == Target_AutoHostAndNATMAP) - { - DomainAuthInfo *AuthInfo; - AuthInfo = GetAuthInfoForName(m, newRR->resrec.name); - if (AuthInfo && AuthInfo->AutoTunnel) - { - domainname *t = GetRRDomainNameTarget(&newRR->resrec); - LogMsg("RecordRegistrationGotZoneData: ERROR!! AutoTunnel has Target_AutoHostAndNATMAP for %s", ARDisplayString(m, newRR)); - if (t) t->c[0] = 0; - newRR->resrec.rdlength = newRR->resrec.rdestimate = 0; - newRR->state = regState_NoTarget; - CancelGetZoneData(m, newRR->nta); - newRR->nta = mDNSNULL; - return; - } - // During network transitions, we are called multiple times in different states. Setup NAT - // state just once for this record. - if (!newRR->NATinfo.clientContext) - { - LogInfo("RecordRegistrationGotZoneData StartRecordNatMap %s", ARDisplayString(m, newRR)); - newRR->state = regState_NATMap; - StartRecordNatMap(m, newRR); - return; - } - else LogInfo("RecordRegistrationGotZoneData: StartRecordNatMap for %s, state %d, context %p", ARDisplayString(m, newRR), newRR->state, newRR->NATinfo.clientContext); - } - mDNS_Lock(m); - // We want IsRecordMergeable to check whether it is a record whose update can be - // sent with others. We set the time before we call IsRecordMergeable, so that - // it does not fail this record based on time. We are interested in other checks - // at this time. If a previous update resulted in error, then don't reset the - // interval. Preserve the back-off so that we don't keep retrying aggressively. - if (newRR->updateError == mStatus_NoError) - { - newRR->ThisAPInterval = INIT_RECORD_REG_INTERVAL; - newRR->LastAPTime = m->timenow - INIT_RECORD_REG_INTERVAL; - } - if (IsRecordMergeable(m, newRR, m->timenow + MERGE_DELAY_TIME)) - { - // Delay the record registration by MERGE_DELAY_TIME so that we can merge them - // into one update - LogInfo("RecordRegistrationGotZoneData: Delayed registration for %s", ARDisplayString(m, newRR)); - newRR->LastAPTime += MERGE_DELAY_TIME; - } - mDNS_Unlock(m); - } - -mDNSlocal void SendRecordDeregistration(mDNS *m, AuthRecord *rr) - { - mDNSu8 *ptr = m->omsg.data; - mDNSu8 *limit; - DomainAuthInfo *AuthInfo; - - if (m->mDNS_busy != m->mDNS_reentrancy+1) - LogMsg("SendRecordDeRegistration: Lock not held! mDNS_busy (%ld) mDNS_reentrancy (%ld)", m->mDNS_busy, m->mDNS_reentrancy); - - if (!rr->nta || mDNSIPv4AddressIsZero(rr->nta->Addr.ip.v4)) - { - LogMsg("SendRecordDeRegistration: No zone info for Resource record %s RecordType %d", ARDisplayString(m, rr), rr->resrec.RecordType); - return; - } - - limit = ptr + AbsoluteMaxDNSMessageData; - AuthInfo = GetAuthInfoForName_internal(m, rr->resrec.name); - limit -= RRAdditionalSize(m, AuthInfo); - - rr->updateid = mDNS_NewMessageID(m); - InitializeDNSMessage(&m->omsg.h, rr->updateid, UpdateReqFlags); - - // set zone - ptr = putZone(&m->omsg, ptr, limit, rr->zone, mDNSOpaque16fromIntVal(rr->resrec.rrclass)); - if (!ptr) goto exit; - - ptr = BuildUpdateMessage(m, ptr, rr, limit); - - if (!ptr) goto exit; - - if (rr->Private) - { - LogInfo("SendRecordDeregistration TCP %p %s", rr->tcp, ARDisplayString(m, rr)); - if (rr->tcp) LogInfo("SendRecordDeregistration: Disposing existing TCP connection for %s", ARDisplayString(m, rr)); - if (rr->tcp) { DisposeTCPConn(rr->tcp); rr->tcp = mDNSNULL; } - if (!rr->nta) { LogMsg("SendRecordDeregistration:Private:ERROR!! nta is NULL for %s", ARDisplayString(m, rr)); return; } - rr->tcp = MakeTCPConn(m, &m->omsg, ptr, kTCPSocketFlags_UseTLS, &rr->nta->Addr, rr->nta->Port, &rr->nta->Host, mDNSNULL, rr); - } - else - { - mStatus err; - LogInfo("SendRecordDeregistration UDP %s", ARDisplayString(m, rr)); - if (!rr->nta) { LogMsg("SendRecordDeregistration:ERROR!! nta is NULL for %s", ARDisplayString(m, rr)); return; } - err = mDNSSendDNSMessage(m, &m->omsg, ptr, mDNSInterface_Any, mDNSNULL, &rr->nta->Addr, rr->nta->Port, mDNSNULL, GetAuthInfoForName_internal(m, rr->resrec.name)); - if (err) debugf("ERROR: SendRecordDeregistration - mDNSSendDNSMessage - %d", err); - //if (rr->state == regState_DeregPending) CompleteDeregistration(m, rr); // Don't touch rr after this - } - SetRecordRetry(m, rr, 0); - return; -exit: - LogMsg("SendRecordDeregistration: Error formatting message for %s", ARDisplayString(m, rr)); - } - -mDNSexport mStatus uDNS_DeregisterRecord(mDNS *const m, AuthRecord *const rr) - { - DomainAuthInfo *info; - - LogInfo("uDNS_DeregisterRecord: Resource Record %s, state %d", ARDisplayString(m, rr), rr->state); - - switch (rr->state) - { - case regState_Refresh: - case regState_Pending: - case regState_UpdatePending: - case regState_Registered: break; - case regState_DeregPending: break; - - case regState_NATError: - case regState_NATMap: - // A record could be in NoTarget to start with if the corresponding SRV record could not find a target. - // It is also possible to reenter the NoTarget state when we move to a network with a NAT that has - // no NAT-PMP/UPnP support. In that case before we entered NoTarget, we already deregistered with - // the server. - case regState_NoTarget: - case regState_Unregistered: - case regState_Zero: - default: - LogInfo("uDNS_DeregisterRecord: State %d for %##s type %s", rr->state, rr->resrec.name->c, DNSTypeName(rr->resrec.rrtype)); - // This function may be called during sleep when there are no sleep proxy servers - if (rr->resrec.RecordType == kDNSRecordTypeDeregistering) CompleteDeregistration(m, rr); - return mStatus_NoError; - } - - // If a current group registration is pending, we can't send this deregisration till that registration - // has reached the server i.e., the ordering is important. Previously, if we did not send this - // registration in a group, then the previous connection will be torn down as part of sending the - // deregistration. If we send this in a group, we need to locate the resource record that was used - // to send this registration and terminate that connection. This means all the updates on that might - // be lost (assuming the response is not waiting for us at the socket) and the retry will send the - // update again sometime in the near future. - // - // NOTE: SSL handshake failures normally free the TCP connection immediately. Hence, you may not - // find the TCP below there. This case can happen only when tcp is trying to actively retransmit - // the request or SSL negotiation taking time i.e resource record is actively trying to get the - // message to the server. During that time a deregister has to happen. - - if (!mDNSOpaque16IsZero(rr->updateid)) - { - AuthRecord *anchorRR; - mDNSBool found = mDNSfalse; - for (anchorRR = m->ResourceRecords; anchorRR; anchorRR = anchorRR->next) - { - if (AuthRecord_uDNS(rr) && mDNSSameOpaque16(anchorRR->updateid, rr->updateid) && anchorRR->tcp) - { - LogInfo("uDNS_DeregisterRecord: Found Anchor RR %s terminated", ARDisplayString(m, anchorRR)); - if (found) - LogMsg("uDNS_DeregisterRecord: ERROR: Another anchorRR %s found", ARDisplayString(m, anchorRR)); - DisposeTCPConn(anchorRR->tcp); - anchorRR->tcp = mDNSNULL; - found = mDNStrue; - } - } - if (!found) LogInfo("uDNSDeregisterRecord: Cannot find the anchor Resource Record for %s, not an error", ARDisplayString(m, rr)); - } - - // Retry logic for deregistration should be no different from sending registration the first time. - // Currently ThisAPInterval most likely is set to the refresh interval - rr->state = regState_DeregPending; - rr->ThisAPInterval = INIT_RECORD_REG_INTERVAL; - rr->LastAPTime = m->timenow - INIT_RECORD_REG_INTERVAL; - info = GetAuthInfoForName_internal(m, rr->resrec.name); - if (IsRecordMergeable(m, rr, m->timenow + MERGE_DELAY_TIME)) - { - // Delay the record deregistration by MERGE_DELAY_TIME so that we can merge them - // into one update. If the domain is being deleted, delay by 2 * MERGE_DELAY_TIME - // so that we can merge all the AutoTunnel records and the service records in - // one update (they get deregistered a little apart) - if (info && info->deltime) rr->LastAPTime += (2 * MERGE_DELAY_TIME); - else rr->LastAPTime += MERGE_DELAY_TIME; - } - // IsRecordMergeable could have returned false for several reasons e.g., DontMerge is set or - // no zone information. Most likely it is the latter, CheckRecordUpdates will fetch the zone - // data when it encounters this record. - - if (m->NextuDNSEvent - (rr->LastAPTime + rr->ThisAPInterval) >= 0) - m->NextuDNSEvent = (rr->LastAPTime + rr->ThisAPInterval); - - return mStatus_NoError; - } - -mDNSexport mStatus uDNS_UpdateRecord(mDNS *m, AuthRecord *rr) - { - LogInfo("uDNS_UpdateRecord: Resource Record %##s, state %d", rr->resrec.name->c, rr->state); - switch(rr->state) - { - case regState_DeregPending: - case regState_Unregistered: - // not actively registered - goto unreg_error; - - case regState_NATMap: - case regState_NoTarget: - // change rdata directly since it hasn't been sent yet - if (rr->UpdateCallback) rr->UpdateCallback(m, rr, rr->resrec.rdata, rr->resrec.rdlength); - SetNewRData(&rr->resrec, rr->NewRData, rr->newrdlength); - rr->NewRData = mDNSNULL; - return mStatus_NoError; - - case regState_Pending: - case regState_Refresh: - case regState_UpdatePending: - // registration in-flight. queue rdata and return - if (rr->QueuedRData && rr->UpdateCallback) - // if unsent rdata is already queued, free it before we replace it - rr->UpdateCallback(m, rr, rr->QueuedRData, rr->QueuedRDLen); - rr->QueuedRData = rr->NewRData; - rr->QueuedRDLen = rr->newrdlength; - rr->NewRData = mDNSNULL; - return mStatus_NoError; - - case regState_Registered: - rr->OrigRData = rr->resrec.rdata; - rr->OrigRDLen = rr->resrec.rdlength; - rr->InFlightRData = rr->NewRData; - rr->InFlightRDLen = rr->newrdlength; - rr->NewRData = mDNSNULL; - rr->state = regState_UpdatePending; - rr->ThisAPInterval = INIT_RECORD_REG_INTERVAL; - rr->LastAPTime = m->timenow - INIT_RECORD_REG_INTERVAL; - return mStatus_NoError; - - case regState_NATError: - LogMsg("ERROR: uDNS_UpdateRecord called for record %##s with bad state regState_NATError", rr->resrec.name->c); - return mStatus_UnknownErr; // states for service records only - - default: LogMsg("uDNS_UpdateRecord: Unknown state %d for %##s", rr->state, rr->resrec.name->c); - } - - unreg_error: - LogMsg("uDNS_UpdateRecord: Requested update of record %##s type %d, in erroneous state %d", - rr->resrec.name->c, rr->resrec.rrtype, rr->state); - return mStatus_Invalid; - } - -// *************************************************************************** -#if COMPILER_LIKES_PRAGMA_MARK -#pragma mark - Periodic Execution Routines -#endif - -// The question to be checked is not passed in as an explicit parameter; -// instead it is implicit that the question to be checked is m->CurrentQuestion. -mDNSexport void uDNS_CheckCurrentQuestion(mDNS *const m) - { - DNSQuestion *q = m->CurrentQuestion; - if (m->timenow - NextQSendTime(q) < 0) return; - - if (q->LongLived) - { - switch (q->state) - { - case LLQ_InitialRequest: startLLQHandshake(m, q); break; - case LLQ_SecondaryRequest: - // For PrivateQueries, we need to start the handshake again as we don't do the Challenge/Response step - if (PrivateQuery(q)) - startLLQHandshake(m, q); - else - sendChallengeResponse(m, q, mDNSNULL); - break; - case LLQ_Established: sendLLQRefresh(m, q); break; - case LLQ_Poll: break; // Do nothing (handled below) - } - } - - // We repeat the check above (rather than just making this the "else" case) because startLLQHandshake can change q->state to LLQ_Poll - if (!(q->LongLived && q->state != LLQ_Poll)) - { - if (q->unansweredQueries >= MAX_UCAST_UNANSWERED_QUERIES) - { - DNSServer *orig = q->qDNSServer; - if (orig) - LogInfo("uDNS_CheckCurrentQuestion: Sent %d unanswered queries for %##s (%s) to %#a:%d (%##s)", - q->unansweredQueries, q->qname.c, DNSTypeName(q->qtype), &orig->addr, mDNSVal16(orig->port), orig->domain.c); - - PenalizeDNSServer(m, q); - q->noServerResponse = 1; - } - // There are two cases here. - // - // 1. We have only one DNS server for this question. It is not responding even after we sent MAX_UCAST_UNANSWERED_QUERIES. - // In that case, we need to keep retrying till we get a response. But we need to backoff as we retry. We set - // noServerResponse in the block above and below we do not touch the question interval. When we come here, we - // already waited for the response. We need to send another query right at this moment. We do that below by - // reinitializing dns servers and reissuing the query. - // - // 2. We have more than one DNS server. If at least one server did not respond, we would have set noServerResponse - // either now (the last server in the list) or before (non-last server in the list). In either case, if we have - // reached the end of DNS server list, we need to try again from the beginning. Ideally we should try just the - // servers that did not respond, but for simplicity we try all the servers. Once we reached the end of list, we - // set triedAllServersOnce so that we don't try all the servers aggressively. See PenalizeDNSServer. - if (!q->qDNSServer && q->noServerResponse) - { - DNSServer *new; - DNSQuestion *qptr; - q->triedAllServersOnce = 1; - // Re-initialize all DNS servers for this question. If we have a DNSServer, DNSServerChangeForQuestion will - // handle all the work including setting the new DNS server. - SetValidDNSServers(m, q); - new = GetServerForQuestion(m, q); - if (new) - { - LogInfo("uDNS_checkCurrentQuestion: Retrying question %p %##s (%s) DNS Server %#a:%d ThisQInterval %d", - q, q->qname.c, DNSTypeName(q->qtype), new ? &new->addr : mDNSNULL, mDNSVal16(new ? new->port : zeroIPPort), q->ThisQInterval); - DNSServerChangeForQuestion(m, q, new); - } - for (qptr = q->next ; qptr; qptr = qptr->next) - if (qptr->DuplicateOf == q) { qptr->validDNSServers = q->validDNSServers; qptr->qDNSServer = q->qDNSServer; } - } - if (q->qDNSServer && q->qDNSServer->teststate != DNSServer_Disabled) - { - mDNSu8 *end = m->omsg.data; - mStatus err = mStatus_NoError; - mDNSBool private = mDNSfalse; - - InitializeDNSMessage(&m->omsg.h, q->TargetQID, uQueryFlags); - - if (q->qDNSServer->teststate != DNSServer_Untested || NoTestQuery(q)) - { - end = putQuestion(&m->omsg, m->omsg.data, m->omsg.data + AbsoluteMaxDNSMessageData, &q->qname, q->qtype, q->qclass); - private = PrivateQuery(q); - } - else if (m->timenow - q->qDNSServer->lasttest >= INIT_UCAST_POLL_INTERVAL) // Make sure at least three seconds has elapsed since last test query - { - LogInfo("Sending DNS test query to %#a:%d", &q->qDNSServer->addr, mDNSVal16(q->qDNSServer->port)); - q->ThisQInterval = INIT_UCAST_POLL_INTERVAL / QuestionIntervalStep; - q->qDNSServer->lasttest = m->timenow; - end = putQuestion(&m->omsg, m->omsg.data, m->omsg.data + AbsoluteMaxDNSMessageData, DNSRelayTestQuestion, kDNSType_PTR, kDNSClass_IN); - q->qDNSServer->testid = m->omsg.h.id; - } - - if (end > m->omsg.data && (q->qDNSServer->teststate != DNSServer_Failed || NoTestQuery(q))) - { - //LogMsg("uDNS_CheckCurrentQuestion %p %d %p %##s (%s)", q, NextQSendTime(q) - m->timenow, private, q->qname.c, DNSTypeName(q->qtype)); - if (private) - { - if (q->nta) CancelGetZoneData(m, q->nta); - q->nta = StartGetZoneData(m, &q->qname, q->LongLived ? ZoneServiceLLQ : ZoneServiceQuery, PrivateQueryGotZoneData, q); - if (q->state == LLQ_Poll) q->ThisQInterval = (LLQ_POLL_INTERVAL + mDNSRandom(LLQ_POLL_INTERVAL/10)) / QuestionIntervalStep; - } - else - { - debugf("uDNS_CheckCurrentQuestion sending %p %##s (%s) %#a:%d UnansweredQueries %d", - q, q->qname.c, DNSTypeName(q->qtype), - q->qDNSServer ? &q->qDNSServer->addr : mDNSNULL, mDNSVal16(q->qDNSServer ? q->qDNSServer->port : zeroIPPort), q->unansweredQueries); - if (!q->LocalSocket) q->LocalSocket = mDNSPlatformUDPSocket(m, zeroIPPort); - if (!q->LocalSocket) err = mStatus_NoMemoryErr; // If failed to make socket (should be very rare), we'll try again next time - else err = mDNSSendDNSMessage(m, &m->omsg, end, q->qDNSServer->interface, q->LocalSocket, &q->qDNSServer->addr, q->qDNSServer->port, mDNSNULL, mDNSNULL); - } - } - - if (err) debugf("ERROR: uDNS_idle - mDNSSendDNSMessage - %d", err); // surpress syslog messages if we have no network - else - { - q->ThisQInterval = q->ThisQInterval * QuestionIntervalStep; // Only increase interval if send succeeded - q->unansweredQueries++; - if (q->ThisQInterval > MAX_UCAST_POLL_INTERVAL) - q->ThisQInterval = MAX_UCAST_POLL_INTERVAL; - if (private && q->state != LLQ_Poll) - { - // We don't want to retransmit too soon. Hence, we always schedule our first - // retransmisson at 3 seconds rather than one second - if (q->ThisQInterval < (3 * mDNSPlatformOneSecond)) - q->ThisQInterval = q->ThisQInterval * QuestionIntervalStep; - if (q->ThisQInterval > LLQ_POLL_INTERVAL) - q->ThisQInterval = LLQ_POLL_INTERVAL; - LogInfo("uDNS_CheckCurrentQuestion: private non polling question for %##s (%s) will be retried in %d ms", q->qname.c, DNSTypeName(q->qtype), q->ThisQInterval); - } - if (q->qDNSServer->cellIntf) - { - // We don't want to retransmit too soon. Schedule our first retransmisson at - // MIN_UCAST_RETRANS_TIMEOUT seconds. - if (q->ThisQInterval < MIN_UCAST_RETRANS_TIMEOUT) - q->ThisQInterval = MIN_UCAST_RETRANS_TIMEOUT; - } - debugf("uDNS_CheckCurrentQuestion: Increased ThisQInterval to %d for %##s (%s), cell %d", q->ThisQInterval, q->qname.c, DNSTypeName(q->qtype), q->qDNSServer->cellIntf); - } - q->LastQTime = m->timenow; - SetNextQueryTime(m, q); - } - else - { - // If we have no server for this query, or the only server is a disabled one, then we deliver - // a transient failure indication to the client. This is important for things like iPhone - // where we want to return timely feedback to the user when no network is available. - // After calling MakeNegativeCacheRecord() we store the resulting record in the - // cache so that it will be visible to other clients asking the same question. - // (When we have a group of identical questions, only the active representative of the group gets - // passed to uDNS_CheckCurrentQuestion -- we only want one set of query packets hitting the wire -- - // but we want *all* of the questions to get answer callbacks.) - - CacheRecord *rr; - const mDNSu32 slot = HashSlot(&q->qname); - CacheGroup *const cg = CacheGroupForName(m, slot, q->qnamehash, &q->qname); - if (cg) - for (rr = cg->members; rr; rr=rr->next) - if (SameNameRecordAnswersQuestion(&rr->resrec, q)) mDNS_PurgeCacheResourceRecord(m, rr); - - if (!q->qDNSServer) - { - if (!mDNSOpaque64IsZero(&q->validDNSServers)) - LogMsg("uDNS_CheckCurrentQuestion: ERROR!!: valid DNSServer bits not zero 0x%x, 0x%x for question %##s (%s)", - q->validDNSServers.l[1], q->validDNSServers.l[0], q->qname.c, DNSTypeName(q->qtype)); - // If we reached the end of list while picking DNS servers, then we don't want to deactivate the - // question. Try after 60 seconds. We find this by looking for valid DNSServers for this question, - // if we find any, then we must have tried them before we came here. This avoids maintaining - // another state variable to see if we had valid DNS servers for this question. - SetValidDNSServers(m, q); - if (mDNSOpaque64IsZero(&q->validDNSServers)) - { - LogInfo("uDNS_CheckCurrentQuestion: no DNS server for %##s (%s)", q->qname.c, DNSTypeName(q->qtype)); - q->ThisQInterval = 0; - } - else - { - DNSQuestion *qptr; - // Pretend that we sent this question. As this is an ActiveQuestion, the NextScheduledQuery should - // be set properly. Also, we need to properly backoff in cases where we don't set the question to - // MaxQuestionInterval when we answer the question e.g., LongLived, we need to keep backing off - q->ThisQInterval = q->ThisQInterval * QuestionIntervalStep; - q->LastQTime = m->timenow; - SetNextQueryTime(m, q); - // Pick a new DNS server now. Otherwise, when the cache is 80% of its expiry, we will try - // to send a query and come back to the same place here and log the above message. - q->qDNSServer = GetServerForQuestion(m, q); - for (qptr = q->next ; qptr; qptr = qptr->next) - if (qptr->DuplicateOf == q) { qptr->validDNSServers = q->validDNSServers; qptr->qDNSServer = q->qDNSServer; } - LogInfo("uDNS_checkCurrentQuestion: Tried all DNS servers, retry question %p SuppressUnusable %d %##s (%s) with DNS Server %#a:%d after 60 seconds, ThisQInterval %d", - q, q->SuppressUnusable, q->qname.c, DNSTypeName(q->qtype), - q->qDNSServer ? &q->qDNSServer->addr : mDNSNULL, mDNSVal16(q->qDNSServer ? q->qDNSServer->port : zeroIPPort), q->ThisQInterval); - } - } - else - { - q->ThisQInterval = 0; - LogMsg("uDNS_CheckCurrentQuestion DNS server %#a:%d for %##s is disabled", &q->qDNSServer->addr, mDNSVal16(q->qDNSServer->port), q->qname.c); - } - - // For some of the WAB queries that we generate form within the mDNSResponder, most of the home routers - // don't understand and return ServFail/NXDomain. In those cases, we don't want to try too often. We try - // every fifteen minutes in that case - MakeNegativeCacheRecord(m, &m->rec.r, &q->qname, q->qnamehash, q->qtype, q->qclass, (DomainEnumQuery(&q->qname) ? 60 * 15 : 60), mDNSInterface_Any, q->qDNSServer); - q->unansweredQueries = 0; - // We're already using the m->CurrentQuestion pointer, so CacheRecordAdd can't use it to walk the question list. - // To solve this problem we set rr->DelayDelivery to a nonzero value (which happens to be 'now') so that we - // momentarily defer generating answer callbacks until mDNS_Execute time. - CreateNewCacheEntry(m, slot, cg, NonZeroTime(m->timenow)); - ScheduleNextCacheCheckTime(m, slot, NonZeroTime(m->timenow)); - m->rec.r.resrec.RecordType = 0; // Clear RecordType to show we're not still using it - // MUST NOT touch m->CurrentQuestion (or q) after this -- client callback could have deleted it - } - } - } - -mDNSexport void CheckNATMappings(mDNS *m) - { - mStatus err = mStatus_NoError; - mDNSBool rfc1918 = mDNSv4AddrIsRFC1918(&m->AdvertisedV4.ip.v4); - mDNSBool HaveRoutable = !rfc1918 && !mDNSIPv4AddressIsZero(m->AdvertisedV4.ip.v4); - m->NextScheduledNATOp = m->timenow + 0x3FFFFFFF; - - if (HaveRoutable) m->ExternalAddress = m->AdvertisedV4.ip.v4; - - if (m->NATTraversals && rfc1918) // Do we need to open NAT-PMP socket to receive multicast announcements from router? - { - if (m->NATMcastRecvskt == mDNSNULL) // If we are behind a NAT and the socket hasn't been opened yet, open it - { - // we need to log a message if we can't get our socket, but only the first time (after success) - static mDNSBool needLog = mDNStrue; - m->NATMcastRecvskt = mDNSPlatformUDPSocket(m, NATPMPAnnouncementPort); - if (!m->NATMcastRecvskt) - { - if (needLog) - { - LogMsg("CheckNATMappings: Failed to allocate port 5350 UDP multicast socket for NAT-PMP announcements"); - needLog = mDNSfalse; - } - } - else - needLog = mDNStrue; - } - } - else // else, we don't want to listen for announcements, so close them if they're open - { - if (m->NATMcastRecvskt) { mDNSPlatformUDPClose(m->NATMcastRecvskt); m->NATMcastRecvskt = mDNSNULL; } - if (m->SSDPSocket) { debugf("CheckNATMappings destroying SSDPSocket %p", &m->SSDPSocket); mDNSPlatformUDPClose(m->SSDPSocket); m->SSDPSocket = mDNSNULL; } - } - - if (!m->NATTraversals) - m->retryGetAddr = m->timenow + 0x78000000; - else - { - if (m->timenow - m->retryGetAddr >= 0) - { - err = uDNS_SendNATMsg(m, mDNSNULL); // Will also do UPnP discovery for us, if necessary - if (!err) - { - if (m->retryIntervalGetAddr < NATMAP_INIT_RETRY) m->retryIntervalGetAddr = NATMAP_INIT_RETRY; - else if (m->retryIntervalGetAddr < NATMAP_MAX_RETRY_INTERVAL / 2) m->retryIntervalGetAddr *= 2; - else m->retryIntervalGetAddr = NATMAP_MAX_RETRY_INTERVAL; - } - LogInfo("CheckNATMappings retryGetAddr sent address request err %d interval %d", err, m->retryIntervalGetAddr); - - // Always update m->retryGetAddr, even if we fail to send the packet. Otherwise in cases where we can't send the packet - // (like when we have no active interfaces) we'll spin in an infinite loop repeatedly failing to send the packet - m->retryGetAddr = m->timenow + m->retryIntervalGetAddr; - } - // Even when we didn't send the GetAddr packet, still need to make sure NextScheduledNATOp is set correctly - if (m->NextScheduledNATOp - m->retryGetAddr > 0) - m->NextScheduledNATOp = m->retryGetAddr; - } - - if (m->CurrentNATTraversal) LogMsg("WARNING m->CurrentNATTraversal already in use"); - m->CurrentNATTraversal = m->NATTraversals; - - while (m->CurrentNATTraversal) - { - NATTraversalInfo *cur = m->CurrentNATTraversal; - m->CurrentNATTraversal = m->CurrentNATTraversal->next; - - if (HaveRoutable) // If not RFC 1918 address, our own address and port are effectively our external address and port - { - cur->ExpiryTime = 0; - cur->NewResult = mStatus_NoError; - } - else if (cur->Protocol) // Check if it's time to send port mapping packets - { - if (m->timenow - cur->retryPortMap >= 0) // Time to do something with this mapping - { - if (cur->ExpiryTime && cur->ExpiryTime - m->timenow < 0) // Mapping has expired - { - cur->ExpiryTime = 0; - cur->retryInterval = NATMAP_INIT_RETRY; - } - - //LogMsg("uDNS_SendNATMsg"); - err = uDNS_SendNATMsg(m, cur); - - if (cur->ExpiryTime) // If have active mapping then set next renewal time halfway to expiry - NATSetNextRenewalTime(m, cur); - else // else no mapping; use exponential backoff sequence - { - if (cur->retryInterval < NATMAP_INIT_RETRY ) cur->retryInterval = NATMAP_INIT_RETRY; - else if (cur->retryInterval < NATMAP_MAX_RETRY_INTERVAL / 2) cur->retryInterval *= 2; - else cur->retryInterval = NATMAP_MAX_RETRY_INTERVAL; - cur->retryPortMap = m->timenow + cur->retryInterval; - } - } - - if (m->NextScheduledNATOp - cur->retryPortMap > 0) - m->NextScheduledNATOp = cur->retryPortMap; - } - - // Notify the client if necessary. We invoke the callback if: - // (1) we have an ExternalAddress, or we've tried and failed a couple of times to discover it - // and (2) the client doesn't want a mapping, or the client won't need a mapping, or the client has a successful mapping, or we've tried and failed a couple of times - // and (3) we have new data to give the client that's changed since the last callback - // Time line is: Send, Wait 500ms, Send, Wait 1sec, Send, Wait 2sec, Send - // At this point we've sent three requests without an answer, we've just sent our fourth request, - // retryIntervalGetAddr is now 4 seconds, which is greater than NATMAP_INIT_RETRY * 8 (2 seconds), - // so we return an error result to the caller. - if (!mDNSIPv4AddressIsZero(m->ExternalAddress) || m->retryIntervalGetAddr > NATMAP_INIT_RETRY * 8) - { - const mStatus EffectiveResult = cur->NewResult ? cur->NewResult : mDNSv4AddrIsRFC1918(&m->ExternalAddress) ? mStatus_DoubleNAT : mStatus_NoError; - const mDNSIPPort ExternalPort = HaveRoutable ? cur->IntPort : - !mDNSIPv4AddressIsZero(m->ExternalAddress) && cur->ExpiryTime ? cur->RequestedPort : zeroIPPort; - if (!cur->Protocol || HaveRoutable || cur->ExpiryTime || cur->retryInterval > NATMAP_INIT_RETRY * 8) - if (!mDNSSameIPv4Address(cur->ExternalAddress, m->ExternalAddress) || - !mDNSSameIPPort (cur->ExternalPort, ExternalPort) || - cur->Result != EffectiveResult) - { - //LogMsg("NAT callback %d %d %d", cur->Protocol, cur->ExpiryTime, cur->retryInterval); - if (cur->Protocol && mDNSIPPortIsZero(ExternalPort) && !mDNSIPv4AddressIsZero(m->Router.ip.v4)) - { - if (!EffectiveResult) - LogInfo("CheckNATMapping: Failed to obtain NAT port mapping %p from router %#a external address %.4a internal port %5d interval %d error %d", - cur, &m->Router, &m->ExternalAddress, mDNSVal16(cur->IntPort), cur->retryInterval, EffectiveResult); - else - LogMsg("CheckNATMapping: Failed to obtain NAT port mapping %p from router %#a external address %.4a internal port %5d interval %d error %d", - cur, &m->Router, &m->ExternalAddress, mDNSVal16(cur->IntPort), cur->retryInterval, EffectiveResult); - } - - cur->ExternalAddress = m->ExternalAddress; - cur->ExternalPort = ExternalPort; - cur->Lifetime = cur->ExpiryTime && !mDNSIPPortIsZero(ExternalPort) ? - (cur->ExpiryTime - m->timenow + mDNSPlatformOneSecond/2) / mDNSPlatformOneSecond : 0; - cur->Result = EffectiveResult; - mDNS_DropLockBeforeCallback(); // Allow client to legally make mDNS API calls from the callback - if (cur->clientCallback) - cur->clientCallback(m, cur); - mDNS_ReclaimLockAfterCallback(); // Decrement mDNS_reentrancy to block mDNS API calls again - // MUST NOT touch cur after invoking the callback - } - } - } - } - -mDNSlocal mDNSs32 CheckRecordUpdates(mDNS *m) - { - AuthRecord *rr; - mDNSs32 nextevent = m->timenow + 0x3FFFFFFF; - - CheckGroupRecordUpdates(m); - - for (rr = m->ResourceRecords; rr; rr = rr->next) - { - if (!AuthRecord_uDNS(rr)) continue; - if (rr->state == regState_NoTarget) {debugf("CheckRecordUpdates: Record %##s in NoTarget", rr->resrec.name->c); continue;} - // While we are waiting for the port mapping, we have nothing to do. The port mapping callback - // will take care of this - if (rr->state == regState_NATMap) {debugf("CheckRecordUpdates: Record %##s in NATMap", rr->resrec.name->c); continue;} - if (rr->state == regState_Pending || rr->state == regState_DeregPending || rr->state == regState_UpdatePending || - rr->state == regState_Refresh || rr->state == regState_Registered) - { - if (rr->LastAPTime + rr->ThisAPInterval - m->timenow <= 0) - { - if (rr->tcp) { DisposeTCPConn(rr->tcp); rr->tcp = mDNSNULL; } - if (!rr->nta || mDNSIPv4AddressIsZero(rr->nta->Addr.ip.v4)) - { - // Zero out the updateid so that if we have a pending response from the server, it won't - // be accepted as a valid response. If we accept the response, we might free the new "nta" - if (rr->nta) { rr->updateid = zeroID; CancelGetZoneData(m, rr->nta); } - rr->nta = StartGetZoneData(m, rr->resrec.name, ZoneServiceUpdate, RecordRegistrationGotZoneData, rr); - - // We have just started the GetZoneData. We need to wait for it to finish. SetRecordRetry here - // schedules the update timer to fire in the future. - // - // There are three cases. - // - // 1) When the updates are sent the first time, the first retry is intended to be at three seconds - // in the future. But by calling SetRecordRetry here we set it to nine seconds. But it does not - // matter because when the answer comes back, RecordRegistrationGotZoneData resets the interval - // back to INIT_RECORD_REG_INTERVAL. This also gives enough time for the query. - // - // 2) In the case of update errors (updateError), this causes further backoff as - // RecordRegistrationGotZoneData does not reset the timer. This is intentional as in the case of - // errors, we don't want to update aggressively. - // - // 3) We might be refreshing the update. This is very similar to case (1). RecordRegistrationGotZoneData - // resets it back to INIT_RECORD_REG_INTERVAL. - // - SetRecordRetry(m, rr, 0); - } - else if (rr->state == regState_DeregPending) SendRecordDeregistration(m, rr); - else SendRecordRegistration(m, rr); - } - } - if (nextevent - (rr->LastAPTime + rr->ThisAPInterval) > 0) - nextevent = (rr->LastAPTime + rr->ThisAPInterval); - } - return nextevent; - } - -mDNSexport void uDNS_Tasks(mDNS *const m) - { - mDNSs32 nexte; - DNSServer *d; - - m->NextuDNSEvent = m->timenow + 0x3FFFFFFF; - - nexte = CheckRecordUpdates(m); - if (m->NextuDNSEvent - nexte > 0) - m->NextuDNSEvent = nexte; - - for (d = m->DNSServers; d; d=d->next) - if (d->penaltyTime) - { - if (m->timenow - d->penaltyTime >= 0) - { - LogInfo("DNS server %#a:%d out of penalty box", &d->addr, mDNSVal16(d->port)); - d->penaltyTime = 0; - } - else - if (m->NextuDNSEvent - d->penaltyTime > 0) - m->NextuDNSEvent = d->penaltyTime; - } - - if (m->CurrentQuestion) - LogMsg("uDNS_Tasks ERROR m->CurrentQuestion already set: %##s (%s)", m->CurrentQuestion->qname.c, DNSTypeName(m->CurrentQuestion->qtype)); - m->CurrentQuestion = m->Questions; - while (m->CurrentQuestion && m->CurrentQuestion != m->NewQuestions) - { - DNSQuestion *const q = m->CurrentQuestion; - if (ActiveQuestion(q) && !mDNSOpaque16IsZero(q->TargetQID)) - { - uDNS_CheckCurrentQuestion(m); - if (q == m->CurrentQuestion) - if (m->NextuDNSEvent - NextQSendTime(q) > 0) - m->NextuDNSEvent = NextQSendTime(q); - } - // If m->CurrentQuestion wasn't modified out from under us, advance it now - // We can't do this at the start of the loop because uDNS_CheckCurrentQuestion() - // depends on having m->CurrentQuestion point to the right question - if (m->CurrentQuestion == q) - m->CurrentQuestion = q->next; - } - m->CurrentQuestion = mDNSNULL; - } - -// *************************************************************************** -#if COMPILER_LIKES_PRAGMA_MARK -#pragma mark - Startup, Shutdown, and Sleep -#endif - -mDNSexport void SleepRecordRegistrations(mDNS *m) - { - AuthRecord *rr; - for (rr = m->ResourceRecords; rr; rr=rr->next) - { - if (AuthRecord_uDNS(rr)) - { - // Zero out the updateid so that if we have a pending response from the server, it won't - // be accepted as a valid response. - if (rr->nta) { rr->updateid = zeroID; CancelGetZoneData(m, rr->nta); rr->nta = mDNSNULL; } - - if (rr->NATinfo.clientContext) - { - mDNS_StopNATOperation_internal(m, &rr->NATinfo); - rr->NATinfo.clientContext = mDNSNULL; - } - // We are waiting to update the resource record. The original data of the record is - // in OrigRData and the updated value is in InFlightRData. Free the old and the new - // one will be registered when we come back. - if (rr->state == regState_UpdatePending) - { - // act as if the update succeeded, since we're about to delete the name anyway - rr->state = regState_Registered; - // deallocate old RData - if (rr->UpdateCallback) rr->UpdateCallback(m, rr, rr->OrigRData, rr->OrigRDLen); - SetNewRData(&rr->resrec, rr->InFlightRData, rr->InFlightRDLen); - rr->OrigRData = mDNSNULL; - rr->InFlightRData = mDNSNULL; - } - - // If we have not begun the registration process i.e., never sent a registration packet, - // then uDNS_DeregisterRecord will not send a deregistration - uDNS_DeregisterRecord(m, rr); - - // When we wake, we call ActivateUnicastRegistration which starts at StartGetZoneData - } - } - } - -mDNSexport void mDNS_AddSearchDomain(const domainname *const domain, mDNSInterfaceID InterfaceID) - { - SearchListElem **p; - SearchListElem *tmp = mDNSNULL; - - // Check to see if we already have this domain in our list - for (p = &SearchList; *p; p = &(*p)->next) - if (((*p)->InterfaceID == InterfaceID) && SameDomainName(&(*p)->domain, domain)) - { - // If domain is already in list, and marked for deletion, unmark the delete - // Be careful not to touch the other flags that may be present - LogInfo("mDNS_AddSearchDomain already in list %##s", domain->c); - if ((*p)->flag & SLE_DELETE) (*p)->flag &= ~SLE_DELETE; - tmp = *p; - *p = tmp->next; - tmp->next = mDNSNULL; - break; - } - - - // move to end of list so that we maintain the same order - while (*p) p = &(*p)->next; - - if (tmp) *p = tmp; - else - { - // if domain not in list, add to list, mark as add (1) - *p = mDNSPlatformMemAllocate(sizeof(SearchListElem)); - if (!*p) { LogMsg("ERROR: mDNS_AddSearchDomain - malloc"); return; } - mDNSPlatformMemZero(*p, sizeof(SearchListElem)); - AssignDomainName(&(*p)->domain, domain); - (*p)->next = mDNSNULL; - (*p)->InterfaceID = InterfaceID; - LogInfo("mDNS_AddSearchDomain created new %##s, InterfaceID %p", domain->c, InterfaceID); - } - } - -mDNSlocal void FreeARElemCallback(mDNS *const m, AuthRecord *const rr, mStatus result) - { - (void)m; // unused - if (result == mStatus_MemFree) mDNSPlatformMemFree(rr->RecordContext); - } - -mDNSlocal void FoundDomain(mDNS *const m, DNSQuestion *question, const ResourceRecord *const answer, QC_result AddRecord) - { - SearchListElem *slElem = question->QuestionContext; - mStatus err; - const char *name; - - if (answer->rrtype != kDNSType_PTR) return; - if (answer->RecordType == kDNSRecordTypePacketNegative) return; - if (answer->InterfaceID == mDNSInterface_LocalOnly) return; - - if (question == &slElem->BrowseQ) name = mDNS_DomainTypeNames[mDNS_DomainTypeBrowse]; - else if (question == &slElem->DefBrowseQ) name = mDNS_DomainTypeNames[mDNS_DomainTypeBrowseDefault]; - else if (question == &slElem->AutomaticBrowseQ) name = mDNS_DomainTypeNames[mDNS_DomainTypeBrowseAutomatic]; - else if (question == &slElem->RegisterQ) name = mDNS_DomainTypeNames[mDNS_DomainTypeRegistration]; - else if (question == &slElem->DefRegisterQ) name = mDNS_DomainTypeNames[mDNS_DomainTypeRegistrationDefault]; - else { LogMsg("FoundDomain - unknown question"); return; } - - LogInfo("FoundDomain: %p %s %s Q %##s A %s", answer->InterfaceID, AddRecord ? "Add" : "Rmv", name, question->qname.c, RRDisplayString(m, answer)); - - if (AddRecord) - { - ARListElem *arElem = mDNSPlatformMemAllocate(sizeof(ARListElem)); - if (!arElem) { LogMsg("ERROR: FoundDomain out of memory"); return; } - mDNS_SetupResourceRecord(&arElem->ar, mDNSNULL, mDNSInterface_LocalOnly, kDNSType_PTR, 7200, kDNSRecordTypeShared, AuthRecordLocalOnly, FreeARElemCallback, arElem); - MakeDomainNameFromDNSNameString(&arElem->ar.namestorage, name); - AppendDNSNameString (&arElem->ar.namestorage, "local"); - AssignDomainName(&arElem->ar.resrec.rdata->u.name, &answer->rdata->u.name); - LogInfo("FoundDomain: Registering %s", ARDisplayString(m, &arElem->ar)); - err = mDNS_Register(m, &arElem->ar); - if (err) { LogMsg("ERROR: FoundDomain - mDNS_Register returned %d", err); mDNSPlatformMemFree(arElem); return; } - arElem->next = slElem->AuthRecs; - slElem->AuthRecs = arElem; - } - else - { - ARListElem **ptr = &slElem->AuthRecs; - while (*ptr) - { - if (SameDomainName(&(*ptr)->ar.resrec.rdata->u.name, &answer->rdata->u.name)) - { - ARListElem *dereg = *ptr; - *ptr = (*ptr)->next; - LogInfo("FoundDomain: Deregistering %s", ARDisplayString(m, &dereg->ar)); - err = mDNS_Deregister(m, &dereg->ar); - if (err) LogMsg("ERROR: FoundDomain - mDNS_Deregister returned %d", err); - // Memory will be freed in the FreeARElemCallback - } - else - ptr = &(*ptr)->next; - } - } - } - -#if APPLE_OSX_mDNSResponder && MACOSX_MDNS_MALLOC_DEBUGGING -mDNSexport void udns_validatelists(void *const v) - { - mDNS *const m = v; - - NATTraversalInfo *n; - for (n = m->NATTraversals; n; n=n->next) - if (n->next == (NATTraversalInfo *)~0 || n->clientCallback == (NATTraversalClientCallback)~0) - LogMemCorruption("m->NATTraversals: %p is garbage", n); - - DNSServer *d; - for (d = m->DNSServers; d; d=d->next) - if (d->next == (DNSServer *)~0 || d->teststate > DNSServer_Disabled) - LogMemCorruption("m->DNSServers: %p is garbage (%d)", d, d->teststate); - - DomainAuthInfo *info; - for (info = m->AuthInfoList; info; info = info->next) - if (info->next == (DomainAuthInfo *)~0 || info->AutoTunnel == (const char*)~0) - LogMemCorruption("m->AuthInfoList: %p is garbage (%X)", info, info->AutoTunnel); - - HostnameInfo *hi; - for (hi = m->Hostnames; hi; hi = hi->next) - if (hi->next == (HostnameInfo *)~0 || hi->StatusCallback == (mDNSRecordCallback*)~0) - LogMemCorruption("m->Hostnames: %p is garbage", n); - - SearchListElem *ptr; - for (ptr = SearchList; ptr; ptr = ptr->next) - if (ptr->next == (SearchListElem *)~0 || ptr->AuthRecs == (void*)~0) - LogMemCorruption("SearchList: %p is garbage (%X)", ptr, ptr->AuthRecs); - } -#endif - -// This should probably move to the UDS daemon -- the concept of legacy clients and automatic registration / automatic browsing -// is really a UDS API issue, not something intrinsic to uDNS - -mDNSexport mStatus uDNS_SetupSearchDomains(mDNS *const m, int action) - { - SearchListElem **p = &SearchList, *ptr; - mStatus err; - - // step 1: mark each element for removal - for (ptr = SearchList; ptr; ptr = ptr->next) ptr->flag |= SLE_DELETE; - - // Make sure we have the search domains from the platform layer so that if we start the WAB - // queries below, we have the latest information - mDNS_Lock(m); - mDNSPlatformSetDNSConfig(m, mDNSfalse, mDNStrue, mDNSNULL, mDNSNULL, mDNSNULL); - mDNS_Unlock(m); - - if (action & UDNS_START_WAB_QUERY) - m->StartWABQueries = mDNStrue; - - // delete elems marked for removal, do queries for elems marked add - while (*p) - { - ptr = *p; - LogInfo("uDNS_SetupSearchDomains:action %d: Flags %d, AuthRecs %p, InterfaceID %p %##s", action, ptr->flag, ptr->AuthRecs, ptr->InterfaceID, ptr->domain.c); - if (ptr->flag & SLE_DELETE) - { - ARListElem *arList = ptr->AuthRecs; - ptr->AuthRecs = mDNSNULL; - *p = ptr->next; - - // If the user has "local" in their DNS searchlist, we ignore that for the purposes of domain enumeration queries - // We suppressed the domain enumeration for scoped search domains below. When we enable that - // enable this. - if ((ptr->flag & SLE_WAB_QUERY_STARTED) && - !SameDomainName(&ptr->domain, &localdomain) && (ptr->InterfaceID == mDNSInterface_Any)) - { - mDNS_StopGetDomains(m, &ptr->BrowseQ); - mDNS_StopGetDomains(m, &ptr->RegisterQ); - mDNS_StopGetDomains(m, &ptr->DefBrowseQ); - mDNS_StopGetDomains(m, &ptr->DefRegisterQ); - mDNS_StopGetDomains(m, &ptr->AutomaticBrowseQ); - } - - mDNSPlatformMemFree(ptr); - - // deregister records generated from answers to the query - while (arList) - { - ARListElem *dereg = arList; - arList = arList->next; - debugf("Deregistering PTR %##s -> %##s", dereg->ar.resrec.name->c, dereg->ar.resrec.rdata->u.name.c); - err = mDNS_Deregister(m, &dereg->ar); - if (err) LogMsg("uDNS_SetupSearchDomains:: ERROR!! mDNS_Deregister returned %d", err); - // Memory will be freed in the FreeARElemCallback - } - continue; - } - - if ((action & UDNS_START_WAB_QUERY) && !(ptr->flag & SLE_WAB_QUERY_STARTED)) - { - // If the user has "local" in their DNS searchlist, we ignore that for the purposes of domain enumeration queries. - // Also, suppress the domain enumeration for scoped search domains for now until there is a need. - if (!SameDomainName(&ptr->domain, &localdomain) && (ptr->InterfaceID == mDNSInterface_Any)) - { - mStatus err1, err2, err3, err4, err5; - err1 = mDNS_GetDomains(m, &ptr->BrowseQ, mDNS_DomainTypeBrowse, &ptr->domain, ptr->InterfaceID, FoundDomain, ptr); - err2 = mDNS_GetDomains(m, &ptr->DefBrowseQ, mDNS_DomainTypeBrowseDefault, &ptr->domain, ptr->InterfaceID, FoundDomain, ptr); - err3 = mDNS_GetDomains(m, &ptr->RegisterQ, mDNS_DomainTypeRegistration, &ptr->domain, ptr->InterfaceID, FoundDomain, ptr); - err4 = mDNS_GetDomains(m, &ptr->DefRegisterQ, mDNS_DomainTypeRegistrationDefault, &ptr->domain, ptr->InterfaceID, FoundDomain, ptr); - err5 = mDNS_GetDomains(m, &ptr->AutomaticBrowseQ, mDNS_DomainTypeBrowseAutomatic, &ptr->domain, ptr->InterfaceID, FoundDomain, ptr); - if (err1 || err2 || err3 || err4 || err5) - LogMsg("uDNS_SetupSearchDomains: GetDomains for domain %##s returned error(s):\n" - "%d (mDNS_DomainTypeBrowse)\n" - "%d (mDNS_DomainTypeBrowseDefault)\n" - "%d (mDNS_DomainTypeRegistration)\n" - "%d (mDNS_DomainTypeRegistrationDefault)" - "%d (mDNS_DomainTypeBrowseAutomatic)\n", - ptr->domain.c, err1, err2, err3, err4, err5); - ptr->flag |= SLE_WAB_QUERY_STARTED; - } - } - - p = &ptr->next; - } - return mStatus_NoError; - } - -mDNSexport domainname *uDNS_GetNextSearchDomain(mDNS *const m, mDNSInterfaceID InterfaceID, mDNSs8 *searchIndex, mDNSBool ignoreDotLocal) - { - SearchListElem *p = SearchList; - int count = *searchIndex; - (void) m; // unused - - if (count < 0) { LogMsg("uDNS_GetNextSearchDomain: count %d less than zero", count); return mDNSNULL; } - - // skip the domains that we already looked at before - for (; count; count--) p = p->next; - - while (p) - { - int labels = CountLabels(&p->domain); - if (labels > 0) - { - const domainname *d = SkipLeadingLabels(&p->domain, labels - 1); - if (SameDomainLabel(d->c, (const mDNSu8 *)"\x4""arpa")) - { - LogInfo("uDNS_GetNextSearchDomain: skipping search domain %##s, InterfaceID %p", p->domain.c, p->InterfaceID); - (*searchIndex)++; - p = p->next; - continue; - } - if (ignoreDotLocal && SameDomainLabel(d->c, (const mDNSu8 *)"\x5""local")) - { - LogInfo("uDNS_GetNextSearchDomain: skipping local domain %##s, InterfaceID %p", p->domain.c, p->InterfaceID); - (*searchIndex)++; - p = p->next; - continue; - } - } - // Point to the next one in the list which we will look at next time. - (*searchIndex)++; - // When we are appending search domains in a ActiveDirectory domain, the question's InterfaceID - // set to mDNSInterface_Unicast. Match the unscoped entries in that case. - if (((InterfaceID == mDNSInterface_Unicast) && (p->InterfaceID == mDNSInterface_Any)) || - p->InterfaceID == InterfaceID) - { - LogInfo("uDNS_GetNextSearchDomain returning domain %##s, InterfaceID %p", p->domain.c, p->InterfaceID); - return &p->domain; - } - LogInfo("uDNS_GetNextSearchDomain skipping domain %##s, InterfaceID %p", p->domain.c, p->InterfaceID); - p = p->next; - } - return mDNSNULL; - } - -mDNSlocal void FlushAddressCacheRecords(mDNS *const m) - { - mDNSu32 slot; - CacheGroup *cg; - CacheRecord *cr; - FORALL_CACHERECORDS(slot, cg, cr) - { - if (cr->resrec.InterfaceID) continue; - - // If a resource record can answer A or AAAA, they need to be flushed so that we will - // never used to deliver an ADD or RMV - if (RRTypeAnswersQuestionType(&cr->resrec, kDNSType_A) || - RRTypeAnswersQuestionType(&cr->resrec, kDNSType_AAAA)) - { - LogInfo("FlushAddressCacheRecords: Purging Resourcerecord %s", CRDisplayString(m, cr)); - mDNS_PurgeCacheResourceRecord(m, cr); - } - } - } - -// Retry questions which has seach domains appended -mDNSexport void RetrySearchDomainQuestions(mDNS *const m) - { - // Purge all the A/AAAA cache records and restart the queries. mDNSCoreRestartAddressQueries - // does this. When we restart the question, we first want to try the new search domains rather - // than use the entries that is already in the cache. When we appended search domains, we might - // have created cache entries which is no longer valid as there are new search domains now - - LogInfo("RetrySearchDomainQuestions: Calling mDNSCoreRestartAddressQueries"); - mDNSCoreRestartAddressQueries(m, mDNStrue, FlushAddressCacheRecords, mDNSNULL, mDNSNULL); - } - -// Construction of Default Browse domain list (i.e. when clients pass NULL) is as follows: -// 1) query for b._dns-sd._udp.local on LocalOnly interface -// (.local manually generated via explicit callback) -// 2) for each search domain (from prefs pane), query for b._dns-sd._udp.. -// 3) for each result from (2), register LocalOnly PTR record b._dns-sd._udp.local. -> -// 4) result above should generate a callback from question in (1). result added to global list -// 5) global list delivered to client via GetSearchDomainList() -// 6) client calls to enumerate domains now go over LocalOnly interface -// (!!!KRS may add outgoing interface in addition) - -struct CompileTimeAssertionChecks_uDNS - { - // Check our structures are reasonable sizes. Including overly-large buffers, or embedding - // other overly-large structures instead of having a pointer to them, can inadvertently - // cause structure sizes (and therefore memory usage) to balloon unreasonably. - char sizecheck_tcpInfo_t [(sizeof(tcpInfo_t) <= 9056) ? 1 : -1]; - char sizecheck_SearchListElem[(sizeof(SearchListElem) <= 5000) ? 1 : -1]; - }; diff -Nru qtcreator-2.5.0/src/tools/mdnssd/uDNS.h qtcreator-2.5.2/src/tools/mdnssd/uDNS.h --- qtcreator-2.5.0/src/tools/mdnssd/uDNS.h 2012-05-09 12:13:19.000000000 +0000 +++ qtcreator-2.5.2/src/tools/mdnssd/uDNS.h 1970-01-01 00:00:00.000000000 +0000 @@ -1,136 +0,0 @@ -/* -*- Mode: C; tab-width: 4 -*- - * - * Copyright (c) 2002-2003 Apple Computer, Inc. All rights reserved. - * - * Licensed under the Apache License, Version 2.0 (the "License"); - * you may not use this file except in compliance with the License. - * You may obtain a copy of the License at - * - * http://www.apache.org/licenses/LICENSE-2.0 - * - * Unless required by applicable law or agreed to in writing, software - * distributed under the License is distributed on an "AS IS" BASIS, - * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. - * See the License for the specific language governing permissions and - * limitations under the License. - */ - -#ifndef __UDNS_H_ -#define __UDNS_H_ - -#include "mDNSEmbeddedAPI.h" -#include "DNSCommon.h" - -#ifdef __cplusplus - extern "C" { -#endif - -#define RESTART_GOODBYE_DELAY (6 * mDNSPlatformOneSecond) // delay after restarting LLQ before nuking previous known answers (avoids flutter if we restart before we have networking up) -#define INIT_UCAST_POLL_INTERVAL (3 * mDNSPlatformOneSecond) // this interval is used after send failures on network transitions - // which typically heal quickly, so we start agressively and exponentially back off -#define MAX_UCAST_POLL_INTERVAL (60 * 60 * mDNSPlatformOneSecond) -//#define MAX_UCAST_POLL_INTERVAL (1 * 60 * mDNSPlatformOneSecond) -#define LLQ_POLL_INTERVAL (15 * 60 * mDNSPlatformOneSecond) // Polling interval for zones w/ an advertised LLQ port (ie not static zones) if LLQ fails due to NAT, etc. -#define RESPONSE_WINDOW (60 * mDNSPlatformOneSecond) // require server responses within one minute of request -#define MAX_UCAST_UNANSWERED_QUERIES 2 // the number of unanswered queries from any one uDNS server before trying another server -#define DNSSERVER_PENALTY_TIME (60 * mDNSPlatformOneSecond) // number of seconds for which new questions don't pick this server - -// On some interfaces, we want to delay the first retransmission to a minimum of 2 seconds -// rather than the default (1 second). -#define MIN_UCAST_RETRANS_TIMEOUT (2 * mDNSPlatformOneSecond) - -#define DEFAULT_UPDATE_LEASE 7200 - -#define QuestionIntervalStep 3 -#define QuestionIntervalStep2 (QuestionIntervalStep*QuestionIntervalStep) -#define QuestionIntervalStep3 (QuestionIntervalStep*QuestionIntervalStep*QuestionIntervalStep) -#define InitialQuestionInterval ((mDNSPlatformOneSecond + QuestionIntervalStep-1) / QuestionIntervalStep) - -// For Unicast record registrations, we initialize the interval to 1 second. When we send any query for -// the record registration e.g., GetZoneData, we always back off by QuestionIntervalStep -// so that the first retry does not happen until 3 seconds which should be enough for TCP/TLS to be done. -#define INIT_RECORD_REG_INTERVAL (1 * mDNSPlatformOneSecond) -#define MAX_RECORD_REG_INTERVAL (15 * 60 * mDNSPlatformOneSecond) -#define MERGE_DELAY_TIME (1 * mDNSPlatformOneSecond) - -// If we are refreshing, we do it at least 5 times with a min update frequency of -// 5 minutes -#define MAX_UPDATE_REFRESH_COUNT 5 -#define MIN_UPDATE_REFRESH_TIME (5 * 60 * mDNSPlatformOneSecond) - -// For questions that use kDNSServiceFlagsTimeout and we don't have a matching resolver e.g., no dns servers, -// then use the default value of 30 seconds -#define DEFAULT_UDNS_TIMEOUT 30 // in seconds - -// Entry points into unicast-specific routines - -extern void LLQGotZoneData(mDNS *const m, mStatus err, const ZoneData *zoneInfo); -extern void startLLQHandshake(mDNS *m, DNSQuestion *q); -extern void sendLLQRefresh(mDNS *m, DNSQuestion *q); - -extern void SleepRecordRegistrations(mDNS *m); - -// uDNS_UpdateRecord -// following fields must be set, and the update validated, upon entry. -// rr->NewRData -// rr->newrdlength -// rr->UpdateCallback - -extern mStatus uDNS_UpdateRecord(mDNS *m, AuthRecord *rr); - -extern void SetNextQueryTime(mDNS *const m, const DNSQuestion *const q); -extern CacheGroup *CacheGroupForName(const mDNS *const m, const mDNSu32 slot, const mDNSu32 namehash, const domainname *const name); -extern mStatus mDNS_Register_internal(mDNS *const m, AuthRecord *const rr); -extern mStatus mDNS_Deregister_internal(mDNS *const m, AuthRecord *const rr, mDNS_Dereg_type drt); -extern mStatus mDNS_StartQuery_internal(mDNS *const m, DNSQuestion *const question); -extern mStatus mDNS_StopQuery_internal(mDNS *const m, DNSQuestion *const question); -extern mStatus mDNS_StartNATOperation_internal(mDNS *const m, NATTraversalInfo *traversal); - -extern void RecordRegistrationGotZoneData(mDNS *const m, mStatus err, const ZoneData *zoneData); -extern mStatus uDNS_DeregisterRecord(mDNS *const m, AuthRecord *const rr); -extern const domainname *GetServiceTarget(mDNS *m, AuthRecord *const rr); -extern void uDNS_CheckCurrentQuestion(mDNS *const m); - -// integer fields of msg header must be in HOST byte order before calling this routine -extern void uDNS_ReceiveMsg(mDNS *const m, DNSMessage *const msg, const mDNSu8 *const end, - const mDNSAddr *const srcaddr, const mDNSIPPort srcport); - -extern void uDNS_Tasks(mDNS *const m); -extern void UpdateAllSRVRecords(mDNS *m); -extern void CheckNATMappings(mDNS *m); - -extern mStatus uDNS_SetupDNSConfig(mDNS *const m); - -// uDNS_SetupSearchDomains by default adds search domains. It also can be called with one or -// more values for "action" which does the following: -// -// -UDNS_START_WAB_QUERY - start Wide Area Bonjour (domain enumeration) queries - -#define UDNS_START_WAB_QUERY 0x00000001 - -extern mStatus uDNS_SetupSearchDomains(mDNS *const m, int action); -extern domainname *uDNS_GetNextSearchDomain(mDNS *const m, mDNSInterfaceID InterfaceID, mDNSs8 *searchIndex, mDNSBool ignoreDotLocal); - -typedef enum - { - uDNS_LLQ_Not = 0, // Normal uDNS answer: Flush any stale records from cache, and respect record TTL - uDNS_LLQ_Ignore, // LLQ initial challenge packet: ignore -- has no useful records for us - uDNS_LLQ_Entire, // LLQ initial set of answers: Flush any stale records from cache, but assume TTL is 2 x LLQ refresh interval - uDNS_LLQ_Events // LLQ event packet: don't flush cache; assume TTL is 2 x LLQ refresh interval - } uDNS_LLQType; - -extern uDNS_LLQType uDNS_recvLLQResponse(mDNS *const m, const DNSMessage *const msg, const mDNSu8 *const end, const mDNSAddr *const srcaddr, const mDNSIPPort srcport, DNSQuestion **matchQuestion); -extern DomainAuthInfo *GetAuthInfoForName_internal(mDNS *m, const domainname *const name); -extern DomainAuthInfo *GetAuthInfoForQuestion(mDNS *m, const DNSQuestion *const q); -extern void DisposeTCPConn(struct tcpInfo_t *tcp); - -// NAT traversal -extern void uDNS_ReceiveNATPMPPacket(mDNS *m, const mDNSInterfaceID InterfaceID, mDNSu8 *pkt, mDNSu16 len); // Called for each received NAT-PMP packet -extern void natTraversalHandleAddressReply(mDNS *const m, mDNSu16 err, mDNSv4Addr ExtAddr); -extern void natTraversalHandlePortMapReply(mDNS *const m, NATTraversalInfo *n, const mDNSInterfaceID InterfaceID, mDNSu16 err, mDNSIPPort extport, mDNSu32 lease); - -#ifdef __cplusplus - } -#endif - -#endif // __UDNS_H_ diff -Nru qtcreator-2.5.0/src/tools/mdnssd/uds_daemon.c qtcreator-2.5.2/src/tools/mdnssd/uds_daemon.c --- qtcreator-2.5.0/src/tools/mdnssd/uds_daemon.c 2012-05-09 12:13:19.000000000 +0000 +++ qtcreator-2.5.2/src/tools/mdnssd/uds_daemon.c 1970-01-01 00:00:00.000000000 +0000 @@ -1,4716 +0,0 @@ -/* -*- Mode: C; tab-width: 4 -*- - * - * Copyright (c) 2003-2006 Apple Computer, Inc. All rights reserved. - * - * Licensed under the Apache License, Version 2.0 (the "License"); - * you may not use this file except in compliance with the License. - * You may obtain a copy of the License at - * - * http://www.apache.org/licenses/LICENSE-2.0 - * - * Unless required by applicable law or agreed to in writing, software - * distributed under the License is distributed on an "AS IS" BASIS, - * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. - * See the License for the specific language governing permissions and - * limitations under the License. - */ - -#if defined(_WIN32) -#include -#define usleep(X) Sleep(((X)+999)/1000) -#else -#include -#include -#include -#include -#include -#include -#endif - -#include -#include - -#include "mDNSEmbeddedAPI.h" -#include "DNSCommon.h" -#include "uDNS.h" -#include "uds_daemon.h" - -// Normally we append search domains only for queries with a single label that are not -// fully qualified. This can be overridden to apply search domains for queries (that are -// not fully qualified) with any number of labels e.g., moon, moon.cs, moon.cs.be, etc. -mDNSBool AlwaysAppendSearchDomains = mDNSfalse; - -// Apple-specific functionality, not required for other platforms -#if APPLE_OSX_mDNSResponder -#include -#ifndef PID_FILE -#define PID_FILE "" -#endif -#endif - -#if APPLE_OSX_mDNSResponder -#include - -#if ! NO_WCF - -int WCFIsServerRunning(WCFConnection *conn) __attribute__((weak_import)); -int WCFNameResolvesToAddr(WCFConnection *conn, char* domainName, struct sockaddr* address, uid_t userid) __attribute__((weak_import)); -int WCFNameResolvesToName(WCFConnection *conn, char* fromName, char* toName, uid_t userid) __attribute__((weak_import)); - -// Do we really need to define a macro for "if"? -#define CHECK_WCF_FUNCTION(X) if (X) -#endif // ! NO_WCF - -#else -#define NO_WCF 1 -#endif // APPLE_OSX_mDNSResponder - -// User IDs 0-500 are system-wide processes, not actual users in the usual sense -// User IDs for real user accounts start at 501 and count up from there -#define SystemUID(X) ((X) <= 500) - -// *************************************************************************** -#if COMPILER_LIKES_PRAGMA_MARK -#pragma mark - -#pragma mark - Types and Data Structures -#endif - -typedef enum - { - t_uninitialized, - t_morecoming, - t_complete, - t_error, - t_terminated - } transfer_state; - -typedef struct request_state request_state; - -typedef void (*req_termination_fn)(request_state *request); - -typedef struct registered_record_entry - { - struct registered_record_entry *next; - mDNSu32 key; - client_context_t regrec_client_context; - request_state *request; - mDNSBool external_advertise; - mDNSInterfaceID origInterfaceID; - AuthRecord *rr; // Pointer to variable-sized AuthRecord (Why a pointer? Why not just embed it here?) - } registered_record_entry; - -// A single registered service: ServiceRecordSet + bookkeeping -// Note that we duplicate some fields from parent service_info object -// to facilitate cleanup, when instances and parent may be deallocated at different times. -typedef struct service_instance - { - struct service_instance *next; - request_state *request; - AuthRecord *subtypes; - mDNSBool renameonmemfree; // Set on config change when we deregister original name - mDNSBool clientnotified; // Has client been notified of successful registration yet? - mDNSBool default_local; // is this the "local." from an empty-string registration? - mDNSBool external_advertise; // is this is being advertised externally? - domainname domain; - ServiceRecordSet srs; // note -- variable-sized object -- must be last field in struct - } service_instance; - -// for multi-domain default browsing -typedef struct browser_t - { - struct browser_t *next; - domainname domain; - DNSQuestion q; - } browser_t; - -struct request_state - { - request_state *next; - request_state *primary; // If this operation is on a shared socket, pointer to primary - // request_state for the original DNSServiceCreateConnection() operation - dnssd_sock_t sd; - dnssd_sock_t errsd; - mDNSu32 uid; - void * platform_data; - - // Note: On a shared connection these fields in the primary structure, including hdr, are re-used - // for each new request. This is because, until we've read the ipc_msg_hdr to find out what the - // operation is, we don't know if we're going to need to allocate a new request_state or not. - transfer_state ts; - mDNSu32 hdr_bytes; // bytes of header already read - ipc_msg_hdr hdr; - mDNSu32 data_bytes; // bytes of message data already read - char *msgbuf; // pointer to data storage to pass to free() - const char *msgptr; // pointer to data to be read from (may be modified) - char *msgend; // pointer to byte after last byte of message - - // reply, termination, error, and client context info - int no_reply; // don't send asynchronous replies to client - mDNSs32 time_blocked; // record time of a blocked client - int unresponsiveness_reports; - struct reply_state *replies; // corresponding (active) reply list - req_termination_fn terminate; - DNSServiceFlags flags; - - union - { - registered_record_entry *reg_recs; // list of registrations for a connection-oriented request - struct - { - mDNSInterfaceID interface_id; - mDNSBool default_domain; - mDNSBool ForceMCast; - domainname regtype; - browser_t *browsers; - } browser; - struct - { - mDNSInterfaceID InterfaceID; - mDNSu16 txtlen; - void *txtdata; - mDNSIPPort port; - domainlabel name; - char type_as_string[MAX_ESCAPED_DOMAIN_NAME]; - domainname type; - mDNSBool default_domain; - domainname host; - mDNSBool autoname; // Set if this name is tied to the Computer Name - mDNSBool autorename; // Set if this client wants us to automatically rename on conflict - mDNSBool allowremotequery; // Respond to unicast queries from outside the local link? - int num_subtypes; - service_instance *instances; - } servicereg; - struct - { - mDNSInterfaceID interface_id; - mDNSu32 flags; - mDNSu32 protocol; - DNSQuestion q4; - DNSQuestion *q42; - DNSQuestion q6; - DNSQuestion *q62; - } addrinfo; - struct - { - mDNSIPPort ReqExt; // External port we originally requested, for logging purposes - NATTraversalInfo NATinfo; - } pm; - struct - { -#if 0 - DNSServiceFlags flags; -#endif - DNSQuestion q_all; - DNSQuestion q_default; - } enumeration; - struct - { - DNSQuestion q; - DNSQuestion *q2; - } queryrecord; - struct - { - DNSQuestion qtxt; - DNSQuestion qsrv; - const ResourceRecord *txt; - const ResourceRecord *srv; - mDNSs32 ReportTime; - mDNSBool external_advertise; - } resolve; - } u; - }; - -// struct physically sits between ipc message header and call-specific fields in the message buffer -typedef struct - { - DNSServiceFlags flags; // Note: This field is in NETWORK byte order - mDNSu32 ifi; // Note: This field is in NETWORK byte order - DNSServiceErrorType error; // Note: This field is in NETWORK byte order - } reply_hdr; - -typedef struct reply_state - { - struct reply_state *next; // If there are multiple unsent replies - mDNSu32 totallen; - mDNSu32 nwriten; - ipc_msg_hdr mhdr[1]; - reply_hdr rhdr[1]; - } reply_state; - -// *************************************************************************** -#if COMPILER_LIKES_PRAGMA_MARK -#pragma mark - -#pragma mark - Globals -#endif - -// globals -mDNSexport mDNS mDNSStorage; -mDNSexport const char ProgramName[] = "mDNSResponder"; - -static dnssd_sock_t listenfd = dnssd_InvalidSocket; -static request_state *all_requests = NULL; - -// Note asymmetry here between registration and browsing. -// For service registrations we only automatically register in domains that explicitly appear in local configuration data -// (so AutoRegistrationDomains could equally well be called SCPrefRegDomains) -// For service browsing we also learn automatic browsing domains from the network, so for that case we have: -// 1. SCPrefBrowseDomains (local configuration data) -// 2. LocalDomainEnumRecords (locally-generated local-only PTR records -- equivalent to slElem->AuthRecs in uDNS.c) -// 3. AutoBrowseDomains, which is populated by tracking add/rmv events in AutomaticBrowseDomainChange, the callback function for our mDNS_GetDomains call. -// By creating and removing our own LocalDomainEnumRecords, we trigger AutomaticBrowseDomainChange callbacks just like domains learned from the network would. - -mDNSexport DNameListElem *AutoRegistrationDomains; // Domains where we automatically register for empty-string registrations - -static DNameListElem *SCPrefBrowseDomains; // List of automatic browsing domains read from SCPreferences for "empty string" browsing -static ARListElem *LocalDomainEnumRecords; // List of locally-generated PTR records to augment those we learn from the network -mDNSexport DNameListElem *AutoBrowseDomains; // List created from those local-only PTR records plus records we get from the network - -#define MSG_PAD_BYTES 5 // pad message buffer (read from client) with n zero'd bytes to guarantee - // n get_string() calls w/o buffer overrun -// initialization, setup/teardown functions - -// If a platform specifies its own PID file name, we use that -#ifndef PID_FILE -#define PID_FILE "/var/run/mDNSResponder.pid" -#endif - -// *************************************************************************** -#if COMPILER_LIKES_PRAGMA_MARK -#pragma mark - -#pragma mark - General Utility Functions -#endif - -mDNSlocal void FatalError(char *errmsg) - { - LogMsg("%s: %s", errmsg, dnssd_strerror(dnssd_errno)); - *(long*)0 = 0; // On OS X abort() doesn't generate a crash log, but writing to zero does - abort(); // On platforms where writing to zero doesn't generate an exception, abort instead - } - -mDNSlocal mDNSu32 dnssd_htonl(mDNSu32 l) - { - mDNSu32 ret; - char *data = (char*) &ret; - put_uint32(l, &data); - return ret; - } - -// hack to search-replace perror's to LogMsg's -mDNSlocal void my_perror(char *errmsg) - { - LogMsg("%s: %d (%s)", errmsg, dnssd_errno, dnssd_strerror(dnssd_errno)); - } - -mDNSlocal void abort_request(request_state *req) - { - if (req->terminate == (req_termination_fn)~0) - { LogMsg("abort_request: ERROR: Attempt to abort operation %p with req->terminate %p", req, req->terminate); return; } - - // First stop whatever mDNSCore operation we were doing - // If this is actually a shared connection operation, then its req->terminate function will scan - // the all_requests list and terminate any subbordinate operations sharing this file descriptor - if (req->terminate) req->terminate(req); - - if (!dnssd_SocketValid(req->sd)) - { LogMsg("abort_request: ERROR: Attempt to abort operation %p with invalid fd %d", req, req->sd); return; } - - // Now, if this request_state is not subordinate to some other primary, close file descriptor and discard replies - if (!req->primary) - { - if (req->errsd != req->sd) LogOperation("%3d: Removing FD and closing errsd %d", req->sd, req->errsd); - else LogOperation("%3d: Removing FD", req->sd); - udsSupportRemoveFDFromEventLoop(req->sd, req->platform_data); // Note: This also closes file descriptor req->sd for us - if (req->errsd != req->sd) { dnssd_close(req->errsd); req->errsd = req->sd; } - - while (req->replies) // free pending replies - { - reply_state *ptr = req->replies; - req->replies = req->replies->next; - freeL("reply_state (abort)", ptr); - } - } - - // Set req->sd to something invalid, so that udsserver_idle knows to unlink and free this structure -#if APPLE_OSX_mDNSResponder && MACOSX_MDNS_MALLOC_DEBUGGING - // Don't use dnssd_InvalidSocket (-1) because that's the sentinel value MACOSX_MDNS_MALLOC_DEBUGGING uses - // for detecting when the memory for an object is inadvertently freed while the object is still on some list - req->sd = req->errsd = -2; -#else - req->sd = req->errsd = dnssd_InvalidSocket; -#endif - // We also set req->terminate to a bogus value so we know if abort_request() gets called again for this request - req->terminate = (req_termination_fn)~0; - } - -mDNSlocal void AbortUnlinkAndFree(request_state *req) - { - request_state **p = &all_requests; - abort_request(req); - while (*p && *p != req) p=&(*p)->next; - if (*p) { *p = req->next; freeL("request_state/AbortUnlinkAndFree", req); } - else LogMsg("AbortUnlinkAndFree: ERROR: Attempt to abort operation %p not in list", req); - } - -mDNSlocal reply_state *create_reply(const reply_op_t op, const size_t datalen, request_state *const request) - { - reply_state *reply; - - if ((unsigned)datalen < sizeof(reply_hdr)) - { - LogMsg("ERROR: create_reply - data length less than length of required fields"); - return NULL; - } - - reply = mallocL("reply_state", sizeof(reply_state) + datalen - sizeof(reply_hdr)); - if (!reply) FatalError("ERROR: malloc"); - - reply->next = mDNSNULL; - reply->totallen = (mDNSu32)datalen + sizeof(ipc_msg_hdr); - reply->nwriten = 0; - - reply->mhdr->version = VERSION; - reply->mhdr->datalen = (mDNSu32)datalen; - reply->mhdr->ipc_flags = 0; - reply->mhdr->op = op; - reply->mhdr->client_context = request->hdr.client_context; - reply->mhdr->reg_index = 0; - - return reply; - } - -// Append a reply to the list in a request object -// If our request is sharing a connection, then we append our reply_state onto the primary's list -mDNSlocal void append_reply(request_state *req, reply_state *rep) - { - request_state *r = req->primary ? req->primary : req; - reply_state **ptr = &r->replies; - while (*ptr) ptr = &(*ptr)->next; - *ptr = rep; - rep->next = NULL; - } - -// Generates a response message giving name, type, domain, plus interface index, -// suitable for a browse result or service registration result. -// On successful completion rep is set to point to a malloc'd reply_state struct -mDNSlocal mStatus GenerateNTDResponse(const domainname *const servicename, const mDNSInterfaceID id, - request_state *const request, reply_state **const rep, reply_op_t op, DNSServiceFlags flags, mStatus err) - { - domainlabel name; - domainname type, dom; - *rep = NULL; - if (!DeconstructServiceName(servicename, &name, &type, &dom)) - return kDNSServiceErr_Invalid; - else - { - char namestr[MAX_DOMAIN_LABEL+1]; - char typestr[MAX_ESCAPED_DOMAIN_NAME]; - char domstr [MAX_ESCAPED_DOMAIN_NAME]; - int len; - char *data; - - ConvertDomainLabelToCString_unescaped(&name, namestr); - ConvertDomainNameToCString(&type, typestr); - ConvertDomainNameToCString(&dom, domstr); - - // Calculate reply data length - len = sizeof(DNSServiceFlags); - len += sizeof(mDNSu32); // if index - len += sizeof(DNSServiceErrorType); - len += (int) (strlen(namestr) + 1); - len += (int) (strlen(typestr) + 1); - len += (int) (strlen(domstr) + 1); - - // Build reply header - *rep = create_reply(op, len, request); - (*rep)->rhdr->flags = dnssd_htonl(flags); - (*rep)->rhdr->ifi = dnssd_htonl(mDNSPlatformInterfaceIndexfromInterfaceID(&mDNSStorage, id, mDNSfalse)); - (*rep)->rhdr->error = dnssd_htonl(err); - - // Build reply body - data = (char *)&(*rep)->rhdr[1]; - put_string(namestr, &data); - put_string(typestr, &data); - put_string(domstr, &data); - - return mStatus_NoError; - } - } - -// Special support to enable the DNSServiceBrowse call made by Bonjour Browser -// Remove after Bonjour Browser is updated to use DNSServiceQueryRecord instead of DNSServiceBrowse -mDNSlocal void GenerateBonjourBrowserResponse(const domainname *const servicename, const mDNSInterfaceID id, - request_state *const request, reply_state **const rep, reply_op_t op, DNSServiceFlags flags, mStatus err) - { - char namestr[MAX_DOMAIN_LABEL+1]; - char typestr[MAX_ESCAPED_DOMAIN_NAME]; - static const char domstr[] = "."; - int len; - char *data; - - *rep = NULL; - - // 1. Put first label in namestr - ConvertDomainLabelToCString_unescaped((const domainlabel *)servicename, namestr); - - // 2. Put second label and "local" into typestr - mDNS_snprintf(typestr, sizeof(typestr), "%#s.local.", SecondLabel(servicename)); - - // Calculate reply data length - len = sizeof(DNSServiceFlags); - len += sizeof(mDNSu32); // if index - len += sizeof(DNSServiceErrorType); - len += (int) (strlen(namestr) + 1); - len += (int) (strlen(typestr) + 1); - len += (int) (strlen(domstr) + 1); - - // Build reply header - *rep = create_reply(op, len, request); - (*rep)->rhdr->flags = dnssd_htonl(flags); - (*rep)->rhdr->ifi = dnssd_htonl(mDNSPlatformInterfaceIndexfromInterfaceID(&mDNSStorage, id, mDNSfalse)); - (*rep)->rhdr->error = dnssd_htonl(err); - - // Build reply body - data = (char *)&(*rep)->rhdr[1]; - put_string(namestr, &data); - put_string(typestr, &data); - put_string(domstr, &data); - } - -// Returns a resource record (allocated w/ malloc) containing the data found in an IPC message -// Data must be in the following format: flags, interfaceIndex, name, rrtype, rrclass, rdlen, rdata, (optional) ttl -// (ttl only extracted/set if ttl argument is non-zero). Returns NULL for a bad-parameter error -mDNSlocal AuthRecord *read_rr_from_ipc_msg(request_state *request, int GetTTL, int validate_flags) - { - DNSServiceFlags flags = get_flags(&request->msgptr, request->msgend); - mDNSu32 interfaceIndex = get_uint32(&request->msgptr, request->msgend); - char name[256]; - int str_err = get_string(&request->msgptr, request->msgend, name, sizeof(name)); - mDNSu16 type = get_uint16(&request->msgptr, request->msgend); - mDNSu16 class = get_uint16(&request->msgptr, request->msgend); - mDNSu16 rdlen = get_uint16(&request->msgptr, request->msgend); - const char *rdata = get_rdata (&request->msgptr, request->msgend, rdlen); - mDNSu32 ttl = GetTTL ? get_uint32(&request->msgptr, request->msgend) : 0; - int storage_size = rdlen > sizeof(RDataBody) ? rdlen : sizeof(RDataBody); - AuthRecord *rr; - mDNSInterfaceID InterfaceID; - AuthRecType artype; - - request->flags = flags; - - if (str_err) { LogMsg("ERROR: read_rr_from_ipc_msg - get_string"); return NULL; } - - if (!request->msgptr) { LogMsg("Error reading Resource Record from client"); return NULL; } - - if (validate_flags && - !((flags & kDNSServiceFlagsShared) == kDNSServiceFlagsShared) && - !((flags & kDNSServiceFlagsUnique) == kDNSServiceFlagsUnique)) - { - LogMsg("ERROR: Bad resource record flags (must be kDNSServiceFlagsShared or kDNSServiceFlagsUnique)"); - return NULL; - } - - rr = mallocL("AuthRecord/read_rr_from_ipc_msg", sizeof(AuthRecord) - sizeof(RDataBody) + storage_size); - if (!rr) FatalError("ERROR: malloc"); - - InterfaceID = mDNSPlatformInterfaceIDfromInterfaceIndex(&mDNSStorage, interfaceIndex); - if (InterfaceID == mDNSInterface_LocalOnly) - artype = AuthRecordLocalOnly; - else if (InterfaceID == mDNSInterface_P2P) - artype = AuthRecordP2P; - else if ((InterfaceID == mDNSInterface_Any) && (flags & kDNSServiceFlagsIncludeP2P)) - artype = AuthRecordAnyIncludeP2P; - else - artype = AuthRecordAny; - - mDNS_SetupResourceRecord(rr, mDNSNULL, InterfaceID, type, 0, - (mDNSu8) ((flags & kDNSServiceFlagsShared) ? kDNSRecordTypeShared : kDNSRecordTypeUnique), artype, mDNSNULL, mDNSNULL); - - if (!MakeDomainNameFromDNSNameString(&rr->namestorage, name)) - { - LogMsg("ERROR: bad name: %s", name); - freeL("AuthRecord/read_rr_from_ipc_msg", rr); - return NULL; - } - - if (flags & kDNSServiceFlagsAllowRemoteQuery) rr->AllowRemoteQuery = mDNStrue; - rr->resrec.rrclass = class; - rr->resrec.rdlength = rdlen; - rr->resrec.rdata->MaxRDLength = rdlen; - mDNSPlatformMemCopy(rr->resrec.rdata->u.data, rdata, rdlen); - if (GetTTL) rr->resrec.rroriginalttl = ttl; - rr->resrec.namehash = DomainNameHashValue(rr->resrec.name); - SetNewRData(&rr->resrec, mDNSNULL, 0); // Sets rr->rdatahash for us - return rr; - } - -mDNSlocal int build_domainname_from_strings(domainname *srv, char *name, char *regtype, char *domain) - { - domainlabel n; - domainname d, t; - - if (!MakeDomainLabelFromLiteralString(&n, name)) return -1; - if (!MakeDomainNameFromDNSNameString(&t, regtype)) return -1; - if (!MakeDomainNameFromDNSNameString(&d, domain)) return -1; - if (!ConstructServiceName(srv, &n, &t, &d)) return -1; - return 0; - } - -mDNSlocal void send_all(dnssd_sock_t s, const char *ptr, int len) - { - int n = send(s, ptr, len, 0); - // On a freshly-created Unix Domain Socket, the kernel should *never* fail to buffer a small write for us - // (four bytes for a typical error code return, 12 bytes for DNSServiceGetProperty(DaemonVersion)). - // If it does fail, we don't attempt to handle this failure, but we do log it so we know something is wrong. - if (n < len) - LogMsg("ERROR: send_all(%d) wrote %d of %d errno %d (%s)", - s, n, len, dnssd_errno, dnssd_strerror(dnssd_errno)); - } - -#if 0 -mDNSlocal mDNSBool AuthorizedDomain(const request_state * const request, const domainname * const d, const DNameListElem * const doms) -{ - const DNameListElem *delem = mDNSNULL; - int bestDelta = -1; // the delta of the best match, lower is better - int dLabels = 0; - mDNSBool allow = mDNSfalse; - - if (SystemUID(request->uid)) return mDNStrue; - - dLabels = CountLabels(d); - for (delem = doms; delem; delem = delem->next) - { - if (delem->uid) - { - int delemLabels = CountLabels(&delem->name); - int delta = dLabels - delemLabels; - if ((bestDelta == -1 || delta <= bestDelta) && SameDomainName(&delem->name, SkipLeadingLabels(d, delta))) - { - bestDelta = delta; - allow = (allow || (delem->uid == request->uid)); - } - } - } - - return bestDelta == -1 ? mDNStrue : allow; -} -#endif - -// *************************************************************************** -#if COMPILER_LIKES_PRAGMA_MARK -#pragma mark - -#pragma mark - external helpers -#endif - -mDNSlocal void external_start_advertising_helper(service_instance *const instance) - { - AuthRecord *st = instance->subtypes; - ExtraResourceRecord *e; - int i; - - if (mDNSIPPortIsZero(instance->request->u.servicereg.port)) - { - LogInfo("external_start_advertising_helper: Not registering service with port number zero"); - return; - } - -#if APPLE_OSX_mDNSResponder - // Update packet filter if p2p interface already exists, otherwise, - // if will be updated when we get the KEV_DL_IF_ATTACHED event for - // the interface. Called here since we don't call external_start_advertising_service() - // with the SRV record when advertising a service. - mDNSInitPacketFilter(); -#endif // APPLE_OSX_mDNSResponder - - if (instance->external_advertise) LogMsg("external_start_advertising_helper: external_advertise already set!"); - - for ( i = 0; i < instance->request->u.servicereg.num_subtypes; i++) - external_start_advertising_service(&st[i].resrec); - - external_start_advertising_service(&instance->srs.RR_PTR.resrec); - external_start_advertising_service(&instance->srs.RR_TXT.resrec); - - for (e = instance->srs.Extras; e; e = e->next) - external_start_advertising_service(&e->r.resrec); - - instance->external_advertise = mDNStrue; - } - -mDNSlocal void external_stop_advertising_helper(service_instance *const instance) - { - AuthRecord *st = instance->subtypes; - ExtraResourceRecord *e; - int i; - - if (!instance->external_advertise) return; - - LogInfo("external_stop_advertising_helper: calling external_stop_advertising_service"); - - for ( i = 0; i < instance->request->u.servicereg.num_subtypes; i++) - external_stop_advertising_service(&st[i].resrec); - - external_stop_advertising_service(&instance->srs.RR_PTR.resrec); - external_stop_advertising_service(&instance->srs.RR_TXT.resrec); - - for (e = instance->srs.Extras; e; e = e->next) - external_stop_advertising_service(&e->r.resrec); - - instance->external_advertise = mDNSfalse; - } - -// *************************************************************************** -#if COMPILER_LIKES_PRAGMA_MARK -#pragma mark - -#pragma mark - DNSServiceRegister -#endif - -mDNSexport void FreeExtraRR(mDNS *const m, AuthRecord *const rr, mStatus result) - { - ExtraResourceRecord *extra = (ExtraResourceRecord *)rr->RecordContext; - (void)m; // Unused - - if (result != mStatus_MemFree) { LogMsg("Error: FreeExtraRR invoked with unexpected error %d", result); return; } - - LogInfo(" FreeExtraRR %s", RRDisplayString(m, &rr->resrec)); - - if (rr->resrec.rdata != &rr->rdatastorage) - freeL("Extra RData", rr->resrec.rdata); - freeL("ExtraResourceRecord/FreeExtraRR", extra); - } - -mDNSlocal void unlink_and_free_service_instance(service_instance *srv) - { - ExtraResourceRecord *e = srv->srs.Extras, *tmp; - - external_stop_advertising_helper(srv); - - // clear pointers from parent struct - if (srv->request) - { - service_instance **p = &srv->request->u.servicereg.instances; - while (*p) - { - if (*p == srv) { *p = (*p)->next; break; } - p = &(*p)->next; - } - } - - while (e) - { - e->r.RecordContext = e; - tmp = e; - e = e->next; - FreeExtraRR(&mDNSStorage, &tmp->r, mStatus_MemFree); - } - - if (srv->srs.RR_TXT.resrec.rdata != &srv->srs.RR_TXT.rdatastorage) - freeL("TXT RData", srv->srs.RR_TXT.resrec.rdata); - - if (srv->subtypes) { freeL("ServiceSubTypes", srv->subtypes); srv->subtypes = NULL; } - freeL("service_instance", srv); - } - -// Count how many other service records we have locally with the same name, but different rdata. -// For auto-named services, we can have at most one per machine -- if we allowed two auto-named services of -// the same type on the same machine, we'd get into an infinite autoimmune-response loop of continuous renaming. -mDNSexport int CountPeerRegistrations(mDNS *const m, ServiceRecordSet *const srs) - { - int count = 0; - ResourceRecord *r = &srs->RR_SRV.resrec; - AuthRecord *rr; - - for (rr = m->ResourceRecords; rr; rr=rr->next) - if (rr->resrec.rrtype == kDNSType_SRV && SameDomainName(rr->resrec.name, r->name) && !IdenticalSameNameRecord(&rr->resrec, r)) - count++; - - verbosedebugf("%d peer registrations for %##s", count, r->name->c); - return(count); - } - -mDNSexport int CountExistingRegistrations(domainname *srv, mDNSIPPort port) - { - int count = 0; - AuthRecord *rr; - for (rr = mDNSStorage.ResourceRecords; rr; rr=rr->next) - if (rr->resrec.rrtype == kDNSType_SRV && - mDNSSameIPPort(rr->resrec.rdata->u.srv.port, port) && - SameDomainName(rr->resrec.name, srv)) - count++; - return(count); - } - -mDNSlocal void SendServiceRemovalNotification(ServiceRecordSet *const srs) - { - reply_state *rep; - service_instance *instance = srs->ServiceContext; - if (GenerateNTDResponse(srs->RR_SRV.resrec.name, srs->RR_SRV.resrec.InterfaceID, instance->request, &rep, reg_service_reply_op, 0, mStatus_NoError) != mStatus_NoError) - LogMsg("%3d: SendServiceRemovalNotification: %##s is not valid DNS-SD SRV name", instance->request->sd, srs->RR_SRV.resrec.name->c); - else { append_reply(instance->request, rep); instance->clientnotified = mDNSfalse; } - } - -// service registration callback performs three duties - frees memory for deregistered services, -// handles name conflicts, and delivers completed registration information to the client -mDNSlocal void regservice_callback(mDNS *const m, ServiceRecordSet *const srs, mStatus result) - { - mStatus err; - mDNSBool SuppressError = mDNSfalse; - service_instance *instance; - reply_state *rep; - (void)m; // Unused - - if (!srs) { LogMsg("regservice_callback: srs is NULL %d", result); return; } - - instance = srs->ServiceContext; - if (!instance) { LogMsg("regservice_callback: srs->ServiceContext is NULL %d", result); return; } - - // don't send errors up to client for wide-area, empty-string registrations - if (instance->request && - instance->request->u.servicereg.default_domain && - !instance->default_local) - SuppressError = mDNStrue; - - if (mDNS_LoggingEnabled) - { - const char *const fmt = - (result == mStatus_NoError) ? "%s DNSServiceRegister(%##s, %u) REGISTERED" : - (result == mStatus_MemFree) ? "%s DNSServiceRegister(%##s, %u) DEREGISTERED" : - (result == mStatus_NameConflict) ? "%s DNSServiceRegister(%##s, %u) NAME CONFLICT" : - "%s DNSServiceRegister(%##s, %u) %s %d"; - char prefix[16] = "---:"; - if (instance->request) mDNS_snprintf(prefix, sizeof(prefix), "%3d:", instance->request->sd); - LogOperation(fmt, prefix, srs->RR_SRV.resrec.name->c, mDNSVal16(srs->RR_SRV.resrec.rdata->u.srv.port), - SuppressError ? "suppressed error" : "CALLBACK", result); - } - - if (!instance->request && result != mStatus_MemFree) { LogMsg("regservice_callback: instance->request is NULL %d", result); return; } - - if (result == mStatus_NoError) - { - if (instance->request->u.servicereg.allowremotequery) - { - ExtraResourceRecord *e; - srs->RR_ADV.AllowRemoteQuery = mDNStrue; - srs->RR_PTR.AllowRemoteQuery = mDNStrue; - srs->RR_SRV.AllowRemoteQuery = mDNStrue; - srs->RR_TXT.AllowRemoteQuery = mDNStrue; - for (e = instance->srs.Extras; e; e = e->next) e->r.AllowRemoteQuery = mDNStrue; - } - - if (GenerateNTDResponse(srs->RR_SRV.resrec.name, srs->RR_SRV.resrec.InterfaceID, instance->request, &rep, reg_service_reply_op, kDNSServiceFlagsAdd, result) != mStatus_NoError) - LogMsg("%3d: regservice_callback: %##s is not valid DNS-SD SRV name", instance->request->sd, srs->RR_SRV.resrec.name->c); - else { append_reply(instance->request, rep); instance->clientnotified = mDNStrue; } - - if (instance->request->u.servicereg.InterfaceID == mDNSInterface_P2P || (!instance->request->u.servicereg.InterfaceID && SameDomainName(&instance->domain, &localdomain) && (instance->request->flags & kDNSServiceFlagsIncludeP2P))) - { - LogInfo("regservice_callback: calling external_start_advertising_helper()"); - external_start_advertising_helper(instance); - } - if (instance->request->u.servicereg.autoname && CountPeerRegistrations(m, srs) == 0) - RecordUpdatedNiceLabel(m, 0); // Successfully got new name, tell user immediately - } - else if (result == mStatus_MemFree) - { - if (instance->request && instance->renameonmemfree) - { - external_stop_advertising_helper(instance); - instance->renameonmemfree = 0; - err = mDNS_RenameAndReregisterService(m, srs, &instance->request->u.servicereg.name); - if (err) LogMsg("ERROR: regservice_callback - RenameAndReregisterService returned %d", err); - // error should never happen - safest to log and continue - } - else - unlink_and_free_service_instance(instance); - } - else if (result == mStatus_NameConflict) - { - if (instance->request->u.servicereg.autorename) - { - external_stop_advertising_helper(instance); - if (instance->request->u.servicereg.autoname && CountPeerRegistrations(m, srs) == 0) - { - // On conflict for an autoname service, rename and reregister *all* autoname services - IncrementLabelSuffix(&m->nicelabel, mDNStrue); - mDNS_ConfigChanged(m); // Will call back into udsserver_handle_configchange() - } - else // On conflict for a non-autoname service, rename and reregister just that one service - { - if (instance->clientnotified) SendServiceRemovalNotification(srs); - mDNS_RenameAndReregisterService(m, srs, mDNSNULL); - } - } - else - { - if (!SuppressError) - { - if (GenerateNTDResponse(srs->RR_SRV.resrec.name, srs->RR_SRV.resrec.InterfaceID, instance->request, &rep, reg_service_reply_op, kDNSServiceFlagsAdd, result) != mStatus_NoError) - LogMsg("%3d: regservice_callback: %##s is not valid DNS-SD SRV name", instance->request->sd, srs->RR_SRV.resrec.name->c); - else { append_reply(instance->request, rep); instance->clientnotified = mDNStrue; } - } - unlink_and_free_service_instance(instance); - } - } - else // Not mStatus_NoError, mStatus_MemFree, or mStatus_NameConflict - { - if (!SuppressError) - { - if (GenerateNTDResponse(srs->RR_SRV.resrec.name, srs->RR_SRV.resrec.InterfaceID, instance->request, &rep, reg_service_reply_op, kDNSServiceFlagsAdd, result) != mStatus_NoError) - LogMsg("%3d: regservice_callback: %##s is not valid DNS-SD SRV name", instance->request->sd, srs->RR_SRV.resrec.name->c); - else { append_reply(instance->request, rep); instance->clientnotified = mDNStrue; } - } - } - } - -mDNSlocal void regrecord_callback(mDNS *const m, AuthRecord *rr, mStatus result) - { - (void)m; // Unused - if (!rr->RecordContext) // parent struct already freed by termination callback - { - if (result == mStatus_NoError) - LogMsg("Error: regrecord_callback: successful registration of orphaned record %s", ARDisplayString(m, rr)); - else - { - if (result != mStatus_MemFree) LogMsg("regrecord_callback: error %d received after parent termination", result); - - // We come here when the record is being deregistered either from DNSServiceRemoveRecord or connection_termination. - // If the record has been updated, we need to free the rdata. Everytime we call mDNS_Update, it calls update_callback - // with the old rdata (so that we can free it) and stores the new rdata in "rr->resrec.rdata". This means, we need - // to free the latest rdata for which the update_callback was never called with. - if (rr->resrec.rdata != &rr->rdatastorage) freeL("RData/regrecord_callback", rr->resrec.rdata); - freeL("AuthRecord/regrecord_callback", rr); - } - } - else - { - registered_record_entry *re = rr->RecordContext; - request_state *request = re->request; - - if (mDNS_LoggingEnabled) - { - char *fmt = (result == mStatus_NoError) ? "%3d: DNSServiceRegisterRecord(%u %s) REGISTERED" : - (result == mStatus_MemFree) ? "%3d: DNSServiceRegisterRecord(%u %s) DEREGISTERED" : - (result == mStatus_NameConflict) ? "%3d: DNSServiceRegisterRecord(%u %s) NAME CONFLICT" : - "%3d: DNSServiceRegisterRecord(%u %s) %d"; - LogOperation(fmt, request->sd, re->key, RRDisplayString(m, &rr->resrec), result); - } - - if (result != mStatus_MemFree) - { - int len = sizeof(DNSServiceFlags) + sizeof(mDNSu32) + sizeof(DNSServiceErrorType); - reply_state *reply = create_reply(reg_record_reply_op, len, request); - reply->mhdr->client_context = re->regrec_client_context; - reply->rhdr->flags = dnssd_htonl(0); - reply->rhdr->ifi = dnssd_htonl(mDNSPlatformInterfaceIndexfromInterfaceID(m, rr->resrec.InterfaceID, mDNSfalse)); - reply->rhdr->error = dnssd_htonl(result); - append_reply(request, reply); - } - - if (result) - { - // unlink from list, free memory - registered_record_entry **ptr = &request->u.reg_recs; - while (*ptr && (*ptr) != re) ptr = &(*ptr)->next; - if (!*ptr) { LogMsg("regrecord_callback - record not in list!"); return; } - *ptr = (*ptr)->next; - freeL("registered_record_entry AuthRecord regrecord_callback", re->rr); - freeL("registered_record_entry regrecord_callback", re); - } - else - { - if (re->external_advertise) LogMsg("regrecord_callback: external_advertise already set!"); - - if (re->origInterfaceID == mDNSInterface_P2P || (!re->origInterfaceID && IsLocalDomain(&rr->namestorage) && (request->flags & kDNSServiceFlagsIncludeP2P))) - { - LogInfo("regrecord_callback: calling external_start_advertising_service"); - external_start_advertising_service(&rr->resrec); - re->external_advertise = mDNStrue; - } - } - } - } - -mDNSlocal void connection_termination(request_state *request) - { - // When terminating a shared connection, we need to scan the all_requests list - // and terminate any subbordinate operations sharing this file descriptor - request_state **req = &all_requests; - - LogOperation("%3d: DNSServiceCreateConnection STOP", request->sd); - - while (*req) - { - if ((*req)->primary == request) - { - // Since we're already doing a list traversal, we unlink the request directly instead of using AbortUnlinkAndFree() - request_state *tmp = *req; - if (tmp->primary == tmp) LogMsg("connection_termination ERROR (*req)->primary == *req for %p %d", tmp, tmp->sd); - if (tmp->replies) LogMsg("connection_termination ERROR How can subordinate req %p %d have replies queued?", tmp, tmp->sd); - abort_request(tmp); - *req = tmp->next; - freeL("request_state/connection_termination", tmp); - } - else - req = &(*req)->next; - } - - while (request->u.reg_recs) - { - registered_record_entry *ptr = request->u.reg_recs; - LogOperation("%3d: DNSServiceRegisterRecord(%u %s) STOP", request->sd, ptr->key, RRDisplayString(&mDNSStorage, &ptr->rr->resrec)); - request->u.reg_recs = request->u.reg_recs->next; - ptr->rr->RecordContext = NULL; - if (ptr->external_advertise) - { - ptr->external_advertise = mDNSfalse; - external_stop_advertising_service(&ptr->rr->resrec); - } - mDNS_Deregister(&mDNSStorage, ptr->rr); // Will free ptr->rr for us - freeL("registered_record_entry/connection_termination", ptr); - } - } - -mDNSlocal void handle_cancel_request(request_state *request) - { - request_state **req = &all_requests; - LogOperation("%3d: Cancel %08X %08X", request->sd, request->hdr.client_context.u32[1], request->hdr.client_context.u32[0]); - while (*req) - { - if ((*req)->primary == request && - (*req)->hdr.client_context.u32[0] == request->hdr.client_context.u32[0] && - (*req)->hdr.client_context.u32[1] == request->hdr.client_context.u32[1]) - { - // Since we're already doing a list traversal, we unlink the request directly instead of using AbortUnlinkAndFree() - request_state *tmp = *req; - abort_request(tmp); - *req = tmp->next; - freeL("request_state/handle_cancel_request", tmp); - } - else - req = &(*req)->next; - } - } - -mDNSlocal mStatus handle_regrecord_request(request_state *request) - { - mStatus err = mStatus_BadParamErr; - AuthRecord *rr = read_rr_from_ipc_msg(request, 1, 1); - if (rr) - { - registered_record_entry *re; - // Don't allow non-local domains to be regsitered as LocalOnly. Allowing this would permit - // clients to register records such as www.bigbank.com A w.x.y.z to redirect Safari. - if (rr->resrec.InterfaceID == mDNSInterface_LocalOnly && !IsLocalDomain(rr->resrec.name) && - rr->resrec.rrclass == kDNSClass_IN && (rr->resrec.rrtype == kDNSType_A || rr->resrec.rrtype == kDNSType_AAAA || - rr->resrec.rrtype == kDNSType_CNAME)) - { - freeL("AuthRecord/handle_regrecord_request", rr); - return (mStatus_BadParamErr); - } - // allocate registration entry, link into list - re = mallocL("registered_record_entry", sizeof(registered_record_entry)); - if (!re) FatalError("ERROR: malloc"); - re->key = request->hdr.reg_index; - re->rr = rr; - re->regrec_client_context = request->hdr.client_context; - re->request = request; - re->external_advertise = mDNSfalse; - rr->RecordContext = re; - rr->RecordCallback = regrecord_callback; - - re->origInterfaceID = rr->resrec.InterfaceID; - if (rr->resrec.InterfaceID == mDNSInterface_P2P) rr->resrec.InterfaceID = mDNSInterface_Any; -#if 0 - if (!AuthorizedDomain(request, rr->resrec.name, AutoRegistrationDomains)) return (mStatus_NoError); -#endif - if (rr->resrec.rroriginalttl == 0) - rr->resrec.rroriginalttl = DefaultTTLforRRType(rr->resrec.rrtype); - - LogOperation("%3d: DNSServiceRegisterRecord(%u %s) START", request->sd, re->key, RRDisplayString(&mDNSStorage, &rr->resrec)); - err = mDNS_Register(&mDNSStorage, rr); - if (err) - { - LogOperation("%3d: DNSServiceRegisterRecord(%u %s) ERROR (%d)", request->sd, re->key, RRDisplayString(&mDNSStorage, &rr->resrec), err); - freeL("registered_record_entry", re); - freeL("registered_record_entry/AuthRecord", rr); - } - else - { - re->next = request->u.reg_recs; - request->u.reg_recs = re; - } - } - return(err); - } - -mDNSlocal void UpdateDeviceInfoRecord(mDNS *const m); - -mDNSlocal void regservice_termination_callback(request_state *request) - { - if (!request) { LogMsg("regservice_termination_callback context is NULL"); return; } - while (request->u.servicereg.instances) - { - service_instance *p = request->u.servicereg.instances; - request->u.servicereg.instances = request->u.servicereg.instances->next; - // only safe to free memory if registration is not valid, i.e. deregister fails (which invalidates p) - LogOperation("%3d: DNSServiceRegister(%##s, %u) STOP", - request->sd, p->srs.RR_SRV.resrec.name->c, mDNSVal16(p->srs.RR_SRV.resrec.rdata->u.srv.port)); - - external_stop_advertising_helper(p); - - // Clear backpointer *before* calling mDNS_DeregisterService/unlink_and_free_service_instance - // We don't need unlink_and_free_service_instance to cut its element from the list, because we're already advancing - // request->u.servicereg.instances as we work our way through the list, implicitly cutting one element at a time - // We can't clear p->request *after* the calling mDNS_DeregisterService/unlink_and_free_service_instance - // because by then we might have already freed p - p->request = NULL; - if (mDNS_DeregisterService(&mDNSStorage, &p->srs)) unlink_and_free_service_instance(p); - // Don't touch service_instance *p after this -- it's likely to have been freed already - } - if (request->u.servicereg.txtdata) - { freeL("service_info txtdata", request->u.servicereg.txtdata); request->u.servicereg.txtdata = NULL; } - if (request->u.servicereg.autoname) - { - // Clear autoname before calling UpdateDeviceInfoRecord() so it doesn't mistakenly include this in its count of active autoname registrations - request->u.servicereg.autoname = mDNSfalse; - UpdateDeviceInfoRecord(&mDNSStorage); - } - } - -mDNSlocal request_state *LocateSubordinateRequest(request_state *request) - { - request_state *req; - for (req = all_requests; req; req = req->next) - if (req->primary == request && - req->hdr.client_context.u32[0] == request->hdr.client_context.u32[0] && - req->hdr.client_context.u32[1] == request->hdr.client_context.u32[1]) return(req); - return(request); - } - -mDNSlocal mStatus add_record_to_service(request_state *request, service_instance *instance, mDNSu16 rrtype, mDNSu16 rdlen, const char *rdata, mDNSu32 ttl) - { - ServiceRecordSet *srs = &instance->srs; - mStatus result; - int size = rdlen > sizeof(RDataBody) ? rdlen : sizeof(RDataBody); - ExtraResourceRecord *extra = mallocL("ExtraResourceRecord", sizeof(*extra) - sizeof(RDataBody) + size); - if (!extra) { my_perror("ERROR: malloc"); return mStatus_NoMemoryErr; } - - mDNSPlatformMemZero(extra, sizeof(ExtraResourceRecord)); // OK if oversized rdata not zero'd - extra->r.resrec.rrtype = rrtype; - extra->r.rdatastorage.MaxRDLength = (mDNSu16) size; - extra->r.resrec.rdlength = rdlen; - mDNSPlatformMemCopy(&extra->r.rdatastorage.u.data, rdata, rdlen); - - result = mDNS_AddRecordToService(&mDNSStorage, srs, extra, &extra->r.rdatastorage, ttl, - (request->flags & kDNSServiceFlagsIncludeP2P) ? 1: 0); - if (result) { freeL("ExtraResourceRecord/add_record_to_service", extra); return result; } - - extra->ClientID = request->hdr.reg_index; - if (instance->external_advertise && (instance->request->u.servicereg.InterfaceID == mDNSInterface_P2P || (!instance->request->u.servicereg.InterfaceID && SameDomainName(&instance->domain, &localdomain) && (instance->request->flags & kDNSServiceFlagsIncludeP2P)))) - { - LogInfo("add_record_to_service: calling external_start_advertising_service"); - external_start_advertising_service(&extra->r.resrec); - } - return result; - } - -mDNSlocal mStatus handle_add_request(request_state *request) - { - service_instance *i; - mStatus result = mStatus_UnknownErr; - DNSServiceFlags flags = get_flags (&request->msgptr, request->msgend); - mDNSu16 rrtype = get_uint16(&request->msgptr, request->msgend); - mDNSu16 rdlen = get_uint16(&request->msgptr, request->msgend); - const char *rdata = get_rdata (&request->msgptr, request->msgend, rdlen); - mDNSu32 ttl = get_uint32(&request->msgptr, request->msgend); - if (!ttl) ttl = DefaultTTLforRRType(rrtype); - (void)flags; // Unused - - if (!request->msgptr) { LogMsg("%3d: DNSServiceAddRecord(unreadable parameters)", request->sd); return(mStatus_BadParamErr); } - - // If this is a shared connection, check if the operation actually applies to a subordinate request_state object - if (request->terminate == connection_termination) request = LocateSubordinateRequest(request); - - if (request->terminate != regservice_termination_callback) - { LogMsg("%3d: DNSServiceAddRecord(not a registered service ref)", request->sd); return(mStatus_BadParamErr); } - - // For a service registered with zero port, don't allow adding records. This mostly happens due to a bug - // in the application. See radar://9165807. - if (mDNSIPPortIsZero(request->u.servicereg.port)) - { LogMsg("%3d: DNSServiceAddRecord: adding record to a service registered with zero port", request->sd); return(mStatus_BadParamErr); } - - LogOperation("%3d: DNSServiceAddRecord(%X, %##s, %s, %d)", request->sd, flags, - (request->u.servicereg.instances) ? request->u.servicereg.instances->srs.RR_SRV.resrec.name->c : NULL, DNSTypeName(rrtype), rdlen); - - for (i = request->u.servicereg.instances; i; i = i->next) - { - result = add_record_to_service(request, i, rrtype, rdlen, rdata, ttl); - if (result && i->default_local) break; - else result = mStatus_NoError; // suppress non-local default errors - } - - return(result); - } - -mDNSlocal void update_callback(mDNS *const m, AuthRecord *const rr, RData *oldrd, mDNSu16 oldrdlen) - { - mDNSBool external_advertise = (rr->UpdateContext) ? *((mDNSBool *)rr->UpdateContext) : mDNSfalse; - (void)m; // Unused - - // There are three cases. - // - // 1. We have updated the primary TXT record of the service - // 2. We have updated the TXT record that was added to the service using DNSServiceAddRecord - // 3. We have updated the TXT record that was registered using DNSServiceRegisterRecord - // - // external_advertise is set if we have advertised at least once during the initial addition - // of the record in all of the three cases above. We should have checked for InterfaceID/LocalDomain - // checks during the first time and hence we don't do any checks here - if (external_advertise) - { - ResourceRecord ext = rr->resrec; - if (ext.rdlength == oldrdlen && mDNSPlatformMemSame(&ext.rdata->u, &oldrd->u, oldrdlen)) goto exit; - SetNewRData(&ext, oldrd, oldrdlen); - external_stop_advertising_service(&ext); - LogInfo("update_callback: calling external_start_advertising_service"); - external_start_advertising_service(&rr->resrec); - } -exit: - if (oldrd != &rr->rdatastorage) freeL("RData/update_callback", oldrd); - } - -mDNSlocal mStatus update_record(AuthRecord *rr, mDNSu16 rdlen, const char *rdata, mDNSu32 ttl, const mDNSBool *const external_advertise) - { - mStatus result; - const int rdsize = rdlen > sizeof(RDataBody) ? rdlen : sizeof(RDataBody); - RData *newrd = mallocL("RData/update_record", sizeof(RData) - sizeof(RDataBody) + rdsize); - if (!newrd) FatalError("ERROR: malloc"); - newrd->MaxRDLength = (mDNSu16) rdsize; - mDNSPlatformMemCopy(&newrd->u, rdata, rdlen); - - // BIND named (name daemon) doesn't allow TXT records with zero-length rdata. This is strictly speaking correct, - // since RFC 1035 specifies a TXT record as "One or more s", not "Zero or more s". - // Since some legacy apps try to create zero-length TXT records, we'll silently correct it here. - if (rr->resrec.rrtype == kDNSType_TXT && rdlen == 0) { rdlen = 1; newrd->u.txt.c[0] = 0; } - - if (external_advertise) rr->UpdateContext = (void *)external_advertise; - - result = mDNS_Update(&mDNSStorage, rr, ttl, rdlen, newrd, update_callback); - if (result) { LogMsg("update_record: Error %d for %s", (int)result, ARDisplayString(&mDNSStorage, rr)); freeL("RData/update_record", newrd); } - return result; - } - -mDNSlocal mStatus handle_update_request(request_state *request) - { - const ipc_msg_hdr *const hdr = &request->hdr; - mStatus result = mStatus_BadReferenceErr; - service_instance *i; - AuthRecord *rr = NULL; - - // get the message data - DNSServiceFlags flags = get_flags (&request->msgptr, request->msgend); // flags unused - mDNSu16 rdlen = get_uint16(&request->msgptr, request->msgend); - const char *rdata = get_rdata (&request->msgptr, request->msgend, rdlen); - mDNSu32 ttl = get_uint32(&request->msgptr, request->msgend); - (void)flags; // Unused - - if (!request->msgptr) { LogMsg("%3d: DNSServiceUpdateRecord(unreadable parameters)", request->sd); return(mStatus_BadParamErr); } - - // If this is a shared connection, check if the operation actually applies to a subordinate request_state object - if (request->terminate == connection_termination) request = LocateSubordinateRequest(request); - - if (request->terminate == connection_termination) - { - // update an individually registered record - registered_record_entry *reptr; - for (reptr = request->u.reg_recs; reptr; reptr = reptr->next) - { - if (reptr->key == hdr->reg_index) - { - result = update_record(reptr->rr, rdlen, rdata, ttl, &reptr->external_advertise); - LogOperation("%3d: DNSServiceUpdateRecord(%##s, %s)", - request->sd, reptr->rr->resrec.name->c, reptr->rr ? DNSTypeName(reptr->rr->resrec.rrtype) : ""); - goto end; - } - } - result = mStatus_BadReferenceErr; - goto end; - } - - if (request->terminate != regservice_termination_callback) - { LogMsg("%3d: DNSServiceUpdateRecord(not a registered service ref)", request->sd); return(mStatus_BadParamErr); } - - // For a service registered with zero port, only SRV record is initialized. Don't allow any updates. - if (mDNSIPPortIsZero(request->u.servicereg.port)) - { LogMsg("%3d: DNSServiceUpdateRecord: updating the record of a service registered with zero port", request->sd); return(mStatus_BadParamErr); } - - // update the saved off TXT data for the service - if (hdr->reg_index == TXT_RECORD_INDEX) - { - if (request->u.servicereg.txtdata) - { freeL("service_info txtdata", request->u.servicereg.txtdata); request->u.servicereg.txtdata = NULL; } - if (rdlen > 0) - { - request->u.servicereg.txtdata = mallocL("service_info txtdata", rdlen); - if (!request->u.servicereg.txtdata) FatalError("ERROR: handle_update_request - malloc"); - mDNSPlatformMemCopy(request->u.servicereg.txtdata, rdata, rdlen); - } - request->u.servicereg.txtlen = rdlen; - } - - // update a record from a service record set - for (i = request->u.servicereg.instances; i; i = i->next) - { - if (hdr->reg_index == TXT_RECORD_INDEX) rr = &i->srs.RR_TXT; - else - { - ExtraResourceRecord *e; - for (e = i->srs.Extras; e; e = e->next) - if (e->ClientID == hdr->reg_index) { rr = &e->r; break; } - } - - if (!rr) { result = mStatus_BadReferenceErr; goto end; } - result = update_record(rr, rdlen, rdata, ttl, &i->external_advertise); - if (result && i->default_local) goto end; - else result = mStatus_NoError; // suppress non-local default errors - } - -end: - if (request->terminate == regservice_termination_callback) - LogOperation("%3d: DNSServiceUpdateRecord(%##s, %s)", request->sd, - (request->u.servicereg.instances) ? request->u.servicereg.instances->srs.RR_SRV.resrec.name->c : NULL, - rr ? DNSTypeName(rr->resrec.rrtype) : ""); - - return(result); - } - -// remove a resource record registered via DNSServiceRegisterRecord() -mDNSlocal mStatus remove_record(request_state *request) - { - mStatus err = mStatus_UnknownErr; - registered_record_entry *e, **ptr = &request->u.reg_recs; - - while (*ptr && (*ptr)->key != request->hdr.reg_index) ptr = &(*ptr)->next; - if (!*ptr) { LogMsg("%3d: DNSServiceRemoveRecord(%u) not found", request->sd, request->hdr.reg_index); return mStatus_BadReferenceErr; } - e = *ptr; - *ptr = e->next; // unlink - - LogOperation("%3d: DNSServiceRemoveRecord(%u %s)", request->sd, e->key, RRDisplayString(&mDNSStorage, &e->rr->resrec)); - e->rr->RecordContext = NULL; - if (e->external_advertise) - { - external_stop_advertising_service(&e->rr->resrec); - e->external_advertise = mDNSfalse; - } - err = mDNS_Deregister(&mDNSStorage, e->rr); // Will free e->rr for us; we're responsible for freeing e - if (err) - { - LogMsg("ERROR: remove_record, mDNS_Deregister: %d", err); - freeL("registered_record_entry AuthRecord remove_record", e->rr); - } - - freeL("registered_record_entry remove_record", e); - return err; - } - -mDNSlocal mStatus remove_extra(const request_state *const request, service_instance *const serv, mDNSu16 *const rrtype) - { - mStatus err = mStatus_BadReferenceErr; - ExtraResourceRecord *ptr; - - for (ptr = serv->srs.Extras; ptr; ptr = ptr->next) - { - if (ptr->ClientID == request->hdr.reg_index) // found match - { - *rrtype = ptr->r.resrec.rrtype; - if (serv->external_advertise) external_stop_advertising_service(&ptr->r.resrec); - err = mDNS_RemoveRecordFromService(&mDNSStorage, &serv->srs, ptr, FreeExtraRR, ptr); - break; - } - } - return err; - } - -mDNSlocal mStatus handle_removerecord_request(request_state *request) - { - mStatus err = mStatus_BadReferenceErr; - get_flags(&request->msgptr, request->msgend); // flags unused - - if (!request->msgptr) { LogMsg("%3d: DNSServiceRemoveRecord(unreadable parameters)", request->sd); return(mStatus_BadParamErr); } - - // If this is a shared connection, check if the operation actually applies to a subordinate request_state object - if (request->terminate == connection_termination) request = LocateSubordinateRequest(request); - - if (request->terminate == connection_termination) - err = remove_record(request); // remove individually registered record - else if (request->terminate != regservice_termination_callback) - { LogMsg("%3d: DNSServiceRemoveRecord(not a registered service ref)", request->sd); return(mStatus_BadParamErr); } - else - { - service_instance *i; - mDNSu16 rrtype = 0; - LogOperation("%3d: DNSServiceRemoveRecord(%##s, %s)", request->sd, - (request->u.servicereg.instances) ? request->u.servicereg.instances->srs.RR_SRV.resrec.name->c : NULL, - rrtype ? DNSTypeName(rrtype) : ""); - for (i = request->u.servicereg.instances; i; i = i->next) - { - err = remove_extra(request, i, &rrtype); - if (err && i->default_local) break; - else err = mStatus_NoError; // suppress non-local default errors - } - } - - return(err); - } - -// If there's a comma followed by another character, -// FindFirstSubType overwrites the comma with a nul and returns the pointer to the next character. -// Otherwise, it returns a pointer to the final nul at the end of the string -mDNSlocal char *FindFirstSubType(char *p) - { - while (*p) - { - if (p[0] == '\\' && p[1]) p += 2; - else if (p[0] == ',' && p[1]) { *p++ = 0; return(p); } - else p++; - } - return(p); - } - -// If there's a comma followed by another character, -// FindNextSubType overwrites the comma with a nul and returns the pointer to the next character. -// If it finds an illegal unescaped dot in the subtype name, it returns mDNSNULL -// Otherwise, it returns a pointer to the final nul at the end of the string -mDNSlocal char *FindNextSubType(char *p) - { - while (*p) - { - if (p[0] == '\\' && p[1]) // If escape character - p += 2; // ignore following character - else if (p[0] == ',') // If we found a comma - { - if (p[1]) *p++ = 0; - return(p); - } - else if (p[0] == '.') - return(mDNSNULL); - else p++; - } - return(p); - } - -// Returns -1 if illegal subtype found -mDNSexport mDNSs32 ChopSubTypes(char *regtype) - { - mDNSs32 NumSubTypes = 0; - char *stp = FindFirstSubType(regtype); - while (stp && *stp) // If we found a comma... - { - if (*stp == ',') return(-1); - NumSubTypes++; - stp = FindNextSubType(stp); - } - if (!stp) return(-1); - return(NumSubTypes); - } - -mDNSexport AuthRecord *AllocateSubTypes(mDNSs32 NumSubTypes, char *p) - { - AuthRecord *st = mDNSNULL; - if (NumSubTypes) - { - mDNSs32 i; - st = mallocL("ServiceSubTypes", NumSubTypes * sizeof(AuthRecord)); - if (!st) return(mDNSNULL); - for (i = 0; i < NumSubTypes; i++) - { - mDNS_SetupResourceRecord(&st[i], mDNSNULL, mDNSInterface_Any, kDNSQType_ANY, kStandardTTL, 0, AuthRecordAny, mDNSNULL, mDNSNULL); - while (*p) p++; - p++; - if (!MakeDomainNameFromDNSNameString(&st[i].namestorage, p)) - { freeL("ServiceSubTypes", st); return(mDNSNULL); } - } - } - return(st); - } - -mDNSlocal mStatus register_service_instance(request_state *request, const domainname *domain) - { - service_instance **ptr, *instance; - const int extra_size = (request->u.servicereg.txtlen > sizeof(RDataBody)) ? (request->u.servicereg.txtlen - sizeof(RDataBody)) : 0; - const mDNSBool DomainIsLocal = SameDomainName(domain, &localdomain); - mStatus result; - mDNSInterfaceID interfaceID = request->u.servicereg.InterfaceID; - mDNSu32 regFlags = 0; - - if (interfaceID == mDNSInterface_P2P) - { - interfaceID = mDNSInterface_Any; - regFlags |= regFlagIncludeP2P; - } - else if (request->flags & kDNSServiceFlagsIncludeP2P) - regFlags |= regFlagIncludeP2P; - - // client guarantees that record names are unique - if (request->flags & kDNSServiceFlagsForce) - regFlags |= regFlagKnownUnique; - - // If the client specified an interface, but no domain, then we honor the specified interface for the "local" (mDNS) - // registration but for the wide-area registrations we don't (currently) have any concept of a wide-area unicast - // registrations scoped to a specific interface, so for the automatic domains we add we must *not* specify an interface. - // (Specifying an interface with an apparently wide-area domain (i.e. something other than "local") - // currently forces the registration to use mDNS multicast despite the apparently wide-area domain.) - if (request->u.servicereg.default_domain && !DomainIsLocal) interfaceID = mDNSInterface_Any; - - for (ptr = &request->u.servicereg.instances; *ptr; ptr = &(*ptr)->next) - { - if (SameDomainName(&(*ptr)->domain, domain)) - { - LogMsg("register_service_instance: domain %##s already registered for %#s.%##s", - domain->c, &request->u.servicereg.name, &request->u.servicereg.type); - return mStatus_AlreadyRegistered; - } - } - - if (mDNSStorage.KnownBugs & mDNS_KnownBug_LimitedIPv6) - { - // Special-case hack: On Mac OS X 10.6.x and earlier we don't advertise SMB service in AutoTunnel domains, - // because AutoTunnel services have to support IPv6, and in Mac OS X 10.6.x the SMB server does not. - // BTMM: Don't advertise SMB with BTMM because it doesn't support IPv6 - if (SameDomainName(&request->u.servicereg.type, (const domainname *) "\x4" "_smb" "\x4" "_tcp")) - { - DomainAuthInfo *AuthInfo = GetAuthInfoForName(&mDNSStorage, domain); - if (AuthInfo && AuthInfo->AutoTunnel) return(kDNSServiceErr_Unsupported); - } - } - - instance = mallocL("service_instance", sizeof(*instance) + extra_size); - if (!instance) { my_perror("ERROR: malloc"); return mStatus_NoMemoryErr; } - - instance->next = mDNSNULL; - instance->request = request; - instance->subtypes = AllocateSubTypes(request->u.servicereg.num_subtypes, request->u.servicereg.type_as_string); - instance->renameonmemfree = 0; - instance->clientnotified = mDNSfalse; - instance->default_local = (request->u.servicereg.default_domain && DomainIsLocal); - instance->external_advertise = mDNSfalse; - AssignDomainName(&instance->domain, domain); - - if (request->u.servicereg.num_subtypes && !instance->subtypes) - { unlink_and_free_service_instance(instance); instance = NULL; FatalError("ERROR: malloc"); } - - result = mDNS_RegisterService(&mDNSStorage, &instance->srs, - &request->u.servicereg.name, &request->u.servicereg.type, domain, - request->u.servicereg.host.c[0] ? &request->u.servicereg.host : NULL, - request->u.servicereg.port, - request->u.servicereg.txtdata, request->u.servicereg.txtlen, - instance->subtypes, request->u.servicereg.num_subtypes, - interfaceID, regservice_callback, instance, regFlags); - - if (!result) - { - *ptr = instance; // Append this to the end of our request->u.servicereg.instances list - LogOperation("%3d: DNSServiceRegister(%##s, %u) ADDED", - instance->request->sd, instance->srs.RR_SRV.resrec.name->c, mDNSVal16(request->u.servicereg.port)); - } - else - { - LogMsg("register_service_instance %#s.%##s%##s error %d", - &request->u.servicereg.name, &request->u.servicereg.type, domain->c, result); - unlink_and_free_service_instance(instance); - } - - return result; - } - -mDNSlocal void udsserver_default_reg_domain_changed(const DNameListElem *const d, const mDNSBool add) - { - request_state *request; - -#if APPLE_OSX_mDNSResponder - machserver_automatic_registration_domain_changed(&d->name, add); -#endif // APPLE_OSX_mDNSResponder - - LogMsg("%s registration domain %##s", add ? "Adding" : "Removing", d->name.c); - for (request = all_requests; request; request = request->next) - { - if (request->terminate != regservice_termination_callback) continue; - if (!request->u.servicereg.default_domain) continue; - if (!d->uid || SystemUID(request->uid) || request->uid == d->uid) - { - service_instance **ptr = &request->u.servicereg.instances; - while (*ptr && !SameDomainName(&(*ptr)->domain, &d->name)) ptr = &(*ptr)->next; - if (add) - { - // If we don't already have this domain in our list for this registration, add it now - if (!*ptr) register_service_instance(request, &d->name); - else debugf("udsserver_default_reg_domain_changed %##s already in list, not re-adding", &d->name); - } - else - { - // Normally we should not fail to find the specified instance - // One case where this can happen is if a uDNS update fails for some reason, - // and regservice_callback then calls unlink_and_free_service_instance and disposes of that instance. - if (!*ptr) - LogMsg("udsserver_default_reg_domain_changed domain %##s not found for service %#s type %s", - &d->name, request->u.servicereg.name.c, request->u.servicereg.type_as_string); - else - { - DNameListElem *p; - for (p = AutoRegistrationDomains; p; p=p->next) - if (!p->uid || SystemUID(request->uid) || request->uid == p->uid) - if (SameDomainName(&d->name, &p->name)) break; - if (p) debugf("udsserver_default_reg_domain_changed %##s still in list, not removing", &d->name); - else - { - mStatus err; - service_instance *si = *ptr; - *ptr = si->next; - if (si->clientnotified) SendServiceRemovalNotification(&si->srs); // Do this *before* clearing si->request backpointer - // Now that we've cut this service_instance from the list, we MUST clear the si->request backpointer. - // Otherwise what can happen is this: While our mDNS_DeregisterService is in the - // process of completing asynchronously, the client cancels the entire operation, so - // regservice_termination_callback then runs through the whole list deregistering each - // instance, clearing the backpointers, and then disposing the parent request_state object. - // However, because this service_instance isn't in the list any more, regservice_termination_callback - // has no way to find it and clear its backpointer, and then when our mDNS_DeregisterService finally - // completes later with a mStatus_MemFree message, it calls unlink_and_free_service_instance() with - // a service_instance with a stale si->request backpointer pointing to memory that's already been freed. - si->request = NULL; - err = mDNS_DeregisterService(&mDNSStorage, &si->srs); - if (err) { LogMsg("udsserver_default_reg_domain_changed err %d", err); unlink_and_free_service_instance(si); } - } - } - } - } - } - } - -mDNSlocal mStatus handle_regservice_request(request_state *request) - { - char name[256]; // Lots of spare space for extra-long names that we'll auto-truncate down to 63 bytes - char domain[MAX_ESCAPED_DOMAIN_NAME], host[MAX_ESCAPED_DOMAIN_NAME]; - char type_as_string[MAX_ESCAPED_DOMAIN_NAME]; - domainname d, srv; - mStatus err; - - DNSServiceFlags flags = get_flags(&request->msgptr, request->msgend); - mDNSu32 interfaceIndex = get_uint32(&request->msgptr, request->msgend); - mDNSInterfaceID InterfaceID = mDNSPlatformInterfaceIDfromInterfaceIndex(&mDNSStorage, interfaceIndex); - if (interfaceIndex && !InterfaceID) - { LogMsg("ERROR: handle_regservice_request - Couldn't find interfaceIndex %d", interfaceIndex); return(mStatus_BadParamErr); } - - if (get_string(&request->msgptr, request->msgend, name, sizeof(name)) < 0 || - get_string(&request->msgptr, request->msgend, type_as_string, MAX_ESCAPED_DOMAIN_NAME) < 0 || - get_string(&request->msgptr, request->msgend, domain, MAX_ESCAPED_DOMAIN_NAME) < 0 || - get_string(&request->msgptr, request->msgend, host, MAX_ESCAPED_DOMAIN_NAME) < 0) - { LogMsg("ERROR: handle_regservice_request - Couldn't read name/regtype/domain"); return(mStatus_BadParamErr); } - - request->flags = flags; - request->u.servicereg.InterfaceID = InterfaceID; - request->u.servicereg.instances = NULL; - request->u.servicereg.txtlen = 0; - request->u.servicereg.txtdata = NULL; - mDNSPlatformStrCopy(request->u.servicereg.type_as_string, type_as_string); - - if (request->msgptr + 2 > request->msgend) request->msgptr = NULL; - else - { - request->u.servicereg.port.b[0] = *request->msgptr++; - request->u.servicereg.port.b[1] = *request->msgptr++; - } - - request->u.servicereg.txtlen = get_uint16(&request->msgptr, request->msgend); - if (request->u.servicereg.txtlen) - { - request->u.servicereg.txtdata = mallocL("service_info txtdata", request->u.servicereg.txtlen); - if (!request->u.servicereg.txtdata) FatalError("ERROR: handle_regservice_request - malloc"); - mDNSPlatformMemCopy(request->u.servicereg.txtdata, get_rdata(&request->msgptr, request->msgend, request->u.servicereg.txtlen), request->u.servicereg.txtlen); - } - - if (!request->msgptr) { LogMsg("%3d: DNSServiceRegister(unreadable parameters)", request->sd); return(mStatus_BadParamErr); } - - // Check for sub-types after the service type - request->u.servicereg.num_subtypes = ChopSubTypes(request->u.servicereg.type_as_string); // Note: Modifies regtype string to remove trailing subtypes - if (request->u.servicereg.num_subtypes < 0) - { LogMsg("ERROR: handle_regservice_request - ChopSubTypes failed %s", request->u.servicereg.type_as_string); return(mStatus_BadParamErr); } - - // Don't try to construct "domainname t" until *after* ChopSubTypes has worked its magic - if (!*request->u.servicereg.type_as_string || !MakeDomainNameFromDNSNameString(&request->u.servicereg.type, request->u.servicereg.type_as_string)) - { LogMsg("ERROR: handle_regservice_request - type_as_string bad %s", request->u.servicereg.type_as_string); return(mStatus_BadParamErr); } - - if (!name[0]) - { - request->u.servicereg.name = mDNSStorage.nicelabel; - request->u.servicereg.autoname = mDNStrue; - } - else - { - // If the client is allowing AutoRename, then truncate name to legal length before converting it to a DomainLabel - if ((flags & kDNSServiceFlagsNoAutoRename) == 0) - { - int newlen = TruncateUTF8ToLength((mDNSu8*)name, mDNSPlatformStrLen(name), MAX_DOMAIN_LABEL); - name[newlen] = 0; - } - if (!MakeDomainLabelFromLiteralString(&request->u.servicereg.name, name)) - { LogMsg("ERROR: handle_regservice_request - name bad %s", name); return(mStatus_BadParamErr); } - request->u.servicereg.autoname = mDNSfalse; - } - - if (*domain) - { - request->u.servicereg.default_domain = mDNSfalse; - if (!MakeDomainNameFromDNSNameString(&d, domain)) - { LogMsg("ERROR: handle_regservice_request - domain bad %s", domain); return(mStatus_BadParamErr); } - } - else - { - request->u.servicereg.default_domain = mDNStrue; - MakeDomainNameFromDNSNameString(&d, "local."); - } - - if (!ConstructServiceName(&srv, &request->u.servicereg.name, &request->u.servicereg.type, &d)) - { - LogMsg("ERROR: handle_regservice_request - Couldn't ConstructServiceName from, '%#s' '%##s' '%##s'", - request->u.servicereg.name.c, request->u.servicereg.type.c, d.c); return(mStatus_BadParamErr); - } - - if (!MakeDomainNameFromDNSNameString(&request->u.servicereg.host, host)) - { LogMsg("ERROR: handle_regservice_request - host bad %s", host); return(mStatus_BadParamErr); } - request->u.servicereg.autorename = (flags & kDNSServiceFlagsNoAutoRename ) == 0; - request->u.servicereg.allowremotequery = (flags & kDNSServiceFlagsAllowRemoteQuery) != 0; - - // Some clients use mDNS for lightweight copy protection, registering a pseudo-service with - // a port number of zero. When two instances of the protected client are allowed to run on one - // machine, we don't want to see misleading "Bogus client" messages in syslog and the console. - if (!mDNSIPPortIsZero(request->u.servicereg.port)) - { - int count = CountExistingRegistrations(&srv, request->u.servicereg.port); - if (count) - LogMsg("Client application registered %d identical instances of service %##s port %u.", - count+1, srv.c, mDNSVal16(request->u.servicereg.port)); - } - - LogOperation("%3d: DNSServiceRegister(%X, %d, \"%s\", \"%s\", \"%s\", \"%s\", %u) START", - request->sd, flags, interfaceIndex, name, request->u.servicereg.type_as_string, domain, host, mDNSVal16(request->u.servicereg.port)); - - // We need to unconditionally set request->terminate, because even if we didn't successfully - // start any registrations right now, subsequent configuration changes may cause successful - // registrations to be added, and we'll need to cancel them before freeing this memory. - // We also need to set request->terminate first, before adding additional service instances, - // because the uds_validatelists uses the request->terminate function pointer to determine - // what kind of request this is, and therefore what kind of list validation is required. - request->terminate = regservice_termination_callback; - - err = register_service_instance(request, &d); - -#if 0 - err = AuthorizedDomain(request, &d, AutoRegistrationDomains) ? register_service_instance(request, &d) : mStatus_NoError; -#endif - if (!err) - { - if (request->u.servicereg.autoname) UpdateDeviceInfoRecord(&mDNSStorage); - - if (!*domain) - { - DNameListElem *ptr; - // Note that we don't report errors for non-local, non-explicit domains - for (ptr = AutoRegistrationDomains; ptr; ptr = ptr->next) - if (!ptr->uid || SystemUID(request->uid) || request->uid == ptr->uid) - register_service_instance(request, &ptr->name); - } - } - - return(err); - } - -// *************************************************************************** -#if COMPILER_LIKES_PRAGMA_MARK -#pragma mark - -#pragma mark - DNSServiceBrowse -#endif - -mDNSlocal void FoundInstance(mDNS *const m, DNSQuestion *question, const ResourceRecord *const answer, QC_result AddRecord) - { - const DNSServiceFlags flags = AddRecord ? kDNSServiceFlagsAdd : 0; - request_state *req = question->QuestionContext; - reply_state *rep; - (void)m; // Unused - - if (answer->rrtype != kDNSType_PTR) - { LogMsg("%3d: FoundInstance: Should not be called with rrtype %d (not a PTR record)", req->sd, answer->rrtype); return; } - - if (GenerateNTDResponse(&answer->rdata->u.name, answer->InterfaceID, req, &rep, browse_reply_op, flags, mStatus_NoError) != mStatus_NoError) - { - if (SameDomainName(&req->u.browser.regtype, (const domainname*)"\x09_services\x07_dns-sd\x04_udp")) - { - // Special support to enable the DNSServiceBrowse call made by Bonjour Browser - // Remove after Bonjour Browser is updated to use DNSServiceQueryRecord instead of DNSServiceBrowse - GenerateBonjourBrowserResponse(&answer->rdata->u.name, answer->InterfaceID, req, &rep, browse_reply_op, flags, mStatus_NoError); - goto bonjourbrowserhack; - } - - LogMsg("%3d: FoundInstance: %##s PTR %##s received from network is not valid DNS-SD service pointer", - req->sd, answer->name->c, answer->rdata->u.name.c); - return; - } - -bonjourbrowserhack: - - LogOperation("%3d: DNSServiceBrowse(%##s, %s) RESULT %s %d: %s", - req->sd, question->qname.c, DNSTypeName(question->qtype), AddRecord ? "Add" : "Rmv", - mDNSPlatformInterfaceIndexfromInterfaceID(m, answer->InterfaceID, mDNSfalse), RRDisplayString(m, answer)); - - append_reply(req, rep); - } - -mDNSlocal mStatus add_domain_to_browser(request_state *info, const domainname *d) - { - browser_t *b, *p; - mStatus err; - - for (p = info->u.browser.browsers; p; p = p->next) - { - if (SameDomainName(&p->domain, d)) - { debugf("add_domain_to_browser %##s already in list", d->c); return mStatus_AlreadyRegistered; } - } - - b = mallocL("browser_t", sizeof(*b)); - if (!b) return mStatus_NoMemoryErr; - AssignDomainName(&b->domain, d); - err = mDNS_StartBrowse(&mDNSStorage, &b->q, - &info->u.browser.regtype, d, info->u.browser.interface_id, info->u.browser.ForceMCast, FoundInstance, info); - if (err) - { - LogMsg("mDNS_StartBrowse returned %d for type %##s domain %##s", err, info->u.browser.regtype.c, d->c); - freeL("browser_t/add_domain_to_browser", b); - } - else - { - b->next = info->u.browser.browsers; - info->u.browser.browsers = b; - LogOperation("%3d: DNSServiceBrowse(%##s) START", info->sd, b->q.qname.c); - if (info->u.browser.interface_id == mDNSInterface_P2P || (!info->u.browser.interface_id && SameDomainName(&b->domain, &localdomain) && (info->flags & kDNSServiceFlagsIncludeP2P))) - { - domainname tmp; - ConstructServiceName(&tmp, NULL, &info->u.browser.regtype, &b->domain); - LogInfo("add_domain_to_browser: calling external_start_browsing_for_service()"); - external_start_browsing_for_service(&mDNSStorage, &tmp, kDNSType_PTR); - } - } - return err; - } - -mDNSlocal void browse_termination_callback(request_state *info) - { - while (info->u.browser.browsers) - { - browser_t *ptr = info->u.browser.browsers; - - if (info->u.browser.interface_id == mDNSInterface_P2P || (!info->u.browser.interface_id && SameDomainName(&ptr->domain, &localdomain) && (info->flags & kDNSServiceFlagsIncludeP2P))) - { - domainname tmp; - ConstructServiceName(&tmp, NULL, &info->u.browser.regtype, &ptr->domain); - LogInfo("browse_termination_callback: calling external_stop_browsing_for_service()"); - external_stop_browsing_for_service(&mDNSStorage, &tmp, kDNSType_PTR); - } - - info->u.browser.browsers = ptr->next; - LogOperation("%3d: DNSServiceBrowse(%##s) STOP", info->sd, ptr->q.qname.c); - mDNS_StopBrowse(&mDNSStorage, &ptr->q); // no need to error-check result - freeL("browser_t/browse_termination_callback", ptr); - } - } - -mDNSlocal void udsserver_automatic_browse_domain_changed(const DNameListElem *const d, const mDNSBool add) - { - request_state *request; - debugf("udsserver_automatic_browse_domain_changed: %s default browse domain %##s", add ? "Adding" : "Removing", d->name.c); - -#if APPLE_OSX_mDNSResponder - machserver_automatic_browse_domain_changed(&d->name, add); -#endif // APPLE_OSX_mDNSResponder - - for (request = all_requests; request; request = request->next) - { - if (request->terminate != browse_termination_callback) continue; // Not a browse operation - if (!request->u.browser.default_domain) continue; // Not an auto-browse operation - if (!d->uid || SystemUID(request->uid) || request->uid == d->uid) - { - browser_t **ptr = &request->u.browser.browsers; - while (*ptr && !SameDomainName(&(*ptr)->domain, &d->name)) ptr = &(*ptr)->next; - if (add) - { - // If we don't already have this domain in our list for this browse operation, add it now - if (!*ptr) add_domain_to_browser(request, &d->name); - else debugf("udsserver_automatic_browse_domain_changed %##s already in list, not re-adding", &d->name); - } - else - { - if (!*ptr) LogMsg("udsserver_automatic_browse_domain_changed ERROR %##s not found", &d->name); - else - { - DNameListElem *p; - for (p = AutoBrowseDomains; p; p=p->next) - if (!p->uid || SystemUID(request->uid) || request->uid == p->uid) - if (SameDomainName(&d->name, &p->name)) break; - if (p) debugf("udsserver_automatic_browse_domain_changed %##s still in list, not removing", &d->name); - else - { - browser_t *rem = *ptr; - *ptr = (*ptr)->next; - mDNS_StopQueryWithRemoves(&mDNSStorage, &rem->q); - freeL("browser_t/udsserver_automatic_browse_domain_changed", rem); - } - } - } - } - } - } - -mDNSlocal void FreeARElemCallback(mDNS *const m, AuthRecord *const rr, mStatus result) - { - (void)m; // unused - if (result == mStatus_MemFree) - { - // On shutdown, mDNS_Close automatically deregisters all records - // Since in this case no one has called DeregisterLocalOnlyDomainEnumPTR to cut the record - // from the LocalDomainEnumRecords list, we do this here before we free the memory. - // (This should actually no longer be necessary, now that we do the proper cleanup in - // udsserver_exit. To confirm this, we'll log an error message if we do find a record that - // hasn't been cut from the list yet. If these messages don't appear, we can delete this code.) - ARListElem **ptr = &LocalDomainEnumRecords; - while (*ptr && &(*ptr)->ar != rr) ptr = &(*ptr)->next; - if (*ptr) { *ptr = (*ptr)->next; LogMsg("FreeARElemCallback: Have to cut %s", ARDisplayString(m, rr)); } - mDNSPlatformMemFree(rr->RecordContext); - } - } - -// RegisterLocalOnlyDomainEnumPTR and DeregisterLocalOnlyDomainEnumPTR largely duplicate code in -// "FoundDomain" in uDNS.c for creating and destroying these special mDNSInterface_LocalOnly records. -// We may want to turn the common code into a subroutine. - -mDNSlocal void RegisterLocalOnlyDomainEnumPTR(mDNS *m, const domainname *d, int type) - { - // allocate/register legacy and non-legacy _browse PTR record - mStatus err; - ARListElem *ptr = mDNSPlatformMemAllocate(sizeof(*ptr)); - - debugf("Incrementing %s refcount for %##s", - (type == mDNS_DomainTypeBrowse ) ? "browse domain " : - (type == mDNS_DomainTypeRegistration ) ? "registration dom" : - (type == mDNS_DomainTypeBrowseAutomatic) ? "automatic browse" : "?", d->c); - - mDNS_SetupResourceRecord(&ptr->ar, mDNSNULL, mDNSInterface_LocalOnly, kDNSType_PTR, 7200, kDNSRecordTypeShared, AuthRecordLocalOnly, FreeARElemCallback, ptr); - MakeDomainNameFromDNSNameString(&ptr->ar.namestorage, mDNS_DomainTypeNames[type]); - AppendDNSNameString (&ptr->ar.namestorage, "local"); - AssignDomainName(&ptr->ar.resrec.rdata->u.name, d); - err = mDNS_Register(m, &ptr->ar); - if (err) - { - LogMsg("SetSCPrefsBrowseDomain: mDNS_Register returned error %d", err); - mDNSPlatformMemFree(ptr); - } - else - { - ptr->next = LocalDomainEnumRecords; - LocalDomainEnumRecords = ptr; - } - } - -mDNSlocal void DeregisterLocalOnlyDomainEnumPTR(mDNS *m, const domainname *d, int type) - { - ARListElem **ptr = &LocalDomainEnumRecords; - domainname lhs; // left-hand side of PTR, for comparison - - debugf("Decrementing %s refcount for %##s", - (type == mDNS_DomainTypeBrowse ) ? "browse domain " : - (type == mDNS_DomainTypeRegistration ) ? "registration dom" : - (type == mDNS_DomainTypeBrowseAutomatic) ? "automatic browse" : "?", d->c); - - MakeDomainNameFromDNSNameString(&lhs, mDNS_DomainTypeNames[type]); - AppendDNSNameString (&lhs, "local"); - - while (*ptr) - { - if (SameDomainName(&(*ptr)->ar.resrec.rdata->u.name, d) && SameDomainName((*ptr)->ar.resrec.name, &lhs)) - { - ARListElem *rem = *ptr; - *ptr = (*ptr)->next; - mDNS_Deregister(m, &rem->ar); - return; - } - else ptr = &(*ptr)->next; - } - } - -mDNSlocal void AddAutoBrowseDomain(const mDNSu32 uid, const domainname *const name) - { - DNameListElem *new = mDNSPlatformMemAllocate(sizeof(DNameListElem)); - if (!new) { LogMsg("ERROR: malloc"); return; } - AssignDomainName(&new->name, name); - new->uid = uid; - new->next = AutoBrowseDomains; - AutoBrowseDomains = new; - udsserver_automatic_browse_domain_changed(new, mDNStrue); - } - -mDNSlocal void RmvAutoBrowseDomain(const mDNSu32 uid, const domainname *const name) - { - DNameListElem **p = &AutoBrowseDomains; - while (*p && (!SameDomainName(&(*p)->name, name) || (*p)->uid != uid)) p = &(*p)->next; - if (!*p) LogMsg("RmvAutoBrowseDomain: Got remove event for domain %##s not in list", name->c); - else - { - DNameListElem *ptr = *p; - *p = ptr->next; - udsserver_automatic_browse_domain_changed(ptr, mDNSfalse); - mDNSPlatformMemFree(ptr); - } - } - -mDNSlocal void SetPrefsBrowseDomains(mDNS *m, DNameListElem *browseDomains, mDNSBool add) - { - DNameListElem *d; - for (d = browseDomains; d; d = d->next) - { - if (add) - { - RegisterLocalOnlyDomainEnumPTR(m, &d->name, mDNS_DomainTypeBrowse); - AddAutoBrowseDomain(d->uid, &d->name); - } - else - { - DeregisterLocalOnlyDomainEnumPTR(m, &d->name, mDNS_DomainTypeBrowse); - RmvAutoBrowseDomain(d->uid, &d->name); - } - } - } - -mDNSlocal void UpdateDeviceInfoRecord(mDNS *const m) - { - int num_autoname = 0; - request_state *req; - for (req = all_requests; req; req = req->next) - if (req->terminate == regservice_termination_callback && req->u.servicereg.autoname) - num_autoname++; - - // If DeviceInfo record is currently registered, see if we need to deregister it - if (m->DeviceInfo.resrec.RecordType != kDNSRecordTypeUnregistered) - if (num_autoname == 0 || !SameDomainLabelCS(m->DeviceInfo.resrec.name->c, m->nicelabel.c)) - { - LogOperation("UpdateDeviceInfoRecord Deregister %##s", m->DeviceInfo.resrec.name); - mDNS_Deregister(m, &m->DeviceInfo); - } - - // If DeviceInfo record is not currently registered, see if we need to register it - if (m->DeviceInfo.resrec.RecordType == kDNSRecordTypeUnregistered) - if (num_autoname > 0) - { - mDNSu8 len = m->HIHardware.c[0] < 255 - 6 ? m->HIHardware.c[0] : 255 - 6; - mDNS_SetupResourceRecord(&m->DeviceInfo, mDNSNULL, mDNSNULL, kDNSType_TXT, kStandardTTL, kDNSRecordTypeAdvisory, AuthRecordAny, mDNSNULL, mDNSNULL); - ConstructServiceName(&m->DeviceInfo.namestorage, &m->nicelabel, &DeviceInfoName, &localdomain); - mDNSPlatformMemCopy(m->DeviceInfo.resrec.rdata->u.data + 1, "model=", 6); - mDNSPlatformMemCopy(m->DeviceInfo.resrec.rdata->u.data + 7, m->HIHardware.c + 1, len); - m->DeviceInfo.resrec.rdata->u.data[0] = 6 + len; // "model=" plus the device string - m->DeviceInfo.resrec.rdlength = 7 + len; // One extra for the length byte at the start of the string - LogOperation("UpdateDeviceInfoRecord Register %##s", m->DeviceInfo.resrec.name); - mDNS_Register(m, &m->DeviceInfo); - } - } - -mDNSexport void udsserver_handle_configchange(mDNS *const m) - { - request_state *req; - service_instance *ptr; - DNameListElem *RegDomains = NULL; - DNameListElem *BrowseDomains = NULL; - DNameListElem *p; - - UpdateDeviceInfoRecord(m); - - // For autoname services, see if the default service name has changed, necessitating an automatic update - for (req = all_requests; req; req = req->next) - if (req->terminate == regservice_termination_callback) - if (req->u.servicereg.autoname && !SameDomainLabelCS(req->u.servicereg.name.c, m->nicelabel.c)) - { - req->u.servicereg.name = m->nicelabel; - for (ptr = req->u.servicereg.instances; ptr; ptr = ptr->next) - { - ptr->renameonmemfree = 1; - if (ptr->clientnotified) SendServiceRemovalNotification(&ptr->srs); - LogInfo("udsserver_handle_configchange: Calling deregister for Service %##s", ptr->srs.RR_PTR.resrec.name->c); - if (mDNS_DeregisterService_drt(m, &ptr->srs, mDNS_Dereg_rapid)) - regservice_callback(m, &ptr->srs, mStatus_MemFree); // If service deregistered already, we can re-register immediately - } - } - - // Let the platform layer get the current DNS information - mDNS_Lock(m); - mDNSPlatformSetDNSConfig(m, mDNSfalse, mDNSfalse, mDNSNULL, &RegDomains, &BrowseDomains); - mDNS_Unlock(m); - - // Any automatic registration domains are also implicitly automatic browsing domains - if (RegDomains) SetPrefsBrowseDomains(m, RegDomains, mDNStrue); // Add the new list first - if (AutoRegistrationDomains) SetPrefsBrowseDomains(m, AutoRegistrationDomains, mDNSfalse); // Then clear the old list - - // Add any new domains not already in our AutoRegistrationDomains list - for (p=RegDomains; p; p=p->next) - { - DNameListElem **pp = &AutoRegistrationDomains; - while (*pp && ((*pp)->uid != p->uid || !SameDomainName(&(*pp)->name, &p->name))) pp = &(*pp)->next; - if (!*pp) // If not found in our existing list, this is a new default registration domain - { - RegisterLocalOnlyDomainEnumPTR(m, &p->name, mDNS_DomainTypeRegistration); - udsserver_default_reg_domain_changed(p, mDNStrue); - } - else // else found same domainname in both old and new lists, so no change, just delete old copy - { - DNameListElem *del = *pp; - *pp = (*pp)->next; - mDNSPlatformMemFree(del); - } - } - - // Delete any domains in our old AutoRegistrationDomains list that are now gone - while (AutoRegistrationDomains) - { - DNameListElem *del = AutoRegistrationDomains; - AutoRegistrationDomains = AutoRegistrationDomains->next; // Cut record from list FIRST, - DeregisterLocalOnlyDomainEnumPTR(m, &del->name, mDNS_DomainTypeRegistration); - udsserver_default_reg_domain_changed(del, mDNSfalse); // before calling udsserver_default_reg_domain_changed() - mDNSPlatformMemFree(del); - } - - // Now we have our new updated automatic registration domain list - AutoRegistrationDomains = RegDomains; - - // Add new browse domains to internal list - if (BrowseDomains) SetPrefsBrowseDomains(m, BrowseDomains, mDNStrue); - - // Remove old browse domains from internal list - if (SCPrefBrowseDomains) - { - SetPrefsBrowseDomains(m, SCPrefBrowseDomains, mDNSfalse); - while (SCPrefBrowseDomains) - { - DNameListElem *fptr = SCPrefBrowseDomains; - SCPrefBrowseDomains = SCPrefBrowseDomains->next; - mDNSPlatformMemFree(fptr); - } - } - - // Replace the old browse domains array with the new array - SCPrefBrowseDomains = BrowseDomains; - } - -mDNSlocal void AutomaticBrowseDomainChange(mDNS *const m, DNSQuestion *q, const ResourceRecord *const answer, QC_result AddRecord) - { - (void)m; // unused; - (void)q; // unused - - LogOperation("AutomaticBrowseDomainChange: %s automatic browse domain %##s", - AddRecord ? "Adding" : "Removing", answer->rdata->u.name.c); - - if (AddRecord) AddAutoBrowseDomain(0, &answer->rdata->u.name); - else RmvAutoBrowseDomain(0, &answer->rdata->u.name); - } - -mDNSlocal mStatus handle_browse_request(request_state *request) - { - char regtype[MAX_ESCAPED_DOMAIN_NAME], domain[MAX_ESCAPED_DOMAIN_NAME]; - domainname typedn, d, temp; - mDNSs32 NumSubTypes; - mStatus err = mStatus_NoError; - - DNSServiceFlags flags = get_flags(&request->msgptr, request->msgend); - mDNSu32 interfaceIndex = get_uint32(&request->msgptr, request->msgend); - mDNSInterfaceID InterfaceID = mDNSPlatformInterfaceIDfromInterfaceIndex(&mDNSStorage, interfaceIndex); - if (interfaceIndex && !InterfaceID) return(mStatus_BadParamErr); - - if (get_string(&request->msgptr, request->msgend, regtype, MAX_ESCAPED_DOMAIN_NAME) < 0 || - get_string(&request->msgptr, request->msgend, domain, MAX_ESCAPED_DOMAIN_NAME) < 0) return(mStatus_BadParamErr); - - if (!request->msgptr) { LogMsg("%3d: DNSServiceBrowse(unreadable parameters)", request->sd); return(mStatus_BadParamErr); } - - if (domain[0] == '\0') uDNS_SetupSearchDomains(&mDNSStorage, UDNS_START_WAB_QUERY); - - request->flags = flags; - typedn.c[0] = 0; - NumSubTypes = ChopSubTypes(regtype); // Note: Modifies regtype string to remove trailing subtypes - if (NumSubTypes < 0 || NumSubTypes > 1) return(mStatus_BadParamErr); - if (NumSubTypes == 1 && !AppendDNSNameString(&typedn, regtype + strlen(regtype) + 1)) return(mStatus_BadParamErr); - - if (!regtype[0] || !AppendDNSNameString(&typedn, regtype)) return(mStatus_BadParamErr); - - if (!MakeDomainNameFromDNSNameString(&temp, regtype)) return(mStatus_BadParamErr); - // For over-long service types, we only allow domain "local" - if (temp.c[0] > 15 && domain[0] == 0) mDNSPlatformStrCopy(domain, "local."); - - // Set up browser info - request->u.browser.ForceMCast = (flags & kDNSServiceFlagsForceMulticast) != 0; - request->u.browser.interface_id = InterfaceID; - AssignDomainName(&request->u.browser.regtype, &typedn); - request->u.browser.default_domain = !domain[0]; - request->u.browser.browsers = NULL; - - LogOperation("%3d: DNSServiceBrowse(%X, %d, \"%##s\", \"%s\") START", - request->sd, request->flags, interfaceIndex, request->u.browser.regtype.c, domain); - - // We need to unconditionally set request->terminate, because even if we didn't successfully - // start any browses right now, subsequent configuration changes may cause successful - // browses to be added, and we'll need to cancel them before freeing this memory. - request->terminate = browse_termination_callback; - - if (domain[0]) - { - if (!MakeDomainNameFromDNSNameString(&d, domain)) return(mStatus_BadParamErr); - err = add_domain_to_browser(request, &d); -#if 0 - err = AuthorizedDomain(request, &d, AutoBrowseDomains) ? add_domain_to_browser(request, &d) : mStatus_NoError; -#endif - } - else - { - DNameListElem *sdom; - for (sdom = AutoBrowseDomains; sdom; sdom = sdom->next) - if (!sdom->uid || SystemUID(request->uid) || request->uid == sdom->uid) - { - err = add_domain_to_browser(request, &sdom->name); - if (err) - { - if (SameDomainName(&sdom->name, &localdomain)) break; - else err = mStatus_NoError; // suppress errors for non-local "default" domains - } - } - } - - return(err); - } - -// *************************************************************************** -#if COMPILER_LIKES_PRAGMA_MARK -#pragma mark - -#pragma mark - DNSServiceResolve -#endif - -mDNSlocal void resolve_result_callback(mDNS *const m, DNSQuestion *question, const ResourceRecord *const answer, QC_result AddRecord) - { - size_t len = 0; - char fullname[MAX_ESCAPED_DOMAIN_NAME], target[MAX_ESCAPED_DOMAIN_NAME]; - char *data; - reply_state *rep; - request_state *req = question->QuestionContext; - (void)m; // Unused - - LogOperation("%3d: DNSServiceResolve(%##s) %s %s", req->sd, question->qname.c, AddRecord ? "ADD" : "RMV", RRDisplayString(m, answer)); - - if (!AddRecord) - { - if (req->u.resolve.srv == answer) req->u.resolve.srv = mDNSNULL; - if (req->u.resolve.txt == answer) req->u.resolve.txt = mDNSNULL; - return; - } - - if (answer->rrtype == kDNSType_SRV) req->u.resolve.srv = answer; - if (answer->rrtype == kDNSType_TXT) req->u.resolve.txt = answer; - - if (!req->u.resolve.txt || !req->u.resolve.srv) return; // only deliver result to client if we have both answers - - ConvertDomainNameToCString(answer->name, fullname); - ConvertDomainNameToCString(&req->u.resolve.srv->rdata->u.srv.target, target); - - // calculate reply length - len += sizeof(DNSServiceFlags); - len += sizeof(mDNSu32); // interface index - len += sizeof(DNSServiceErrorType); - len += strlen(fullname) + 1; - len += strlen(target) + 1; - len += 2 * sizeof(mDNSu16); // port, txtLen - len += req->u.resolve.txt->rdlength; - - // allocate/init reply header - rep = create_reply(resolve_reply_op, len, req); - rep->rhdr->flags = dnssd_htonl(0); - rep->rhdr->ifi = dnssd_htonl(mDNSPlatformInterfaceIndexfromInterfaceID(m, answer->InterfaceID, mDNSfalse)); - rep->rhdr->error = dnssd_htonl(kDNSServiceErr_NoError); - - data = (char *)&rep->rhdr[1]; - - // write reply data to message - put_string(fullname, &data); - put_string(target, &data); - *data++ = req->u.resolve.srv->rdata->u.srv.port.b[0]; - *data++ = req->u.resolve.srv->rdata->u.srv.port.b[1]; - put_uint16(req->u.resolve.txt->rdlength, &data); - put_rdata (req->u.resolve.txt->rdlength, req->u.resolve.txt->rdata->u.data, &data); - - LogOperation("%3d: DNSServiceResolve(%s) RESULT %s:%d", req->sd, fullname, target, mDNSVal16(req->u.resolve.srv->rdata->u.srv.port)); - append_reply(req, rep); - } - -mDNSlocal void resolve_termination_callback(request_state *request) - { - LogOperation("%3d: DNSServiceResolve(%##s) STOP", request->sd, request->u.resolve.qtxt.qname.c); - mDNS_StopQuery(&mDNSStorage, &request->u.resolve.qtxt); - mDNS_StopQuery(&mDNSStorage, &request->u.resolve.qsrv); - if (request->u.resolve.external_advertise) external_stop_resolving_service(&request->u.resolve.qsrv.qname); - } - -mDNSlocal mStatus handle_resolve_request(request_state *request) - { - char name[256], regtype[MAX_ESCAPED_DOMAIN_NAME], domain[MAX_ESCAPED_DOMAIN_NAME]; - domainname fqdn; - mStatus err; - - // extract the data from the message - DNSServiceFlags flags = get_flags(&request->msgptr, request->msgend); - mDNSu32 interfaceIndex = get_uint32(&request->msgptr, request->msgend); - mDNSInterfaceID InterfaceID; - mDNSBool wasP2P = (interfaceIndex == kDNSServiceInterfaceIndexP2P); - - - request->flags = flags; - if (wasP2P) interfaceIndex = kDNSServiceInterfaceIndexAny; - - InterfaceID = mDNSPlatformInterfaceIDfromInterfaceIndex(&mDNSStorage, interfaceIndex); - if (interfaceIndex && !InterfaceID) - { LogMsg("ERROR: handle_resolve_request bad interfaceIndex %d", interfaceIndex); return(mStatus_BadParamErr); } - - if (get_string(&request->msgptr, request->msgend, name, 256) < 0 || - get_string(&request->msgptr, request->msgend, regtype, MAX_ESCAPED_DOMAIN_NAME) < 0 || - get_string(&request->msgptr, request->msgend, domain, MAX_ESCAPED_DOMAIN_NAME) < 0) - { LogMsg("ERROR: handle_resolve_request - Couldn't read name/regtype/domain"); return(mStatus_BadParamErr); } - - if (!request->msgptr) { LogMsg("%3d: DNSServiceResolve(unreadable parameters)", request->sd); return(mStatus_BadParamErr); } - - if (build_domainname_from_strings(&fqdn, name, regtype, domain) < 0) - { LogMsg("ERROR: handle_resolve_request bad '%s' '%s' '%s'", name, regtype, domain); return(mStatus_BadParamErr); } - - mDNSPlatformMemZero(&request->u.resolve, sizeof(request->u.resolve)); - - // format questions - request->u.resolve.qsrv.InterfaceID = InterfaceID; - request->u.resolve.qsrv.Target = zeroAddr; - AssignDomainName(&request->u.resolve.qsrv.qname, &fqdn); - request->u.resolve.qsrv.qtype = kDNSType_SRV; - request->u.resolve.qsrv.qclass = kDNSClass_IN; - request->u.resolve.qsrv.LongLived = (flags & kDNSServiceFlagsLongLivedQuery ) != 0; - request->u.resolve.qsrv.ExpectUnique = mDNStrue; - request->u.resolve.qsrv.ForceMCast = (flags & kDNSServiceFlagsForceMulticast ) != 0; - request->u.resolve.qsrv.ReturnIntermed = (flags & kDNSServiceFlagsReturnIntermediates) != 0; - request->u.resolve.qsrv.SuppressUnusable = mDNSfalse; - request->u.resolve.qsrv.SearchListIndex = 0; - request->u.resolve.qsrv.AppendSearchDomains = 0; - request->u.resolve.qsrv.RetryWithSearchDomains = mDNSfalse; - request->u.resolve.qsrv.TimeoutQuestion = 0; - request->u.resolve.qsrv.WakeOnResolve = (flags & kDNSServiceFlagsWakeOnResolve) != 0; - request->u.resolve.qsrv.qnameOrig = mDNSNULL; - request->u.resolve.qsrv.QuestionCallback = resolve_result_callback; - request->u.resolve.qsrv.QuestionContext = request; - - request->u.resolve.qtxt.InterfaceID = InterfaceID; - request->u.resolve.qtxt.Target = zeroAddr; - AssignDomainName(&request->u.resolve.qtxt.qname, &fqdn); - request->u.resolve.qtxt.qtype = kDNSType_TXT; - request->u.resolve.qtxt.qclass = kDNSClass_IN; - request->u.resolve.qtxt.LongLived = (flags & kDNSServiceFlagsLongLivedQuery ) != 0; - request->u.resolve.qtxt.ExpectUnique = mDNStrue; - request->u.resolve.qtxt.ForceMCast = (flags & kDNSServiceFlagsForceMulticast ) != 0; - request->u.resolve.qtxt.ReturnIntermed = (flags & kDNSServiceFlagsReturnIntermediates) != 0; - request->u.resolve.qtxt.SuppressUnusable = mDNSfalse; - request->u.resolve.qtxt.SearchListIndex = 0; - request->u.resolve.qtxt.AppendSearchDomains = 0; - request->u.resolve.qtxt.RetryWithSearchDomains = mDNSfalse; - request->u.resolve.qtxt.TimeoutQuestion = 0; - request->u.resolve.qtxt.WakeOnResolve = 0; - request->u.resolve.qtxt.qnameOrig = mDNSNULL; - request->u.resolve.qtxt.QuestionCallback = resolve_result_callback; - request->u.resolve.qtxt.QuestionContext = request; - - request->u.resolve.ReportTime = NonZeroTime(mDNS_TimeNow(&mDNSStorage) + 130 * mDNSPlatformOneSecond); - - request->u.resolve.external_advertise = mDNSfalse; - -#if 0 - if (!AuthorizedDomain(request, &fqdn, AutoBrowseDomains)) return(mStatus_NoError); -#endif - - // ask the questions - LogOperation("%3d: DNSServiceResolve(%##s) START", request->sd, request->u.resolve.qsrv.qname.c); - err = mDNS_StartQuery(&mDNSStorage, &request->u.resolve.qsrv); - if (!err) - { - err = mDNS_StartQuery(&mDNSStorage, &request->u.resolve.qtxt); - if (err) mDNS_StopQuery(&mDNSStorage, &request->u.resolve.qsrv); - else - { - request->terminate = resolve_termination_callback; - // If the user explicitly passed in P2P, we don't restrict the domain in which we resolve. - if (wasP2P || (!InterfaceID && IsLocalDomain(&fqdn) && (request->flags & kDNSServiceFlagsIncludeP2P))) - { - request->u.resolve.external_advertise = mDNStrue; - LogInfo("handle_resolve_request: calling external_start_resolving_service()"); - external_start_resolving_service(&fqdn); - } - } - } - - return(err); - } - -// *************************************************************************** -#if COMPILER_LIKES_PRAGMA_MARK -#pragma mark - -#pragma mark - DNSServiceQueryRecord -#endif - -// mDNS operation functions. Each operation has 3 associated functions - a request handler that parses -// the client's request and makes the appropriate mDNSCore call, a result handler (passed as a callback -// to the mDNSCore routine) that sends results back to the client, and a termination routine that aborts -// the mDNSCore operation if the client dies or closes its socket. - -// Returns -1 to tell the caller that it should not try to reissue the query anymore -// Returns 1 on successfully appending a search domain and the caller should reissue the new query -// Returns 0 when there are no more search domains and the caller should reissue the query -mDNSlocal int AppendNewSearchDomain(mDNS *const m, DNSQuestion *question) - { - domainname *sd; - mStatus err; - - // Sanity check: The caller already checks this. We use -1 to indicate that we have searched all - // the domains and should try the single label query directly on the wire. - if (question->SearchListIndex == -1) - { - LogMsg("AppendNewSearchDomain: question %##s (%s) SearchListIndex is -1", question->qname.c, DNSTypeName(question->qtype)); - return -1; - } - - if (!question->AppendSearchDomains) - { - LogMsg("AppendNewSearchDomain: question %##s (%s) AppendSearchDoamins is 0", question->qname.c, DNSTypeName(question->qtype)); - return -1; - } - - // Save the original name, before we modify them below. - if (!question->qnameOrig) - { - question->qnameOrig = mallocL("AppendNewSearchDomain", sizeof(domainname)); - if (!question->qnameOrig) { LogMsg("AppendNewSearchDomain: ERROR!! malloc failure"); return -1; } - question->qnameOrig->c[0] = 0; - AssignDomainName(question->qnameOrig, &question->qname); - LogInfo("AppendSearchDomain: qnameOrig %##s", question->qnameOrig->c); - } - - sd = uDNS_GetNextSearchDomain(m, question->InterfaceID, &question->SearchListIndex, !question->AppendLocalSearchDomains); - // We use -1 to indicate that we have searched all the domains and should try the single label - // query directly on the wire. uDNS_GetNextSearchDomain should never return a negative value - if (question->SearchListIndex == -1) - { - LogMsg("AppendNewSearchDomain: ERROR!! uDNS_GetNextSearchDomain returned -1"); - return -1; - } - - // Not a common case. Perhaps, we should try the next search domain if it exceeds ? - if (sd && (DomainNameLength(question->qnameOrig) + DomainNameLength(sd)) > MAX_DOMAIN_NAME) - { - LogMsg("AppendNewSearchDomain: ERROR!! exceeding max domain length for %##s (%s) SearchDomain %##s length %d, Question name length %d", question->qnameOrig->c, DNSTypeName(question->qtype), sd->c, DomainNameLength(question->qnameOrig), DomainNameLength(sd)); - return -1; - } - - // if there are no more search domains and we have already tried this question - // without appending search domains, then we are done. - if (!sd && !ApplySearchDomainsFirst(question)) - { - LogInfo("AppnedNewSearchDomain: No more search domains for question with name %##s (%s), not trying anymore", question->qname.c, DNSTypeName(question->qtype)); - return -1; - } - - // Stop the question before changing the name as negative cache entries could be pointing at this question. - // Even if we don't change the question in the case of returning 0, the caller is going to restart the - // question. - err = mDNS_StopQuery(&mDNSStorage, question); - if (err) { LogMsg("AppendNewSearchDomain: ERROR!! %##s %s mDNS_StopQuery: %d, while retrying with search domains", question->qname.c, DNSTypeName(question->qtype), (int)err); } - - AssignDomainName(&question->qname, question->qnameOrig); - if (sd) - { - AppendDomainName(&question->qname, sd); - LogInfo("AppnedNewSearchDomain: Returning question with name %##s, SearchListIndex %d", question->qname.c, question->SearchListIndex); - return 1; - } - - // Try the question as single label - LogInfo("AppnedNewSearchDomain: No more search domains for question with name %##s (%s), trying one last time", question->qname.c, DNSTypeName(question->qtype)); - return 0; - } - -#if APPLE_OSX_mDNSResponder - -mDNSlocal mDNSBool DomainInSearchList(domainname *domain) - { - const SearchListElem *s; - for (s=SearchList; s; s=s->next) - if (SameDomainName(&s->domain, domain)) return mDNStrue; - return mDNSfalse; - } - -// Workaround for networks using Microsoft Active Directory using "local" as a private internal -// top-level domain -mDNSlocal mStatus SendAdditionalQuery(DNSQuestion *q, request_state *request, mStatus err) - { - extern domainname ActiveDirectoryPrimaryDomain; - DNSQuestion **question2; - #define VALID_MSAD_SRV_TRANSPORT(T) (SameDomainLabel((T)->c, (const mDNSu8 *)"\x4_tcp") || SameDomainLabel((T)->c, (const mDNSu8 *)"\x4_udp")) - #define VALID_MSAD_SRV(Q) ((Q)->qtype == kDNSType_SRV && VALID_MSAD_SRV_TRANSPORT(SecondLabel(&(Q)->qname))) - - question2 = mDNSNULL; - if (request->hdr.op == query_request) - question2 = &request->u.queryrecord.q2; - else if (request->hdr.op == addrinfo_request) - { - if (q->qtype == kDNSType_A) - question2 = &request->u.addrinfo.q42; - else if (q->qtype == kDNSType_AAAA) - question2 = &request->u.addrinfo.q62; - } - if (!question2) - { - LogMsg("SendAdditionalQuery: question2 NULL for %##s (%s)", q->qname.c, DNSTypeName(q->qtype)); - return mStatus_BadParamErr; - } - - // Sanity check: If we already sent an additonal query, we don't need to send one more. - // - // 1. When the application calls DNSServiceQueryRecord or DNSServiceGetAddrInfo with a .local name, this function - // is called to see whether a unicast query should be sent or not. - // - // 2. As a result of appending search domains, the question may be end up with a .local suffix even though it - // was not a .local name to start with. In that case, queryrecord_result_callback calls this function to - // send the additional query. - // - // Thus, it should not be called more than once. - if (*question2) - { - LogInfo("SendAdditionalQuery: question2 already sent for %##s (%s), no more q2", q->qname.c, DNSTypeName(q->qtype)); - return err; - } - - if (!q->ForceMCast && SameDomainLabel(LastLabel(&q->qname), (const mDNSu8 *)&localdomain)) - if (q->qtype == kDNSType_A || q->qtype == kDNSType_AAAA || VALID_MSAD_SRV(q)) - { - DNSQuestion *q2; - int labels = CountLabels(&q->qname); - q2 = mallocL("DNSQuestion", sizeof(DNSQuestion)); - if (!q2) FatalError("ERROR: SendAdditionalQuery malloc"); - *question2 = q2; - *q2 = *q; - q2->InterfaceID = mDNSInterface_Unicast; - q2->ExpectUnique = mDNStrue; - // If the query starts as a single label e.g., somehost, and we have search domains with .local, - // queryrecord_result_callback calls this function when .local is appended to "somehost". - // At that time, the name in "q" is pointing at somehost.local and its qnameOrig pointing at - // "somehost". We need to copy that information so that when we retry with a different search - // domain e.g., mycompany.local, we get "somehost.mycompany.local". - if (q->qnameOrig) - { - (*question2)->qnameOrig = mallocL("SendAdditionalQuery", DomainNameLength(q->qnameOrig)); - if (!(*question2)->qnameOrig) { LogMsg("SendAdditionalQuery: ERROR!! malloc failure"); return mStatus_NoMemoryErr; } - (*question2)->qnameOrig->c[0] = 0; - AssignDomainName((*question2)->qnameOrig, q->qnameOrig); - LogInfo("SendAdditionalQuery: qnameOrig %##s", (*question2)->qnameOrig->c); - } - // For names of the form ".bar.local." we always do a second unicast query in parallel. - // For names of the form ".local." it's less clear whether we should do a unicast query. - // If the name being queried is exactly the same as the name in the DHCP "domain" option (e.g. the DHCP - // "domain" is my-small-company.local, and the user types "my-small-company.local" into their web browser) - // then that's a hint that it's worth doing a unicast query. Otherwise, we first check to see if the - // site's DNS server claims there's an SOA record for "local", and if so, that's also a hint that queries - // for names in the "local" domain will be safely answered privately before they hit the root name servers. - // Note that in the "my-small-company.local" example above there will typically be an SOA record for - // "my-small-company.local" but *not* for "local", which is why the "local SOA" check would fail in that case. - // We need to check against both ActiveDirectoryPrimaryDomain and SearchList. If it matches against either - // of those, we don't want do the SOA check for the local - if (labels == 2 && !SameDomainName(&q->qname, &ActiveDirectoryPrimaryDomain) && !DomainInSearchList(&q->qname)) - { - AssignDomainName(&q2->qname, &localdomain); - q2->qtype = kDNSType_SOA; - q2->LongLived = mDNSfalse; - q2->ForceMCast = mDNSfalse; - q2->ReturnIntermed = mDNStrue; - // Don't append search domains for the .local SOA query - q2->AppendSearchDomains = 0; - q2->AppendLocalSearchDomains = 0; - q2->RetryWithSearchDomains = mDNSfalse; - q2->SearchListIndex = 0; - q2->TimeoutQuestion = 0; - } - LogOperation("%3d: DNSServiceQueryRecord(%##s, %s) unicast", request->sd, q2->qname.c, DNSTypeName(q2->qtype)); - err = mDNS_StartQuery(&mDNSStorage, q2); - if (err) LogMsg("%3d: ERROR: DNSServiceQueryRecord %##s %s mDNS_StartQuery: %d", request->sd, q2->qname.c, DNSTypeName(q2->qtype), (int)err); - } - return(err); - } -#endif // APPLE_OSX_mDNSResponder - -// This function tries to append a search domain if valid and possible. If so, returns true. -mDNSlocal mDNSBool RetryQuestionWithSearchDomains(mDNS *const m, DNSQuestion *question, request_state *req) - { - int result; - // RetryWithSearchDomains tells the core to call us back so that we can retry with search domains if there is no - // answer in the cache or /etc/hosts. In the first call back from the core, we clear RetryWithSearchDomains so - // that we don't get called back repeatedly. If we got an answer from the cache or /etc/hosts, we don't touch - // RetryWithSearchDomains which may or may not be set. - // - // If we get e.g., NXDOMAIN and the query is neither suppressed nor exhausted the domain search list and - // is a valid question for appending search domains, retry by appending domains - - if (!question->SuppressQuery && question->SearchListIndex != -1 && question->AppendSearchDomains) - { - question->RetryWithSearchDomains = 0; - result = AppendNewSearchDomain(m, question); - // As long as the result is either zero or 1, we retry the question. If we exahaust the search - // domains (result is zero) we try the original query (as it was before appending the search - // domains) as such on the wire as a last resort if we have not tried them before. For queries - // with more than one label, we have already tried them before appending search domains and - // hence don't retry again - if (result != -1) - { - mStatus err; - err = mDNS_StartQuery(m, question); - if (!err) - { - LogOperation("%3d: RetryQuestionWithSearchDomains(%##s, %s), retrying after appending search domain", req->sd, question->qname.c, DNSTypeName(question->qtype)); - // If the result was zero, it meant that there are no search domains and we just retried the question - // as a single label and we should not retry with search domains anymore. - if (!result) question->SearchListIndex = -1; - return mDNStrue; - } - else - { - LogMsg("%3d: ERROR: RetryQuestionWithSearchDomains %##s %s mDNS_StartQuery: %d, while retrying with search domains", req->sd, question->qname.c, DNSTypeName(question->qtype), (int)err); - // We have already stopped the query and could not restart. Reset the appropriate pointers - // so that we don't call stop again when the question terminates - question->QuestionContext = mDNSNULL; - } - } - } - else - { - LogInfo("%3d: RetryQuestionWithSearchDomains: Not appending search domains - SuppressQuery %d, SearchListIndex %d, AppendSearchDomains %d", req->sd, question->SuppressQuery, question->SearchListIndex, question->AppendSearchDomains); - } - return mDNSfalse; - } - -mDNSlocal void queryrecord_result_callback(mDNS *const m, DNSQuestion *question, const ResourceRecord *const answer, QC_result AddRecord) - { - char name[MAX_ESCAPED_DOMAIN_NAME]; - request_state *req = question->QuestionContext; - reply_state *rep; - char *data; - size_t len; - DNSServiceErrorType error = kDNSServiceErr_NoError; - DNSQuestion *q = mDNSNULL; - -#if APPLE_OSX_mDNSResponder - { - // Sanity check: QuestionContext is set to NULL after we stop the question and hence we should not - // get any callbacks from the core after this. - if (!req) - { - LogMsg("queryrecord_result_callback: ERROR!! QuestionContext NULL for %##s (%s)", question->qname.c, DNSTypeName(question->qtype)); - return; - } - if (req->hdr.op == query_request && question == req->u.queryrecord.q2) - q = &req->u.queryrecord.q; - else if (req->hdr.op == addrinfo_request && question == req->u.addrinfo.q42) - q = &req->u.addrinfo.q4; - else if (req->hdr.op == addrinfo_request && question == req->u.addrinfo.q62) - q = &req->u.addrinfo.q6; - - if (q && question->qtype != q->qtype && !SameDomainName(&question->qname, &q->qname)) - { - mStatus err; - domainname *orig = question->qnameOrig; - - LogInfo("queryrecord_result_callback: Stopping q2 local %##s", question->qname.c); - mDNS_StopQuery(m, question); - question->QuestionContext = mDNSNULL; - - // We got a negative response for the SOA record indicating that .local does not exist. - // But we might have other search domains (that does not end in .local) that can be - // appended to this question. In that case, we want to retry the question. Otherwise, - // we don't want to try this question as unicast. - if (answer->RecordType == kDNSRecordTypePacketNegative && !q->AppendSearchDomains) - { - LogInfo("queryrecord_result_callback: question %##s AppendSearchDomains zero", q->qname.c); - return; - } - - // If we got a non-negative answer for our "local SOA" test query, start an additional parallel unicast query - // - // Note: When we copy the original question, we copy everything including the AppendSearchDomains, - // RetryWithSearchDomains except for qnameOrig which can be non-NULL if the original question is - // e.g., somehost and then we appended e.g., ".local" and retried that question. See comment in - // SendAdditionalQuery as to how qnameOrig gets initialized. - *question = *q; - question->InterfaceID = mDNSInterface_Unicast; - question->ExpectUnique = mDNStrue; - question->qnameOrig = orig; - - LogOperation("%3d: DNSServiceQueryRecord(%##s, %s) unicast, context %p", req->sd, question->qname.c, DNSTypeName(question->qtype), question->QuestionContext); - - // If the original question timed out, its QuestionContext would already be set to NULL and that's what we copied above. - // Hence, we need to set it explicitly here. - question->QuestionContext = req; - err = mDNS_StartQuery(m, question); - if (err) LogMsg("%3d: ERROR: queryrecord_result_callback %##s %s mDNS_StartQuery: %d", req->sd, question->qname.c, DNSTypeName(question->qtype), (int)err); - - // If we got a positive response to local SOA, then try the .local question as unicast - if (answer->RecordType != kDNSRecordTypePacketNegative) return; - - // Fall through and get the next search domain. The question is pointing at .local - // and we don't want to try that. Try the next search domain. Don't try with local - // search domains for the unicast question anymore. - // - // Note: we started the question above which will be stopped immediately (never sent on the wire) - // before we pick the next search domain below. RetryQuestionWithSearchDomains assumes that the - // question has already started. - question->AppendLocalSearchDomains = 0; - } - - if (q && AddRecord && (question->InterfaceID == mDNSInterface_Unicast) && !answer->rdlength) - { - // If we get a negative response to the unicast query that we sent above, retry after appending search domains - // Note: We could have appended search domains below (where do it for regular unicast questions) instead of doing it here. - // As we ignore negative unicast answers below, we would never reach the code where the search domains are appended. - // To keep things simple, we handle unicast ".local" separately here. - LogInfo("queryrecord_result_callback: Retrying .local question %##s (%s) as unicast after appending search domains", question->qname.c, DNSTypeName(question->qtype)); - if (RetryQuestionWithSearchDomains(m, question, req)) - return; - if (question->AppendSearchDomains && !question->AppendLocalSearchDomains && IsLocalDomain(&question->qname)) - { - // If "local" is the last search domain, we need to stop the question so that we don't send the "local" - // question on the wire as we got a negative response for the local SOA. But, we can't stop the question - // yet as we may have to timeout the question (done by the "core") for which we need to leave the question - // in the list. We leave it disabled so that it does not hit the wire. - LogInfo("queryrecord_result_callback: Disabling .local question %##s (%s)", question->qname.c, DNSTypeName(question->qtype)); - question->ThisQInterval = 0; - } - } - // If we are here it means that either "question" is not "q2" OR we got a positive response for "q2" OR we have no more search - // domains to append for "q2". In all cases, fall through and deliver the response - } -#endif // APPLE_OSX_mDNSResponder - - if (answer->RecordType == kDNSRecordTypePacketNegative) - { - // If this question needs to be timed out and we have reached the stop time, mark - // the error as timeout. It is possible that we might get a negative response from an - // external DNS server at the same time when this question reaches its stop time. We - // can't tell the difference as there is no indication in the callback. This should - // be okay as we will be timing out this query anyway. - mDNS_Lock(m); - if (question->TimeoutQuestion) - { - if ((m->timenow - question->StopTime) >= 0) - { - LogInfo("queryrecord_result_callback:Question %##s (%s) timing out, InterfaceID %p", question->qname.c, DNSTypeName(question->qtype), question->InterfaceID); - error = kDNSServiceErr_Timeout; - } - } - mDNS_Unlock(m); - // When we're doing parallel unicast and multicast queries for dot-local names (for supporting Microsoft - // Active Directory sites) we need to ignore negative unicast answers. Otherwise we'll generate negative - // answers for just about every single multicast name we ever look up, since the Microsoft Active Directory - // server is going to assert that pretty much every single multicast name doesn't exist. - // - // If we are timing out this query, we need to deliver the negative answer to the application - if (error != kDNSServiceErr_Timeout) - { - if (!answer->InterfaceID && IsLocalDomain(answer->name)) - { - LogInfo("queryrecord_result_callback:Question %##s (%s) answering local with unicast", question->qname.c, DNSTypeName(question->qtype)); - return; - } - error = kDNSServiceErr_NoSuchRecord; - } - AddRecord = mDNStrue; - } - // If we get a negative answer, try appending search domains. Don't append search domains - // - if we are timing out this question - // - if the negative response was received as a result of a multicast query - // - if this is an additional query (q2), we already appended search domains above (indicated by "!q" below) - if (error != kDNSServiceErr_Timeout) - { - if (!q && !answer->InterfaceID && !answer->rdlength && AddRecord) - { - // If the original question did not end in .local, we did not send an SOA query - // to figure out whether we should send an additional unicast query or not. If we just - // appended .local, we need to see if we need to send an additional query. This should - // normally happen just once because after we append .local, we ignore all negative - // responses for .local above. - LogInfo("queryrecord_result_callback: Retrying question %##s (%s) after appending search domains", question->qname.c, DNSTypeName(question->qtype)); - if (RetryQuestionWithSearchDomains(m, question, req)) - { - // Note: We need to call SendAdditionalQuery every time after appending a search domain as .local could - // be anywhere in the search domain list. -#if APPLE_OSX_mDNSResponder - mStatus err = mStatus_NoError; - err = SendAdditionalQuery(question, req, err); - if (err) LogMsg("queryrecord_result_callback: Sending .local SOA query failed, after appending domains"); -#endif // APPLE_OSX_mDNSResponder - return; - } - } - } - - ConvertDomainNameToCString(answer->name, name); - - LogOperation("%3d: %s(%##s, %s) %s %s", req->sd, - req->hdr.op == query_request ? "DNSServiceQueryRecord" : "DNSServiceGetAddrInfo", - question->qname.c, DNSTypeName(question->qtype), AddRecord ? "ADD" : "RMV", RRDisplayString(m, answer)); - - len = sizeof(DNSServiceFlags); // calculate reply data length - len += sizeof(mDNSu32); // interface index - len += sizeof(DNSServiceErrorType); - len += strlen(name) + 1; - len += 3 * sizeof(mDNSu16); // type, class, rdlen - len += answer->rdlength; - len += sizeof(mDNSu32); // TTL - - rep = create_reply(req->hdr.op == query_request ? query_reply_op : addrinfo_reply_op, len, req); - - rep->rhdr->flags = dnssd_htonl(AddRecord ? kDNSServiceFlagsAdd : 0); - // Call mDNSPlatformInterfaceIndexfromInterfaceID, but suppressNetworkChange (last argument). Otherwise, if the - // InterfaceID is not valid, then it simulates a "NetworkChanged" which in turn makes questions - // to be stopped and started including *this* one. Normally the InterfaceID is valid. But when we - // are using the /etc/hosts entries to answer a question, the InterfaceID may not be known to the - // mDNS core . Eventually, we should remove the calls to "NetworkChanged" in - // mDNSPlatformInterfaceIndexfromInterfaceID when it can't find InterfaceID as ResourceRecords - // should not have existed to answer this question if the corresponding interface is not valid. - rep->rhdr->ifi = dnssd_htonl(mDNSPlatformInterfaceIndexfromInterfaceID(m, answer->InterfaceID, mDNStrue)); - rep->rhdr->error = dnssd_htonl(error); - - data = (char *)&rep->rhdr[1]; - - put_string(name, &data); - put_uint16(answer->rrtype, &data); - put_uint16(answer->rrclass, &data); - put_uint16(answer->rdlength, &data); - // We need to use putRData here instead of the crude put_rdata function, because the crude put_rdata - // function just does a blind memory copy without regard to structures that may have holes in them. - if (answer->rdlength) - if (!putRData(mDNSNULL, (mDNSu8 *)data, (mDNSu8 *)rep->rhdr + len, answer)) - LogMsg("queryrecord_result_callback putRData failed %d", (mDNSu8 *)rep->rhdr + len - (mDNSu8 *)data); - data += answer->rdlength; - put_uint32(AddRecord ? answer->rroriginalttl : 0, &data); - - append_reply(req, rep); - // Stop the question, if we just timed out - if (error == kDNSServiceErr_Timeout) - { - mDNS_StopQuery(m, question); - // Reset the pointers so that we don't call stop on termination - question->QuestionContext = mDNSNULL; - } -#if APPLE_OSX_mDNSResponder -#if ! NO_WCF - CHECK_WCF_FUNCTION(WCFIsServerRunning) - { - struct xucred x; - socklen_t xucredlen = sizeof(x); - - if (WCFIsServerRunning((WCFConnection *)m->WCF) && answer->rdlength != 0) - { - if (getsockopt(req->sd, 0, LOCAL_PEERCRED, &x, &xucredlen) >= 0 && - (x.cr_version == XUCRED_VERSION)) - { - struct sockaddr_storage addr; - const RDataBody2 *const rdb = (RDataBody2 *)answer->rdata->u.data; - addr.ss_len = 0; - if (answer->rrtype == kDNSType_A || answer->rrtype == kDNSType_AAAA) - { - if (answer->rrtype == kDNSType_A) - { - struct sockaddr_in *sin = (struct sockaddr_in *)&addr; - sin->sin_port = 0; - if (!putRData(mDNSNULL, (mDNSu8 *)&sin->sin_addr, (mDNSu8 *)(&sin->sin_addr + sizeof(rdb->ipv4)), answer)) - LogMsg("queryrecord_result_callback: WCF AF_INET putRData failed"); - else - { - addr.ss_len = sizeof (struct sockaddr_in); - addr.ss_family = AF_INET; - } - } - else if (answer->rrtype == kDNSType_AAAA) - { - struct sockaddr_in6 *sin6 = (struct sockaddr_in6 *)&addr; - sin6->sin6_port = 0; - if (!putRData(mDNSNULL, (mDNSu8 *)&sin6->sin6_addr, (mDNSu8 *)(&sin6->sin6_addr + sizeof(rdb->ipv6)), answer)) - LogMsg("queryrecord_result_callback: WCF AF_INET6 putRData failed"); - else - { - addr.ss_len = sizeof (struct sockaddr_in6); - addr.ss_family = AF_INET6; - } - } - if (addr.ss_len) - { - debugf("queryrecord_result_callback: Name %s, uid %u, addr length %d", name, x.cr_uid, addr.ss_len); - CHECK_WCF_FUNCTION((WCFConnection *)WCFNameResolvesToAddr) - { - WCFNameResolvesToAddr(m->WCF, name, (struct sockaddr *)&addr, x.cr_uid); - } - } - } - else if (answer->rrtype == kDNSType_CNAME) - { - domainname cname; - char cname_cstr[MAX_ESCAPED_DOMAIN_NAME]; - if (!putRData(mDNSNULL, cname.c, (mDNSu8 *)(cname.c + MAX_DOMAIN_NAME), answer)) - LogMsg("queryrecord_result_callback: WCF CNAME putRData failed"); - else - { - ConvertDomainNameToCString(&cname, cname_cstr); - CHECK_WCF_FUNCTION((WCFConnection *)WCFNameResolvesToAddr) - { - WCFNameResolvesToName(m->WCF, name, cname_cstr, x.cr_uid); - } - } - } - } - else my_perror("queryrecord_result_callback: ERROR: getsockopt LOCAL_PEERCRED"); - } - } -#endif -#endif - } - -mDNSlocal void queryrecord_termination_callback(request_state *request) - { - LogOperation("%3d: DNSServiceQueryRecord(%##s, %s) STOP", - request->sd, request->u.queryrecord.q.qname.c, DNSTypeName(request->u.queryrecord.q.qtype)); - if (request->u.queryrecord.q.QuestionContext) - { - mDNS_StopQuery(&mDNSStorage, &request->u.queryrecord.q); // no need to error check - request->u.queryrecord.q.QuestionContext = mDNSNULL; - } - else - { - DNSQuestion *question = &request->u.queryrecord.q; - LogInfo("queryrecord_termination_callback: question %##s (%s) already stopped, InterfaceID %p", question->qname.c, DNSTypeName(question->qtype), question->InterfaceID); - } - - if (request->u.queryrecord.q.qnameOrig) - { - freeL("QueryTermination", request->u.queryrecord.q.qnameOrig); - request->u.queryrecord.q.qnameOrig = mDNSNULL; - } - if (request->u.queryrecord.q.InterfaceID == mDNSInterface_P2P || (!request->u.queryrecord.q.InterfaceID && SameDomainName((const domainname *)LastLabel(&request->u.queryrecord.q.qname), &localdomain) && (request->flags & kDNSServiceFlagsIncludeP2P))) - { - LogInfo("queryrecord_termination_callback: calling external_stop_browsing_for_service()"); - external_stop_browsing_for_service(&mDNSStorage, &request->u.queryrecord.q.qname, request->u.queryrecord.q.qtype); - } - if (request->u.queryrecord.q2) - { - if (request->u.queryrecord.q2->QuestionContext) - { - LogInfo("queryrecord_termination_callback: Stopping q2 %##s", request->u.queryrecord.q2->qname.c); - mDNS_StopQuery(&mDNSStorage, request->u.queryrecord.q2); - } - else - { - DNSQuestion *question = request->u.queryrecord.q2; - LogInfo("queryrecord_termination_callback: q2 %##s (%s) already stopped, InterfaceID %p", question->qname.c, DNSTypeName(question->qtype), question->InterfaceID); - } - if (request->u.queryrecord.q2->qnameOrig) - { - LogInfo("queryrecord_termination_callback: freeing q2 qnameOrig %##s", request->u.queryrecord.q2->qnameOrig->c); - freeL("QueryTermination q2", request->u.queryrecord.q2->qnameOrig); - request->u.queryrecord.q2->qnameOrig = mDNSNULL; - } - freeL("queryrecord Q2", request->u.queryrecord.q2); - request->u.queryrecord.q2 = mDNSNULL; - } - } - -mDNSlocal mStatus handle_queryrecord_request(request_state *request) - { - DNSQuestion *const q = &request->u.queryrecord.q; - char name[256]; - mDNSu16 rrtype, rrclass; - mStatus err; - - DNSServiceFlags flags = get_flags(&request->msgptr, request->msgend); - mDNSu32 interfaceIndex = get_uint32(&request->msgptr, request->msgend); - mDNSInterfaceID InterfaceID = mDNSPlatformInterfaceIDfromInterfaceIndex(&mDNSStorage, interfaceIndex); - if (interfaceIndex && !InterfaceID) return(mStatus_BadParamErr); - - if (get_string(&request->msgptr, request->msgend, name, 256) < 0) return(mStatus_BadParamErr); - rrtype = get_uint16(&request->msgptr, request->msgend); - rrclass = get_uint16(&request->msgptr, request->msgend); - - if (!request->msgptr) - { LogMsg("%3d: DNSServiceQueryRecord(unreadable parameters)", request->sd); return(mStatus_BadParamErr); } - - request->flags = flags; - mDNSPlatformMemZero(&request->u.queryrecord, sizeof(request->u.queryrecord)); - - q->InterfaceID = InterfaceID; - q->Target = zeroAddr; - if (!MakeDomainNameFromDNSNameString(&q->qname, name)) return(mStatus_BadParamErr); -#if 0 - if (!AuthorizedDomain(request, &q->qname, AutoBrowseDomains)) return (mStatus_NoError); -#endif - q->qtype = rrtype; - q->qclass = rrclass; - q->LongLived = (flags & kDNSServiceFlagsLongLivedQuery ) != 0; - q->ExpectUnique = mDNSfalse; - q->ForceMCast = (flags & kDNSServiceFlagsForceMulticast ) != 0; - q->ReturnIntermed = (flags & kDNSServiceFlagsReturnIntermediates) != 0; - q->SuppressUnusable = (flags & kDNSServiceFlagsSuppressUnusable ) != 0; - q->TimeoutQuestion = (flags & kDNSServiceFlagsTimeout ) != 0; - q->WakeOnResolve = 0; - q->QuestionCallback = queryrecord_result_callback; - q->QuestionContext = request; - q->SearchListIndex = 0; - - // Don't append search domains for fully qualified domain names including queries - // such as e.g., "abc." that has only one label. We convert all names to FQDNs as internally - // we only deal with FQDNs. Hence, we cannot look at qname to figure out whether we should - // append search domains or not. So, we record that information in AppendSearchDomains. - // - // We append search domains only for queries that are a single label. If overriden using - // command line argument "AlwaysAppendSearchDomains", then we do it for any query which - // is not fully qualified. - - if ((rrtype == kDNSType_A || rrtype == kDNSType_AAAA) && name[strlen(name) - 1] != '.' && - (AlwaysAppendSearchDomains || CountLabels(&q->qname) == 1)) - { - q->AppendSearchDomains = 1; - q->AppendLocalSearchDomains = 1; - } - else - { - q->AppendSearchDomains = 0; - q->AppendLocalSearchDomains = 0; - } - - // For single label queries that are not fully qualified, look at /etc/hosts, cache and try - // search domains before trying them on the wire as a single label query. RetryWithSearchDomains - // tell the core to call back into the UDS layer if there is no valid response in /etc/hosts or - // the cache - q->RetryWithSearchDomains = ApplySearchDomainsFirst(q) ? 1 : 0; - q->qnameOrig = mDNSNULL; - - LogOperation("%3d: DNSServiceQueryRecord(%X, %d, %##s, %s) START", request->sd, flags, interfaceIndex, q->qname.c, DNSTypeName(q->qtype)); - err = mDNS_StartQuery(&mDNSStorage, q); - if (err) LogMsg("%3d: ERROR: DNSServiceQueryRecord %##s %s mDNS_StartQuery: %d", request->sd, q->qname.c, DNSTypeName(q->qtype), (int)err); - else - { - request->terminate = queryrecord_termination_callback; - if (q->InterfaceID == mDNSInterface_P2P || (!q->InterfaceID && SameDomainName((const domainname *)LastLabel(&q->qname), &localdomain) && (flags & kDNSServiceFlagsIncludeP2P))) - { - LogInfo("handle_queryrecord_request: calling external_start_browsing_for_service()"); - external_start_browsing_for_service(&mDNSStorage, &q->qname, q->qtype); - } - } - -#if APPLE_OSX_mDNSResponder - err = SendAdditionalQuery(q, request, err); -#endif // APPLE_OSX_mDNSResponder - - return(err); - } - -// *************************************************************************** -#if COMPILER_LIKES_PRAGMA_MARK -#pragma mark - -#pragma mark - DNSServiceEnumerateDomains -#endif - -mDNSlocal reply_state *format_enumeration_reply(request_state *request, - const char *domain, DNSServiceFlags flags, mDNSu32 ifi, DNSServiceErrorType err) - { - size_t len; - reply_state *reply; - char *data; - - len = sizeof(DNSServiceFlags); - len += sizeof(mDNSu32); - len += sizeof(DNSServiceErrorType); - len += strlen(domain) + 1; - - reply = create_reply(enumeration_reply_op, len, request); - reply->rhdr->flags = dnssd_htonl(flags); - reply->rhdr->ifi = dnssd_htonl(ifi); - reply->rhdr->error = dnssd_htonl(err); - data = (char *)&reply->rhdr[1]; - put_string(domain, &data); - return reply; - } - -mDNSlocal void enum_termination_callback(request_state *request) - { - mDNS_StopGetDomains(&mDNSStorage, &request->u.enumeration.q_all); - mDNS_StopGetDomains(&mDNSStorage, &request->u.enumeration.q_default); - } - -mDNSlocal void enum_result_callback(mDNS *const m, - DNSQuestion *const question, const ResourceRecord *const answer, QC_result AddRecord) - { - char domain[MAX_ESCAPED_DOMAIN_NAME]; - request_state *request = question->QuestionContext; - DNSServiceFlags flags = 0; - reply_state *reply; - (void)m; // Unused - - if (answer->rrtype != kDNSType_PTR) return; - -#if 0 - if (!AuthorizedDomain(request, &answer->rdata->u.name, request->u.enumeration.flags ? AutoRegistrationDomains : AutoBrowseDomains)) return; -#endif - - // We only return add/remove events for the browse and registration lists - // For the default browse and registration answers, we only give an "ADD" event - if (question == &request->u.enumeration.q_default && !AddRecord) return; - - if (AddRecord) - { - flags |= kDNSServiceFlagsAdd; - if (question == &request->u.enumeration.q_default) flags |= kDNSServiceFlagsDefault; - } - - ConvertDomainNameToCString(&answer->rdata->u.name, domain); - // Note that we do NOT propagate specific interface indexes to the client - for example, a domain we learn from - // a machine's system preferences may be discovered on the LocalOnly interface, but should be browsed on the - // network, so we just pass kDNSServiceInterfaceIndexAny - reply = format_enumeration_reply(request, domain, flags, kDNSServiceInterfaceIndexAny, kDNSServiceErr_NoError); - if (!reply) { LogMsg("ERROR: enum_result_callback, format_enumeration_reply"); return; } - - LogOperation("%3d: DNSServiceEnumerateDomains(%#2s) RESULT %s: %s", request->sd, question->qname.c, AddRecord ? "Add" : "Rmv", domain); - - append_reply(request, reply); - } - -mDNSlocal mStatus handle_enum_request(request_state *request) - { - mStatus err; - DNSServiceFlags flags = get_flags(&request->msgptr, request->msgend); - DNSServiceFlags reg = flags & kDNSServiceFlagsRegistrationDomains; - mDNS_DomainType t_all = reg ? mDNS_DomainTypeRegistration : mDNS_DomainTypeBrowse; - mDNS_DomainType t_default = reg ? mDNS_DomainTypeRegistrationDefault : mDNS_DomainTypeBrowseDefault; - mDNSu32 interfaceIndex = get_uint32(&request->msgptr, request->msgend); - mDNSInterfaceID InterfaceID = mDNSPlatformInterfaceIDfromInterfaceIndex(&mDNSStorage, interfaceIndex); - if (interfaceIndex && !InterfaceID) return(mStatus_BadParamErr); - - if (!request->msgptr) - { LogMsg("%3d: DNSServiceEnumerateDomains(unreadable parameters)", request->sd); return(mStatus_BadParamErr); } - - // allocate context structures - uDNS_SetupSearchDomains(&mDNSStorage, UDNS_START_WAB_QUERY); - -#if 0 - // mark which kind of enumeration we're doing so we can (de)authorize certain domains - request->u.enumeration.flags = reg; -#endif - - // enumeration requires multiple questions, so we must link all the context pointers so that - // necessary context can be reached from the callbacks - request->u.enumeration.q_all .QuestionContext = request; - request->u.enumeration.q_default.QuestionContext = request; - - // if the caller hasn't specified an explicit interface, we use local-only to get the system-wide list. - if (!InterfaceID) InterfaceID = mDNSInterface_LocalOnly; - - // make the calls - LogOperation("%3d: DNSServiceEnumerateDomains(%X=%s)", request->sd, flags, - (flags & kDNSServiceFlagsBrowseDomains ) ? "kDNSServiceFlagsBrowseDomains" : - (flags & kDNSServiceFlagsRegistrationDomains) ? "kDNSServiceFlagsRegistrationDomains" : "<>"); - err = mDNS_GetDomains(&mDNSStorage, &request->u.enumeration.q_all, t_all, NULL, InterfaceID, enum_result_callback, request); - if (!err) - { - err = mDNS_GetDomains(&mDNSStorage, &request->u.enumeration.q_default, t_default, NULL, InterfaceID, enum_result_callback, request); - if (err) mDNS_StopGetDomains(&mDNSStorage, &request->u.enumeration.q_all); - else request->terminate = enum_termination_callback; - } - - return(err); - } - -// *************************************************************************** -#if COMPILER_LIKES_PRAGMA_MARK -#pragma mark - -#pragma mark - DNSServiceReconfirmRecord & Misc -#endif - -mDNSlocal mStatus handle_reconfirm_request(request_state *request) - { - mStatus status = mStatus_BadParamErr; - AuthRecord *rr = read_rr_from_ipc_msg(request, 0, 0); - if (rr) - { - status = mDNS_ReconfirmByValue(&mDNSStorage, &rr->resrec); - LogOperation( - (status == mStatus_NoError) ? - "%3d: DNSServiceReconfirmRecord(%s) interface %d initiated" : - "%3d: DNSServiceReconfirmRecord(%s) interface %d failed: %d", - request->sd, RRDisplayString(&mDNSStorage, &rr->resrec), - mDNSPlatformInterfaceIndexfromInterfaceID(&mDNSStorage, rr->resrec.InterfaceID, mDNSfalse), status); - freeL("AuthRecord/handle_reconfirm_request", rr); - } - return(status); - } - -mDNSlocal mStatus handle_setdomain_request(request_state *request) - { - char domainstr[MAX_ESCAPED_DOMAIN_NAME]; - domainname domain; - DNSServiceFlags flags = get_flags(&request->msgptr, request->msgend); - (void)flags; // Unused - if (get_string(&request->msgptr, request->msgend, domainstr, MAX_ESCAPED_DOMAIN_NAME) < 0 || - !MakeDomainNameFromDNSNameString(&domain, domainstr)) - { LogMsg("%3d: DNSServiceSetDefaultDomainForUser(unreadable parameters)", request->sd); return(mStatus_BadParamErr); } - - LogOperation("%3d: DNSServiceSetDefaultDomainForUser(%##s)", request->sd, domain.c); - return(mStatus_NoError); - } - -typedef packedstruct - { - mStatus err; - mDNSu32 len; - mDNSu32 vers; - } DaemonVersionReply; - -mDNSlocal void handle_getproperty_request(request_state *request) - { - const mStatus BadParamErr = dnssd_htonl((mDNSu32)mStatus_BadParamErr); - char prop[256]; - if (get_string(&request->msgptr, request->msgend, prop, sizeof(prop)) >= 0) - { - LogOperation("%3d: DNSServiceGetProperty(%s)", request->sd, prop); - if (!strcmp(prop, kDNSServiceProperty_DaemonVersion)) - { - DaemonVersionReply x = { 0, dnssd_htonl(4), dnssd_htonl(_DNS_SD_H) }; - send_all(request->sd, (const char *)&x, sizeof(x)); - return; - } - } - - // If we didn't recogize the requested property name, return BadParamErr - send_all(request->sd, (const char *)&BadParamErr, sizeof(BadParamErr)); - } - -// *************************************************************************** -#if COMPILER_LIKES_PRAGMA_MARK -#pragma mark - -#pragma mark - DNSServiceNATPortMappingCreate -#endif - -#define DNSServiceProtocol(X) ((X) == NATOp_AddrRequest ? 0 : (X) == NATOp_MapUDP ? kDNSServiceProtocol_UDP : kDNSServiceProtocol_TCP) - -mDNSlocal void port_mapping_termination_callback(request_state *request) - { - LogOperation("%3d: DNSServiceNATPortMappingCreate(%X, %u, %u, %d) STOP", request->sd, - DNSServiceProtocol(request->u.pm.NATinfo.Protocol), - mDNSVal16(request->u.pm.NATinfo.IntPort), mDNSVal16(request->u.pm.ReqExt), request->u.pm.NATinfo.NATLease); - mDNS_StopNATOperation(&mDNSStorage, &request->u.pm.NATinfo); - } - -// Called via function pointer when we get a NAT-PMP address request or port mapping response -mDNSlocal void port_mapping_create_request_callback(mDNS *m, NATTraversalInfo *n) - { - request_state *request = (request_state *)n->clientContext; - reply_state *rep; - int replyLen; - char *data; - - if (!request) { LogMsg("port_mapping_create_request_callback called with unknown request_state object"); return; } - - // calculate reply data length - replyLen = sizeof(DNSServiceFlags); - replyLen += 3 * sizeof(mDNSu32); // if index + addr + ttl - replyLen += sizeof(DNSServiceErrorType); - replyLen += 2 * sizeof(mDNSu16); // Internal Port + External Port - replyLen += sizeof(mDNSu8); // protocol - - rep = create_reply(port_mapping_reply_op, replyLen, request); - - rep->rhdr->flags = dnssd_htonl(0); - rep->rhdr->ifi = dnssd_htonl(mDNSPlatformInterfaceIndexfromInterfaceID(m, n->InterfaceID, mDNSfalse)); - rep->rhdr->error = dnssd_htonl(n->Result); - - data = (char *)&rep->rhdr[1]; - - *data++ = request->u.pm.NATinfo.ExternalAddress.b[0]; - *data++ = request->u.pm.NATinfo.ExternalAddress.b[1]; - *data++ = request->u.pm.NATinfo.ExternalAddress.b[2]; - *data++ = request->u.pm.NATinfo.ExternalAddress.b[3]; - *data++ = DNSServiceProtocol(request->u.pm.NATinfo.Protocol); - *data++ = request->u.pm.NATinfo.IntPort.b[0]; - *data++ = request->u.pm.NATinfo.IntPort.b[1]; - *data++ = request->u.pm.NATinfo.ExternalPort.b[0]; - *data++ = request->u.pm.NATinfo.ExternalPort.b[1]; - put_uint32(request->u.pm.NATinfo.Lifetime, &data); - - LogOperation("%3d: DNSServiceNATPortMappingCreate(%X, %u, %u, %d) RESULT %.4a:%u TTL %u", request->sd, - DNSServiceProtocol(request->u.pm.NATinfo.Protocol), - mDNSVal16(request->u.pm.NATinfo.IntPort), mDNSVal16(request->u.pm.ReqExt), request->u.pm.NATinfo.NATLease, - &request->u.pm.NATinfo.ExternalAddress, mDNSVal16(request->u.pm.NATinfo.ExternalPort), request->u.pm.NATinfo.Lifetime); - - append_reply(request, rep); - } - -mDNSlocal mStatus handle_port_mapping_request(request_state *request) - { - mDNSu32 ttl = 0; - mStatus err = mStatus_NoError; - - DNSServiceFlags flags = get_flags(&request->msgptr, request->msgend); - mDNSu32 interfaceIndex = get_uint32(&request->msgptr, request->msgend); - mDNSInterfaceID InterfaceID = mDNSPlatformInterfaceIDfromInterfaceIndex(&mDNSStorage, interfaceIndex); - mDNSu8 protocol = (mDNSu8)get_uint32(&request->msgptr, request->msgend); - (void)flags; // Unused - if (interfaceIndex && !InterfaceID) return(mStatus_BadParamErr); - if (request->msgptr + 8 > request->msgend) request->msgptr = NULL; - else - { - request->u.pm.NATinfo.IntPort.b[0] = *request->msgptr++; - request->u.pm.NATinfo.IntPort.b[1] = *request->msgptr++; - request->u.pm.ReqExt.b[0] = *request->msgptr++; - request->u.pm.ReqExt.b[1] = *request->msgptr++; - ttl = get_uint32(&request->msgptr, request->msgend); - } - - if (!request->msgptr) - { LogMsg("%3d: DNSServiceNATPortMappingCreate(unreadable parameters)", request->sd); return(mStatus_BadParamErr); } - - if (protocol == 0) // If protocol == 0 (i.e. just request public address) then IntPort, ExtPort, ttl must be zero too - { - if (!mDNSIPPortIsZero(request->u.pm.NATinfo.IntPort) || !mDNSIPPortIsZero(request->u.pm.ReqExt) || ttl) return(mStatus_BadParamErr); - } - else - { - if (mDNSIPPortIsZero(request->u.pm.NATinfo.IntPort)) return(mStatus_BadParamErr); - if (!(protocol & (kDNSServiceProtocol_UDP | kDNSServiceProtocol_TCP))) return(mStatus_BadParamErr); - } - - request->u.pm.NATinfo.Protocol = !protocol ? NATOp_AddrRequest : (protocol == kDNSServiceProtocol_UDP) ? NATOp_MapUDP : NATOp_MapTCP; - // u.pm.NATinfo.IntPort = already set above - request->u.pm.NATinfo.RequestedPort = request->u.pm.ReqExt; - request->u.pm.NATinfo.NATLease = ttl; - request->u.pm.NATinfo.clientCallback = port_mapping_create_request_callback; - request->u.pm.NATinfo.clientContext = request; - - LogOperation("%3d: DNSServiceNATPortMappingCreate(%X, %u, %u, %d) START", request->sd, - protocol, mDNSVal16(request->u.pm.NATinfo.IntPort), mDNSVal16(request->u.pm.ReqExt), request->u.pm.NATinfo.NATLease); - err = mDNS_StartNATOperation(&mDNSStorage, &request->u.pm.NATinfo); - if (err) LogMsg("ERROR: mDNS_StartNATOperation: %d", (int)err); - else request->terminate = port_mapping_termination_callback; - - return(err); - } - -// *************************************************************************** -#if COMPILER_LIKES_PRAGMA_MARK -#pragma mark - -#pragma mark - DNSServiceGetAddrInfo -#endif - -mDNSlocal void addrinfo_termination_callback(request_state *request) - { - LogOperation("%3d: DNSServiceGetAddrInfo(%##s) STOP", request->sd, request->u.addrinfo.q4.qname.c); - - if (request->u.addrinfo.q4.QuestionContext) - { - mDNS_StopQuery(&mDNSStorage, &request->u.addrinfo.q4); - request->u.addrinfo.q4.QuestionContext = mDNSNULL; - } - if (request->u.addrinfo.q4.qnameOrig) - { - freeL("QueryTermination", request->u.addrinfo.q4.qnameOrig); - request->u.addrinfo.q4.qnameOrig = mDNSNULL; - } - if (request->u.addrinfo.q42) - { - if (request->u.addrinfo.q42->QuestionContext) - { - LogInfo("addrinfo_termination_callback: Stopping q42 %##s", request->u.addrinfo.q42->qname.c); - mDNS_StopQuery(&mDNSStorage, request->u.addrinfo.q42); - } - if (request->u.addrinfo.q42->qnameOrig) - { - LogInfo("addrinfo_termination_callback: freeing q42 qnameOrig %##s", request->u.addrinfo.q42->qnameOrig->c); - freeL("QueryTermination q42", request->u.addrinfo.q42->qnameOrig); - request->u.addrinfo.q42->qnameOrig = mDNSNULL; - } - freeL("addrinfo Q42", request->u.addrinfo.q42); - request->u.addrinfo.q42 = mDNSNULL; - } - - if (request->u.addrinfo.q6.QuestionContext) - { - mDNS_StopQuery(&mDNSStorage, &request->u.addrinfo.q6); - request->u.addrinfo.q6.QuestionContext = mDNSNULL; - } - if (request->u.addrinfo.q6.qnameOrig) - { - freeL("QueryTermination", request->u.addrinfo.q6.qnameOrig); - request->u.addrinfo.q6.qnameOrig = mDNSNULL; - } - if (request->u.addrinfo.q62) - { - if (request->u.addrinfo.q62->QuestionContext) - { - LogInfo("addrinfo_termination_callback: Stopping q62 %##s", request->u.addrinfo.q62->qname.c); - mDNS_StopQuery(&mDNSStorage, request->u.addrinfo.q62); - } - if (request->u.addrinfo.q62->qnameOrig) - { - LogInfo("addrinfo_termination_callback: freeing q62 qnameOrig %##s", request->u.addrinfo.q62->qnameOrig->c); - freeL("QueryTermination q62", request->u.addrinfo.q62->qnameOrig); - request->u.addrinfo.q62->qnameOrig = mDNSNULL; - } - freeL("addrinfo Q62", request->u.addrinfo.q62); - request->u.addrinfo.q62 = mDNSNULL; - } - } - -mDNSlocal mStatus handle_addrinfo_request(request_state *request) - { - char hostname[256]; - domainname d; - mStatus err = 0; - - DNSServiceFlags flags = get_flags(&request->msgptr, request->msgend); - mDNSu32 interfaceIndex = get_uint32(&request->msgptr, request->msgend); - - mDNSPlatformMemZero(&request->u.addrinfo, sizeof(request->u.addrinfo)); - request->u.addrinfo.interface_id = mDNSPlatformInterfaceIDfromInterfaceIndex(&mDNSStorage, interfaceIndex); - request->u.addrinfo.flags = flags; - request->u.addrinfo.protocol = get_uint32(&request->msgptr, request->msgend); - - if (interfaceIndex && !request->u.addrinfo.interface_id) return(mStatus_BadParamErr); - if (request->u.addrinfo.protocol > (kDNSServiceProtocol_IPv4|kDNSServiceProtocol_IPv6)) return(mStatus_BadParamErr); - - if (get_string(&request->msgptr, request->msgend, hostname, 256) < 0) return(mStatus_BadParamErr); - - if (!request->msgptr) { LogMsg("%3d: DNSServiceGetAddrInfo(unreadable parameters)", request->sd); return(mStatus_BadParamErr); } - - if (!MakeDomainNameFromDNSNameString(&d, hostname)) - { LogMsg("ERROR: handle_addrinfo_request: bad hostname: %s", hostname); return(mStatus_BadParamErr); } - -#if 0 - if (!AuthorizedDomain(request, &d, AutoBrowseDomains)) return (mStatus_NoError); -#endif - - if (!request->u.addrinfo.protocol) - { - flags |= kDNSServiceFlagsSuppressUnusable; - request->u.addrinfo.protocol = (kDNSServiceProtocol_IPv4 | kDNSServiceProtocol_IPv6); - } - - request->u.addrinfo.q4.InterfaceID = request->u.addrinfo.q6.InterfaceID = request->u.addrinfo.interface_id; - request->u.addrinfo.q4.Target = request->u.addrinfo.q6.Target = zeroAddr; - request->u.addrinfo.q4.qname = request->u.addrinfo.q6.qname = d; - request->u.addrinfo.q4.qclass = request->u.addrinfo.q6.qclass = kDNSServiceClass_IN; - request->u.addrinfo.q4.LongLived = request->u.addrinfo.q6.LongLived = (flags & kDNSServiceFlagsLongLivedQuery ) != 0; - request->u.addrinfo.q4.ExpectUnique = request->u.addrinfo.q6.ExpectUnique = mDNSfalse; - request->u.addrinfo.q4.ForceMCast = request->u.addrinfo.q6.ForceMCast = (flags & kDNSServiceFlagsForceMulticast ) != 0; - request->u.addrinfo.q4.ReturnIntermed = request->u.addrinfo.q6.ReturnIntermed = (flags & kDNSServiceFlagsReturnIntermediates) != 0; - request->u.addrinfo.q4.SuppressUnusable = request->u.addrinfo.q6.SuppressUnusable = (flags & kDNSServiceFlagsSuppressUnusable ) != 0; - request->u.addrinfo.q4.TimeoutQuestion = request->u.addrinfo.q6.TimeoutQuestion = (flags & kDNSServiceFlagsTimeout ) != 0; - request->u.addrinfo.q4.WakeOnResolve = request->u.addrinfo.q6.WakeOnResolve = 0; - request->u.addrinfo.q4.qnameOrig = request->u.addrinfo.q6.qnameOrig = mDNSNULL; - - if (request->u.addrinfo.protocol & kDNSServiceProtocol_IPv4) - { - request->u.addrinfo.q4.qtype = kDNSServiceType_A; - request->u.addrinfo.q4.SearchListIndex = 0; - - // We append search domains only for queries that are a single label. If overriden using - // command line argument "AlwaysAppendSearchDomains", then we do it for any query which - // is not fully qualified. - if (hostname[strlen(hostname) - 1] != '.' && (AlwaysAppendSearchDomains || CountLabels(&d) == 1)) - { - request->u.addrinfo.q4.AppendSearchDomains = 1; - request->u.addrinfo.q4.AppendLocalSearchDomains = 1; - } - else - { - request->u.addrinfo.q4.AppendSearchDomains = 0; - request->u.addrinfo.q4.AppendLocalSearchDomains = 0; - } - request->u.addrinfo.q4.RetryWithSearchDomains = (ApplySearchDomainsFirst(&request->u.addrinfo.q4) ? 1 : 0); - request->u.addrinfo.q4.QuestionCallback = queryrecord_result_callback; - request->u.addrinfo.q4.QuestionContext = request; - err = mDNS_StartQuery(&mDNSStorage, &request->u.addrinfo.q4); - if (err != mStatus_NoError) - { - LogMsg("ERROR: mDNS_StartQuery: %d", (int)err); - request->u.addrinfo.q4.QuestionContext = mDNSNULL; - } - #if APPLE_OSX_mDNSResponder - err = SendAdditionalQuery(&request->u.addrinfo.q4, request, err); - #endif // APPLE_OSX_mDNSResponder - } - - if (!err && (request->u.addrinfo.protocol & kDNSServiceProtocol_IPv6)) - { - request->u.addrinfo.q6.qtype = kDNSServiceType_AAAA; - request->u.addrinfo.q6.SearchListIndex = 0; - if (hostname[strlen(hostname) - 1] != '.' && (AlwaysAppendSearchDomains || CountLabels(&d) == 1)) - { - request->u.addrinfo.q6.AppendSearchDomains = 1; - request->u.addrinfo.q6.AppendLocalSearchDomains = 1; - } - else - { - request->u.addrinfo.q6.AppendSearchDomains = 0; - request->u.addrinfo.q6.AppendLocalSearchDomains = 0; - } - request->u.addrinfo.q6.RetryWithSearchDomains = (ApplySearchDomainsFirst(&request->u.addrinfo.q6) ? 1 : 0); - request->u.addrinfo.q6.QuestionCallback = queryrecord_result_callback; - request->u.addrinfo.q6.QuestionContext = request; - err = mDNS_StartQuery(&mDNSStorage, &request->u.addrinfo.q6); - if (err != mStatus_NoError) - { - LogMsg("ERROR: mDNS_StartQuery: %d", (int)err); - request->u.addrinfo.q6.QuestionContext = mDNSNULL; - if (request->u.addrinfo.protocol & kDNSServiceProtocol_IPv4) - { - // If we started a query for IPv4, we need to cancel it - mDNS_StopQuery(&mDNSStorage, &request->u.addrinfo.q4); - request->u.addrinfo.q4.QuestionContext = mDNSNULL; - } - } - #if APPLE_OSX_mDNSResponder - err = SendAdditionalQuery(&request->u.addrinfo.q6, request, err); - #endif // APPLE_OSX_mDNSResponder - } - - LogOperation("%3d: DNSServiceGetAddrInfo(%X, %d, %d, %##s) START", - request->sd, flags, interfaceIndex, request->u.addrinfo.protocol, d.c); - - if (!err) request->terminate = addrinfo_termination_callback; - - return(err); - } - -// *************************************************************************** -#if COMPILER_LIKES_PRAGMA_MARK -#pragma mark - -#pragma mark - Main Request Handler etc. -#endif - -mDNSlocal request_state *NewRequest(void) - { - request_state **p = &all_requests; - while (*p) p=&(*p)->next; - *p = mallocL("request_state", sizeof(request_state)); - if (!*p) FatalError("ERROR: malloc"); - mDNSPlatformMemZero(*p, sizeof(request_state)); - return(*p); - } - -// read_msg may be called any time when the transfer state (req->ts) is t_morecoming. -// if there is no data on the socket, the socket will be closed and t_terminated will be returned -mDNSlocal void read_msg(request_state *req) - { - if (req->ts == t_terminated || req->ts == t_error) - { LogMsg("%3d: ERROR: read_msg called with transfer state terminated or error", req->sd); req->ts = t_error; return; } - - if (req->ts == t_complete) // this must be death or something is wrong - { - char buf[4]; // dummy for death notification - int nread = udsSupportReadFD(req->sd, buf, 4, 0, req->platform_data); - if (!nread) { req->ts = t_terminated; return; } - if (nread < 0) goto rerror; - LogMsg("%3d: ERROR: read data from a completed request", req->sd); - req->ts = t_error; - return; - } - - if (req->ts != t_morecoming) - { LogMsg("%3d: ERROR: read_msg called with invalid transfer state (%d)", req->sd, req->ts); req->ts = t_error; return; } - - if (req->hdr_bytes < sizeof(ipc_msg_hdr)) - { - mDNSu32 nleft = sizeof(ipc_msg_hdr) - req->hdr_bytes; - int nread = udsSupportReadFD(req->sd, (char *)&req->hdr + req->hdr_bytes, nleft, 0, req->platform_data); - if (nread == 0) { req->ts = t_terminated; return; } - if (nread < 0) goto rerror; - req->hdr_bytes += nread; - if (req->hdr_bytes > sizeof(ipc_msg_hdr)) - { LogMsg("%3d: ERROR: read_msg - read too many header bytes", req->sd); req->ts = t_error; return; } - - // only read data if header is complete - if (req->hdr_bytes == sizeof(ipc_msg_hdr)) - { - ConvertHeaderBytes(&req->hdr); - if (req->hdr.version != VERSION) - { LogMsg("%3d: ERROR: client version 0x%08X daemon version 0x%08X", req->sd, req->hdr.version, VERSION); req->ts = t_error; return; } - - // Largest conceivable single request is a DNSServiceRegisterRecord() or DNSServiceAddRecord() - // with 64kB of rdata. Adding 1009 byte for a maximal domain name, plus a safety margin - // for other overhead, this means any message above 70kB is definitely bogus. - if (req->hdr.datalen > 70000) - { LogMsg("%3d: ERROR: read_msg: hdr.datalen %u (0x%X) > 70000", req->sd, req->hdr.datalen, req->hdr.datalen); req->ts = t_error; return; } - req->msgbuf = mallocL("request_state msgbuf", req->hdr.datalen + MSG_PAD_BYTES); - if (!req->msgbuf) { my_perror("ERROR: malloc"); req->ts = t_error; return; } - req->msgptr = req->msgbuf; - req->msgend = req->msgbuf + req->hdr.datalen; - mDNSPlatformMemZero(req->msgbuf, req->hdr.datalen + MSG_PAD_BYTES); - } - } - - // If our header is complete, but we're still needing more body data, then try to read it now - // Note: For cancel_request req->hdr.datalen == 0, but there's no error return socket for cancel_request - // Any time we need to get the error return socket we know we'll have at least one data byte - // (even if only the one-byte empty C string placeholder for the old ctrl_path parameter) - if (req->hdr_bytes == sizeof(ipc_msg_hdr) && req->data_bytes < req->hdr.datalen) - { - mDNSu32 nleft = req->hdr.datalen - req->data_bytes; - int nread; -#if !defined(_WIN32) - struct iovec vec = { req->msgbuf + req->data_bytes, nleft }; // Tell recvmsg where we want the bytes put - struct msghdr msg; - struct cmsghdr *cmsg; - char cbuf[CMSG_SPACE(sizeof(dnssd_sock_t))]; - msg.msg_name = 0; - msg.msg_namelen = 0; - msg.msg_iov = &vec; - msg.msg_iovlen = 1; - msg.msg_control = cbuf; - msg.msg_controllen = sizeof(cbuf); - msg.msg_flags = 0; - nread = recvmsg(req->sd, &msg, 0); -#else - nread = udsSupportReadFD(req->sd, (char *)req->msgbuf + req->data_bytes, nleft, 0, req->platform_data); -#endif - if (nread == 0) { req->ts = t_terminated; return; } - if (nread < 0) goto rerror; - req->data_bytes += nread; - if (req->data_bytes > req->hdr.datalen) - { LogMsg("%3d: ERROR: read_msg - read too many data bytes", req->sd); req->ts = t_error; return; } -#if !defined(_WIN32) - cmsg = CMSG_FIRSTHDR(&msg); -#if DEBUG_64BIT_SCM_RIGHTS - LogMsg("%3d: Expecting %d %d %d %d", req->sd, sizeof(cbuf), sizeof(cbuf), SOL_SOCKET, SCM_RIGHTS); - LogMsg("%3d: Got %d %d %d %d", req->sd, msg.msg_controllen, cmsg->cmsg_len, cmsg->cmsg_level, cmsg->cmsg_type); -#endif // DEBUG_64BIT_SCM_RIGHTS - if (msg.msg_controllen == sizeof(cbuf) && - cmsg->cmsg_len == CMSG_LEN(sizeof(dnssd_sock_t)) && - cmsg->cmsg_level == SOL_SOCKET && - cmsg->cmsg_type == SCM_RIGHTS) - { -#if APPLE_OSX_mDNSResponder - // Strictly speaking BPF_fd belongs solely in the platform support layer, but because - // of privilege separation on Mac OS X we need to get BPF_fd from mDNSResponderHelper, - // and it's convenient to repurpose the existing fd-passing code here for that task - if (req->hdr.op == send_bpf) - { - dnssd_sock_t x = *(dnssd_sock_t *)CMSG_DATA(cmsg); - LogOperation("%3d: Got BPF %d", req->sd, x); - mDNSPlatformReceiveBPF_fd(&mDNSStorage, x); - } - else -#endif // APPLE_OSX_mDNSResponder - req->errsd = *(dnssd_sock_t *)CMSG_DATA(cmsg); -#if DEBUG_64BIT_SCM_RIGHTS - LogMsg("%3d: read req->errsd %d", req->sd, req->errsd); -#endif // DEBUG_64BIT_SCM_RIGHTS - if (req->data_bytes < req->hdr.datalen) - { - LogMsg("%3d: Client sent error socket %d via SCM_RIGHTS with req->data_bytes %d < req->hdr.datalen %d", - req->sd, req->errsd, req->data_bytes, req->hdr.datalen); - req->ts = t_error; - return; - } - } -#endif - } - - // If our header and data are both complete, see if we need to make our separate error return socket - if (req->hdr_bytes == sizeof(ipc_msg_hdr) && req->data_bytes == req->hdr.datalen) - { - if (req->terminate && req->hdr.op != cancel_request) - { - dnssd_sockaddr_t cliaddr; -#if defined(USE_TCP_LOOPBACK) - mDNSOpaque16 port; - u_long opt = 1; - port.b[0] = req->msgptr[0]; - port.b[1] = req->msgptr[1]; - req->msgptr += 2; - cliaddr.sin_family = AF_INET; - cliaddr.sin_port = port.NotAnInteger; - cliaddr.sin_addr.s_addr = inet_addr(MDNS_TCP_SERVERADDR); -#else - char ctrl_path[MAX_CTLPATH]; - get_string(&req->msgptr, req->msgend, ctrl_path, MAX_CTLPATH); // path is first element in message buffer - mDNSPlatformMemZero(&cliaddr, sizeof(cliaddr)); - cliaddr.sun_family = AF_LOCAL; - mDNSPlatformStrCopy(cliaddr.sun_path, ctrl_path); - // If the error return path UDS name is empty string, that tells us - // that this is a new version of the library that's going to pass us - // the error return path socket via sendmsg/recvmsg - if (ctrl_path[0] == 0) - { - if (req->errsd == req->sd) - { LogMsg("%3d: read_msg: ERROR failed to get errsd via SCM_RIGHTS", req->sd); req->ts = t_error; return; } - goto got_errfd; - } -#endif - - req->errsd = socket(AF_DNSSD, SOCK_STREAM, 0); - if (!dnssd_SocketValid(req->errsd)) { my_perror("ERROR: socket"); req->ts = t_error; return; } - - if (connect(req->errsd, (struct sockaddr *)&cliaddr, sizeof(cliaddr)) < 0) - { -#if !defined(USE_TCP_LOOPBACK) - struct stat sb; - LogMsg("%3d: read_msg: Couldn't connect to error return path socket '%s' errno %d (%s)", - req->sd, cliaddr.sun_path, dnssd_errno, dnssd_strerror(dnssd_errno)); - if (stat(cliaddr.sun_path, &sb) < 0) - LogMsg("%3d: read_msg: stat failed '%s' errno %d (%s)", req->sd, cliaddr.sun_path, dnssd_errno, dnssd_strerror(dnssd_errno)); - else - LogMsg("%3d: read_msg: file '%s' mode %o (octal) uid %d gid %d", req->sd, cliaddr.sun_path, sb.st_mode, sb.st_uid, sb.st_gid); -#endif - req->ts = t_error; - return; - } - -#if !defined(USE_TCP_LOOPBACK) -got_errfd: -#endif - LogOperation("%3d: Error socket %d created %08X %08X", req->sd, req->errsd, req->hdr.client_context.u32[1], req->hdr.client_context.u32[0]); -#if defined(_WIN32) - if (ioctlsocket(req->errsd, FIONBIO, &opt) != 0) -#else - if (fcntl(req->errsd, F_SETFL, fcntl(req->errsd, F_GETFL, 0) | O_NONBLOCK) != 0) -#endif - { - LogMsg("%3d: ERROR: could not set control socket to non-blocking mode errno %d (%s)", - req->sd, dnssd_errno, dnssd_strerror(dnssd_errno)); - req->ts = t_error; - return; - } - } - - req->ts = t_complete; - } - - return; - -rerror: - if (dnssd_errno == dnssd_EWOULDBLOCK || dnssd_errno == dnssd_EINTR) return; - LogMsg("%3d: ERROR: read_msg errno %d (%s)", req->sd, dnssd_errno, dnssd_strerror(dnssd_errno)); - req->ts = t_error; - } - -#define RecordOrientedOp(X) \ - ((X) == reg_record_request || (X) == add_record_request || (X) == update_record_request || (X) == remove_record_request) - -// The lightweight operations are the ones that don't need a dedicated request_state structure allocated for them -#define LightweightOp(X) (RecordOrientedOp(X) || (X) == cancel_request) - -mDNSlocal void request_callback(int fd, short filter, void *info) - { - mStatus err = 0; - request_state *req = info; - mDNSs32 min_size = sizeof(DNSServiceFlags); - (void)fd; // Unused - (void)filter; // Unused - - for (;;) - { - read_msg(req); - if (req->ts == t_morecoming) return; - if (req->ts == t_terminated || req->ts == t_error) { AbortUnlinkAndFree(req); return; } - if (req->ts != t_complete) { LogMsg("req->ts %d != t_complete", req->ts); AbortUnlinkAndFree(req); return; } - - if (req->hdr.version != VERSION) - { - LogMsg("ERROR: client version %d incompatible with daemon version %d", req->hdr.version, VERSION); - AbortUnlinkAndFree(req); - return; - } - - switch(req->hdr.op) // Interface + other data - { - case connection_request: min_size = 0; break; - case reg_service_request: min_size += sizeof(mDNSu32) + 4 /* name, type, domain, host */ + 4 /* port, textlen */; break; - case add_record_request: min_size += 4 /* type, rdlen */ + 4 /* ttl */; break; - case update_record_request: min_size += 2 /* rdlen */ + 4 /* ttl */; break; - case remove_record_request: break; - case browse_request: min_size += sizeof(mDNSu32) + 2 /* type, domain */; break; - case resolve_request: min_size += sizeof(mDNSu32) + 3 /* type, type, domain */; break; - case query_request: min_size += sizeof(mDNSu32) + 1 /* name */ + 4 /* type, class*/; break; - case enumeration_request: min_size += sizeof(mDNSu32); break; - case reg_record_request: min_size += sizeof(mDNSu32) + 1 /* name */ + 6 /* type, class, rdlen */ + 4 /* ttl */; break; - case reconfirm_record_request: min_size += sizeof(mDNSu32) + 1 /* name */ + 6 /* type, class, rdlen */; break; - case setdomain_request: min_size += 1 /* domain */; break; - case getproperty_request: min_size = 2; break; - case port_mapping_request: min_size += sizeof(mDNSu32) + 4 /* udp/tcp */ + 4 /* int/ext port */ + 4 /* ttl */; break; - case addrinfo_request: min_size += sizeof(mDNSu32) + 4 /* v4/v6 */ + 1 /* hostname */; break; - case send_bpf: // Same as cancel_request below - case cancel_request: min_size = 0; break; - default: LogMsg("ERROR: validate_message - unsupported req type: %d", req->hdr.op); min_size = -1; break; - } - - if ((mDNSs32)req->data_bytes < min_size) - { LogMsg("Invalid message %d bytes; min for %d is %d", req->data_bytes, req->hdr.op, min_size); AbortUnlinkAndFree(req); return; } - - if (LightweightOp(req->hdr.op) && !req->terminate) - { LogMsg("Reg/Add/Update/Remove %d require existing connection", req->hdr.op); AbortUnlinkAndFree(req); return; } - - // check if client wants silent operation - if (req->hdr.ipc_flags & IPC_FLAGS_NOREPLY) req->no_reply = 1; - - // If req->terminate is already set, this means this operation is sharing an existing connection - if (req->terminate && !LightweightOp(req->hdr.op)) - { - request_state *newreq = NewRequest(); - newreq->primary = req; - newreq->sd = req->sd; - newreq->errsd = req->errsd; - newreq->uid = req->uid; - newreq->hdr = req->hdr; - newreq->msgbuf = req->msgbuf; - newreq->msgptr = req->msgptr; - newreq->msgend = req->msgend; - req = newreq; - } - - // If we're shutting down, don't allow new client requests - // We do allow "cancel" and "getproperty" during shutdown - if (mDNSStorage.ShutdownTime && req->hdr.op != cancel_request && req->hdr.op != getproperty_request) - { - err = mStatus_ServiceNotRunning; - } - else switch(req->hdr.op) - { - // These are all operations that have their own first-class request_state object - case connection_request: LogOperation("%3d: DNSServiceCreateConnection START", req->sd); - req->terminate = connection_termination; break; - case resolve_request: err = handle_resolve_request (req); break; - case query_request: err = handle_queryrecord_request (req); break; - case browse_request: err = handle_browse_request (req); break; - case reg_service_request: err = handle_regservice_request (req); break; - case enumeration_request: err = handle_enum_request (req); break; - case reconfirm_record_request: err = handle_reconfirm_request (req); break; - case setdomain_request: err = handle_setdomain_request (req); break; - case getproperty_request: handle_getproperty_request (req); break; - case port_mapping_request: err = handle_port_mapping_request(req); break; - case addrinfo_request: err = handle_addrinfo_request (req); break; - case send_bpf: /* Do nothing for send_bpf */ break; - - // These are all operations that work with an existing request_state object - case reg_record_request: err = handle_regrecord_request (req); break; - case add_record_request: err = handle_add_request (req); break; - case update_record_request: err = handle_update_request (req); break; - case remove_record_request: err = handle_removerecord_request(req); break; - case cancel_request: handle_cancel_request (req); break; - default: LogMsg("%3d: ERROR: Unsupported UDS req: %d", req->sd, req->hdr.op); - } - - // req->msgbuf may be NULL, e.g. for connection_request or remove_record_request - if (req->msgbuf) freeL("request_state msgbuf", req->msgbuf); - - // There's no return data for a cancel request (DNSServiceRefDeallocate returns no result) - // For a DNSServiceGetProperty call, the handler already generated the response, so no need to do it again here - if (req->hdr.op != cancel_request && req->hdr.op != getproperty_request && req->hdr.op != send_bpf) - { - const mStatus err_netorder = dnssd_htonl(err); - send_all(req->errsd, (const char *)&err_netorder, sizeof(err_netorder)); - if (req->errsd != req->sd) - { - LogOperation("%3d: Error socket %d closed %08X %08X (%d)", - req->sd, req->errsd, req->hdr.client_context.u32[1], req->hdr.client_context.u32[0], err); - dnssd_close(req->errsd); - req->errsd = req->sd; - // Also need to reset the parent's errsd, if this is a subordinate operation - if (req->primary) req->primary->errsd = req->primary->sd; - } - } - - // Reset ready to accept the next req on this pipe - if (req->primary) req = req->primary; - req->ts = t_morecoming; - req->hdr_bytes = 0; - req->data_bytes = 0; - req->msgbuf = mDNSNULL; - req->msgptr = mDNSNULL; - req->msgend = 0; - } - } - -mDNSlocal void connect_callback(int fd, short filter, void *info) - { - dnssd_sockaddr_t cliaddr; - dnssd_socklen_t len = (dnssd_socklen_t) sizeof(cliaddr); - dnssd_sock_t sd = accept(fd, (struct sockaddr*) &cliaddr, &len); -#if defined(SO_NOSIGPIPE) || defined(_WIN32) - unsigned long optval = 1; -#endif - - (void)filter; // Unused - (void)info; // Unused - - if (!dnssd_SocketValid(sd)) - { - if (dnssd_errno != dnssd_EWOULDBLOCK) my_perror("ERROR: accept"); - return; - } - -#ifdef SO_NOSIGPIPE - // Some environments (e.g. OS X) support turning off SIGPIPE for a socket - if (setsockopt(sd, SOL_SOCKET, SO_NOSIGPIPE, &optval, sizeof(optval)) < 0) - LogMsg("%3d: WARNING: setsockopt - SO_NOSIGPIPE %d (%s)", sd, dnssd_errno, dnssd_strerror(dnssd_errno)); -#endif - -#if defined(_WIN32) - if (ioctlsocket(sd, FIONBIO, &optval) != 0) -#else - if (fcntl(sd, F_SETFL, fcntl(sd, F_GETFL, 0) | O_NONBLOCK) != 0) -#endif - { - my_perror("ERROR: fcntl(sd, F_SETFL, O_NONBLOCK) - aborting client"); - dnssd_close(sd); - return; - } - else - { - request_state *request = NewRequest(); - request->ts = t_morecoming; - request->sd = sd; - request->errsd = sd; -#if APPLE_OSX_mDNSResponder - struct xucred x; - socklen_t xucredlen = sizeof(x); - if (getsockopt(sd, 0, LOCAL_PEERCRED, &x, &xucredlen) >= 0 && x.cr_version == XUCRED_VERSION) request->uid = x.cr_uid; - else my_perror("ERROR: getsockopt, LOCAL_PEERCRED"); - debugf("LOCAL_PEERCRED %d %u %u %d", xucredlen, x.cr_version, x.cr_uid, x.cr_ngroups); -#endif // APPLE_OSX_mDNSResponder - LogOperation("%3d: Adding FD for uid %u", request->sd, request->uid); - udsSupportAddFDToEventLoop(sd, request_callback, request, &request->platform_data); - } - } - -mDNSlocal mDNSBool uds_socket_setup(dnssd_sock_t skt) - { -#if defined(SO_NP_EXTENSIONS) - struct so_np_extensions sonpx; - socklen_t optlen = sizeof(struct so_np_extensions); - sonpx.npx_flags = SONPX_SETOPTSHUT; - sonpx.npx_mask = SONPX_SETOPTSHUT; - if (setsockopt(skt, SOL_SOCKET, SO_NP_EXTENSIONS, &sonpx, optlen) < 0) - my_perror("WARNING: could not set sockopt - SO_NP_EXTENSIONS"); -#endif -#if defined(_WIN32) - // SEH: do we even need to do this on windows? - // This socket will be given to WSAEventSelect which will automatically set it to non-blocking - u_long opt = 1; - if (ioctlsocket(skt, FIONBIO, &opt) != 0) -#else - if (fcntl(skt, F_SETFL, fcntl(skt, F_GETFL, 0) | O_NONBLOCK) != 0) -#endif - { - my_perror("ERROR: could not set listen socket to non-blocking mode"); - return mDNSfalse; - } - - if (listen(skt, LISTENQ) != 0) - { - my_perror("ERROR: could not listen on listen socket"); - return mDNSfalse; - } - - if (mStatus_NoError != udsSupportAddFDToEventLoop(skt, connect_callback, (void *) NULL, (void **) NULL)) - { - my_perror("ERROR: could not add listen socket to event loop"); - return mDNSfalse; - } - else LogOperation("%3d: Listening for incoming Unix Domain Socket client requests", skt); - - return mDNStrue; - } - -mDNSexport int udsserver_init(dnssd_sock_t skts[], mDNSu32 count) - { - dnssd_sockaddr_t laddr; - int ret; - mDNSu32 i = 0; - - LogInfo("udsserver_init"); - - // If a particular platform wants to opt out of having a PID file, define PID_FILE to be "" - if (PID_FILE[0]) - { - FILE *fp = fopen(PID_FILE, "w"); - if (fp != NULL) - { - fprintf(fp, "%d\n", getpid()); - fclose(fp); - } - } - - if (skts) - { - for (i = 0; i < count; i++) - if (dnssd_SocketValid(skts[i]) && !uds_socket_setup(skts[i])) - goto error; - } - else - { - listenfd = socket(AF_DNSSD, SOCK_STREAM, 0); - if (!dnssd_SocketValid(listenfd)) - { - my_perror("ERROR: socket(AF_DNSSD, SOCK_STREAM, 0); failed"); - goto error; - } - - mDNSPlatformMemZero(&laddr, sizeof(laddr)); - - #if defined(USE_TCP_LOOPBACK) - { - laddr.sin_family = AF_INET; - laddr.sin_port = htons(MDNS_TCP_SERVERPORT); - laddr.sin_addr.s_addr = inet_addr(MDNS_TCP_SERVERADDR); - ret = bind(listenfd, (struct sockaddr *) &laddr, sizeof(laddr)); - if (ret < 0) - { - my_perror("ERROR: bind(listenfd, (struct sockaddr *) &laddr, sizeof(laddr)); failed"); - goto error; - } - } - #else - { - mode_t mask = umask(0); - unlink(MDNS_UDS_SERVERPATH); // OK if this fails - laddr.sun_family = AF_LOCAL; - #ifndef NOT_HAVE_SA_LEN - // According to Stevens (section 3.2), there is no portable way to - // determine whether sa_len is defined on a particular platform. - laddr.sun_len = sizeof(struct sockaddr_un); - #endif - mDNSPlatformStrCopy(laddr.sun_path, MDNS_UDS_SERVERPATH); - ret = bind(listenfd, (struct sockaddr *) &laddr, sizeof(laddr)); - umask(mask); - if (ret < 0) - { - my_perror("ERROR: bind(listenfd, (struct sockaddr *) &laddr, sizeof(laddr)); failed"); - goto error; - } - } - #endif - - if (!uds_socket_setup(listenfd)) goto error; - } - -#if !defined(PLATFORM_NO_RLIMIT) - { - // Set maximum number of open file descriptors - #define MIN_OPENFILES 10240 - struct rlimit maxfds, newfds; - - // Due to bugs in OS X (, , ) - // you have to get and set rlimits once before getrlimit will return sensible values - if (getrlimit(RLIMIT_NOFILE, &maxfds) < 0) { my_perror("ERROR: Unable to get file descriptor limit"); return 0; } - if (setrlimit(RLIMIT_NOFILE, &maxfds) < 0) my_perror("ERROR: Unable to set maximum file descriptor limit"); - - if (getrlimit(RLIMIT_NOFILE, &maxfds) < 0) { my_perror("ERROR: Unable to get file descriptor limit"); return 0; } - newfds.rlim_max = (maxfds.rlim_max > MIN_OPENFILES) ? maxfds.rlim_max : MIN_OPENFILES; - newfds.rlim_cur = (maxfds.rlim_cur > MIN_OPENFILES) ? maxfds.rlim_cur : MIN_OPENFILES; - if (newfds.rlim_max != maxfds.rlim_max || newfds.rlim_cur != maxfds.rlim_cur) - if (setrlimit(RLIMIT_NOFILE, &newfds) < 0) my_perror("ERROR: Unable to set maximum file descriptor limit"); - - if (getrlimit(RLIMIT_NOFILE, &maxfds) < 0) { my_perror("ERROR: Unable to get file descriptor limit"); return 0; } - debugf("maxfds.rlim_max %d", (long)maxfds.rlim_max); - debugf("maxfds.rlim_cur %d", (long)maxfds.rlim_cur); - } -#endif - - // We start a "LocalOnly" query looking for Automatic Browse Domain records. - // When Domain Enumeration in uDNS.c finds an "lb" record from the network, its "FoundDomain" routine - // creates a "LocalOnly" record, which results in our AutomaticBrowseDomainChange callback being invoked - mDNS_GetDomains(&mDNSStorage, &mDNSStorage.AutomaticBrowseDomainQ, mDNS_DomainTypeBrowseAutomatic, - mDNSNULL, mDNSInterface_LocalOnly, AutomaticBrowseDomainChange, mDNSNULL); - - // Add "local" as recommended registration domain ("dns-sd -E"), recommended browsing domain ("dns-sd -F"), and automatic browsing domain - RegisterLocalOnlyDomainEnumPTR(&mDNSStorage, &localdomain, mDNS_DomainTypeRegistration); - RegisterLocalOnlyDomainEnumPTR(&mDNSStorage, &localdomain, mDNS_DomainTypeBrowse); - AddAutoBrowseDomain(0, &localdomain); - - udsserver_handle_configchange(&mDNSStorage); - return 0; - -error: - - my_perror("ERROR: udsserver_init"); - return -1; - } - -mDNSexport int udsserver_exit(void) - { - // Cancel all outstanding client requests - while (all_requests) AbortUnlinkAndFree(all_requests); - - // Clean up any special mDNSInterface_LocalOnly records we created, both the entries for "local" we - // created in udsserver_init, and others we created as a result of reading local configuration data - while (LocalDomainEnumRecords) - { - ARListElem *rem = LocalDomainEnumRecords; - LocalDomainEnumRecords = LocalDomainEnumRecords->next; - mDNS_Deregister(&mDNSStorage, &rem->ar); - } - - // If the launching environment created no listening socket, - // that means we created it ourselves, so we should clean it up on exit - if (dnssd_SocketValid(listenfd)) - { - dnssd_close(listenfd); -#if !defined(USE_TCP_LOOPBACK) - // Currently, we're unable to remove /var/run/mdnsd because we've changed to userid "nobody" - // to give up unnecessary privilege, but we need to be root to remove this Unix Domain Socket. - // It would be nice if we could find a solution to this problem - if (unlink(MDNS_UDS_SERVERPATH)) - debugf("Unable to remove %s", MDNS_UDS_SERVERPATH); -#endif - } - - if (PID_FILE[0]) unlink(PID_FILE); - - return 0; - } - -mDNSlocal void LogClientInfo(mDNS *const m, const request_state *req) - { - char prefix[16]; - if (req->primary) mDNS_snprintf(prefix, sizeof(prefix), " -> "); - else mDNS_snprintf(prefix, sizeof(prefix), "%3d:", req->sd); - - usleep((m->KnownBugs & mDNS_KnownBug_LossySyslog) ? 3333 : 1000); - - if (!req->terminate) - LogMsgNoIdent("%s No operation yet on this socket", prefix); - else if (req->terminate == connection_termination) - { - int num_records = 0, num_ops = 0; - const registered_record_entry *p; - const request_state *r; - for (p = req->u.reg_recs; p; p=p->next) num_records++; - for (r = req->next; r; r=r->next) if (r->primary == req) num_ops++; - LogMsgNoIdent("%s DNSServiceCreateConnection: %d registered record%s, %d kDNSServiceFlagsShareConnection operation%s", prefix, - num_records, num_records != 1 ? "s" : "", - num_ops, num_ops != 1 ? "s" : ""); - for (p = req->u.reg_recs; p; p=p->next) - LogMsgNoIdent(" -> DNSServiceRegisterRecord %3d %s", p->key, ARDisplayString(m, p->rr)); - for (r = req->next; r; r=r->next) if (r->primary == req) LogClientInfo(m, r); - } - else if (req->terminate == regservice_termination_callback) - { - service_instance *ptr; - for (ptr = req->u.servicereg.instances; ptr; ptr = ptr->next) - LogMsgNoIdent("%s DNSServiceRegister %##s %u/%u", - (ptr == req->u.servicereg.instances) ? prefix : " ", - ptr->srs.RR_SRV.resrec.name->c, mDNSVal16(req->u.servicereg.port), SRS_PORT(&ptr->srs)); - } - else if (req->terminate == browse_termination_callback) - { - browser_t *blist; - for (blist = req->u.browser.browsers; blist; blist = blist->next) - LogMsgNoIdent("%s DNSServiceBrowse %##s", (blist == req->u.browser.browsers) ? prefix : " ", blist->q.qname.c); - } - else if (req->terminate == resolve_termination_callback) - LogMsgNoIdent("%s DNSServiceResolve %##s", prefix, req->u.resolve.qsrv.qname.c); - else if (req->terminate == queryrecord_termination_callback) - LogMsgNoIdent("%s DNSServiceQueryRecord %##s (%s)", prefix, req->u.queryrecord.q.qname.c, DNSTypeName(req->u.queryrecord.q.qtype)); - else if (req->terminate == enum_termination_callback) - LogMsgNoIdent("%s DNSServiceEnumerateDomains %##s", prefix, req->u.enumeration.q_all.qname.c); - else if (req->terminate == port_mapping_termination_callback) - LogMsgNoIdent("%s DNSServiceNATPortMapping %.4a %s%s Int %d Req %d Ext %d Req TTL %d Granted TTL %d", - prefix, - &req->u.pm.NATinfo.ExternalAddress, - req->u.pm.NATinfo.Protocol & NATOp_MapTCP ? "TCP" : " ", - req->u.pm.NATinfo.Protocol & NATOp_MapUDP ? "UDP" : " ", - mDNSVal16(req->u.pm.NATinfo.IntPort), - mDNSVal16(req->u.pm.ReqExt), - mDNSVal16(req->u.pm.NATinfo.ExternalPort), - req->u.pm.NATinfo.NATLease, - req->u.pm.NATinfo.Lifetime); - else if (req->terminate == addrinfo_termination_callback) - LogMsgNoIdent("%s DNSServiceGetAddrInfo %s%s %##s", prefix, - req->u.addrinfo.protocol & kDNSServiceProtocol_IPv4 ? "v4" : " ", - req->u.addrinfo.protocol & kDNSServiceProtocol_IPv6 ? "v6" : " ", - req->u.addrinfo.q4.qname.c); - else - LogMsgNoIdent("%s Unrecognized operation %p", prefix, req->terminate); - } - -mDNSlocal char *RecordTypeName(mDNSu8 rtype) - { - switch (rtype) - { - case kDNSRecordTypeUnregistered: return ("Unregistered "); - case kDNSRecordTypeDeregistering: return ("Deregistering"); - case kDNSRecordTypeUnique: return ("Unique "); - case kDNSRecordTypeAdvisory: return ("Advisory "); - case kDNSRecordTypeShared: return ("Shared "); - case kDNSRecordTypeVerified: return ("Verified "); - case kDNSRecordTypeKnownUnique: return ("KnownUnique "); - default: return("Unknown"); - } - } - -mDNSlocal void LogEtcHosts(mDNS *const m) - { - mDNSBool showheader = mDNStrue; - const AuthRecord *ar; - mDNSu32 slot; - AuthGroup *ag; - int count = 0; - int authslot = 0; - mDNSBool truncated = 0; - - for (slot = 0; slot < AUTH_HASH_SLOTS; slot++) - { - if (m->rrauth.rrauth_hash[slot]) authslot++; - for (ag = m->rrauth.rrauth_hash[slot]; ag; ag = ag->next) - for (ar = ag->members; ar; ar = ar->next) - { - if (ar->RecordCallback != FreeEtcHosts) continue; - if (showheader) { showheader = mDNSfalse; LogMsgNoIdent(" State Interface"); } - - // Print a maximum of 50 records - if (count++ >= 50) { truncated = mDNStrue; continue; } - if (ar->ARType == AuthRecordLocalOnly) - { - if (ar->resrec.InterfaceID == mDNSInterface_LocalOnly) - LogMsgNoIdent(" %s LO %s", RecordTypeName(ar->resrec.RecordType), ARDisplayString(m, ar)); - else - { - mDNSu32 scopeid = (mDNSu32)(uintptr_t)ar->resrec.InterfaceID; - LogMsgNoIdent(" %s %u %s", RecordTypeName(ar->resrec.RecordType), scopeid, ARDisplayString(m, ar)); - } - } - usleep((m->KnownBugs & mDNS_KnownBug_LossySyslog) ? 3333 : 1000); - } - } - - if (showheader) LogMsgNoIdent(""); - else if (truncated) LogMsgNoIdent("", count, m->rrauth.rrauth_totalused, authslot); - } - -mDNSlocal void LogLocalOnlyAuthRecords(mDNS *const m) - { - mDNSBool showheader = mDNStrue; - const AuthRecord *ar; - mDNSu32 slot; - AuthGroup *ag; - - for (slot = 0; slot < AUTH_HASH_SLOTS; slot++) - { - for (ag = m->rrauth.rrauth_hash[slot]; ag; ag = ag->next) - for (ar = ag->members; ar; ar = ar->next) - { - if (ar->RecordCallback == FreeEtcHosts) continue; - if (showheader) { showheader = mDNSfalse; LogMsgNoIdent(" State Interface"); } - - // Print a maximum of 400 records - if (ar->ARType == AuthRecordLocalOnly) - LogMsgNoIdent(" %s LO %s", RecordTypeName(ar->resrec.RecordType), ARDisplayString(m, ar)); - else if (ar->ARType == AuthRecordP2P) - LogMsgNoIdent(" %s PP %s", RecordTypeName(ar->resrec.RecordType), ARDisplayString(m, ar)); - usleep((m->KnownBugs & mDNS_KnownBug_LossySyslog) ? 3333 : 1000); - } - } - - if (showheader) LogMsgNoIdent(""); - } - -mDNSlocal void LogAuthRecords(mDNS *const m, const mDNSs32 now, AuthRecord *ResourceRecords, int *proxy) - { - mDNSBool showheader = mDNStrue; - const AuthRecord *ar; - OwnerOptData owner = zeroOwner; - for (ar = ResourceRecords; ar; ar=ar->next) - { - const char *const ifname = InterfaceNameForID(m, ar->resrec.InterfaceID); - if ((ar->WakeUp.HMAC.l[0] != 0) == (proxy != mDNSNULL)) - { - if (showheader) { showheader = mDNSfalse; LogMsgNoIdent(" Int Next Expire State"); } - if (proxy) (*proxy)++; - if (!mDNSPlatformMemSame(&owner, &ar->WakeUp, sizeof(owner))) - { - owner = ar->WakeUp; - if (owner.password.l[0]) - LogMsgNoIdent("Proxying for H-MAC %.6a I-MAC %.6a Password %.6a seq %d", &owner.HMAC, &owner.IMAC, &owner.password, owner.seq); - else if (!mDNSSameEthAddress(&owner.HMAC, &owner.IMAC)) - LogMsgNoIdent("Proxying for H-MAC %.6a I-MAC %.6a seq %d", &owner.HMAC, &owner.IMAC, owner.seq); - else - LogMsgNoIdent("Proxying for %.6a seq %d", &owner.HMAC, owner.seq); - } - if (AuthRecord_uDNS(ar)) - LogMsgNoIdent("%7d %7d %7d %7d %s", - ar->ThisAPInterval / mDNSPlatformOneSecond, - (ar->LastAPTime + ar->ThisAPInterval - now) / mDNSPlatformOneSecond, - ar->expire ? (ar->expire - now) / mDNSPlatformOneSecond : 0, - ar->state, ARDisplayString(m, ar)); - else if (ar->ARType == AuthRecordLocalOnly) - LogMsgNoIdent(" LO %s", ARDisplayString(m, ar)); - else if (ar->ARType == AuthRecordP2P) - LogMsgNoIdent(" PP %s", ARDisplayString(m, ar)); - else - LogMsgNoIdent("%7d %7d %7d %7s %s", - ar->ThisAPInterval / mDNSPlatformOneSecond, - ar->AnnounceCount ? (ar->LastAPTime + ar->ThisAPInterval - now) / mDNSPlatformOneSecond : 0, - ar->TimeExpire ? (ar->TimeExpire - now) / mDNSPlatformOneSecond : 0, - ifname ? ifname : "ALL", - ARDisplayString(m, ar)); - usleep((m->KnownBugs & mDNS_KnownBug_LossySyslog) ? 3333 : 1000); - } - } - if (showheader) LogMsgNoIdent(""); - } - -mDNSexport void udsserver_info(mDNS *const m) - { - const mDNSs32 now = mDNS_TimeNow(m); - mDNSu32 CacheUsed = 0, CacheActive = 0, slot; - int ProxyA = 0, ProxyD = 0; - const CacheGroup *cg; - const CacheRecord *cr; - const DNSQuestion *q; - const DNameListElem *d; - const SearchListElem *s; - - LogMsgNoIdent("Timenow 0x%08lX (%d)", (mDNSu32)now, now); - - LogMsgNoIdent("------------ Cache -------------"); - LogMsgNoIdent("Slt Q TTL if U Type rdlen"); - for (slot = 0; slot < CACHE_HASH_SLOTS; slot++) - for (cg = m->rrcache_hash[slot]; cg; cg=cg->next) - { - CacheUsed++; // Count one cache entity for the CacheGroup object - for (cr = cg->members; cr; cr=cr->next) - { - const mDNSs32 remain = cr->resrec.rroriginalttl - (now - cr->TimeRcvd) / mDNSPlatformOneSecond; - const char *ifname; - mDNSInterfaceID InterfaceID = cr->resrec.InterfaceID; - if (!InterfaceID && cr->resrec.rDNSServer) - InterfaceID = cr->resrec.rDNSServer->interface; - ifname = InterfaceNameForID(m, InterfaceID); - CacheUsed++; - if (cr->CRActiveQuestion) CacheActive++; - LogMsgNoIdent("%3d %s%8ld %-7s%s %-6s%s", - slot, - cr->CRActiveQuestion ? "*" : " ", - remain, - ifname ? ifname : "-U-", - (cr->resrec.RecordType == kDNSRecordTypePacketNegative) ? "-" : - (cr->resrec.RecordType & kDNSRecordTypePacketUniqueMask) ? " " : "+", - DNSTypeName(cr->resrec.rrtype), - CRDisplayString(m, cr)); - usleep((m->KnownBugs & mDNS_KnownBug_LossySyslog) ? 3333 : 1000); - } - } - - if (m->rrcache_totalused != CacheUsed) - LogMsgNoIdent("Cache use mismatch: rrcache_totalused is %lu, true count %lu", m->rrcache_totalused, CacheUsed); - if (m->rrcache_active != CacheActive) - LogMsgNoIdent("Cache use mismatch: rrcache_active is %lu, true count %lu", m->rrcache_active, CacheActive); - LogMsgNoIdent("Cache currently contains %lu entities; %lu referenced by active questions", CacheUsed, CacheActive); - - LogMsgNoIdent("--------- Auth Records ---------"); - LogAuthRecords(m, now, m->ResourceRecords, mDNSNULL); - - LogMsgNoIdent("--------- LocalOnly, P2P Auth Records ---------"); - LogLocalOnlyAuthRecords(m); - - LogMsgNoIdent("--------- /etc/hosts ---------"); - LogEtcHosts(m); - - LogMsgNoIdent("------ Duplicate Records -------"); - LogAuthRecords(m, now, m->DuplicateRecords, mDNSNULL); - - LogMsgNoIdent("----- Auth Records Proxied -----"); - LogAuthRecords(m, now, m->ResourceRecords, &ProxyA); - - LogMsgNoIdent("-- Duplicate Records Proxied ---"); - LogAuthRecords(m, now, m->DuplicateRecords, &ProxyD); - - LogMsgNoIdent("---------- Questions -----------"); - if (!m->Questions) LogMsgNoIdent(""); - else - { - CacheUsed = 0; - CacheActive = 0; - LogMsgNoIdent(" Int Next if T NumAns VDNS Qptr DupOf SU SQ Type Name"); - for (q = m->Questions; q; q=q->next) - { - mDNSs32 i = q->ThisQInterval / mDNSPlatformOneSecond; - mDNSs32 n = (NextQSendTime(q) - now) / mDNSPlatformOneSecond; - char *ifname = InterfaceNameForID(m, q->InterfaceID); - CacheUsed++; - if (q->ThisQInterval) CacheActive++; - LogMsgNoIdent("%6d%6d %-7s%s%s %5d 0x%x%x 0x%p 0x%p %1d %2d %-5s%##s%s", - i, n, - ifname ? ifname : mDNSOpaque16IsZero(q->TargetQID) ? "" : "-U-", - mDNSOpaque16IsZero(q->TargetQID) ? (q->LongLived ? "l" : " ") : (q->LongLived ? "L" : "O"), - PrivateQuery(q) ? "P" : " ", - q->CurrentAnswers, q->validDNSServers.l[1], q->validDNSServers.l[0], q, q->DuplicateOf, - q->SuppressUnusable, q->SuppressQuery, DNSTypeName(q->qtype), q->qname.c, q->DuplicateOf ? " (dup)" : ""); - usleep((m->KnownBugs & mDNS_KnownBug_LossySyslog) ? 3333 : 1000); - } - LogMsgNoIdent("%lu question%s; %lu active", CacheUsed, CacheUsed > 1 ? "s" : "", CacheActive); - } - - LogMsgNoIdent("----- Local-Only Questions -----"); - if (!m->LocalOnlyQuestions) LogMsgNoIdent(""); - else for (q = m->LocalOnlyQuestions; q; q=q->next) - LogMsgNoIdent(" %5d %-6s%##s%s", - q->CurrentAnswers, DNSTypeName(q->qtype), q->qname.c, q->DuplicateOf ? " (dup)" : ""); - - LogMsgNoIdent("---- Active Client Requests ----"); - if (!all_requests) LogMsgNoIdent(""); - else - { - const request_state *req, *r; - for (req = all_requests; req; req=req->next) - { - if (req->primary) // If this is a subbordinate operation, check that the parent is in the list - { - for (r = all_requests; r && r != req; r=r->next) if (r == req->primary) goto foundparent; - LogMsgNoIdent("%3d: Orhpan operation %p; parent %p not found in request list", req->sd); - } - // For non-subbordinate operations, and subbordinate operations that have lost their parent, write out their info - LogClientInfo(m, req); - foundparent:; - } - } - - LogMsgNoIdent("-------- NAT Traversals --------"); - if (!m->NATTraversals) LogMsgNoIdent(""); - else - { - const NATTraversalInfo *nat; - for (nat = m->NATTraversals; nat; nat=nat->next) - { - if (nat->Protocol) - LogMsgNoIdent("%p %s Int %5d Ext %5d Err %d Retry %5d Interval %5d Expire %5d", - nat, nat->Protocol == NATOp_MapTCP ? "TCP" : "UDP", - mDNSVal16(nat->IntPort), mDNSVal16(nat->ExternalPort), nat->Result, - nat->retryPortMap ? (nat->retryPortMap - now) / mDNSPlatformOneSecond : 0, - nat->retryInterval / mDNSPlatformOneSecond, - nat->ExpiryTime ? (nat->ExpiryTime - now) / mDNSPlatformOneSecond : 0); - else - LogMsgNoIdent("%p Address Request Retry %5d Interval %5d", nat, - (m->retryGetAddr - now) / mDNSPlatformOneSecond, - m->retryIntervalGetAddr / mDNSPlatformOneSecond); - usleep((m->KnownBugs & mDNS_KnownBug_LossySyslog) ? 3333 : 1000); - } - } - - LogMsgNoIdent("--------- AuthInfoList ---------"); - if (!m->AuthInfoList) LogMsgNoIdent(""); - else - { - const DomainAuthInfo *a; - for (a = m->AuthInfoList; a; a = a->next) - LogMsgNoIdent("%##s %##s %##s %d %s", a->domain.c, a->keyname.c, a->hostname.c, (a->port.b[0] << 8 | a->port.b[1]), a->AutoTunnel ? a->AutoTunnel : ""); - } - - #if APPLE_OSX_mDNSResponder - LogMsgNoIdent("--------- TunnelClients --------"); - if (!m->TunnelClients) LogMsgNoIdent(""); - else - { - const ClientTunnel *c; - for (c = m->TunnelClients; c; c = c->next) - LogMsgNoIdent("%s %##s local %.16a %.4a %.16a remote %.16a %.4a %5d %.16a interval %d", - c->prefix, c->dstname.c, &c->loc_inner, &c->loc_outer, &c->loc_outer6, &c->rmt_inner, &c->rmt_outer, mDNSVal16(c->rmt_outer_port), &c->rmt_outer6, c->q.ThisQInterval); - } - #endif // APPLE_OSX_mDNSResponder - - LogMsgNoIdent("---------- Misc State ----------"); - - LogMsgNoIdent("PrimaryMAC: %.6a", &m->PrimaryMAC); - - LogMsgNoIdent("m->SleepState %d (%s) seq %d", - m->SleepState, - m->SleepState == SleepState_Awake ? "Awake" : - m->SleepState == SleepState_Transferring ? "Transferring" : - m->SleepState == SleepState_Sleeping ? "Sleeping" : "?", - m->SleepSeqNum); - - if (!m->SPSSocket) LogMsgNoIdent("Not offering Sleep Proxy Service"); - else LogMsgNoIdent("Offering Sleep Proxy Service: %#s", m->SPSRecords.RR_SRV.resrec.name->c); - - if (m->ProxyRecords == ProxyA + ProxyD) LogMsgNoIdent("ProxyRecords: %d + %d = %d", ProxyA, ProxyD, ProxyA + ProxyD); - else LogMsgNoIdent("ProxyRecords: MISMATCH %d + %d = %d != %d", ProxyA, ProxyD, ProxyA + ProxyD, m->ProxyRecords); - - LogMsgNoIdent("------ Auto Browse Domains -----"); - if (!AutoBrowseDomains) LogMsgNoIdent(""); - else for (d=AutoBrowseDomains; d; d=d->next) LogMsgNoIdent("%##s", d->name.c); - - LogMsgNoIdent("--- Auto Registration Domains --"); - if (!AutoRegistrationDomains) LogMsgNoIdent(""); - else for (d=AutoRegistrationDomains; d; d=d->next) LogMsgNoIdent("%##s", d->name.c); - - LogMsgNoIdent("--- Search Domains --"); - if (!SearchList) LogMsgNoIdent(""); - else - { - for (s=SearchList; s; s=s->next) - { - char *ifname = InterfaceNameForID(m, s->InterfaceID); - LogMsgNoIdent("%##s %s", s->domain.c, ifname ? ifname : ""); - } - } - - LogMsgNoIdent("---- Task Scheduling Timers ----"); - - if (!m->NewQuestions) - LogMsgNoIdent("NewQuestion "); - else - LogMsgNoIdent("NewQuestion DelayAnswering %d %d %##s (%s)", - m->NewQuestions->DelayAnswering, m->NewQuestions->DelayAnswering-now, - m->NewQuestions->qname.c, DNSTypeName(m->NewQuestions->qtype)); - - if (!m->NewLocalOnlyQuestions) - LogMsgNoIdent("NewLocalOnlyQuestions "); - else - LogMsgNoIdent("NewLocalOnlyQuestions %##s (%s)", - m->NewLocalOnlyQuestions->qname.c, DNSTypeName(m->NewLocalOnlyQuestions->qtype)); - - if (!m->NewLocalRecords) - LogMsgNoIdent("NewLocalRecords "); - else - LogMsgNoIdent("NewLocalRecords %02X %s", m->NewLocalRecords->resrec.RecordType, ARDisplayString(m, m->NewLocalRecords)); - - LogMsgNoIdent("SPSProxyListChanged%s", m->SPSProxyListChanged ? "" : " "); - LogMsgNoIdent("LocalRemoveEvents%s", m->LocalRemoveEvents ? "" : " "); - LogMsgNoIdent("m->RegisterAutoTunnel6 %08X", m->RegisterAutoTunnel6); - LogMsgNoIdent("m->AutoTunnelRelayAddrIn %.16a", &m->AutoTunnelRelayAddrIn); - LogMsgNoIdent("m->AutoTunnelRelayAddrOut %.16a", &m->AutoTunnelRelayAddrOut); - -#define LogTimer(MSG,T) LogMsgNoIdent( MSG " %08X %11d %08X %11d", (T), (T), (T)-now, (T)-now) - - LogMsgNoIdent(" ABS (hex) ABS (dec) REL (hex) REL (dec)"); - LogMsgNoIdent("m->timenow %08X %11d", now, now); - LogMsgNoIdent("m->timenow_adjust %08X %11d", m->timenow_adjust, m->timenow_adjust); - LogTimer("m->NextScheduledEvent ", m->NextScheduledEvent); - -#ifndef UNICAST_DISABLED - LogTimer("m->NextuDNSEvent ", m->NextuDNSEvent); - LogTimer("m->NextSRVUpdate ", m->NextSRVUpdate); - LogTimer("m->NextScheduledNATOp ", m->NextScheduledNATOp); - LogTimer("m->retryGetAddr ", m->retryGetAddr); -#endif - - LogTimer("m->NextCacheCheck ", m->NextCacheCheck); - LogTimer("m->NextScheduledSPS ", m->NextScheduledSPS); - LogTimer("m->NextScheduledSPRetry ", m->NextScheduledSPRetry); - LogTimer("m->DelaySleep ", m->DelaySleep); - - LogTimer("m->NextScheduledQuery ", m->NextScheduledQuery); - LogTimer("m->NextScheduledProbe ", m->NextScheduledProbe); - LogTimer("m->NextScheduledResponse", m->NextScheduledResponse); - - LogTimer("m->SuppressSending ", m->SuppressSending); - LogTimer("m->SuppressProbes ", m->SuppressProbes); - LogTimer("m->ProbeFailTime ", m->ProbeFailTime); - LogTimer("m->DelaySleep ", m->DelaySleep); - LogTimer("m->SleepLimit ", m->SleepLimit); - LogTimer("m->NextScheduledStopTime ", m->NextScheduledStopTime); - } - -#if APPLE_OSX_mDNSResponder && MACOSX_MDNS_MALLOC_DEBUGGING -mDNSexport void uds_validatelists(void) - { - const request_state *req, *p; - for (req = all_requests; req; req=req->next) - { - if (req->next == (request_state *)~0 || (req->sd < 0 && req->sd != -2)) - LogMemCorruption("UDS request list: %p is garbage (%d)", req, req->sd); - - if (req->primary == req) - LogMemCorruption("UDS request list: req->primary should not point to self %p/%d", req, req->sd); - - if (req->primary && req->replies) - LogMemCorruption("UDS request list: Subordinate request %p/%d/%p should not have replies (%p)", - req, req->sd, req->primary && req->replies); - - p = req->primary; - if ((long)p & 3) - LogMemCorruption("UDS request list: req %p primary %p is misaligned (%d)", req, p, req->sd); - else if (p && (p->next == (request_state *)~0 || (p->sd < 0 && p->sd != -2))) - LogMemCorruption("UDS request list: req %p primary %p is garbage (%d)", req, p, p->sd); - - reply_state *rep; - for (rep = req->replies; rep; rep=rep->next) - if (rep->next == (reply_state *)~0) - LogMemCorruption("UDS req->replies: %p is garbage", rep); - - if (req->terminate == connection_termination) - { - registered_record_entry *r; - for (r = req->u.reg_recs; r; r=r->next) - if (r->next == (registered_record_entry *)~0) - LogMemCorruption("UDS req->u.reg_recs: %p is garbage", r); - } - else if (req->terminate == regservice_termination_callback) - { - service_instance *s; - for (s = req->u.servicereg.instances; s; s=s->next) - if (s->next == (service_instance *)~0) - LogMemCorruption("UDS req->u.servicereg.instances: %p is garbage", s); - } - else if (req->terminate == browse_termination_callback) - { - browser_t *b; - for (b = req->u.browser.browsers; b; b=b->next) - if (b->next == (browser_t *)~0) - LogMemCorruption("UDS req->u.browser.browsers: %p is garbage", b); - } - } - - DNameListElem *d; - for (d = SCPrefBrowseDomains; d; d=d->next) - if (d->next == (DNameListElem *)~0 || d->name.c[0] > 63) - LogMemCorruption("SCPrefBrowseDomains: %p is garbage (%d)", d, d->name.c[0]); - - ARListElem *b; - for (b = LocalDomainEnumRecords; b; b=b->next) - if (b->next == (ARListElem *)~0 || b->ar.resrec.name->c[0] > 63) - LogMemCorruption("LocalDomainEnumRecords: %p is garbage (%d)", b, b->ar.resrec.name->c[0]); - - for (d = AutoBrowseDomains; d; d=d->next) - if (d->next == (DNameListElem *)~0 || d->name.c[0] > 63) - LogMemCorruption("AutoBrowseDomains: %p is garbage (%d)", d, d->name.c[0]); - - for (d = AutoRegistrationDomains; d; d=d->next) - if (d->next == (DNameListElem *)~0 || d->name.c[0] > 63) - LogMemCorruption("AutoRegistrationDomains: %p is garbage (%d)", d, d->name.c[0]); - } -#endif // APPLE_OSX_mDNSResponder && MACOSX_MDNS_MALLOC_DEBUGGING - -mDNSlocal int send_msg(request_state *const req) - { - reply_state *const rep = req->replies; // Send the first waiting reply - ssize_t nwriten; - if (req->no_reply) return(t_complete); - - ConvertHeaderBytes(rep->mhdr); - nwriten = send(req->sd, (char *)&rep->mhdr + rep->nwriten, rep->totallen - rep->nwriten, 0); - ConvertHeaderBytes(rep->mhdr); - - if (nwriten < 0) - { - if (dnssd_errno == dnssd_EINTR || dnssd_errno == dnssd_EWOULDBLOCK) nwriten = 0; - else - { -#if !defined(PLATFORM_NO_EPIPE) - if (dnssd_errno == EPIPE) - return(req->ts = t_terminated); - else -#endif - { - LogMsg("send_msg ERROR: failed to write %d of %d bytes to fd %d errno %d (%s)", - rep->totallen - rep->nwriten, rep->totallen, req->sd, dnssd_errno, dnssd_strerror(dnssd_errno)); - return(t_error); - } - } - } - rep->nwriten += nwriten; - return (rep->nwriten == rep->totallen) ? t_complete : t_morecoming; - } - -mDNSexport mDNSs32 udsserver_idle(mDNSs32 nextevent) - { - mDNSs32 now = mDNS_TimeNow(&mDNSStorage); - request_state **req = &all_requests; - - while (*req) - { - request_state *const r = *req; - - if (r->terminate == resolve_termination_callback) - if (r->u.resolve.ReportTime && now - r->u.resolve.ReportTime >= 0) - { - r->u.resolve.ReportTime = 0; - LogMsgNoIdent("Client application bug: DNSServiceResolve(%##s) active for over two minutes. " - "This places considerable burden on the network.", r->u.resolve.qsrv.qname.c); - } - - // Note: Only primary req's have reply lists, not subordinate req's. - while (r->replies) // Send queued replies - { - transfer_state result; - if (r->replies->next) r->replies->rhdr->flags |= dnssd_htonl(kDNSServiceFlagsMoreComing); - result = send_msg(r); // Returns t_morecoming if buffer full because client is not reading - if (result == t_complete) - { - reply_state *fptr = r->replies; - r->replies = r->replies->next; - freeL("reply_state/udsserver_idle", fptr); - r->time_blocked = 0; // reset failure counter after successful send - r->unresponsiveness_reports = 0; - continue; - } - else if (result == t_terminated || result == t_error) - { - LogMsg("%3d: Could not write data to client because of error - aborting connection", r->sd); - LogClientInfo(&mDNSStorage, r); - abort_request(r); - } - break; - } - - if (r->replies) // If we failed to send everything, check our time_blocked timer - { - if (nextevent - now > mDNSPlatformOneSecond) nextevent = now + mDNSPlatformOneSecond; - - if (mDNSStorage.SleepState != SleepState_Awake) r->time_blocked = 0; - else if (!r->time_blocked) r->time_blocked = NonZeroTime(now); - else if (now - r->time_blocked >= 10 * mDNSPlatformOneSecond * (r->unresponsiveness_reports+1)) - { - int num = 0; - struct reply_state *x = r->replies; - while (x) { num++; x=x->next; } - LogMsg("%3d: Could not write data to client after %ld seconds, %d repl%s waiting", - r->sd, (now - r->time_blocked) / mDNSPlatformOneSecond, num, num == 1 ? "y" : "ies"); - if (++r->unresponsiveness_reports >= 60) - { - LogMsg("%3d: Client unresponsive; aborting connection", r->sd); - LogClientInfo(&mDNSStorage, r); - abort_request(r); - } - } - } - - if (!dnssd_SocketValid(r->sd)) // If this request is finished, unlink it from the list and free the memory - { - // Since we're already doing a list traversal, we unlink the request directly instead of using AbortUnlinkAndFree() - *req = r->next; - freeL("request_state/udsserver_idle", r); - } - else - req = &r->next; - } - return nextevent; - } - -struct CompileTimeAssertionChecks_uds_daemon - { - // Check our structures are reasonable sizes. Including overly-large buffers, or embedding - // other overly-large structures instead of having a pointer to them, can inadvertently - // cause structure sizes (and therefore memory usage) to balloon unreasonably. - char sizecheck_request_state [(sizeof(request_state) <= 1784) ? 1 : -1]; - char sizecheck_registered_record_entry[(sizeof(registered_record_entry) <= 60) ? 1 : -1]; - char sizecheck_service_instance [(sizeof(service_instance) <= 6552) ? 1 : -1]; - char sizecheck_browser_t [(sizeof(browser_t) <= 1050) ? 1 : -1]; - char sizecheck_reply_hdr [(sizeof(reply_hdr) <= 12) ? 1 : -1]; - char sizecheck_reply_state [(sizeof(reply_state) <= 64) ? 1 : -1]; - }; diff -Nru qtcreator-2.5.0/src/tools/mdnssd/uds_daemon.h qtcreator-2.5.2/src/tools/mdnssd/uds_daemon.h --- qtcreator-2.5.0/src/tools/mdnssd/uds_daemon.h 2012-05-09 12:13:19.000000000 +0000 +++ qtcreator-2.5.2/src/tools/mdnssd/uds_daemon.h 1970-01-01 00:00:00.000000000 +0000 @@ -1,80 +0,0 @@ -/* -*- Mode: C; tab-width: 4 -*- - * - * Copyright (c) 2002-2003 Apple Computer, Inc. All rights reserved. - * - * Licensed under the Apache License, Version 2.0 (the "License"); - * you may not use this file except in compliance with the License. - * You may obtain a copy of the License at - * - * http://www.apache.org/licenses/LICENSE-2.0 - * - * Unless required by applicable law or agreed to in writing, software - * distributed under the License is distributed on an "AS IS" BASIS, - * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. - * See the License for the specific language governing permissions and - * limitations under the License. - - File: uds_daemon.h - - Contains: Interfaces necessary to talk to uds_daemon.c. - - Version: 1.0 - - */ - -#include "mDNSEmbeddedAPI.h" -#include "dnssd_ipc.h" - -/* Client interface: */ - -#define SRS_PORT(S) mDNSVal16((S)->RR_SRV.resrec.rdata->u.srv.port) - -extern int udsserver_init(dnssd_sock_t skts[], mDNSu32 count); -extern mDNSs32 udsserver_idle(mDNSs32 nextevent); -extern void udsserver_info(mDNS *const m); // print out info about current state -extern void udsserver_handle_configchange(mDNS *const m); -extern int udsserver_exit(void); // should be called prior to app exit - -/* Routines that uds_daemon expects to link against: */ - -typedef void (*udsEventCallback)(int fd, short filter, void *context); -extern mStatus udsSupportAddFDToEventLoop(dnssd_sock_t fd, udsEventCallback callback, void *context, void **platform_data); -extern int udsSupportReadFD(dnssd_sock_t fd, char* buf, int len, int flags, void *platform_data); -extern mStatus udsSupportRemoveFDFromEventLoop(dnssd_sock_t fd, void *platform_data); // Note: This also CLOSES the file descriptor as well - -extern void RecordUpdatedNiceLabel(mDNS *const m, mDNSs32 delay); - -// Globals and functions defined in uds_daemon.c and also shared with the old "daemon.c" on OS X - -extern mDNS mDNSStorage; -extern DNameListElem *AutoRegistrationDomains; -extern DNameListElem *AutoBrowseDomains; - -extern mDNSs32 ChopSubTypes(char *regtype); -extern AuthRecord *AllocateSubTypes(mDNSs32 NumSubTypes, char *p); -extern int CountExistingRegistrations(domainname *srv, mDNSIPPort port); -extern void FreeExtraRR(mDNS *const m, AuthRecord *const rr, mStatus result); -extern int CountPeerRegistrations(mDNS *const m, ServiceRecordSet *const srs); - -#if APPLE_OSX_mDNSResponder -extern void machserver_automatic_browse_domain_changed(const domainname *d, mDNSBool add); -extern void machserver_automatic_registration_domain_changed(const domainname *d, mDNSBool add); -// External support -extern void mDNSInitPacketFilter(void); -extern void external_start_browsing_for_service(mDNS *const m, const domainname *const type, DNS_TypeValues qtype); -extern void external_stop_browsing_for_service(mDNS *const m, const domainname *const type, DNS_TypeValues qtype); -extern void external_start_advertising_service(const ResourceRecord *const resourceRecord); -extern void external_stop_advertising_service(const ResourceRecord *const resourceRecord); -extern void external_start_resolving_service(const domainname *const fqdn); -extern void external_stop_resolving_service(const domainname *const fqdn); -#else -#define external_start_browsing_for_service(A,B,C) (void)(A) -#define external_stop_browsing_for_service(A,B,C) (void)(A) -#define external_start_advertising_service(A) (void)(A) -#define external_stop_advertising_service(A) (void)(A) -#define external_start_resolving_service(A) (void)(A) -#define external_stop_resolving_service(A) (void)(A) -#endif // APPLE_OSX_mDNSResponder - -extern const char mDNSResponderVersionString_SCCS[]; -#define mDNSResponderVersionString (mDNSResponderVersionString_SCCS+5) diff -Nru qtcreator-2.5.0/src/tools/tools.pro qtcreator-2.5.2/src/tools/tools.pro --- qtcreator-2.5.0/src/tools/tools.pro 2012-05-09 12:13:19.000000000 +0000 +++ qtcreator-2.5.2/src/tools/tools.pro 2012-08-08 13:47:06.000000000 +0000 @@ -18,10 +18,6 @@ SUBDIRS += valgrindfake } -linux-* { - SUBDIRS += mdnssd -} - QT_BREAKPAD_ROOT_PATH = $$(QT_BREAKPAD_ROOT_PATH) !isEmpty(QT_BREAKPAD_ROOT_PATH) { SUBDIRS += qtcrashhandler diff -Nru qtcreator-2.5.0/tests/system/objects.map qtcreator-2.5.2/tests/system/objects.map --- qtcreator-2.5.0/tests/system/objects.map 2012-05-09 12:13:19.000000000 +0000 +++ qtcreator-2.5.2/tests/system/objects.map 2012-08-08 13:47:06.000000000 +0000 @@ -1,10 +1,7 @@ -:*Qt Creator.Build Project_Core::Internal::FancyToolButton {text='Build Project' type='Core::Internal::FancyToolButton' unnamed='1' visible='1' window=':Qt Creator_Core::Internal::MainWindow'} :*Qt Creator.Continue_Core::Internal::FancyToolButton {text='Continue' type='Core::Internal::FancyToolButton' unnamed='1' visible='1' window=':Qt Creator_Core::Internal::MainWindow'} :*Qt Creator.Interrupt_Core::Internal::FancyToolButton {text='Interrupt' type='Core::Internal::FancyToolButton' unnamed='1' visible='1' window=':Qt Creator_Core::Internal::MainWindow'} -:*Qt Creator.Run_Core::Internal::FancyToolButton {text='Run' type='Core::Internal::FancyToolButton' unnamed='1' visible='1' window=':Qt Creator_Core::Internal::MainWindow'} :*Qt Creator.Start Debugging_Core::Internal::FancyToolButton {text='Start Debugging' type='Core::Internal::FancyToolButton' unnamed='1' visible='1' window=':Qt Creator_Core::Internal::MainWindow'} :*Qt Creator.findEdit_Utils::FilterLineEdit {name='findEdit' type='Utils::FilterLineEdit' visible='1' window=':Qt Creator_Core::Internal::MainWindow'} -:*Qt Creator.qt_tabwidget_stackedwidget_QStackedWidget {name='qt_tabwidget_stackedwidget' type='QStackedWidget' visible='1' window=':Qt Creator_Core::Internal::MainWindow'} :*Qt Creator_Core::Internal::FancyToolButton {occurrence='3' type='Core::Internal::FancyToolButton' unnamed='1' visible='1' window=':Qt Creator_Core::Internal::MainWindow'} :*Qt Creator_Utils::FilterLineEdit {type='Utils::FilterLineEdit' unnamed='1' visible='1' window=':Qt Creator_Core::Internal::MainWindow'} :*Qt Creator_Utils::IconButton {occurrence='4' type='Utils::IconButton' unnamed='1' visible='1' window=':Qt Creator_Core::Internal::MainWindow'} @@ -32,6 +29,8 @@ :Options.qt_tabwidget_tabbar_QTabBar {name='qt_tabwidget_tabbar' type='QTabBar' visible='1' window=':Options_Core::Internal::SettingsDialog'} :Options_Core::Internal::SettingsDialog {type='Core::Internal::SettingsDialog' unnamed='1' visible='1' windowTitle~='(Options|Preferences)'} :Options_QListView {type='QListView' unnamed='1' visible='1' window=':Options_Core::Internal::SettingsDialog'} +:QML Debugging.No_QPushButton {text='No' type='QPushButton' unnamed='1' visible='1' window=':QML Debugging_QMessageBox'} +:QML Debugging_QMessageBox {type='QMessageBox' unnamed='1' visible='1' windowTitle='QML Debugging'} :Qt Creator.Create Build Configurations:_QComboBox {leftWidget=':Qt Creator.Create Build Configurations:_QLabel' type='QComboBox' unnamed='1' visible='1' window=':Qt Creator_Core::Internal::MainWindow'} :Qt Creator.Create Build Configurations:_QLabel {text='Create build configurations:' type='QLabel' unnamed='1' visible='1' window=':Qt Creator_Core::Internal::MainWindow'} :Qt Creator.DebugModeWidget_QSplitter {name='DebugModeWidget' type='QSplitter' visible='1' window=':Qt Creator_Core::Internal::MainWindow'} @@ -45,15 +44,12 @@ :Qt Creator_CppEditor::Internal::CPPEditorWidget {type='CppEditor::Internal::CPPEditorWidget' unnamed='1' visible='1' window=':Qt Creator_Core::Internal::MainWindow'} :Qt Creator_QTableView {type='QTableView' unnamed='1' visible='1' window=':Qt Creator_Core::Internal::MainWindow'} :Qt Creator_SearchResult_Core::Internal::OutputPaneToggleButton {occurrence='2' type='Core::Internal::OutputPaneToggleButton' unnamed='1' visible='1' window=':Qt Creator_Core::Internal::MainWindow'} -:Qt Creator_Utils::IconButton {occurrence='2' type='Utils::IconButton' unnamed='1' visible='1' window=':Qt Creator_Core::Internal::MainWindow'} :Qt Creator_Utils::NavigationTreeView {type='Utils::NavigationTreeView' unnamed='1' visible='1' window=':Qt Creator_Core::Internal::MainWindow'} :Qt Gui Application.Form file:_QLabel {name='formLabel' text='Form file:' type='QLabel' visible='1' window=':Qt Gui Application_Qt4ProjectManager::Internal::GuiAppWizardDialog'} :Qt Gui Application.Header file:_QLabel {name='headerLabel' text='Header file:' type='QLabel' visible='1' window=':Qt Gui Application_Qt4ProjectManager::Internal::GuiAppWizardDialog'} :Qt Gui Application.Source file:_QLabel {name='sourceLabel' text='Source file:' type='QLabel' visible='1' window=':Qt Gui Application_Qt4ProjectManager::Internal::GuiAppWizardDialog'} -:Qt Gui Application.frame_QFrame {name='frame' type='QFrame' visible='1' window=':Qt Gui Application_Qt4ProjectManager::Internal::GuiAppWizardDialog'} :Qt Gui Application.scrollArea_QScrollArea {name='scrollArea' type='QScrollArea' visible='1'} :Qt Gui Application_Qt4ProjectManager::Internal::GuiAppWizardDialog {type='Qt4ProjectManager::Internal::GuiAppWizardDialog' unnamed='1' visible='1' windowTitle='Qt Gui Application'} -:QtCreator.MenuBar_ProjectExplorer::Internal::MiniProjectTargetSelector {type='ProjectExplorer::Internal::MiniProjectTargetSelector'} :QtSupport__Internal__QtVersionManager.QLabel {container=':qt_tabwidget_stackedwidget.QtSupport__Internal__QtVersionManager_QtSupport::Internal::QtOptionsPageWidget' name='errorLabel' type='QLabel' visible='1'} :QtSupport__Internal__QtVersionManager.qtdirList_QTreeWidget {container=':qt_tabwidget_stackedwidget.QtSupport__Internal__QtVersionManager_QtSupport::Internal::QtOptionsPageWidget' name='qtdirList' type='QTreeWidget' visible='1'} :Symbol Server_Utils::CheckableMessageBox {type='Utils::CheckableMessageBox' unnamed='1' visible='1' windowTitle='Symbol Server'} @@ -61,19 +57,19 @@ :formFileLineEdit_Utils::FileNameValidatingLineEdit {buddy=':Qt Gui Application.Form file:_QLabel' name='formFileLineEdit' type='Utils::FileNameValidatingLineEdit' visible='1'} :frame.templateDescription_QTextBrowser {container=':New.frame_QFrame' name='templateDescription' type='QTextBrowser' visible='1'} :headerFileLineEdit_Utils::FileNameValidatingLineEdit {buddy=':Qt Gui Application.Header file:_QLabel' name='headerFileLineEdit' type='Utils::FileNameValidatingLineEdit' visible='1'} -:projects.projects.pro_QModelIndex {column='0' container=':projects_QModelIndex' text='projects.pro' type='QModelIndex'} -:projects_QModelIndex {column='0' container=':Qt Creator_Utils::NavigationTreeView' text='projects' type='QModelIndex'} :qt_tabwidget_stackedwidget.QtSupport__Internal__QtVersionManager_QtSupport::Internal::QtOptionsPageWidget {container=':Options.qt_tabwidget_stackedwidget_QStackedWidget' name='QtSupport__Internal__QtVersionManager' type='QtSupport::Internal::QtOptionsPageWidget' visible='1'} :scrollArea.Create Build Configurations:_QComboBox_2 {container=':Qt Gui Application.scrollArea_QScrollArea' leftWidget=':scrollArea.Create Build Configurations:_QLabel_2' type='QComboBox' unnamed='1' visible='1'} :scrollArea.Create Build Configurations:_QLabel_2 {container=':Qt Gui Application.scrollArea_QScrollArea' text='Create build configurations:' type='QLabel' unnamed='1' visible='1'} :scrollArea.Details_Utils::DetailsButton {container=':Qt Creator.scrollArea_QScrollArea' text='Details' type='Utils::DetailsButton' unnamed='1' visible='1'} :scrollArea.Edit build configuration:_QComboBox {container=':Qt Creator.scrollArea_QScrollArea' leftWidget=':scrollArea.Edit build configuration:_QLabel' type='QComboBox' unnamed='1' visible='1'} :scrollArea.Edit build configuration:_QLabel {container=':Qt Creator.scrollArea_QScrollArea' text='Edit build configuration:' type='QLabel' unnamed='1' visible='1'} +:scrollArea.Library not available_QLabel {container=':Qt Creator.scrollArea_QScrollArea' name='qmlDebuggingWarningText' text?='Library not available*' type='QLabel' visible='1'} :scrollArea.Qt 4 for Desktop - (Qt SDK) debug_QCheckBox {container=':Qt Gui Application.scrollArea_QScrollArea' text?='*Qt 4.* for *(Qt SDK) debug' type='QCheckBox' unnamed='1' visible='1'} :scrollArea.Qt 4 for Desktop - (Qt SDK) release_QCheckBox {container=':Qt Gui Application.scrollArea_QScrollArea' text?='*Qt 4.* for *(Qt SDK) release' type='QCheckBox' unnamed='1' visible='1'} :scrollArea.Qt Version:_QComboBox {aboveWidget=':scrollArea.Use Shadow Building_QCheckBox' container=':Qt Gui Application.scrollArea_QScrollArea' leftWidget=':scrollArea.Qt Version:_QLabel' type='QComboBox' unnamed='1' visible='1'} :scrollArea.Qt Version:_QLabel {container=':Qt Gui Application.scrollArea_QScrollArea' text='Qt Version:' type='QLabel' unnamed='1' visible='1'} :scrollArea.Use Shadow Building_QCheckBox {container=':Qt Gui Application.scrollArea_QScrollArea' text='Shadow build' type='QCheckBox' unnamed='1' visible='1'} +:scrollArea.qmlDebuggingLibraryCheckBox_QCheckBox {container=':Qt Gui Application.scrollArea_QScrollArea' name='qmlDebuggingLibraryCheckBox' type='QCheckBox' visible='1'} :scrollArea.qtVersionComboBox_QComboBox {container=':Qt Creator.scrollArea_QScrollArea' name='qtVersionComboBox' type='QComboBox' visible='1'} :scrollArea_QTableView {container=':Qt Creator.scrollArea_QScrollArea' type='QTableView' unnamed='1' visible='1'} :sourceFileLineEdit_Utils::FileNameValidatingLineEdit {buddy=':Qt Gui Application.Source file:_QLabel' name='sourceFileLineEdit' type='Utils::FileNameValidatingLineEdit' visible='1'} diff -Nru qtcreator-2.5.0/tests/system/settings/unix/Nokia/qtversion.xml qtcreator-2.5.2/tests/system/settings/unix/Nokia/qtversion.xml --- qtcreator-2.5.0/tests/system/settings/unix/Nokia/qtversion.xml 2012-05-09 12:13:19.000000000 +0000 +++ qtcreator-2.5.2/tests/system/settings/unix/Nokia/qtversion.xml 2012-08-08 13:47:06.000000000 +0000 @@ -5,7 +5,7 @@ QtVersion.0 - 4 + 3 Qt for Fremantle PR1.3 Devices (Qt SDK) ~/QtSDK/Maemo/4.6.2/targets/fremantle-pr13/bin/qmake Qt4ProjectManager.QtVersion.Maemo @@ -15,7 +15,7 @@ QtVersion.1 - 5 + 4 Desktop Qt 4.7.4 for GCC (Qt SDK) ~/QtSDK/Desktop/Qt/474/gcc/bin/qmake Qt4ProjectManager.QtVersion.Desktop @@ -25,7 +25,7 @@ QtVersion.2 - 6 + 5 Simulator Qt for GCC (Qt SDK) ~/QtSDK/Simulator/Qt/gcc/bin/qmake Qt4ProjectManager.QtVersion.Simulator @@ -35,16 +35,6 @@ QtVersion.3 - 8 - 4.7.0 - /usr/bin/qmake - Qt4ProjectManager.QtVersion.Desktop - false - - - - QtVersion.4 - 1 Harmattan Target (Qt SDK) ~/QtSDK/Madde/targets/harmattan_10.2011.34-1/bin/qmake @@ -54,7 +44,7 @@ - QtVersion.5 + QtVersion.4 2 Desktop Qt 4.8 for GCC (Qt SDK) @@ -65,7 +55,7 @@ QtVersion.Count - 6 + 5 Version diff -Nru qtcreator-2.5.0/tests/system/shared/build_utils.py qtcreator-2.5.2/tests/system/shared/build_utils.py --- qtcreator-2.5.0/tests/system/shared/build_utils.py 2012-05-09 12:13:19.000000000 +0000 +++ qtcreator-2.5.2/tests/system/shared/build_utils.py 2012-08-08 13:47:06.000000000 +0000 @@ -164,7 +164,8 @@ "sourceFilesRefreshed(QStringList)") switchViewTo(ViewConstants.EDIT) -def verifyBuildConfig(targetCount, currentTarget, shouldBeDebug=False, enableShadowBuild=False): +# This will not trigger a rebuild. If needed, caller has to do this. +def verifyBuildConfig(targetCount, currentTarget, shouldBeDebug=False, enableShadowBuild=False, enableQmlDebug=False): switchViewTo(ViewConstants.PROJECTS) switchToBuildOrRunSettingsFor(targetCount, currentTarget, ProjectSettings.BUILD) detailsButton = waitForObject(":scrollArea.Details_Utils::DetailsButton") @@ -177,6 +178,18 @@ else: test.compare(buildCfCombo.currentText, 'Release', "Verifying whether it's a release build") try: + libLabel = waitForObject(":scrollArea.Library not available_QLabel", 2000) + mouseClick(libLabel, libLabel.width - 5, 5, 0, Qt.LeftButton) + except: + pass + # Since waitForObject waits for the object to be enabled, + # it will wait here until compilation of the debug libraries has finished. + qmlDebugCheckbox = waitForObject(":scrollArea.qmlDebuggingLibraryCheckBox_QCheckBox", 150000) + if qmlDebugCheckbox.checked != enableQmlDebug: + clickButton(qmlDebugCheckbox) + # Don't rebuild now + clickButton(waitForObject(":QML Debugging.No_QPushButton", 5000)) + try: problemFound = waitForObject("{container=':Qt Creator.scrollArea_QScrollArea' type='QLabel' " "name='problemLabel' visible='1'}", 1000) if problemFound: diff -Nru qtcreator-2.5.0/tests/system/shared/debugger.py qtcreator-2.5.2/tests/system/shared/debugger.py --- qtcreator-2.5.0/tests/system/shared/debugger.py 2012-05-09 12:13:19.000000000 +0000 +++ qtcreator-2.5.2/tests/system/shared/debugger.py 2012-08-08 13:47:06.000000000 +0000 @@ -35,7 +35,7 @@ return debuggerLog # function to set breakpoints for the current project -# on the given file,line pairs inside the given dict +# on the given file,line pairs inside the given list of dicts # the lines are treated as regular expression def setBreakpointsForCurrentProject(filesAndLines): # internal helper for setBreakpointsForCurrentProject @@ -52,18 +52,19 @@ switchViewTo(ViewConstants.DEBUG) removeOldBreakpoints() - if not filesAndLines or not isinstance(filesAndLines, dict): - test.fatal("This function only takes a non-empty dict.") + if not filesAndLines or not isinstance(filesAndLines, (list,tuple)): + test.fatal("This function only takes a non-empty list/tuple holding dicts.") return False navTree = waitForObject("{type='Utils::NavigationTreeView' unnamed='1' visible='1' " "window=':Qt Creator_Core::Internal::MainWindow'}", 20000) - for curFile,curLine in filesAndLines.iteritems(): - fName = __doubleClickFile__(navTree, curFile) - editor = getEditorForFileSuffix(curFile) - if not placeCursorToLine(editor, curLine, True): - return False - invokeMenuItem("Debug", "Toggle Breakpoint") - test.log('Set breakpoint in %s' % fName, curLine) + for current in filesAndLines: + for curFile,curLine in current.iteritems(): + fName = __doubleClickFile__(navTree, curFile) + editor = getEditorForFileSuffix(curFile) + if not placeCursorToLine(editor, curLine, True): + return False + invokeMenuItem("Debug", "Toggle Breakpoint") + test.log('Set breakpoint in %s' % fName, curLine) try: breakPointTreeView = waitForObject("{type='Debugger::Internal::BreakWindow' visible='1' " "windowTitle='Breakpoints' name='Debugger.Docks.Break'}") diff -Nru qtcreator-2.5.0/tests/system/shared/editor_utils.py qtcreator-2.5.2/tests/system/shared/editor_utils.py --- qtcreator-2.5.0/tests/system/shared/editor_utils.py 2012-05-09 12:13:19.000000000 +0000 +++ qtcreator-2.5.2/tests/system/shared/editor_utils.py 2012-08-08 13:47:06.000000000 +0000 @@ -5,7 +5,7 @@ # and goes to the end of the line # line can be a regex - but if so, remember to set isRegex to True # the function returns True if this went fine, False on error -def placeCursorToLine(editor,line,isRegex=False): +def placeCursorToLine(editor, line, isRegex=False): cursor = editor.textCursor() oldPosition = 0 cursor.setPosition(oldPosition) @@ -15,45 +15,21 @@ regex = re.compile(line) while not found: currentLine = str(lineUnderCursor(editor)).strip() - if isRegex: - if regex.match(currentLine): - found = True - else: - moveTextCursor(editor, QTextCursor.Down, QTextCursor.MoveAnchor) - if oldPosition==editor.textCursor().position(): - break - oldPosition = editor.textCursor().position() - else: - if currentLine==line: - found = True - else: - moveTextCursor(editor, QTextCursor.Down, QTextCursor.MoveAnchor) - if oldPosition==editor.textCursor().position(): - break - oldPosition = editor.textCursor().position() + found = isRegex and regex.match(currentLine) or not isRegex and currentLine == line + if not found: + type(editor, "") + newPosition = editor.textCursor().position() + if oldPosition == newPosition: + break + oldPosition = newPosition if not found: test.fatal("Couldn't find line matching\n\n%s\n\nLeaving test..." % line) return False - cursor=editor.textCursor() + cursor = editor.textCursor() cursor.movePosition(QTextCursor.EndOfLine, QTextCursor.MoveAnchor) editor.setTextCursor(cursor) return True -# this function moves the text cursor of an editor by using Qt internal functions -# this is more reliable (especially on Mac) than the type() approach -# param editor an editor object -# param moveOperation a value of enum MoveOperation (of QTextCursor) -# param moveAnchor a value of enum MoveMode (of QTextCursor) -# param n how often repeat the move operation? -def moveTextCursor(editor, moveOperation, moveAnchor, n=1): - if not editor or isinstance(editor, (str,unicode)): - test.fatal("Either got a NoneType or a string instead of an editor object") - return False - textCursor = editor.textCursor() - result = textCursor.movePosition(moveOperation, moveAnchor, n) - editor.setTextCursor(textCursor) - return result - # this function returns True if a QMenu is # popped up above the given editor # param editor is the editor where the menu should appear @@ -231,6 +207,7 @@ "tcc", "tpp", "t++", "c", "cu", "m", "mm", "hh", "hxx", "h++", "hpp", "hp"] qmlEditorSuffixes = ["qml", "qmlproject", "js", "qs", "qtt"] proEditorSuffixes = ["pro", "pri", "prf"] + glslEditorSuffixes= ["frag", "vert", "fsh", "vsh", "glsl", "shader", "gsh"] suffix = __getFileSuffix__(curFile) if suffix in cppEditorSuffixes: editor = waitForObject("{type='CppEditor::Internal::CPPEditorWidget' unnamed='1' " @@ -241,6 +218,9 @@ elif suffix in proEditorSuffixes: editor = waitForObject("{type='Qt4ProjectManager::Internal::ProFileEditorWidget' unnamed='1' " "visible='1' window=':Qt Creator_Core::Internal::MainWindow'}") + elif suffix in glslEditorSuffixes: + editor = waitForObject("{type='GLSLEditor::GLSLTextEditorWidget' unnamed='1' " + "visible='1' window=':Qt Creator_Core::Internal::MainWindow'}") else: test.log("Trying PlainTextEditor (file suffix: %s)" % suffix) try: @@ -259,3 +239,52 @@ return None else: return suffix[1] + +def maskSpecialCharsForSearchResult(filename): + filename = filename.replace("_", "\\_").replace(".","\\.") + return filename + +def validateSearchResult(expectedCount): + searchResult = waitForObject(":Qt Creator_SearchResult_Core::Internal::OutputPaneToggleButton") + ensureChecked(searchResult) + resultTreeView = waitForObject("{type='Find::Internal::SearchResultTreeView' unnamed='1' " + "visible='1' window=':Qt Creator_Core::Internal::MainWindow'}") + counterLabel = waitForObject("{type='QLabel' unnamed='1' visible='1' text?='*matches found.' " + "window=':Qt Creator_Core::Internal::MainWindow'}") + matches = cast((str(counterLabel.text)).split(" ", 1)[0], "int") + test.compare(matches, expectedCount, "Verified match count.") + model = resultTreeView.model() + for row in range(model.rowCount()): + index = model.index(row, 0) + itemText = str(model.data(index).toString()) + doubleClickItem(resultTreeView, maskSpecialCharsForSearchResult(itemText), 5, 5, 0, Qt.LeftButton) + test.log("%d occurrences in %s" % (model.rowCount(index), itemText)) + for chRow in range(model.rowCount(index)): + chIndex = model.index(chRow, 0, index) + resultTreeView.scrollTo(chIndex) + text = str(chIndex.data()) + rect = resultTreeView.visualRect(chIndex) + doubleClick(resultTreeView, rect.x+5, rect.y+5, 0, Qt.LeftButton) + editor = getEditorForFileSuffix(itemText) + waitFor("lineUnderCursor(editor) == text", 2000) + test.compare(lineUnderCursor(editor), text) + +# this function invokes context menu and command from it +def invokeContextMenuItem(editorArea, command1, command2 = None): + ctxtMenu = openContextMenuOnTextCursorPosition(editorArea) + activateItem(waitForObjectItem(objectMap.realName(ctxtMenu), command1, 1000)) + if command2: + activateItem(waitForObjectItem(objectMap.realName(ctxtMenu), command2, 1000)) + +# this function invokes the "Find Usages" item from context menu +# param editor an editor object +# param line a line in editor (content of the line as a string) +# param typeOperation a key to type +# param n how often repeat the type operation? +def invokeFindUsage(editor, line, typeOperation, n=1): + if not placeCursorToLine(editor, line, True): + return False + for i in range(n): + type(editor, typeOperation) + invokeContextMenuItem(editor, "Find Usages") + return True diff -Nru qtcreator-2.5.0/tests/system/shared/project.py qtcreator-2.5.2/tests/system/shared/project.py --- qtcreator-2.5.0/tests/system/shared/project.py 2012-05-09 12:13:19.000000000 +0000 +++ qtcreator-2.5.2/tests/system/shared/project.py 2012-08-08 13:47:06.000000000 +0000 @@ -1,3 +1,4 @@ +import __builtin__ import re processStarted = False @@ -26,6 +27,10 @@ selectFromCombo(waitForObject(":Qt Creator.Create Build Configurations:_QComboBox", 180000), "For Each Qt Version One Debug And One Release") __chooseTargets__(targets) + if targets & QtQuickConstants.Targets.DESKTOP: + ensureChecked("{text='Shadow build' type='QCheckBox' unnamed='1' visible='1' " + "window=':Qt Creator_Core::Internal::MainWindow'}") + configureButton = waitForObject("{text='Configure Project' type='QPushButton' unnamed='1' visible='1'" "window=':Qt Creator_Core::Internal::MainWindow'}", 20000) clickButton(configureButton) @@ -153,7 +158,9 @@ # created for this version. If it is None, all Qt versions will be used # param checks turns tests in the function on if set to True def createProject_Qt_GUI(path, projectName, qtVersion = None, checks = True): - available = __createProjectSelectType__(" Applications", "Qt Gui Application") + template = "Qt Gui Application" + available = __createProjectSelectType__(" Applications", template) + JIRA.performWorkaroundIfStillOpen(6994, JIRA.Bug.CREATOR, template, available) __createProjectSetNameAndPath__(path, projectName, checks) __selectQtVersionDesktop__(qtVersion, checks, available) @@ -271,7 +278,8 @@ available = availableTargets else: # following targets depend on the build environment - added for further/later tests - available = [QtQuickConstants.Targets.MAEMO5, QtQuickConstants.Targets.EMBEDDED_LINUX, + available = [QtQuickConstants.Targets.DESKTOP, + QtQuickConstants.Targets.MAEMO5, QtQuickConstants.Targets.EMBEDDED_LINUX, QtQuickConstants.Targets.SIMULATOR, QtQuickConstants.Targets.HARMATTAN] if platform.system() in ('Windows', 'Microsoft'): available += [QtQuickConstants.Targets.SYMBIAN] @@ -460,3 +468,40 @@ templateDir = os.path.abspath(tempDir() + "/template") shutil.copytree(sourceExample, templateDir) return templateDir + +def __iterateChildren__(model, parent, nestingLevel=0): + children = [] + for currentIndex in [model.index(row, 0, parent) for row in range(model.rowCount(parent))]: + children.append([str(currentIndex.text), nestingLevel]) + if model.hasChildren(currentIndex): + children.extend(__iterateChildren__(model, currentIndex, nestingLevel + 1)) + return children + +# This will write the data to a file which can then be used for comparing +def __writeProjectTreeFile__(projectTree, filename): + f = open(filename, "w+") + f.write('"text"\t"nestinglevel"\n') + for elem in projectTree: + f.write('"%s"\t"%s"\n' % (elem[0], elem[1])) + f.close() + +def __getTestData__(record): + return [testData.field(record, "text"), + __builtin__.int(testData.field(record, "nestinglevel"))] + +def compareProjectTree(rootObject, dataset): + root = waitForObject(rootObject) + tree = __iterateChildren__(root.model(), root) + + # __writeProjectTreeFile__(tree, dataset) + + for i, current in enumerate(map(__getTestData__, testData.dataset(dataset))): + try: + # Just removing everything up to the found item + # Writing a pass would result in truly massive logs + tree = tree[tree.index(current) + 1:] + except ValueError: + test.fail('Could not find "%s" with nesting level %s' % tuple(current), + 'Line %s in dataset' % str(i + 1)) + return + test.passes("No errors found in project tree") diff -Nru qtcreator-2.5.0/tests/system/shared/suites_qtta.py qtcreator-2.5.2/tests/system/shared/suites_qtta.py --- qtcreator-2.5.0/tests/system/shared/suites_qtta.py 2012-05-09 12:13:19.000000000 +0000 +++ qtcreator-2.5.2/tests/system/shared/suites_qtta.py 2012-08-08 13:47:06.000000000 +0000 @@ -35,8 +35,11 @@ return False # wait and verify if object exists/not exists -def checkIfObjectExists(name, shouldExist = True, timeout = 3000): - return waitFor("object.exists(name) == shouldExist", timeout) +def checkIfObjectExists(name, shouldExist = True, timeout = 3000, verboseOnFail = False): + result = waitFor("object.exists(name) == shouldExist", timeout) + if verboseOnFail and not result: + test.log("checkIfObjectExists() failed for '%s'" % name) + return result # change autocomplete options to manual def changeAutocompleteToManual(): @@ -47,4 +50,3 @@ selectFromCombo(":Behavior.completionTrigger_QComboBox", "Manually") verifyEnabled(":Options.OK_QPushButton") clickButton(waitForObject(":Options.OK_QPushButton")) - diff -Nru qtcreator-2.5.0/tests/system/shared/utils.py qtcreator-2.5.2/tests/system/shared/utils.py --- qtcreator-2.5.0/tests/system/shared/utils.py 2012-05-09 12:13:19.000000000 +0000 +++ qtcreator-2.5.2/tests/system/shared/utils.py 2012-08-08 13:47:06.000000000 +0000 @@ -31,7 +31,7 @@ else: state = "unchecked" test.log("New state for QCheckBox: %s" % state) - test.verify(object.checked == shouldBeChecked) + test.verify(waitFor("object.checked == shouldBeChecked", 1000)) return object # verify that an object is in an expected enable state. Returns the object. @@ -198,8 +198,10 @@ try: output = waitForObject("{type='Core::OutputWindow' visible='1' windowTitle='Application Output Window'}", 20000) test.log("Application Output:\n%s" % output.plainText) + return str(output.plainText) except: test.fail("Could not find any Application Output - did the project run?") + return None # get the output from a given cmdline call def getOutputFromCmdline(cmdline): @@ -375,7 +377,8 @@ # param keepOptionsOpen set to True if the Options dialog should stay open when # leaving this function # param additionalFunction pass a function or name of a defined function to execute -# for each item on the list of Qt versions +# for each correctly configured item on the list of Qt versions +# (Qt versions having no assigned toolchain, failing qmake,... will be skipped) # this function must take at least 2 parameters - the first is the target name # and the second the version of the current selected Qt version item # param argsForAdditionalFunc you can specify as much parameters as you want to pass @@ -411,17 +414,19 @@ target = matches.group("target").strip() version = matches.group("version").strip() result.append({target:version}) - if additionalFunction: - try: - if isinstance(additionalFunction, (str, unicode)): - currResult = globals()[additionalFunction](target, version, *argsForAdditionalFunc) - else: - currResult = additionalFunction(target, version, *argsForAdditionalFunc) - except: - currResult = None - test.fatal("Function to additionally execute on Options Dialog could not be found or " - "an exception occured while executing it.") - additionalResult.append(currResult) + if additionalFunction: + try: + if isinstance(additionalFunction, (str, unicode)): + currResult = globals()[additionalFunction](target, version, *argsForAdditionalFunc) + else: + currResult = additionalFunction(target, version, *argsForAdditionalFunc) + except: + import sys + t,v,tb = sys.exc_info() + currResult = None + test.fatal("Function to additionally execute on Options Dialog could not be found or " + "an exception occured while executing it.", "%s(%s)" % (str(t), str(v))) + additionalResult.append(currResult) if not keepOptionsOpen: clickButton(waitForObject(":Options.Cancel_QPushButton")) if additionalFunction: diff -Nru qtcreator-2.5.0/tests/system/shared/workarounds.py qtcreator-2.5.2/tests/system/shared/workarounds.py --- qtcreator-2.5.0/tests/system/shared/workarounds.py 2012-05-09 12:13:19.000000000 +0000 +++ qtcreator-2.5.2/tests/system/shared/workarounds.py 2012-08-08 13:47:06.000000000 +0000 @@ -187,7 +187,10 @@ def _workaroundCreator6994_(self, *args): if args[0] in ('Mobile Qt Application', 'Qt Gui Application', 'Qt Custom Designer Widget'): - args[1].remove('Harmattan') + if QtQuickConstants.Targets.HARMATTAN in args[1]: + args[1].remove(QtQuickConstants.Targets.HARMATTAN) + else: + args[1].remove(QtQuickConstants.getStringForTarget(QtQuickConstants.Targets.HARMATTAN)) test.xverify(False, "Removed Harmattan from expected targets.") def _workaroundCreator6853_(self, *args): diff -Nru qtcreator-2.5.0/tests/system/suite_APTW/envvars qtcreator-2.5.2/tests/system/suite_APTW/envvars --- qtcreator-2.5.0/tests/system/suite_APTW/envvars 1970-01-01 00:00:00.000000000 +0000 +++ qtcreator-2.5.2/tests/system/suite_APTW/envvars 2012-08-08 13:47:06.000000000 +0000 @@ -0,0 +1 @@ +QT_PLATFORM_PLUGIN=nonesuch diff -Nru qtcreator-2.5.0/tests/system/suite_APTW/objects.map qtcreator-2.5.2/tests/system/suite_APTW/objects.map --- qtcreator-2.5.0/tests/system/suite_APTW/objects.map 1970-01-01 00:00:00.000000000 +0000 +++ qtcreator-2.5.2/tests/system/suite_APTW/objects.map 2012-08-08 13:47:06.000000000 +0000 @@ -0,0 +1,24 @@ +:*Qt Creator.Clear_QToolButton {text='Clear' type='QToolButton' unnamed='1' visible='1' window=':Qt Creator_Core::Internal::MainWindow'} +:Next_QPushButton {text~='(Next.*|Continue)' type='QPushButton' visible='1'} +:Options.OK_QPushButton {text='OK' type='QPushButton' unnamed='1' visible='1' window=':Options_Core::Internal::SettingsDialog'} +:Qt Creator.Compile Output_Core::OutputWindow {type='Core::OutputWindow' unnamed='1' visible='1' window=':Qt Creator_Core::Internal::MainWindow' windowTitle='Compile Output'} +:Qt Creator.QtCreator.MenuBar_QMenuBar {name='QtCreator.MenuBar' type='QMenuBar' visible='1' window=':Qt Creator_Core::Internal::MainWindow'} +:Qt Creator.ReRun_QToolButton {toolTip='Re-run this run-configuration' type='QToolButton' unnamed='1' visible='1' window=':Qt Creator_Core::Internal::MainWindow'} +:Qt Creator.Stop_QToolButton {text='Stop' type='QToolButton' unnamed='1' visible='1' window=':Qt Creator_Core::Internal::MainWindow'} +:Qt Creator.scrollArea_QScrollArea {name='scrollArea' type='QScrollArea' visible='1' window=':Qt Creator_Core::Internal::MainWindow'} +:Qt Creator_AppOutput_Core::Internal::OutputPaneToggleButton {occurrence='3' type='Core::Internal::OutputPaneToggleButton' unnamed='1' visible='1' window=':Qt Creator_Core::Internal::MainWindow'} +:Qt Creator_Core::Internal::MainWindow {type='Core::Internal::MainWindow' unnamed='1' visible='1' windowTitle?='*Qt Creator'} +:Qt Gui Application.Form file:_QLabel {name='formLabel' text='Form file:' type='QLabel' visible='1' window=':Qt Gui Application_Qt4ProjectManager::Internal::GuiAppWizardDialog'} +:Qt Gui Application.Header file:_QLabel {name='headerLabel' text='Header file:' type='QLabel' visible='1' window=':Qt Gui Application_Qt4ProjectManager::Internal::GuiAppWizardDialog'} +:Qt Gui Application.Source file:_QLabel {name='sourceLabel' text='Source file:' type='QLabel' visible='1' window=':Qt Gui Application_Qt4ProjectManager::Internal::GuiAppWizardDialog'} +:Qt Gui Application.scrollArea_QScrollArea {name='scrollArea' type='QScrollArea' visible='1'} +:Qt Gui Application_Qt4ProjectManager::Internal::GuiAppWizardDialog {type='Qt4ProjectManager::Internal::GuiAppWizardDialog' unnamed='1' visible='1' windowTitle='Qt Gui Application'} +:addToVersionControlComboBox_QComboBox {name='addToVersionControlComboBox' type='QComboBox' visible='1'} +:formFileLineEdit_Utils::FileNameValidatingLineEdit {buddy=':Qt Gui Application.Form file:_QLabel' name='formFileLineEdit' type='Utils::FileNameValidatingLineEdit' visible='1'} +:headerFileLineEdit_Utils::FileNameValidatingLineEdit {buddy=':Qt Gui Application.Header file:_QLabel' name='headerFileLineEdit' type='Utils::FileNameValidatingLineEdit' visible='1'} +:scrollArea.Create Build Configurations:_QComboBox_2 {container=':Qt Gui Application.scrollArea_QScrollArea' leftWidget=':scrollArea.Create Build Configurations:_QLabel_2' type='QComboBox' unnamed='1' visible='1'} +:scrollArea.Create Build Configurations:_QLabel_2 {container=':Qt Gui Application.scrollArea_QScrollArea' text='Create build configurations:' type='QLabel' unnamed='1' visible='1'} +:scrollArea.Edit build configuration:_QComboBox {container=':Qt Creator.scrollArea_QScrollArea' leftWidget=':scrollArea.Edit build configuration:_QLabel' type='QComboBox' unnamed='1' visible='1'} +:scrollArea.Edit build configuration:_QLabel {container=':Qt Creator.scrollArea_QScrollArea' text='Edit build configuration:' type='QLabel' unnamed='1' visible='1'} +:scrollArea.Use Shadow Building_QCheckBox {container=':Qt Gui Application.scrollArea_QScrollArea' text='Shadow build' type='QCheckBox' unnamed='1' visible='1'} +:sourceFileLineEdit_Utils::FileNameValidatingLineEdit {buddy=':Qt Gui Application.Source file:_QLabel' name='sourceFileLineEdit' type='Utils::FileNameValidatingLineEdit' visible='1'} diff -Nru qtcreator-2.5.0/tests/system/suite_APTW/shared/aptw.py qtcreator-2.5.2/tests/system/suite_APTW/shared/aptw.py --- qtcreator-2.5.0/tests/system/suite_APTW/shared/aptw.py 1970-01-01 00:00:00.000000000 +0000 +++ qtcreator-2.5.2/tests/system/suite_APTW/shared/aptw.py 2012-08-08 13:47:06.000000000 +0000 @@ -0,0 +1,29 @@ +# shared script for APTW suite +# helping to run and close app +# verification + +# verify if building and running of project was successful +def verifyBuildAndRun(): + # check compile output if build successful + checkCompile() + # check application output log + appOutput = logApplicationOutput() + if appOutput: + test.verify(re.search(".*([Pp]rogram).*(unexpectedly).*([Ff]inished).*", str(appOutput)) and + re.search('[Ss]tarting.*', str(appOutput)), + "Verifying if built app started and closed successfully.") + +# pick version 4.7.4 and then run project for debug and release +def pickVersion474runVerify(): + availableConfigs = iterateBuildConfigs(1, 0, ".*4.7.4.*") + if not availableConfigs: + test.fatal("Haven't found needed Qt version (Qt 4.7.4), quitting") + invokeMenuItem("File", "Save All") + invokeMenuItem("File", "Exit") + # select debug configuration + for config in availableConfigs: + selectBuildConfig(1, 0, config) + test.log("Using build config '%s'" % config) + runAndCloseApp() + verifyBuildAndRun() + mouseClick(waitForObject(":*Qt Creator.Clear_QToolButton")) diff -Nru qtcreator-2.5.0/tests/system/suite_APTW/suite.conf qtcreator-2.5.2/tests/system/suite_APTW/suite.conf --- qtcreator-2.5.0/tests/system/suite_APTW/suite.conf 1970-01-01 00:00:00.000000000 +0000 +++ qtcreator-2.5.2/tests/system/suite_APTW/suite.conf 2012-08-08 13:47:06.000000000 +0000 @@ -0,0 +1,10 @@ +AUT=qtcreator +CLASS= +CLASSPATH= +ENVVARS=envvars +HOOK_SUB_PROCESSES=false +IMPLICITAUTSTART=0 +LANGUAGE=Python +TEST_CASES=tst_APTW01 tst_APTW02 +VERSION=2 +WRAPPERS=Qt diff -Nru qtcreator-2.5.0/tests/system/suite_APTW/tst_APTW01/test.py qtcreator-2.5.2/tests/system/suite_APTW/tst_APTW01/test.py --- qtcreator-2.5.0/tests/system/suite_APTW/tst_APTW01/test.py 1970-01-01 00:00:00.000000000 +0000 +++ qtcreator-2.5.2/tests/system/suite_APTW/tst_APTW01/test.py 2012-08-08 13:47:06.000000000 +0000 @@ -0,0 +1,13 @@ +source("../../shared/qtcreator.py") +source("../../shared/suites_qtta.py") +source("../shared/aptw.py") + +# test New Qt Gui Application build and run for release and debug option +def main(): + startApplication("qtcreator" + SettingsPath) + createProject_Qt_GUI(tempDir(), "SampleApp") + # pick version 4.7.4 and then run project for debug and release and verify results + pickVersion474runVerify() + #close Qt creator + invokeMenuItem("File", "Exit") +#no cleanup needed diff -Nru qtcreator-2.5.0/tests/system/suite_APTW/tst_APTW02/test.py qtcreator-2.5.2/tests/system/suite_APTW/tst_APTW02/test.py --- qtcreator-2.5.0/tests/system/suite_APTW/tst_APTW02/test.py 1970-01-01 00:00:00.000000000 +0000 +++ qtcreator-2.5.2/tests/system/suite_APTW/tst_APTW02/test.py 2012-08-08 13:47:06.000000000 +0000 @@ -0,0 +1,14 @@ +source("../../shared/qtcreator.py") +source("../../shared/suites_qtta.py") +source("../shared/aptw.py") + +# test New Qt Quick Application build and run for release and debug option +def main(): + startApplication("qtcreator" + SettingsPath) + createNewQtQuickApplication(tempDir(), "SampleApp") + waitForSignal("{type='CppTools::Internal::CppModelManager' unnamed='1'}", "sourceFilesRefreshed(QStringList)") + # pick version 4.7.4 and then run project for debug and release and verify results + pickVersion474runVerify() + #close Qt creator + invokeMenuItem("File", "Exit") +#no cleanup needed diff -Nru qtcreator-2.5.0/tests/system/suite_CCOM/tst_CCOM01/test.py qtcreator-2.5.2/tests/system/suite_CCOM/tst_CCOM01/test.py --- qtcreator-2.5.0/tests/system/suite_CCOM/tst_CCOM01/test.py 2012-05-09 12:13:19.000000000 +0000 +++ qtcreator-2.5.2/tests/system/suite_CCOM/tst_CCOM01/test.py 2012-08-08 13:47:06.000000000 +0000 @@ -14,7 +14,10 @@ # open example project openQmakeProject(examplePath) # build and wait until finished - on all (except Qt 4.7.0 (would fail)) build configurations - for config in iterateBuildConfigs(1, 0, "(?!.*4\.7\.0.*)"): + availableConfigs = iterateBuildConfigs(1, 0, "(?!.*4\.7\.0.*)") + if not availableConfigs: + test.fatal("Haven't found a suitable Qt version (anything except Qt 4.7.0) - leaving without building.") + for config in availableConfigs: selectBuildConfig(1, 0, config) # try to build project test.log("Testing build configuration: " + config) diff -Nru qtcreator-2.5.0/tests/system/suite_CSUP/objects.map qtcreator-2.5.2/tests/system/suite_CSUP/objects.map --- qtcreator-2.5.0/tests/system/suite_CSUP/objects.map 2012-05-09 12:13:19.000000000 +0000 +++ qtcreator-2.5.2/tests/system/suite_CSUP/objects.map 2012-08-08 13:47:06.000000000 +0000 @@ -1,3 +1,4 @@ +:*Qt Creator.Find_Find::Internal::FindToolBar {name='Find__Internal__FindWidget' type='Find::Internal::FindToolBar' visible='1' window=':Qt Creator_Core::Internal::MainWindow' windowTitle='Find'} :Behavior.Autocomplete common prefix_QCheckBox {container=':CppTools__Internal__CompletionSettingsPage.Behavior_QGroupBox' name='partiallyComplete' text='Autocomplete common prefix' type='QCheckBox' visible='1'} :Behavior.completionTrigger_QComboBox {container=':CppTools__Internal__CompletionSettingsPage.Behavior_QGroupBox' name='completionTrigger' type='QComboBox' visible='1'} :CppTools__Internal__CompletionSettingsPage.Behavior_QGroupBox {container=':qt_tabwidget_stackedwidget.CppTools__Internal__CompletionSettingsPage_QWidget' name='groupBox' title='Behavior' type='QGroupBox' visible='1'} @@ -7,9 +8,17 @@ :Options.qt_tabwidget_tabbar_QTabBar {name='qt_tabwidget_tabbar' type='QTabBar' visible='1' window=':Options_Core::Internal::SettingsDialog'} :Options_Core::Internal::SettingsDialog {type='Core::Internal::SettingsDialog' unnamed='1' visible='1' windowTitle~='(Options|Preferences)'} :Options_QListView {type='QListView' unnamed='1' visible='1' window=':Options_Core::Internal::SettingsDialog'} +:Qt Creator.CloseFind_QToolButton {name='close' type='QToolButton' visible='1' window=':Qt Creator_Core::Internal::MainWindow'} +:Qt Creator.Create Build Configurations:_QComboBox {leftWidget=':Qt Creator.Create Build Configurations:_QLabel' type='QComboBox' unnamed='1' visible='1' window=':Qt Creator_Core::Internal::MainWindow'} +:Qt Creator.Create Build Configurations:_QLabel {text='Create build configurations:' type='QLabel' unnamed='1' visible='1' window=':Qt Creator_Core::Internal::MainWindow'} :Qt Creator.QtCreator.MenuBar_QMenuBar {name='QtCreator.MenuBar' type='QMenuBar' visible='1' window=':Qt Creator_Core::Internal::MainWindow'} +:Qt Creator.Replace All_QToolButton {name='replaceAllButton' text='Replace All' type='QToolButton' visible='1' window=':Qt Creator_Core::Internal::MainWindow'} +:Qt Creator.Replace_QToolButton {name='replaceButton' text='Replace' type='QToolButton' visible='1' window=':Qt Creator_Core::Internal::MainWindow'} +:Qt Creator.findEdit_Utils::FilterLineEdit {name='findEdit' type='Utils::FilterLineEdit' visible='1' window=':Qt Creator_Core::Internal::MainWindow'} +:Qt Creator.replaceEdit_Utils::FilterLineEdit {name='replaceEdit' type='Utils::FilterLineEdit' visible='1' window=':Qt Creator_Core::Internal::MainWindow'} :Qt Creator_Core::Internal::MainWindow {type='Core::Internal::MainWindow' unnamed='1' visible='1' windowTitle?='*Qt Creator'} :Qt Creator_CppEditor::Internal::CPPEditorWidget {type='CppEditor::Internal::CPPEditorWidget' unnamed='1' visible='1' window=':Qt Creator_Core::Internal::MainWindow'} +:Qt Creator_SearchResult_Core::Internal::OutputPaneToggleButton {occurrence='2' type='Core::Internal::OutputPaneToggleButton' unnamed='1' visible='1' window=':Qt Creator_Core::Internal::MainWindow'} :Qt Creator_Utils::NavigationTreeView {type='Utils::NavigationTreeView' unnamed='1' visible='1' window=':Qt Creator_Core::Internal::MainWindow'} :addToVersionControlComboBox_QComboBox {name='addToVersionControlComboBox' type='QComboBox' visible='1'} :m_popupFrame_QListView {container=':m_popupFrame_TextEditor::GenericProposalWidget' type='QListView' unnamed='1' visible='1'} diff -Nru qtcreator-2.5.0/tests/system/suite_CSUP/suite.conf qtcreator-2.5.2/tests/system/suite_CSUP/suite.conf --- qtcreator-2.5.0/tests/system/suite_CSUP/suite.conf 2012-05-09 12:13:19.000000000 +0000 +++ qtcreator-2.5.2/tests/system/suite_CSUP/suite.conf 2012-08-08 13:47:06.000000000 +0000 @@ -5,6 +5,6 @@ HOOK_SUB_PROCESSES=false IMPLICITAUTSTART=0 LANGUAGE=Python -TEST_CASES=tst_CSUP01 tst_CSUP02 +TEST_CASES=tst_CSUP01 tst_CSUP02 tst_CSUP04 tst_CSUP05 VERSION=2 WRAPPERS=Qt diff -Nru qtcreator-2.5.0/tests/system/suite_CSUP/tst_CSUP04/test.py qtcreator-2.5.2/tests/system/suite_CSUP/tst_CSUP04/test.py --- qtcreator-2.5.0/tests/system/suite_CSUP/tst_CSUP04/test.py 1970-01-01 00:00:00.000000000 +0000 +++ qtcreator-2.5.2/tests/system/suite_CSUP/tst_CSUP04/test.py 2012-08-08 13:47:06.000000000 +0000 @@ -0,0 +1,60 @@ +source("../../shared/suites_qtta.py") +source("../../shared/qtcreator.py") + +# entry of test +def main(): + global searchFinished + # prepare example project + sourceExample = os.path.abspath(sdkPath + "/Examples/4.7/declarative/animation/basics/property-animation") + if not neededFilePresent(sourceExample): + return + # copy example project to temp directory + templateDir = prepareTemplate(sourceExample) + examplePath = templateDir + "/propertyanimation.pro" + startApplication("qtcreator" + SettingsPath) + # open example project + openQmakeProject(examplePath) + installLazySignalHandler("{type='Core::FutureProgress' unnamed='1'}", "finished()", "__handleFutureProgress__") + # wait for parsing to complete + waitForSignal("{type='CppTools::Internal::CppModelManager' unnamed='1'}", "sourceFilesRefreshed(QStringList)") + # open test .pro project. + test.verify(waitForObjectItem(":Qt Creator_Utils::NavigationTreeView", "propertyanimation"), + "Verifying if: Project is opened.") + # open .cpp file in editor + doubleClickItem(":Qt Creator_Utils::NavigationTreeView", "propertyanimation.Sources.main\\.cpp", 5, 5, 0, Qt.LeftButton) + test.verify(checkIfObjectExists(":Qt Creator_CppEditor::Internal::CPPEditorWidget"), + "Verifying if: .cpp file is opened in Edit mode.") + # place cursor on line "QmlApplicationViewer viewer;" + editorWidget = findObject(":Qt Creator_CppEditor::Internal::CPPEditorWidget") + searchFinished = False + # invoke find usages from context menu on word "viewer" + if not invokeFindUsage(editorWidget, "QmlApplicationViewer viewer;", "", 10): + invokeMenuItem("File", "Exit") + return + # wait until search finished and verify search results + waitFor("searchFinished") + validateSearchResult(17) + result = re.search("QmlApplicationViewer", str(editorWidget.plainText)) + test.verify(result, "Verifying if: The list of all usages of the selected text is displayed in Search Results. " + "File with used text is opened.") + # move cursor to the other word and test Find Usages function by pressing Ctrl+Shift+U. + doubleClickItem(":Qt Creator_Utils::NavigationTreeView", "propertyanimation.Sources.main\\.cpp", 5, 5, 0, Qt.LeftButton) + if not placeCursorToLine(editorWidget, "viewer.setOrientation(QmlApplicationViewer::ScreenOrientationAuto);"): + return + for i in range(4): + type(editorWidget, "") + searchFinished = False + type(editorWidget, "") + # wait until search finished and verify search results + waitFor("searchFinished") + validateSearchResult(3) + # exit qt creator + invokeMenuItem("File", "Save All") + invokeMenuItem("File", "Exit") +# no cleanup needed, as whole testing directory gets properly removed after test finished + +def __handleFutureProgress__(obj): + global searchFinished + if className(obj) == "Core::FutureProgress": + searchFinished = True + diff -Nru qtcreator-2.5.0/tests/system/suite_CSUP/tst_CSUP05/test.py qtcreator-2.5.2/tests/system/suite_CSUP/tst_CSUP05/test.py --- qtcreator-2.5.0/tests/system/suite_CSUP/tst_CSUP05/test.py 1970-01-01 00:00:00.000000000 +0000 +++ qtcreator-2.5.2/tests/system/suite_CSUP/tst_CSUP05/test.py 2012-08-08 13:47:06.000000000 +0000 @@ -0,0 +1,68 @@ +source("../../shared/suites_qtta.py") +source("../../shared/qtcreator.py") + +# entry of test +def main(): + # prepare example project + sourceExample = os.path.abspath(sdkPath + "/Examples/4.7/declarative/animation/basics/property-animation") + if not neededFilePresent(sourceExample): + return + # copy example project to temp directory + templateDir = prepareTemplate(sourceExample) + examplePath = templateDir + "/propertyanimation.pro" + startApplication("qtcreator" + SettingsPath) + # open example project + openQmakeProject(examplePath) + # wait for parsing to complete + waitForSignal("{type='CppTools::Internal::CppModelManager' unnamed='1'}", "sourceFilesRefreshed(QStringList)") + test.verify(waitForObjectItem(":Qt Creator_Utils::NavigationTreeView", "propertyanimation"), + "Verifying if: Project is opened.") + # open .cpp file in editor + doubleClickItem(":Qt Creator_Utils::NavigationTreeView", "propertyanimation.Sources.main\\.cpp", 5, 5, 0, Qt.LeftButton) + test.verify(checkIfObjectExists(":Qt Creator_CppEditor::Internal::CPPEditorWidget"), + "Verifying if: .cpp file is opened in Edit mode.") + # select some word for example "viewer" and press Ctrl+F. + editorWidget = findObject(":Qt Creator_CppEditor::Internal::CPPEditorWidget") + if not placeCursorToLine(editorWidget, "QmlApplicationViewer viewer;"): + invokeMenuItem("File", "Exit") + return + type(editorWidget, "") + for i in range(6): + type(editorWidget, "") + type(editorWidget, "") + # verify if find toolbar exists and if search text contains selected word + test.verify(checkIfObjectExists(":*Qt Creator.Find_Find::Internal::FindToolBar"), + "Verifying if: Find/Replace pane is displayed at the bottom of the view.") + test.compare(waitForObject(":Qt Creator.findEdit_Utils::FilterLineEdit").displayText, "viewer", + "Verifying if: Find line edit contains 'viewer' text.") + # insert some word to "Replace with:" field and select "Replace All". + replaceEditorContent(waitForObject(":Qt Creator.replaceEdit_Utils::FilterLineEdit"), "find") + oldCodeText = str(editorWidget.plainText) + clickButton(waitForObject(":Qt Creator.Replace All_QToolButton")) + mouseClick(waitForObject(":Qt Creator.replaceEdit_Utils::FilterLineEdit"), 5, 5, 0, Qt.LeftButton) + newCodeText = str(editorWidget.plainText) + test.compare(newCodeText, oldCodeText.replace("viewer", "find").replace("Viewer", "find"), + "Verifying if: Found text is replaced with new word properly.") + # select some other word in .cpp file and select "Edit" -> "Find/Replace". + clickButton(waitForObject(":Qt Creator.CloseFind_QToolButton")) + placeCursorToLine(editorWidget, "find.setOrientation(QmlApplicationfind::ScreenOrientationAuto);") + for i in range(25): + type(editorWidget, "") + for i in range(18): + type(editorWidget, "") + invokeMenuItem("Edit", "Find/Replace", "Find/Replace") + replaceEditorContent(waitForObject(":Qt Creator.replaceEdit_Utils::FilterLineEdit"), "QmlApplicationViewer") + oldCodeText = str(editorWidget.plainText) + clickButton(waitForObject(":Qt Creator.Replace_QToolButton")) + newCodeText = str(editorWidget.plainText) + # "::" is used to replace only one occurrence by python + test.compare(newCodeText, oldCodeText.replace("QmlApplicationfind::", "QmlApplicationViewer::"), + "Verifying if: Only selected word is replaced, the rest of found words are not replaced.") + # close Find/Replace tab. + clickButton(waitForObject(":Qt Creator.CloseFind_QToolButton")) + test.verify(checkIfObjectExists(":*Qt Creator.Find_Find::Internal::FindToolBar", False), + "Verifying if: Find/Replace tab is closed.") + # exit qt creator + invokeMenuItem("File", "Save All") + invokeMenuItem("File", "Exit") +# no cleanup needed, as whole testing directory gets properly removed after test finished diff -Nru qtcreator-2.5.0/tests/system/suite_HELP/envvars qtcreator-2.5.2/tests/system/suite_HELP/envvars --- qtcreator-2.5.0/tests/system/suite_HELP/envvars 1970-01-01 00:00:00.000000000 +0000 +++ qtcreator-2.5.2/tests/system/suite_HELP/envvars 2012-08-08 13:47:06.000000000 +0000 @@ -0,0 +1 @@ +QT_PLATFORM_PLUGIN=nonesuch diff -Nru qtcreator-2.5.0/tests/system/suite_HELP/objects.map qtcreator-2.5.2/tests/system/suite_HELP/objects.map --- qtcreator-2.5.0/tests/system/suite_HELP/objects.map 1970-01-01 00:00:00.000000000 +0000 +++ qtcreator-2.5.2/tests/system/suite_HELP/objects.map 2012-08-08 13:47:06.000000000 +0000 @@ -0,0 +1,55 @@ +:About Qt Creator.Close_QPushButton {text='Close' type='QPushButton' unnamed='1' visible='1' window=':About Qt Creator_Core::Internal::VersionDialog'} +:About Qt Creator.Label {text?='*Qt Creator*' type='QLabel' unnamed='1' visible='1' window=':About Qt Creator_Core::Internal::VersionDialog'} +:About Qt Creator_Core::Internal::VersionDialog {type='Core::Internal::VersionDialog' unnamed='1' visible='1' windowTitle='About Qt Creator'} +:Add Bookmark.+_QToolButton {name='toolButton' text='+' type='QToolButton' visible='1' window=':Add Bookmark_BookmarkDialog'} +:Add Bookmark.New Folder_QPushButton {name='newFolderButton' text='New Folder' type='QPushButton' visible='1' window=':Add Bookmark_BookmarkDialog'} +:Add Bookmark.OK_QPushButton {text='OK' type='QPushButton' unnamed='1' visible='1' window=':Add Bookmark_BookmarkDialog'} +:Add Bookmark.line_QFrame {name='line' type='QFrame' visible='1' window=':Add Bookmark_BookmarkDialog'} +:Add Bookmark.line_QMenu {aboveWidget=':Add Bookmark.line_QFrame' type='QMenu' unnamed='1' visible='1' window=':Add Bookmark_BookmarkDialog'} +:Add Bookmark.treeView_QTreeView {name='treeView' type='QTreeView' visible='1' window=':Add Bookmark_BookmarkDialog'} +:Add Bookmark_BookmarkDialog {name='BookmarkDialog' type='BookmarkDialog' visible='1' windowTitle='Add Bookmark'} +:Folder 1.Folder 2_QModelIndex {column='0' container=':Sample.Folder 1_QModelIndex' text='Folder 2' type='QModelIndex'} +:Folder 2.Qt Creator : Building and Running an Example_QModelIndex {column='0' container=':Folder 1.Folder 2_QModelIndex' text?='Qt Creator : Building and Running an Example*' type='QModelIndex'} +:Getting Started.Building and Running an Example_QModelIndex {column='0' container=':Qt Creator Manual*.Getting Started_QModelIndex' text='Building and Running an Example' type='QModelIndex'} +:Hits_QCLuceneResultWidget {aboveWidget=':Hits_QLabel' type='QCLuceneResultWidget' unnamed='1' visible='1' window=':Qt Creator_Core::Internal::MainWindow'} +:Hits_QLabel {text~='\\\\d+ - \\\\d+ of \\\\d+ Hits' type='QLabel' unnamed='1' visible='1' window=':Qt Creator_Core::Internal::MainWindow'} +:New Folder.Qt Creator : Qt Creator Manual_QModelIndex {column='0' container=':Sample.New Folder_QModelIndex' text='Qt Creator : Qt Creator Manual' type='QModelIndex'} +:Next_QPushButton {text~='(Next.*|Continue)' type='QPushButton' visible='1'} +:Options.OK_QPushButton {text='OK' type='QPushButton' unnamed='1' visible='1' window=':Options_Core::Internal::SettingsDialog'} +:Options.qt_tabwidget_tabbar_QTabBar {name='qt_tabwidget_tabbar' type='QTabBar' visible='1' window=':Options_Core::Internal::SettingsDialog'} +:Options_Core::Internal::SettingsDialog {type='Core::Internal::SettingsDialog' unnamed='1' visible='1' windowTitle~='Options|Preferences'} +:Options_QListView {type='QListView' unnamed='1' visible='1' window=':Options_Core::Internal::SettingsDialog'} +:Qt Creator Manual QModelIndex {column='0' container=':Qt Creator_QHelpContentWidget' text='Qt Creator Manual 2.5.1' type='QModelIndex'} +:Qt Creator Manual*.Getting Started_QModelIndex {column='0' container=':Qt Creator Manual QModelIndex' text='Getting Started' type='QModelIndex'} +:Qt Creator.+_QToolButton {text='+' type='QToolButton' unnamed='1' visible='1' window=':Qt Creator_Core::Internal::MainWindow'} +:Qt Creator.Add Bookmark_QToolButton {text='Add Bookmark' type='QToolButton' unnamed='1' visible='1' window=':Qt Creator_Core::Internal::MainWindow'} +:Qt Creator.QtCreator.MenuBar_QMenuBar {name='QtCreator.MenuBar' type='QMenuBar' visible='1' window=':Qt Creator_Core::Internal::MainWindow'} +:Qt Creator.Search for:_QLabel {text='Search for:' type='QLabel' unnamed='1' visible='1' window=':Qt Creator_Core::Internal::MainWindow'} +:Qt Creator.Search for:_QLineEdit {leftWidget=':Qt Creator.Search for:_QLabel' type='QLineEdit' unnamed='1' visible='1' window=':Qt Creator_Core::Internal::MainWindow'} +:Qt Creator.Search_QPushButton {text='Search' type='QPushButton' unnamed='1' visible='1' window=':Qt Creator_Core::Internal::MainWindow'} +:Qt Creator.with all of the words:_QLabel {text='with all of the words:' type='QLabel' unnamed='1' visible='1' window=':Qt Creator_Core::Internal::MainWindow'} +:Qt Creator.with all of the words:_QLineEdit {leftWidget=':Qt Creator.with all of the words:_QLabel' type='QLineEdit' unnamed='1' visible='1' window=':Qt Creator_Core::Internal::MainWindow'} +:Qt Creator.with at least one of the words:_QLabel {text='with at least one of the words:' type='QLabel' unnamed='1' visible='1' window=':Qt Creator_Core::Internal::MainWindow'} +:Qt Creator.with at least one of the words:_QLineEdit {leftWidget=':Qt Creator.with at least one of the words:_QLabel' type='QLineEdit' unnamed='1' visible='1' window=':Qt Creator_Core::Internal::MainWindow'} +:Qt Creator.with exact phrase:_QLabel {text='with exact phrase:' type='QLabel' unnamed='1' visible='1' window=':Qt Creator_Core::Internal::MainWindow'} +:Qt Creator.with exact phrase:_QLineEdit {leftWidget=':Qt Creator.with exact phrase:_QLabel' type='QLineEdit' unnamed='1' visible='1' window=':Qt Creator_Core::Internal::MainWindow'} +:Qt Creator.without the words:_QLabel {text='without the words:' type='QLabel' unnamed='1' visible='1' window=':Qt Creator_Core::Internal::MainWindow'} +:Qt Creator.without the words:_QLineEdit {leftWidget=':Qt Creator.without the words:_QLabel' type='QLineEdit' unnamed='1' visible='1' window=':Qt Creator_Core::Internal::MainWindow'} +:Qt Creator.words similar to:_QLabel {text='words similar to:' type='QLabel' unnamed='1' visible='1' window=':Qt Creator_Core::Internal::MainWindow'} +:Qt Creator.words similar to:_QLineEdit {leftWidget=':Qt Creator.words similar to:_QLabel' type='QLineEdit' unnamed='1' visible='1' window=':Qt Creator_Core::Internal::MainWindow'} +:Qt Creator_Core::Internal::CommandComboBox {occurrence='2' type='Core::Internal::CommandComboBox' unnamed='1' visible='1' window=':Qt Creator_Core::Internal::MainWindow'} +:Qt Creator_Core::Internal::CommandComboBox_2 {type='Core::Internal::CommandComboBox' unnamed='1' visible='1' window=':Qt Creator_Core::Internal::MainWindow'} +:Qt Creator_Core::Internal::MainWindow {type='Core::Internal::MainWindow' unnamed='1' visible='1' windowTitle?='*Qt Creator'} +:Qt Creator_Help::Internal::HelpViewer {type='Help::Internal::HelpViewer' unnamed='1' visible='1' window=':Qt Creator_Core::Internal::MainWindow'} +:Qt Creator_QHelpContentWidget {type='QHelpContentWidget' unnamed='1' visible='1' window=':Qt Creator_Core::Internal::MainWindow'} +:Qt Creator_QHelpSearchQueryWidget {type='QHelpSearchQueryWidget' unnamed='1' visible='1' window=':Qt Creator_Core::Internal::MainWindow'} +:Qt Creator_QmlJSEditor::QmlJSTextEditorWidget {type='QmlJSEditor::QmlJSTextEditorWidget' unnamed='1' visible='1' window=':Qt Creator_Core::Internal::MainWindow'} +:Qt Creator_TreeView {type='TreeView' unnamed='1' visible='1' window=':Qt Creator_Core::Internal::MainWindow'} +:Qt Reference Documentation_QModelIndex {column='0' container=':Qt Creator_QHelpContentWidget' text='Qt Reference Documentation' type='QModelIndex'} +:Sample.Folder 1_QModelIndex {column='0' container=':Sample_QModelIndex' text='Folder 1' type='QModelIndex'} +:Sample.New Folder_QModelIndex {column='0' container=':Sample_QModelIndex' text?='New Folder*' type='QModelIndex'} +:Sample_QModelIndex {column='0' container=':Qt Creator_TreeView' text='Sample' type='QModelIndex'} +:addToVersionControlComboBox_QComboBox {name='addToVersionControlComboBox' type='QComboBox' visible='1'} +:treeView.Sample_QModelIndex {column='0' container=':Add Bookmark.treeView_QTreeView' text='Sample' type='QModelIndex'} +:treeView.Yes_QPushButton {container=':Add Bookmark.treeView_QTreeView' text='Yes' type='QPushButton' unnamed='1' visible='1'} +:treeView_QExpandingLineEdit {container=':Add Bookmark.treeView_QTreeView' type='QExpandingLineEdit' unnamed='1' visible='1'} diff -Nru qtcreator-2.5.0/tests/system/suite_HELP/suite.conf qtcreator-2.5.2/tests/system/suite_HELP/suite.conf --- qtcreator-2.5.0/tests/system/suite_HELP/suite.conf 1970-01-01 00:00:00.000000000 +0000 +++ qtcreator-2.5.2/tests/system/suite_HELP/suite.conf 2012-08-08 13:47:06.000000000 +0000 @@ -0,0 +1,10 @@ +AUT=qtcreator +CLASS= +CLASSPATH= +ENVVARS=envvars +HOOK_SUB_PROCESSES=false +IMPLICITAUTSTART=0 +LANGUAGE=Python +TEST_CASES=tst_HELP02 tst_HELP04 tst_HELP05 tst_HELP06 +VERSION=2 +WRAPPERS=Qt diff -Nru qtcreator-2.5.0/tests/system/suite_HELP/tst_HELP02/test.py qtcreator-2.5.2/tests/system/suite_HELP/tst_HELP02/test.py --- qtcreator-2.5.0/tests/system/suite_HELP/tst_HELP02/test.py 1970-01-01 00:00:00.000000000 +0000 +++ qtcreator-2.5.2/tests/system/suite_HELP/tst_HELP02/test.py 2012-08-08 13:47:06.000000000 +0000 @@ -0,0 +1,50 @@ +source("../../shared/qtcreator.py") +source("../../shared/suites_qtta.py") + +# test Qt Creator version information from file and dialog +def getQtCreatorVersionFromDialog(): + chk = re.search("(?<=Qt Creator)\s\d+.\d+.\d+" , str(waitForObject(':About Qt Creator.Label' ).text)) + try: + ver = chk.group(0).strip() + return ver + except: + test.fail("Failed to get the exact version from Dialog") + return "" + +def getQtCreatorVersionFromFile(): + qtCreatorPriFileName = "../../../../qtcreator.pri" + # open file and read version + fileText = open(qtCreatorPriFileName, "r").read() + chk = re.search("(?<=QTCREATOR_VERSION =)\s\d+.\d+.\d+", fileText) + try: + ver = chk.group(0).strip() + return ver + except: + test.fail("Failed to get the exact version from File") + return "" + +def main(): + expectedVersion = getQtCreatorVersionFromFile() + if not expectedVersion: + test.fatal("Can't find version from file.") + return + startApplication("qtcreator" + SettingsPath) + if platform.system() == "Darwin": + invokeMenuItem("Help", "About Qt Creator") + else: + invokeMenuItem("Help", "About Qt Creator...") + # verify qt creator version + waitForObject(":About Qt Creator_Core::Internal::VersionDialog") + actualVersion = getQtCreatorVersionFromDialog() + test.verify(actualVersion == expectedVersion, + "Verifying if version is good. Current version is: " + actualVersion + ", expected version is: " + expectedVersion) + # close and verify about dialog closed + clickButton(waitForObject(":About Qt Creator.Close_QPushButton")) + test.verify(checkIfObjectExists(":About Qt Creator_Core::Internal::VersionDialog", False), + "Verifying if About dialog closed.") + # exit qt creator + invokeMenuItem("File", "Exit") + # verify if qt creator closed properly + test.verify(checkIfObjectExists(":Qt Creator_Core::Internal::MainWindow", False), + "Verifying if Qt Creator closed.") +# no cleanup needed diff -Nru qtcreator-2.5.0/tests/system/suite_HELP/tst_HELP04/test.py qtcreator-2.5.2/tests/system/suite_HELP/tst_HELP04/test.py --- qtcreator-2.5.0/tests/system/suite_HELP/tst_HELP04/test.py 1970-01-01 00:00:00.000000000 +0000 +++ qtcreator-2.5.2/tests/system/suite_HELP/tst_HELP04/test.py 2012-08-08 13:47:06.000000000 +0000 @@ -0,0 +1,116 @@ +source("../../shared/qtcreator.py") +source("../../shared/suites_qtta.py") +import re + +# test search in help mode and advanced search +searchKeywordDictionary={ "deployment":True, "deplmint":False, "build":True, "bld":False } + +def __getSelectedText__(): + try: + selText = findObject(":Qt Creator_Help::Internal::HelpViewer").selectedText + if className(selText) != 'instancemethod': + return str(selText) + except: + pass + try: + hv = findObject(":Qt Creator_Help::Internal::HelpViewer") + selText = getHighlightsInHtml(str(hv.toHtml())) + except: + test.warning("Could not get highlighted text.") + selText = '' + return str(selText) + +def __handleTextChanged__(obj): + global textHasChanged + textHasChanged = True + +def getHighlightsInHtml(htmlCode): + pattern = re.compile('color:#ff0000;">(.*?)') + res = "" + for curr in pattern.finditer(htmlCode): + if curr.group(1) in res: + continue + res += "%s " % curr.group(1) + test.log(res) + return res + +# wait for indexing progress bar to appear and disappear +def progressBarWait(): + checkIfObjectExists("{type='Core::Internal::ProgressBar' unnamed='1'}", True, 2000) + checkIfObjectExists("{type='Core::Internal::ProgressBar' unnamed='1'}", False, 60000) + +def main(): + global textHasChanged + noMatch = "Your search did not match any documents." + startApplication("qtcreator" + SettingsPath) + installLazySignalHandler(":Qt Creator_Help::Internal::HelpViewer", "textChanged()", "__handleTextChanged__") + addHelpDocumentationFromSDK() + # switch to help mode + switchViewTo(ViewConstants.HELP) + # verify that search widget is accessible + mouseClick(waitForObjectItem(":Qt Creator_Core::Internal::CommandComboBox_2", "Search")) + test.verify(checkIfObjectExists(":Qt Creator_QHelpSearchQueryWidget"), + "Verifying search widget visibility.") + # try to search empty string + clickButton(waitForObject(":Qt Creator.Search_QPushButton")) + progressBarWait() + test.verify(waitFor("noMatch in " + "str(waitForObject(':Hits_QCLuceneResultWidget').plainText)", 2000), + "Verifying if search did not match anything.") + # workaround for "endless waiting cursor" + mouseClick(waitForObject(":Qt Reference Documentation_QModelIndex")) + # try to search keyword from list + for searchKeyword,shouldFind in searchKeywordDictionary.items(): + mouseClick(waitForObject(":Qt Creator.Search for:_QLineEdit")) + replaceEditorContent(":Qt Creator.Search for:_QLineEdit", searchKeyword) + type(waitForObject(":Qt Creator.Search for:_QLineEdit"), "") + progressBarWait() + if shouldFind: + test.verify(waitFor("re.match('[1-9]\d* - [1-9]\d* of [1-9]\d* Hits'," + "str(findObject(':Hits_QLabel').text))", 2000), + "Verifying if search results found with 1+ hits for: " + searchKeyword) + textHasChanged = False + selText = __getSelectedText__() + # click in the widget, tab to first item and press enter + mouseClick(waitForObject(":Hits_QCLuceneResultWidget"), 1, 1, 0, Qt.LeftButton) + type(waitForObject(":Hits_QCLuceneResultWidget"), "") + type(waitForObject(":Hits_QCLuceneResultWidget"), "") + waitFor("textHasChanged or selText != __getSelectedText__()") + # verify if search keyword is found in results + test.verify(searchKeyword.lower() in __getSelectedText__().lower(), + searchKeyword + " search result can be found") + else: + test.verify(waitFor("noMatch in " + "str(waitForObject(':Hits_QCLuceneResultWidget').plainText)", 1000), + "Verifying if search did not match anything for: " + searchKeyword) + # advanced search - setup + clickButton(waitForObject(":Qt Creator.+_QToolButton")) + type(waitForObject(":Qt Creator.words similar to:_QLineEdit"), "deploy") + type(waitForObject(":Qt Creator.without the words:_QLineEdit"), "bookmark") + type(waitForObject(":Qt Creator.with exact phrase:_QLineEdit"), "sql in qt") + type(waitForObject(":Qt Creator.with all of the words:_QLineEdit"), "designer sql") + type(waitForObject(":Qt Creator.with at least one of the words:_QLineEdit"), "printing") + # advanced search - do search + clickButton(waitForObject(":Qt Creator.Search_QPushButton")) + progressBarWait() + # verify that advanced search results found + test.verify(waitFor("re.search('1 - 2 of 2 Hits'," + "str(findObject(':Hits_QLabel').text))", 3000), + "Verifying if 2 search results found") + resultsView = waitForObject(":Hits_QCLuceneResultWidget") + mouseClick(resultsView, 1, 1, 0, Qt.LeftButton) + type(resultsView, "") + type(resultsView, "") + test.verify("printing" in str(findObject(":Qt Creator_Help::Internal::HelpViewer").selectedText).lower(), + "printing advanced search result can be found") + for i in range(2): + type(resultsView, "") + type(resultsView, "") + test.verify("sql" in str(findObject(":Qt Creator_Help::Internal::HelpViewer").selectedText).lower(), + "sql advanced search result can be found") + # verify if simple search is properly disabled + test.verify(findObject(":Qt Creator.Search for:_QLineEdit").enabled == False, + "Verifying if simple search is not active in advanced mode.") + # exit + invokeMenuItem("File", "Exit") +# no cleanup needed diff -Nru qtcreator-2.5.0/tests/system/suite_HELP/tst_HELP05/test.py qtcreator-2.5.2/tests/system/suite_HELP/tst_HELP05/test.py --- qtcreator-2.5.0/tests/system/suite_HELP/tst_HELP05/test.py 1970-01-01 00:00:00.000000000 +0000 +++ qtcreator-2.5.2/tests/system/suite_HELP/tst_HELP05/test.py 2012-08-08 13:47:06.000000000 +0000 @@ -0,0 +1,34 @@ +source("../../shared/qtcreator.py") +source("../../shared/suites_qtta.py") + +# test context sensitive help in edit mode +# place cursor to keyword, in , and verify help to contain +def verifyInteractiveQMLHelp(lineText, helpText): + editorArea = waitForObject(":Qt Creator_QmlJSEditor::QmlJSTextEditorWidget") + # go to the specified word + placeCursorToLine(editorArea, lineText) + homeKey = "" + if platform.system() == "Darwin": + homeKey = "" + type(editorArea, homeKey) + else: + type(editorArea, homeKey) + # call help + type(editorArea, "") + test.verify(helpText in str(waitForObject(":Qt Creator_Help::Internal::HelpViewer").title), + "Verifying if help is opened with documentation for '%s'." % helpText) + +def main(): + startApplication("qtcreator" + SettingsPath) + addHelpDocumentationFromSDK() + # create qt quick application + createNewQtQuickApplication(tempDir(), "SampleApp") + # verify Rectangle help + verifyInteractiveQMLHelp("Rectangle {", "QML Rectangle Element") + # go back to edit mode + switchViewTo(ViewConstants.EDIT) + # verify MouseArea help + verifyInteractiveQMLHelp("MouseArea {", "QML MouseArea Element") + # exit + invokeMenuItem("File","Exit") +# no cleanup needed diff -Nru qtcreator-2.5.0/tests/system/suite_HELP/tst_HELP06/test.py qtcreator-2.5.2/tests/system/suite_HELP/tst_HELP06/test.py --- qtcreator-2.5.0/tests/system/suite_HELP/tst_HELP06/test.py 1970-01-01 00:00:00.000000000 +0000 +++ qtcreator-2.5.2/tests/system/suite_HELP/tst_HELP06/test.py 2012-08-08 13:47:06.000000000 +0000 @@ -0,0 +1,83 @@ +source("../../shared/qtcreator.py") +source("../../shared/suites_qtta.py") + +# test bookmark functionality +def renameBookmarkFolder(view, item, newName): + openItemContextMenu(view, item, 5, 5, 0) + activateItem(waitForObjectItem(":Add Bookmark.line_QMenu", "Rename Folder")) + replaceEditorContent(waitForObject(":treeView_QExpandingLineEdit"), newName) + type(waitForObject(":treeView_QExpandingLineEdit"), "") + return + +def main(): + startApplication("qtcreator" + SettingsPath) + # goto help mode and click on topic + switchViewTo(ViewConstants.HELP) + doubleClick(":Qt Creator Manual QModelIndex", 5, 5, 0, Qt.LeftButton) + doubleClick(":Qt Creator Manual*.Getting Started_QModelIndex", 5, 5, 0, Qt.LeftButton) + mouseClick(waitForObject(":Getting Started.Building and Running an Example_QModelIndex"), 5, 5, 0, Qt.LeftButton) + # open bookmarks window + clickButton(waitForObject(":Qt Creator.Add Bookmark_QToolButton")) + clickButton(waitForObject(":Add Bookmark.+_QToolButton")) + # create root bookmark directory + clickButton(waitForObject(":Add Bookmark.New Folder_QPushButton")) + # rename root bookmark directory + bookmarkView = waitForObject(":Add Bookmark.treeView_QTreeView") + renameBookmarkFolder(bookmarkView, "New Folder*", "Sample") + # create two more subfolders + clickButton(waitForObject(":Add Bookmark.New Folder_QPushButton")) + renameBookmarkFolder(bookmarkView, "Sample.New Folder*", "Folder 1") + clickButton(waitForObject(":Add Bookmark.New Folder_QPushButton")) + renameBookmarkFolder(bookmarkView, "Sample.Folder 1.New Folder*", "Folder 2") + clickButton(waitForObject(":Add Bookmark.OK_QPushButton")) + mouseClick(":Qt Creator Manual QModelIndex", 5, 5, 0, Qt.LeftButton) + type(waitForObject(":Qt Creator_QHelpContentWidget"), "") + clickButton(waitForObject(":Qt Creator.Add Bookmark_QToolButton")) + clickButton(waitForObject(":Add Bookmark.+_QToolButton")) + # click on "Sample" and create new directory under it + mouseClick(waitForObject(":treeView.Sample_QModelIndex")) + clickButton(waitForObject(":Add Bookmark.New Folder_QPushButton")) + clickButton(waitForObject(":Add Bookmark.OK_QPushButton")) + # choose bookmarks + mouseClick(waitForObjectItem(":Qt Creator_Core::Internal::CommandComboBox", "Bookmarks")) + # verify if all folders are created and bookmarks present + test.verify(checkIfObjectExists(":Sample_QModelIndex", verboseOnFail = True) and + checkIfObjectExists(":Sample.Folder 1_QModelIndex", verboseOnFail = True) and + checkIfObjectExists(":Folder 1.Folder 2_QModelIndex", verboseOnFail = True) and + checkIfObjectExists(":Folder 2.Qt Creator : Building and Running an Example_QModelIndex", verboseOnFail = True) and + checkIfObjectExists(":New Folder.Qt Creator : Qt Creator Manual_QModelIndex", verboseOnFail = True), + "Verifying if all folders and bookmarks are present") + mouseClick(waitForObject(":Qt Creator_TreeView"), 5, 5, 0, Qt.LeftButton) + for i in range(6): + type(waitForObject(":Qt Creator_TreeView"), "") + type(waitForObject(":Qt Creator_TreeView"), "") + test.verify("Qt Creator : Building and Running an Example" in str(waitForObject(":Qt Creator_Help::Internal::HelpViewer").title), + "Verifying if first bookmark is opened") + mouseClick(waitForObject(":Folder 2.Qt Creator : Building and Running an Example_QModelIndex")) + type(waitForObject(":Qt Creator_TreeView"), "") + type(waitForObject(":Qt Creator_TreeView"), "") + type(waitForObject(":Qt Creator_TreeView"), "") + type(waitForObject(":Qt Creator_TreeView"), "") + test.verify("Qt Creator : Qt Creator Manual" in str(waitForObject(":Qt Creator_Help::Internal::HelpViewer").title), + "Verifying if second bookmark is opened") + # delete previously created directory + clickButton(waitForObject(":Qt Creator.Add Bookmark_QToolButton")) + clickButton(waitForObject(":Add Bookmark.+_QToolButton")) + openItemContextMenu(waitForObject(":Add Bookmark.treeView_QTreeView"), "Sample.Folder 1", 5, 5, 0) + activateItem(waitForObjectItem(":Add Bookmark.line_QMenu", "Delete Folder")) + clickButton(waitForObject(":treeView.Yes_QPushButton")) + # close bookmarks + clickButton(waitForObject(":Add Bookmark.OK_QPushButton")) + # choose bookmarks from command combobox (left top corner of qt creator) + mouseClick(waitForObject(":Qt Creator_Core::Internal::CommandComboBox")) + mouseClick(waitForObjectItem(":Qt Creator_Core::Internal::CommandComboBox", "Bookmarks")) + # verify if folders and bookmark deleted + test.verify(checkIfObjectExists(":Sample_QModelIndex", verboseOnFail = True) and + checkIfObjectExists(":Sample.Folder 1_QModelIndex ", shouldExist = False, verboseOnFail = True) and + checkIfObjectExists(":Folder 1.Folder 2_QModelIndex", shouldExist = False, verboseOnFail = True) and + checkIfObjectExists(":Folder 2.Qt Creator : Building and Running an Example_QModelIndex", shouldExist = False, verboseOnFail = True) and + checkIfObjectExists(":New Folder.Qt Creator : Qt Creator Manual_QModelIndex", verboseOnFail = True), + "Verifying if folder 1 and folder 2 deleted including their bookmark") + # exit + invokeMenuItem("File", "Exit") +# no cleanup needed diff -Nru qtcreator-2.5.0/tests/system/suite_QMLS/shared/qmls.py qtcreator-2.5.2/tests/system/suite_QMLS/shared/qmls.py --- qtcreator-2.5.0/tests/system/suite_QMLS/shared/qmls.py 1970-01-01 00:00:00.000000000 +0000 +++ qtcreator-2.5.2/tests/system/suite_QMLS/shared/qmls.py 2012-08-08 13:47:06.000000000 +0000 @@ -0,0 +1,24 @@ +source("../../shared/qtcreator.py") +source("../../shared/suites_qtta.py") + +def startQtCreatorWithNewAppAtQMLEditor(projectDir, projectName, line = None): + startApplication("qtcreator" + SettingsPath) + # create qt quick application + createNewQtQuickApplication(projectDir, projectName) + # open qml file + doubleClickItem(":Qt Creator_Utils::NavigationTreeView", projectName + ".QML.qml/" + + projectName + ".main\\.qml", 5, 5, 0, Qt.LeftButton) + # get editor + editorArea = waitForObject(":Qt Creator_QmlJSEditor::QmlJSTextEditorWidget") + # place to line if needed + if line: + # place cursor to component + if not placeCursorToLine(editorArea, line): + invokeMenuItem("File", "Exit") + return None + return editorArea + +def verifyCurrentLine(editorArea, currentLineExpectedText, verifyMessage): + currentLineText = str(lineUnderCursor(editorArea)).strip(); + return test.compare(currentLineText, currentLineExpectedText, verifyMessage) + diff -Nru qtcreator-2.5.0/tests/system/suite_QMLS/suite.conf qtcreator-2.5.2/tests/system/suite_QMLS/suite.conf --- qtcreator-2.5.0/tests/system/suite_QMLS/suite.conf 2012-05-09 12:13:19.000000000 +0000 +++ qtcreator-2.5.2/tests/system/suite_QMLS/suite.conf 2012-08-08 13:47:06.000000000 +0000 @@ -5,6 +5,6 @@ HOOK_SUB_PROCESSES=false IMPLICITAUTSTART=0 LANGUAGE=Python -TEST_CASES=tst_QMLS01 tst_QMLS02 tst_QMLS03 tst_QMLS04 tst_QMLS05 +TEST_CASES=tst_QMLS01 tst_QMLS02 tst_QMLS03 tst_QMLS04 tst_QMLS05 tst_QMLS06 tst_QMLS07 tst_QMLS08 VERSION=2 WRAPPERS=Qt diff -Nru qtcreator-2.5.0/tests/system/suite_QMLS/tst_QMLS01/test.py qtcreator-2.5.2/tests/system/suite_QMLS/tst_QMLS01/test.py --- qtcreator-2.5.0/tests/system/suite_QMLS/tst_QMLS01/test.py 2012-05-09 12:13:19.000000000 +0000 +++ qtcreator-2.5.2/tests/system/suite_QMLS/tst_QMLS01/test.py 2012-08-08 13:47:06.000000000 +0000 @@ -1,5 +1,4 @@ -source("../../shared/qtcreator.py") -source("../../shared/suites_qtta.py") +source("../shared/qmls.py") # go to proper line, make backup, type needed text def __beginTestSuggestions__(editorArea, lineText, textToType): @@ -84,11 +83,8 @@ invokeMenuItem("File", "Exit") def main(): - startApplication("qtcreator" + SettingsPath) - # create qt quick application - createNewQtQuickApplication(tempDir(), "SampleApp") - # open qml file - doubleClickItem(":Qt Creator_Utils::NavigationTreeView", "SampleApp.QML.qml/SampleApp.main\\.qml", 5, 5, 0, Qt.LeftButton) + if not startQtCreatorWithNewAppAtQMLEditor(tempDir(), "SampleApp"): + return # test "color: " suggestion usage with Enter key if not testSuggestionsAuto("Text {", "col", "color:", ""): saveAndExit() diff -Nru qtcreator-2.5.0/tests/system/suite_QMLS/tst_QMLS02/test.py qtcreator-2.5.2/tests/system/suite_QMLS/tst_QMLS02/test.py --- qtcreator-2.5.0/tests/system/suite_QMLS/tst_QMLS02/test.py 2012-05-09 12:13:19.000000000 +0000 +++ qtcreator-2.5.2/tests/system/suite_QMLS/tst_QMLS02/test.py 2012-08-08 13:47:06.000000000 +0000 @@ -1,16 +1,8 @@ -source("../../shared/qtcreator.py") -source("../../shared/suites_qtta.py") +source("../shared/qmls.py") def main(): - startApplication("qtcreator" + SettingsPath) - # create qt quick application - createNewQtQuickApplication(tempDir(), "SampleApp") - # open qml file - doubleClickItem(":Qt Creator_Utils::NavigationTreeView", "SampleApp.QML.qml/SampleApp.main\\.qml", 5, 5, 0, Qt.LeftButton) - # get editor - editorArea = waitForObject(":Qt Creator_QmlJSEditor::QmlJSTextEditorWidget") - if not placeCursorToLine(editorArea, "Text {"): - invokeMenuItem("File", "Exit") + editorArea = startQtCreatorWithNewAppAtQMLEditor(tempDir(), "SampleApp", "Text {") + if not editorArea: return # write code with error (C should be lower case) testingCodeLine = 'Color : "blue"' @@ -24,8 +16,9 @@ "Verifying if error is properly reported") # repair error - go to written line placeCursorToLine(editorArea, testingCodeLine) - moveTextCursor(editorArea, QTextCursor.Left, QTextCursor.MoveAnchor, 14) - moveTextCursor(editorArea, QTextCursor.Right, QTextCursor.KeepAnchor, 1) + for i in range(14): + type(editorArea, "") + type(editorArea, "") type(editorArea, "c") # invoke QML parsing invokeMenuItem("Tools", "QML/JS", "Run Checks") diff -Nru qtcreator-2.5.0/tests/system/suite_QMLS/tst_QMLS03/test.py qtcreator-2.5.2/tests/system/suite_QMLS/tst_QMLS03/test.py --- qtcreator-2.5.0/tests/system/suite_QMLS/tst_QMLS03/test.py 2012-05-09 12:13:19.000000000 +0000 +++ qtcreator-2.5.2/tests/system/suite_QMLS/tst_QMLS03/test.py 2012-08-08 13:47:06.000000000 +0000 @@ -59,7 +59,8 @@ if not placeCursorToLine(editorArea, "Rectangle {"): invokeMenuItem("File", "Exit") return - moveTextCursor(editorArea, QTextCursor.Left, QTextCursor.MoveAnchor, 5) + for i in range(5): + type(editorArea, "") ctxtMenu = openContextMenuOnTextCursorPosition(editorArea) activateItem(waitForObjectItem(objectMap.realName(ctxtMenu), "Find Usages")) # check if usage was properly found @@ -77,7 +78,8 @@ if not placeCursorToLine(editorArea, "anchors { left: parent.left; top: parent.top; right: parent.right; bottom: parent.verticalCenter }"): invokeMenuItem("File", "Exit") return - moveTextCursor(editorArea, QTextCursor.Left, QTextCursor.MoveAnchor, 87) + for i in range(87): + type(editorArea, "") invokeMenuItem("Tools", "QML/JS", "Find Usages") # check if usage was properly found expectedResults = [ExpectedResult("color-animation.qml", 50, "anchors { left: parent.left; top: parent.top; right: parent.right; bottom: parent.verticalCenter }"), @@ -94,7 +96,8 @@ if not placeCursorToLine(editorArea, "SequentialAnimation on opacity {"): invokeMenuItem("File", "Exit") return - moveTextCursor(editorArea, QTextCursor.Left, QTextCursor.MoveAnchor, 5) + for i in range(5): + type(editorArea, "") type(editorArea, "") # check if usage was properly found expectedResults = [ExpectedResult("color-animation.qml", 87, "SequentialAnimation on opacity {")] diff -Nru qtcreator-2.5.0/tests/system/suite_QMLS/tst_QMLS04/test.py qtcreator-2.5.2/tests/system/suite_QMLS/tst_QMLS04/test.py --- qtcreator-2.5.0/tests/system/suite_QMLS/tst_QMLS04/test.py 2012-05-09 12:13:19.000000000 +0000 +++ qtcreator-2.5.2/tests/system/suite_QMLS/tst_QMLS04/test.py 2012-08-08 13:47:06.000000000 +0000 @@ -1,20 +1,12 @@ -source("../../shared/qtcreator.py") -source("../../shared/suites_qtta.py") +source("../shared/qmls.py") def main(): - startApplication("qtcreator" + SettingsPath) - # create qt quick application projectDir = tempDir() - createNewQtQuickApplication(projectDir, "SampleApp") - # open qml file - doubleClickItem(":Qt Creator_Utils::NavigationTreeView", "SampleApp.QML.qml/SampleApp.main\\.qml", 5, 5, 0, Qt.LeftButton) - # get editor - editorArea = waitForObject(":Qt Creator_QmlJSEditor::QmlJSTextEditorWidget") - # place cursor to component - if not placeCursorToLine(editorArea, "Text {"): - invokeMenuItem("File", "Exit") + editorArea = startQtCreatorWithNewAppAtQMLEditor(projectDir, "SampleApp", "Text {") + if not editorArea: return - moveTextCursor(editorArea, QTextCursor.Left, QTextCursor.MoveAnchor, 5) + for i in range(5): + type(editorArea, "") # invoke Refactoring - Move Component into separate file ctxtMenu = openContextMenuOnTextCursorPosition(editorArea) activateItem(waitForObjectItem(objectMap.realName(ctxtMenu), "Refactoring")) diff -Nru qtcreator-2.5.0/tests/system/suite_QMLS/tst_QMLS05/test.py qtcreator-2.5.2/tests/system/suite_QMLS/tst_QMLS05/test.py --- qtcreator-2.5.0/tests/system/suite_QMLS/tst_QMLS05/test.py 2012-05-09 12:13:19.000000000 +0000 +++ qtcreator-2.5.2/tests/system/suite_QMLS/tst_QMLS05/test.py 2012-08-08 13:47:06.000000000 +0000 @@ -1,29 +1,20 @@ -source("../../shared/qtcreator.py") -source("../../shared/suites_qtta.py") - -def verifyCurrentLine(editorArea, currentLineExpectedText): - verifyMessage = "Verifying split initializer functionality at element line." - currentLineText = str(lineUnderCursor(editorArea)).strip(); - return test.compare(currentLineText, currentLineExpectedText, verifyMessage) +source("../shared/qmls.py") def main(): - startApplication("qtcreator" + SettingsPath) - # create qt quick application - createNewQtQuickApplication(tempDir(), "SampleApp") - # open qml file - doubleClickItem(":Qt Creator_Utils::NavigationTreeView", "SampleApp.QML.qml/SampleApp.main\\.qml", 5, 5, 0, Qt.LeftButton) - # get editor - editorArea = waitForObject(":Qt Creator_QmlJSEditor::QmlJSTextEditorWidget") - # prepare code for test - type one-line element - if not placeCursorToLine(editorArea, "Text {"): - invokeMenuItem("File", "Exit") + editorArea = startQtCreatorWithNewAppAtQMLEditor(tempDir(), "SampleApp", "Text {") + if not editorArea: return - moveTextCursor(editorArea, QTextCursor.StartOfLine, QTextCursor.MoveAnchor) + homeKey = "" + if platform.system() == "Darwin": + homeKey = "" + for i in range(2): + type(editorArea, homeKey) type(editorArea, "") - moveTextCursor(editorArea, QTextCursor.Up, QTextCursor.MoveAnchor) + type(editorArea, "") type(editorArea, "") type(editorArea, "Item { x: 10; y: 20; width: 10 }") - moveTextCursor(editorArea, QTextCursor.Left, QTextCursor.MoveAnchor, 30) + for i in range(30): + type(editorArea, "") invokeMenuItem("File", "Save All") # activate menu and apply 'Refactoring - Split initializer' numLinesExpected = len(str(editorArea.plainText).splitlines()) + 4 @@ -33,16 +24,10 @@ # wait until refactoring ended waitFor("len(str(editorArea.plainText).splitlines()) == numLinesExpected", 5000) # verify if refactoring was properly applied - each part on separate line - verifyCurrentLine(editorArea, "Item {") - moveTextCursor(editorArea, QTextCursor.Down, QTextCursor.MoveAnchor, 1) - verifyCurrentLine(editorArea, "x: 10;") - moveTextCursor(editorArea, QTextCursor.Down, QTextCursor.MoveAnchor, 1) - verifyCurrentLine(editorArea, "y: 20;") - moveTextCursor(editorArea, QTextCursor.Down, QTextCursor.MoveAnchor, 1) - verifyCurrentLine(editorArea, "width: 10") - moveTextCursor(editorArea, QTextCursor.Down, QTextCursor.MoveAnchor, 1) - verifyCurrentLine(editorArea, "}") - moveTextCursor(editorArea, QTextCursor.Down, QTextCursor.MoveAnchor, 1) + verifyMessage = "Verifying split initializer functionality at element line." + for line in ["Item {", "x: 10;", "y: 20;", "width: 10", "}"]: + verifyCurrentLine(editorArea, line, verifyMessage) + type(editorArea, "") #save and exit invokeMenuItem("File", "Save All") invokeMenuItem("File", "Exit") diff -Nru qtcreator-2.5.0/tests/system/suite_QMLS/tst_QMLS06/test.py qtcreator-2.5.2/tests/system/suite_QMLS/tst_QMLS06/test.py --- qtcreator-2.5.0/tests/system/suite_QMLS/tst_QMLS06/test.py 1970-01-01 00:00:00.000000000 +0000 +++ qtcreator-2.5.2/tests/system/suite_QMLS/tst_QMLS06/test.py 2012-08-08 13:47:06.000000000 +0000 @@ -0,0 +1,41 @@ +source("../shared/qmls.py") + +def main(): + editorArea = startQtCreatorWithNewAppAtQMLEditor(tempDir(), "SampleApp", "}") + if not editorArea: + return + type(editorArea, "") + testingItemText = "Item { x: 10; y: 20; width: 10 }" + type(editorArea, testingItemText) + for i in range(30): + type(editorArea, "") + invokeMenuItem("File", "Save All") + # invoke Refactoring - Wrap Component in Loader + numLinesExpected = len(str(editorArea.plainText).splitlines()) + 10 + ctxtMenu = openContextMenuOnTextCursorPosition(editorArea) + activateItem(waitForObjectItem(objectMap.realName(ctxtMenu), "Refactoring")) + activateItem(waitForObjectItem(objectMap.realName(ctxtMenu), "Wrap Component in Loader")) + # wait until refactoring ended + waitFor("len(str(editorArea.plainText).splitlines()) >= numLinesExpected", 5000) + # verify if refactoring was properly applied + verifyMessage = "Verifying wrap component in loader functionality at element line." + type(editorArea, "") + type(editorArea, "") + verifyCurrentLine(editorArea, "Component {", verifyMessage) + type(editorArea, "") + verifyCurrentLine(editorArea, "id: component_Item", verifyMessage) + type(editorArea, "") + verifyCurrentLine(editorArea, testingItemText, verifyMessage) + type(editorArea, "") + verifyCurrentLine(editorArea, "}", verifyMessage) + type(editorArea, "") + verifyCurrentLine(editorArea, "Loader {", verifyMessage) + type(editorArea, "") + verifyCurrentLine(editorArea, "id: loader_Item", verifyMessage) + type(editorArea, "") + verifyCurrentLine(editorArea, "sourceComponent: component_Item", verifyMessage) + type(editorArea, "") + verifyCurrentLine(editorArea, "}", verifyMessage) + # save and exit + invokeMenuItem("File", "Save All") + invokeMenuItem("File", "Exit") diff -Nru qtcreator-2.5.0/tests/system/suite_QMLS/tst_QMLS07/test.py qtcreator-2.5.2/tests/system/suite_QMLS/tst_QMLS07/test.py --- qtcreator-2.5.0/tests/system/suite_QMLS/tst_QMLS07/test.py 1970-01-01 00:00:00.000000000 +0000 +++ qtcreator-2.5.2/tests/system/suite_QMLS/tst_QMLS07/test.py 2012-08-08 13:47:06.000000000 +0000 @@ -0,0 +1,25 @@ +source("../shared/qmls.py") + +def main(): + editorArea = startQtCreatorWithNewAppAtQMLEditor(tempDir(), "SampleApp", "Rectangle {") + if not editorArea: + return + type(editorArea, "") + type(editorArea, "Color") + for i in range(3): + type(editorArea, "") + invokeMenuItem("File", "Save All") + # invoke Refactoring - Add a message suppression comment. + numLinesExpected = len(str(editorArea.plainText).splitlines()) + 1 + ctxtMenu = openContextMenuOnTextCursorPosition(editorArea) + activateItem(waitForObjectItem(objectMap.realName(ctxtMenu), "Refactoring")) + activateItem(waitForObjectItem(objectMap.realName(ctxtMenu), "Add a comment to suppress this message")) + # wait until refactoring ended + waitFor("len(str(editorArea.plainText).splitlines()) >= numLinesExpected", 5000) + # verify if refactoring was properly applied + type(editorArea, "") + test.compare(str(lineUnderCursor(editorArea)).strip(), "// @disable-check M16", + "Verifying 'Add comment to suppress message' refactoring") + # save and exit + invokeMenuItem("File", "Save All") + invokeMenuItem("File", "Exit") diff -Nru qtcreator-2.5.0/tests/system/suite_QMLS/tst_QMLS08/test.py qtcreator-2.5.2/tests/system/suite_QMLS/tst_QMLS08/test.py --- qtcreator-2.5.0/tests/system/suite_QMLS/tst_QMLS08/test.py 1970-01-01 00:00:00.000000000 +0000 +++ qtcreator-2.5.2/tests/system/suite_QMLS/tst_QMLS08/test.py 2012-08-08 13:47:06.000000000 +0000 @@ -0,0 +1,60 @@ +source("../shared/qmls.py") + +def verifyNextLineIndented(editorArea, expectedIndentation): + type(editorArea, "") + rawLine = str(lineUnderCursor(editorArea)) + indentedLine = rawLine.rstrip() + # handle special case if line has only whitespaces inside + if not indentedLine: + indentedLine = rawLine + unindentedLine = indentedLine.lstrip() + numSpaces = len(indentedLine) - len(unindentedLine) + # verify if indentation is valid (multiply of 4) and if has correct depth level + test.compare(numSpaces % 4, 0, "Verifying indentation spaces") + test.compare(numSpaces / 4, expectedIndentation, "Verifying indentation level") + +def verifyIndentation(editorArea): + #verify indentation + if not placeCursorToLine(editorArea, "id: rect"): + invokeMenuItem("File", "Save All") + invokeMenuItem("File", "Exit") + return False + type(editorArea, "") + expectedIndentations = [1,1,1,2,2,2,2,3,3,3,4,3,3,2,1] + for indentation in expectedIndentations: + verifyNextLineIndented(editorArea, indentation) + return True + +def main(): + editorArea = startQtCreatorWithNewAppAtQMLEditor(tempDir(), "SampleApp") + if not editorArea: + return + # prepare code for test - insert unindented code + lines = ['id: rect', 'property bool random: true', 'Text{', + 'anchors.bottom:parent.bottom', 'text: rect.random ? getRandom() : "I\'m fixed."', + '', 'function getRandom(){', 'var result="I\'m random: ";', + 'var chars="ABCDEFGHIJKLMNOPQRSTUVWXYZabcdefghijklmnopqrstuvwxyz0123456789";', + 'for(var i=0;i<8;++i)', 'result += chars.charAt(Math.floor(Math.random() * chars.length));', + 'return result + ".";'] + if not placeCursorToLine(editorArea, "Rectangle {"): + invokeMenuItem("File", "Exit") + return + type(editorArea, "") + typeLines(editorArea, lines) + # verify default indentation + if not verifyIndentation(editorArea): + return + # cancel indentation + type(editorArea, "") + for i in range(5): + type(editorArea, "") + # select unindented block + type(editorArea, "") + # do indentation + type(editorArea, "") + # verify invoked indentation + if not verifyIndentation(editorArea): + return + # save and exit + invokeMenuItem("File", "Save All") + invokeMenuItem("File", "Exit") diff -Nru qtcreator-2.5.0/tests/system/suite_SCOM/tst_SCOM01/test.py qtcreator-2.5.2/tests/system/suite_SCOM/tst_SCOM01/test.py --- qtcreator-2.5.0/tests/system/suite_SCOM/tst_SCOM01/test.py 2012-05-09 12:13:19.000000000 +0000 +++ qtcreator-2.5.2/tests/system/suite_SCOM/tst_SCOM01/test.py 2012-08-08 13:47:06.000000000 +0000 @@ -7,7 +7,10 @@ # create qt quick application createNewQtQuickApplication(tempDir(), "SampleApp") # build it - on all (except Qt 4.7.0 (would fail)) build configurations - for config in iterateBuildConfigs(1, 0, "(?!.*4\.7\.0.*)"): + availableConfigs = iterateBuildConfigs(1, 0, "(?!.*4\.7\.0.*)") + if not availableConfigs: + test.fatal("Haven't found a suitable Qt version (anything except Qt 4.7.0) - leaving without building.") + for config in availableConfigs: selectBuildConfig(1, 0, config) # try to compile test.log("Testing build configuration: " + config) diff -Nru qtcreator-2.5.0/tests/system/suite_SCOM/tst_SCOM04/test.py qtcreator-2.5.2/tests/system/suite_SCOM/tst_SCOM04/test.py --- qtcreator-2.5.0/tests/system/suite_SCOM/tst_SCOM04/test.py 2012-05-09 12:13:19.000000000 +0000 +++ qtcreator-2.5.2/tests/system/suite_SCOM/tst_SCOM04/test.py 2012-08-08 13:47:06.000000000 +0000 @@ -17,7 +17,10 @@ # save all invokeMenuItem("File", "Save All") # build it - on all (except Qt 4.7.0 (would fail)) build configurations - for config in iterateBuildConfigs(1, 0, "(?!.*4\.7\.0.*)"): + availableConfigs = iterateBuildConfigs(1, 0, "(?!.*4\.7\.0.*)") + if not availableConfigs: + test.fatal("Haven't found a suitable Qt version (anything except Qt 4.7.0) - leaving without building.") + for config in availableConfigs: selectBuildConfig(1, 0, config) # try to compile test.log("Testing build configuration: " + config) diff -Nru qtcreator-2.5.0/tests/system/suite_debugger/suite.conf qtcreator-2.5.2/tests/system/suite_debugger/suite.conf --- qtcreator-2.5.0/tests/system/suite_debugger/suite.conf 2012-05-09 12:13:19.000000000 +0000 +++ qtcreator-2.5.2/tests/system/suite_debugger/suite.conf 2012-08-08 13:47:06.000000000 +0000 @@ -7,6 +7,6 @@ IMPLICITAUTSTART=0 LANGUAGE=Python OBJECTMAP=../objects.map -TEST_CASES=tst_build_new_project tst_cli_output_console +TEST_CASES=tst_build_new_project tst_cli_output_console tst_simple_debug VERSION=2 WRAPPERS=Qt diff -Nru qtcreator-2.5.0/tests/system/suite_debugger/tst_build_new_project/test.py qtcreator-2.5.2/tests/system/suite_debugger/tst_build_new_project/test.py --- qtcreator-2.5.0/tests/system/suite_debugger/tst_build_new_project/test.py 2012-05-09 12:13:19.000000000 +0000 +++ qtcreator-2.5.2/tests/system/suite_debugger/tst_build_new_project/test.py 2012-08-08 13:47:06.000000000 +0000 @@ -6,7 +6,10 @@ def main(): startApplication("qtcreator" + SettingsPath) createProject_Qt_Console(projectsPath, project) - for config in iterateBuildConfigs(1, 0): + availableConfigs = iterateBuildConfigs(1, 0) + if not availableConfigs: + test.fatal("Haven't found a suitable Qt version - leaving without building.") + for config in availableConfigs: selectBuildConfig(1, 0, config) test.log("Testing build configuration: " + config) runAndCloseApp() diff -Nru qtcreator-2.5.0/tests/system/suite_debugger/tst_cli_output_console/test.py qtcreator-2.5.2/tests/system/suite_debugger/tst_cli_output_console/test.py --- qtcreator-2.5.0/tests/system/suite_debugger/tst_cli_output_console/test.py 2012-05-09 12:13:19.000000000 +0000 +++ qtcreator-2.5.2/tests/system/suite_debugger/tst_cli_output_console/test.py 2012-08-08 13:47:06.000000000 +0000 @@ -37,7 +37,10 @@ test.verify("CONFIG += console" in str(proEditor.plainText), "Verifying that program is configured with console") setRunInTerminal(1, 0, False) - for config in iterateBuildConfigs(1, 0): + availableConfigs = iterateBuildConfigs(1, 0) + if not availableConfigs: + test.fatal("Haven't found a suitable Qt version - leaving without building.") + for config in availableConfigs: selectBuildConfig(1, 0, config) test.log("Testing build configuration: " + config) diff -Nru qtcreator-2.5.0/tests/system/suite_debugger/tst_simple_debug/test.py qtcreator-2.5.2/tests/system/suite_debugger/tst_simple_debug/test.py --- qtcreator-2.5.0/tests/system/suite_debugger/tst_simple_debug/test.py 2012-05-09 12:13:19.000000000 +0000 +++ qtcreator-2.5.2/tests/system/suite_debugger/tst_simple_debug/test.py 2012-08-08 13:47:06.000000000 +0000 @@ -5,7 +5,7 @@ def main(): global workingDir startApplication("qtcreator" + SettingsPath) - if not checkDebuggingLibrary("4.8.0", [QtQuickConstants.Targets.DESKTOP]): + if not checkDebuggingLibrary("4.7.4", [QtQuickConstants.Targets.DESKTOP]): test.fatal("Error while checking debugging libraries - leaving this test.") invokeMenuItem("File", "Exit") return @@ -20,23 +20,27 @@ if placeCursorToLine(editor, "MouseArea.*", True): type(editor, '') type(editor, '') - type(editor, 'Component.onCompleted: console.log("Break here")') + typeLines(editor, ['Timer {', + 'interval: 1000', + 'running: true', + 'onTriggered: console.log("Break here")']) invokeMenuItem("File", "Save All") - filesAndLines = { - "%s.QML.qml/%s.main\\.qml" % (projectName,projectName) : 'Component.onCompleted.*', - "%s.Sources.main\\.cpp" % projectName : "viewer.setOrientation\\(.+\\);" - } + filesAndLines = [ + { "%s.QML.qml/%s.main\\.qml" % (projectName,projectName) : 'onTriggered.*' }, + { "%s.Sources.main\\.cpp" % projectName : "viewer.setOrientation\\(.+\\);" } + ] test.log("Setting breakpoints") result = setBreakpointsForCurrentProject(filesAndLines) if result: - expectedBreakpointsOrder = [{"main.cpp":9}, {"main.qml":11}] - availableConfigs = iterateBuildConfigs(1, 0, ".*4\.8(\.\d+)?.*$(?= 4.8) - leaving without debugging.") + test.fatal("Haven't found a suitable Qt version (need Qt 4.7.4) - leaving without debugging.") for config in availableConfigs: test.log("Selecting '%s' as build config" % config) selectBuildConfig(1, 0, config) - verifyBuildConfig(1, 0, True) + verifyBuildConfig(1, 0, True, enableQmlDebug=True) # explicitly build before start debugging for adding the executable as allowed program to WinFW invokeMenuItem("Build", "Rebuild All") waitForSignal("{type='ProjectExplorer::BuildManager' unnamed='1'}", diff -Nru qtcreator-2.5.0/tests/system/suite_general/shared/testdata/creator_tree.tsv qtcreator-2.5.2/tests/system/suite_general/shared/testdata/creator_tree.tsv --- qtcreator-2.5.0/tests/system/suite_general/shared/testdata/creator_tree.tsv 2012-05-09 12:13:19.000000000 +0000 +++ qtcreator-2.5.2/tests/system/suite_general/shared/testdata/creator_tree.tsv 1970-01-01 00:00:00.000000000 +0000 @@ -1 +0,0 @@ -"node" "value" diff -Nru qtcreator-2.5.0/tests/system/suite_general/shared/testdata/qt_tree.tsv qtcreator-2.5.2/tests/system/suite_general/shared/testdata/qt_tree.tsv --- qtcreator-2.5.0/tests/system/suite_general/shared/testdata/qt_tree.tsv 2012-05-09 12:13:19.000000000 +0000 +++ qtcreator-2.5.2/tests/system/suite_general/shared/testdata/qt_tree.tsv 1970-01-01 00:00:00.000000000 +0000 @@ -1,3 +0,0 @@ -"node" "value" -":projects.projects.pro_QModelIndex" "projects.pro" -":projects_QModelIndex" "projects" diff -Nru qtcreator-2.5.0/tests/system/suite_general/shared/testdata/speedcrunch_tree.tsv qtcreator-2.5.2/tests/system/suite_general/shared/testdata/speedcrunch_tree.tsv --- qtcreator-2.5.0/tests/system/suite_general/shared/testdata/speedcrunch_tree.tsv 2012-05-09 12:13:19.000000000 +0000 +++ qtcreator-2.5.2/tests/system/suite_general/shared/testdata/speedcrunch_tree.tsv 1970-01-01 00:00:00.000000000 +0000 @@ -1 +0,0 @@ -"node" "value" diff -Nru qtcreator-2.5.0/tests/system/suite_general/tst_build_speedcrunch/test.py qtcreator-2.5.2/tests/system/suite_general/tst_build_speedcrunch/test.py --- qtcreator-2.5.0/tests/system/suite_general/tst_build_speedcrunch/test.py 2012-05-09 12:13:19.000000000 +0000 +++ qtcreator-2.5.2/tests/system/suite_general/tst_build_speedcrunch/test.py 2012-08-08 13:47:06.000000000 +0000 @@ -18,16 +18,13 @@ openQmakeProject(SpeedCrunchPath) waitForSignal("{type='CppTools::Internal::CppModelManager' unnamed='1'}", "sourceFilesRefreshed(QStringList)") - # Test that some of the expected items are in the navigation tree - for row, record in enumerate(testData.dataset("speedcrunch_tree.tsv")): - node = testData.field(record, "node") - value = testData.field(record, "value") - test.compare(waitForObject(node).text, value) - fancyToolButton = waitForObject(":*Qt Creator_Core::Internal::FancyToolButton") qtVerPattern = re.compile("\d\.\d(\.\d+)?") - for config in iterateBuildConfigs(1, 0, "(Desktop )?Qt.*Release"): + availableConfigs = iterateBuildConfigs(1, 0, "(Desktop )?Qt.*Release") + if not availableConfigs: + test.fatal("Haven't found a suitable Qt version (need Release build) - leaving without building.") + for config in availableConfigs: qtVersion = qtVerPattern.search(config) if qtVersion: qtVersion = qtVersion.group() diff -Nru qtcreator-2.5.0/tests/system/suite_general/tst_cmake_speedcrunch/test.py qtcreator-2.5.2/tests/system/suite_general/tst_cmake_speedcrunch/test.py --- qtcreator-2.5.0/tests/system/suite_general/tst_cmake_speedcrunch/test.py 2012-05-09 12:13:19.000000000 +0000 +++ qtcreator-2.5.2/tests/system/suite_general/tst_cmake_speedcrunch/test.py 2012-08-08 13:47:06.000000000 +0000 @@ -23,12 +23,6 @@ return waitForSignal("{type='CppTools::Internal::CppModelManager' unnamed='1'}", "sourceFilesRefreshed(QStringList)") - # Test that some of the expected items are in the navigation tree - for row, record in enumerate(testData.dataset("speedcrunch_tree.tsv")): - node = testData.field(record, "node") - value = testData.field(record, "value") - test.compare(findObject(node).text, value) - # Invoke a rebuild of the application invokeMenuItem("Build", "Rebuild All") diff -Nru qtcreator-2.5.0/tests/system/suite_general/tst_openqt_creator/test.py qtcreator-2.5.2/tests/system/suite_general/tst_openqt_creator/test.py --- qtcreator-2.5.0/tests/system/suite_general/tst_openqt_creator/test.py 2012-05-09 12:13:19.000000000 +0000 +++ qtcreator-2.5.2/tests/system/suite_general/tst_openqt_creator/test.py 2012-08-08 13:47:06.000000000 +0000 @@ -15,16 +15,9 @@ # Wait for parsing to complete waitForSignal("{type='CppTools::Internal::CppModelManager' unnamed='1'}", "sourceFilesRefreshed(QStringList)", 300000) - # Test that some of the expected items are in the navigation tree - for row, record in enumerate(testData.dataset("creator_tree.tsv")): - node = testData.field(record, "node") - value = testData.field(record, "value") - test.compare(waitForObject(node).text, value) - - for row, record in enumerate(testData.dataset("speedcrunch_tree.tsv")): - node = testData.field(record, "node") - value = testData.field(record, "value") - test.compare(waitForObject(node).text, value) + naviTreeView = "{column='0' container=':Qt Creator_Utils::NavigationTreeView' text='%s' type='QModelIndex'}" + compareProjectTree(naviTreeView % "speedcrunch", "projecttree_speedcrunch.tsv") + compareProjectTree(naviTreeView % "qtcreator", "projecttree_creator.tsv") # Now check some basic lookups in the search box selectFromLocator(": Qlist::QList", "QList::QList") diff -Nru qtcreator-2.5.0/tests/system/suite_general/tst_openqt_creator/testdata/projecttree_creator.tsv qtcreator-2.5.2/tests/system/suite_general/tst_openqt_creator/testdata/projecttree_creator.tsv --- qtcreator-2.5.0/tests/system/suite_general/tst_openqt_creator/testdata/projecttree_creator.tsv 1970-01-01 00:00:00.000000000 +0000 +++ qtcreator-2.5.2/tests/system/suite_general/tst_openqt_creator/testdata/projecttree_creator.tsv 2012-08-08 13:47:06.000000000 +0000 @@ -0,0 +1,8262 @@ +"text" "nestinglevel" +"qtcreator.pro" "0" +"bin" "0" +"bin.pro" "1" +"Other files" "1" +"qtcreator.sh" "2" +"doc" "0" +"doc.pri" "1" +"Other files" "1" +"api" "2" +"coding-style.qdoc" "3" +"creating-plugins.qdoc" "3" +"external-tool-spec.qdoc" "3" +"first-plugin.qdoc" "3" +"getting-and-building.qdoc" "3" +"plugin-lifecycle.qdoc" "3" +"plugin-specifications.qdoc" "3" +"qtcreator-api.qdoc" "3" +"qtcreator-dev-wizards.qdoc" "3" +"qtcreator-dev.qdoc" "3" +"qtcreator-dev.qdocconf" "3" +"config" "2" +"compat.qdocconf" "3" +"macros.qdocconf" "3" +"qt-cpp-ignore.qdocconf" "3" +"qt-defines.qdocconf" "3" +"qt-html-default-styles.qdocconf" "3" +"qt-html-templates.qdocconf" "3" +"src" "2" +"qtcreator.qdoc" "3" +"addressbook-sdk.qdoc" "2" +"qtcreator.qdocconf" "2" +"qtcomponents" "0" +"qtcomponents.pro" "1" +"qtcreator" "1" +"qtcreator.pri" "2" +"QML" "1" +"custom" "2" +"behaviors" "3" +"ButtonBehavior.qml" "4" +"ModalPopupBehavior.qml" "4" +"private" "3" +"ChoiceListPopup.qml" "4" +"BasicButton.qml" "3" +"Button.qml" "3" +"ButtonColumn.qml" "3" +"ButtonRow.qml" "3" +"CheckBox.qml" "3" +"ChoiceList.qml" "3" +"GroupBox.qml" "3" +"ProgressBar.qml" "3" +"Slider.qml" "3" +"SpinBox.qml" "3" +"SplitterRow.qml" "3" +"TextField.qml" "3" +"Button.qml" "2" +"ButtonRow.qml" "2" +"CheckBox.qml" "2" +"ChoiceList.qml" "2" +"ContextMenu.qml" "2" +"Dial.qml" "2" +"Frame.qml" "2" +"GroupBox.qml" "2" +"Menu.qml" "2" +"MenuItem.qml" "2" +"ProgressBar.qml" "2" +"RadioButton.qml" "2" +"ScrollArea.qml" "2" +"ScrollBar.qml" "2" +"Slider.qml" "2" +"SpinBox.qml" "2" +"SplitterRow.qml" "2" +"Switch.qml" "2" +"Tab.qml" "2" +"TabBar.qml" "2" +"TabFrame.qml" "2" +"TableColumn.qml" "2" +"TableView.qml" "2" +"TextArea.qml" "2" +"TextField.qml" "2" +"ToolBar.qml" "2" +"ToolButton.qml" "2" +"Other files" "1" +"custom" "2" +"ButtonGroup.js" "3" +"components.pro" "3" +"qmldir" "3" +"images" "2" +"folder_new.png" "3" +"qmldir" "2" +"qtcreator" "0" +"qtcreator.pri" "1" +"share" "0" +"share.pro" "1" +"static" "1" +"static.pro" "2" +"qtcreator" "2" +"qtcreator.pri" "3" +"QML" "2" +"qml/qmlobserver" "3" +"browser" "4" +"Browser.qml" "5" +"startup" "4" +"Logo.qml" "5" +"startup.qml" "5" +"qmldesigner/propertyeditor" "3" +"Qt" "4" +"AlignmentHorizontalButtons.qml" "5" +"AlignmentVerticalButtons.qml" "5" +"AnchorBox.qml" "5" +"AnchorButtons.qml" "5" +"BorderImageSpecifics.qml" "5" +"CheckBox.qml" "5" +"ColorGroupBox.qml" "5" +"ColorLabel.qml" "5" +"ColorScheme.qml" "5" +"ColorTypeButtons.qml" "5" +"ComboBox.qml" "5" +"DoubleSpinBox.qml" "5" +"DoubleSpinBoxAlternate.qml" "5" +"emptyPane.qml" "5" +"ExpressionEditor.qml" "5" +"Extended.qml" "5" +"ExtendedFunctionButton.qml" "5" +"ExtendedPane.qml" "5" +"ExtendedSwitches.qml" "5" +"FlagedButton.qml" "5" +"FlickableGroupBox.qml" "5" +"FlickableSpecifics.qml" "5" +"FlipableSpecifics.qml" "5" +"FlowSpecifics.qml" "5" +"FontComboBox.qml" "5" +"FontGroupBox.qml" "5" +"FontStyleButtons.qml" "5" +"Geometry.qml" "5" +"GridSpecifics.qml" "5" +"GridViewSpecifics.qml" "5" +"GroupBox.qml" "5" +"GroupBoxOption.qml" "5" +"HorizontalLayout.qml" "5" +"HorizontalWhiteLine.qml" "5" +"ImageSpecifics.qml" "5" +"IntEditor.qml" "5" +"ItemPane.qml" "5" +"Label.qml" "5" +"Layout.qml" "5" +"LayoutPane.qml" "5" +"LineEdit.qml" "5" +"ListViewSpecifics.qml" "5" +"Modifiers.qml" "5" +"MouseAreaSpecifics.qml" "5" +"PathViewSpecifics.qml" "5" +"PlaceHolder.qml" "5" +"PropertyFrame.qml" "5" +"RectangleColorGroupBox.qml" "5" +"RectangleSpecifics.qml" "5" +"RowSpecifics.qml" "5" +"ScrollArea.qml" "5" +"SliderWidget.qml" "5" +"SpinBox.qml" "5" +"StandardTextColorGroupBox.qml" "5" +"StandardTextGroupBox.qml" "5" +"Switches.qml" "5" +"TextEditor.qml" "5" +"TextEditSpecifics.qml" "5" +"TextInputGroupBox.qml" "5" +"TextInputSpecifics.qml" "5" +"TextSpecifics.qml" "5" +"Transformation.qml" "5" +"Type.qml" "5" +"VerticalLayout.qml" "5" +"Visibility.qml" "5" +"QtWebKit" "4" +"WebViewSpecifics.qml" "5" +"templates/qtquickapp/qml/app" "3" +"meego10" "4" +"main.qml" "5" +"MainPage.qml" "5" +"qtquick10" "4" +"main.qml" "5" +"symbian11" "4" +"main.qml" "5" +"MainPage.qml" "5" +"welcomescreen" "3" +"dummydata" "4" +"examplesModel.qml" "5" +"pagesModel.qml" "5" +"projectList.qml" "5" +"sessionList.qml" "5" +"tutorialsModel.qml" "5" +"widgets" "4" +"dummydata" "5" +"context" "6" +"ExampleDelegate.qml" "7" +"ExampleGridView.qml" "7" +"examplesModel.qml" "6" +"mockupTags.qml" "6" +"pagesModel.qml" "6" +"tabsModel.qml" "6" +"CustomColors.qml" "5" +"CustomFonts.qml" "5" +"CustomizedGridView.qml" "5" +"CustomTab.qml" "5" +"Delegate.qml" "5" +"Feedback.qml" "5" +"GettingStartedItem.qml" "5" +"LinkedText.qml" "5" +"LinksBar.qml" "5" +"Logo.qml" "5" +"PageCaption.qml" "5" +"PageLoader.qml" "5" +"ProjectItem.qml" "5" +"RecentProjects.qml" "5" +"SearchBar.qml" "5" +"SessionItem.qml" "5" +"Sessions.qml" "5" +"ToolTip.qml" "5" +"develop.qml" "4" +"examples.qml" "4" +"gettingstarted.qml" "4" +"tutorials.qml" "4" +"welcomescreen.qml" "4" +"Other files" "2" +"designer" "3" +"templates" "4" +"Dialog_with_Buttons_Bottom.ui" "5" +"Dialog_with_Buttons_Right.ui" "5" +"Dialog_without_Buttons.ui" "5" +"Main_Window.ui" "5" +"Widget.ui" "5" +"templates.xml" "4" +"dumper" "3" +"test" "4" +"dumpertest.pro" "5" +"main.cpp" "5" +"bridge.py" "4" +"dumper.cpp" "4" +"dumper.h" "4" +"dumper.pro" "4" +"dumper.py" "4" +"dumper_p.h" "4" +"LGPL_EXCEPTION.TXT" "4" +"LICENSE.LGPL" "4" +"pdumper.py" "4" +"qttypes.py" "4" +"generic-highlighter" "3" +"alert.xml" "4" +"autoconf.xml" "4" +"bash.xml" "4" +"cmake.xml" "4" +"css.xml" "4" +"doxygen.xml" "4" +"dtd.xml" "4" +"html.xml" "4" +"ini.xml" "4" +"java.xml" "4" +"javadoc.xml" "4" +"perl.xml" "4" +"ruby.xml" "4" +"valgrind-suppression.xml" "4" +"xml.xml" "4" +"yacc.xml" "4" +"glsl" "3" +"glsl_120.frag" "4" +"glsl_120.vert" "4" +"glsl_120_common.glsl" "4" +"glsl_es_100.frag" "4" +"glsl_es_100.vert" "4" +"glsl_es_100_common.glsl" "4" +"qml" "3" +"qmldump" "4" +"LGPL_EXCEPTION.TXT" "5" +"LICENSE.LGPL" "5" +"main.cpp" "5" +"qmldump.pro" "5" +"qmlstreamwriter.cpp" "5" +"qmlstreamwriter.h" "5" +"qmljsdebugger" "4" +"editor" "5" +"abstractliveedittool.cpp" "6" +"abstractliveedittool.h" "6" +"boundingrecthighlighter.cpp" "6" +"boundingrecthighlighter.h" "6" +"colorpickertool.cpp" "6" +"colorpickertool.h" "6" +"livelayeritem.cpp" "6" +"livelayeritem.h" "6" +"liverubberbandselectionmanipulator.cpp" "6" +"liverubberbandselectionmanipulator.h" "6" +"liveselectionindicator.cpp" "6" +"liveselectionindicator.h" "6" +"liveselectionrectangle.cpp" "6" +"liveselectionrectangle.h" "6" +"liveselectiontool.cpp" "6" +"liveselectiontool.h" "6" +"livesingleselectionmanipulator.cpp" "6" +"livesingleselectionmanipulator.h" "6" +"subcomponentmasklayeritem.cpp" "6" +"subcomponentmasklayeritem.h" "6" +"zoomtool.cpp" "6" +"zoomtool.h" "6" +"include" "5" +"qt_private" "6" +"qdeclarativedebughelper_p.h" "7" +"qdeclarativedebugservice_p.h" "7" +"qdeclarativestate_p.h" "7" +"jsdebuggeragent.h" "6" +"qdeclarativeinspectorservice.h" "6" +"qdeclarativeviewinspector.h" "6" +"qdeclarativeviewobserver.h" "6" +"qmlinspectorconstants.h" "6" +"qmljsdebugger_global.h" "6" +"protocol" "5" +"inspectorprotocol.h" "6" +"protocol.pri" "6" +"jsdebuggeragent.cpp" "5" +"qdeclarativeinspectorservice.cpp" "5" +"qdeclarativeviewinspector.cpp" "5" +"qdeclarativeviewinspector_p.h" "5" +"qmljsdebugger-lib.pri" "5" +"qmljsdebugger-src.pri" "5" +"qmljsdebugger.pro" "5" +"qmlobserver" "4" +"browser" "5" +"images" "6" +"folder.png" "7" +"titlebar.png" "7" +"titlebar.sci" "7" +"up.png" "7" +"browser.qrc" "6" +"startup" "5" +"qt-back.png" "6" +"qt-blue.jpg" "6" +"qt-front.png" "6" +"qt-sketch.jpg" "6" +"qt-text.png" "6" +"quick-blur.png" "6" +"quick-regular.png" "6" +"shadow.png" "6" +"startup.qrc" "6" +"white-star.png" "6" +"deviceorientation.cpp" "5" +"deviceorientation.h" "5" +"deviceorientation_harmattan.cpp" "5" +"deviceorientation_maemo5.cpp" "5" +"deviceorientation_symbian.cpp" "5" +"LGPL_EXCEPTION.TXT" "5" +"LICENSE.LGPL" "5" +"loggerwidget.cpp" "5" +"loggerwidget.h" "5" +"main.cpp" "5" +"proxysettings.cpp" "5" +"proxysettings.h" "5" +"proxysettings.ui" "5" +"proxysettings_maemo5.ui" "5" +"qdeclarativetester.cpp" "5" +"qdeclarativetester.h" "5" +"qml.icns" "5" +"qml.pri" "5" +"qmlobserver.pro" "5" +"qmlruntime.cpp" "5" +"qmlruntime.h" "5" +"recopts.ui" "5" +"recopts_maemo5.ui" "5" +"texteditautoresizer_maemo5.h" "5" +"qmlpuppet" "4" +"commands" "5" +"changeauxiliarycommand.cpp" "6" +"changeauxiliarycommand.h" "6" +"changebindingscommand.cpp" "6" +"changebindingscommand.h" "6" +"changefileurlcommand.cpp" "6" +"changefileurlcommand.h" "6" +"changeidscommand.cpp" "6" +"changeidscommand.h" "6" +"changenodesourcecommand.cpp" "6" +"changenodesourcecommand.h" "6" +"changestatecommand.cpp" "6" +"changestatecommand.h" "6" +"changevaluescommand.cpp" "6" +"changevaluescommand.h" "6" +"childrenchangedcommand.cpp" "6" +"childrenchangedcommand.h" "6" +"clearscenecommand.cpp" "6" +"clearscenecommand.h" "6" +"commands.pri" "6" +"completecomponentcommand.cpp" "6" +"completecomponentcommand.h" "6" +"componentcompletedcommand.cpp" "6" +"componentcompletedcommand.h" "6" +"createinstancescommand.cpp" "6" +"createinstancescommand.h" "6" +"createscenecommand.cpp" "6" +"createscenecommand.h" "6" +"informationchangedcommand.cpp" "6" +"informationchangedcommand.h" "6" +"pixmapchangedcommand.cpp" "6" +"pixmapchangedcommand.h" "6" +"removeinstancescommand.cpp" "6" +"removeinstancescommand.h" "6" +"removepropertiescommand.cpp" "6" +"removepropertiescommand.h" "6" +"reparentinstancescommand.cpp" "6" +"reparentinstancescommand.h" "6" +"statepreviewimagechangedcommand.cpp" "6" +"statepreviewimagechangedcommand.h" "6" +"synchronizecommand.cpp" "6" +"synchronizecommand.h" "6" +"tokencommand.cpp" "6" +"tokencommand.h" "6" +"valueschangedcommand.cpp" "6" +"valueschangedcommand.h" "6" +"container" "5" +"addimportcontainer.cpp" "6" +"addimportcontainer.h" "6" +"container.pri" "6" +"idcontainer.cpp" "6" +"idcontainer.h" "6" +"imagecontainer.cpp" "6" +"imagecontainer.h" "6" +"informationcontainer.cpp" "6" +"informationcontainer.h" "6" +"instancecontainer.cpp" "6" +"instancecontainer.h" "6" +"propertyabstractcontainer.cpp" "6" +"propertyabstractcontainer.h" "6" +"propertybindingcontainer.cpp" "6" +"propertybindingcontainer.h" "6" +"propertyvaluecontainer.cpp" "6" +"propertyvaluecontainer.h" "6" +"reparentcontainer.cpp" "6" +"reparentcontainer.h" "6" +"html" "5" +"welcome.html" "6" +"images" "5" +"template_image.png" "6" +"webkit.png" "6" +"instances" "5" +"anchorchangesnodeinstance.cpp" "6" +"anchorchangesnodeinstance.h" "6" +"behaviornodeinstance.cpp" "6" +"behaviornodeinstance.h" "6" +"childrenchangeeventfilter.cpp" "6" +"childrenchangeeventfilter.h" "6" +"componentnodeinstance.cpp" "6" +"componentnodeinstance.h" "6" +"dummycontextobject.cpp" "6" +"dummycontextobject.h" "6" +"dummynodeinstance.cpp" "6" +"dummynodeinstance.h" "6" +"instances.pri" "6" +"nodeinstanceclientproxy.cpp" "6" +"nodeinstanceclientproxy.h" "6" +"nodeinstancemetaobject.cpp" "6" +"nodeinstancemetaobject.h" "6" +"nodeinstanceserver.cpp" "6" +"nodeinstanceserver.h" "6" +"nodeinstancesignalspy.cpp" "6" +"nodeinstancesignalspy.h" "6" +"objectnodeinstance.cpp" "6" +"objectnodeinstance.h" "6" +"qmlpropertychangesnodeinstance.cpp" "6" +"qmlpropertychangesnodeinstance.h" "6" +"qmlstatenodeinstance.cpp" "6" +"qmlstatenodeinstance.h" "6" +"qmltransitionnodeinstance.cpp" "6" +"qmltransitionnodeinstance.h" "6" +"servernodeinstance.cpp" "6" +"servernodeinstance.h" "6" +"interfaces" "5" +"commondefines.h" "6" +"interfaces.pri" "6" +"nodeinstanceclientinterface.h" "6" +"nodeinstanceserverinterface.cpp" "6" +"nodeinstanceserverinterface.h" "6" +"qml2puppet" "5" +"instances" "6" +"instances.pri" "7" +"qt5informationnodeinstanceserver.cpp" "7" +"qt5informationnodeinstanceserver.h" "7" +"qt5nodeinstanceclientproxy.cpp" "7" +"qt5nodeinstanceclientproxy.h" "7" +"qt5nodeinstanceserver.cpp" "7" +"qt5nodeinstanceserver.h" "7" +"qt5previewnodeinstanceserver.cpp" "7" +"qt5previewnodeinstanceserver.h" "7" +"qt5rendernodeinstanceserver.cpp" "7" +"qt5rendernodeinstanceserver.h" "7" +"sgitemnodeinstance.cpp" "7" +"sgitemnodeinstance.h" "7" +"main.cpp" "6" +"qml2puppet.pro" "6" +"qmlpuppet" "5" +"instances" "6" +"graphicsobjectnodeinstance.cpp" "7" +"graphicsobjectnodeinstance.h" "7" +"instances.pri" "7" +"positionernodeinstance.cpp" "7" +"positionernodeinstance.h" "7" +"qmlgraphicsitemnodeinstance.cpp" "7" +"qmlgraphicsitemnodeinstance.h" "7" +"qt4informationnodeinstanceserver.cpp" "7" +"qt4informationnodeinstanceserver.h" "7" +"qt4nodeinstanceclientproxy.cpp" "7" +"qt4nodeinstanceclientproxy.h" "7" +"qt4nodeinstanceserver.cpp" "7" +"qt4nodeinstanceserver.h" "7" +"qt4previewnodeinstanceserver.cpp" "7" +"qt4previewnodeinstanceserver.h" "7" +"qt4rendernodeinstanceserver.cpp" "7" +"qt4rendernodeinstanceserver.h" "7" +"main.cpp" "6" +"qmlpuppet.pri" "6" +"qmlpuppet.pro" "6" +"qmlpuppet.pro" "5" +"qmlpuppet.qrc" "5" +"qmlpuppet_utilities.pri" "5" +"qml-type-descriptions" "3" +"builtins.qmltypes" "4" +"qmlproject.qmltypes" "4" +"qmlruntime.qmltypes" "4" +"qt-labs-folderlistmodel.qmltypes" "4" +"qt-labs-gestures.qmltypes" "4" +"qt-labs-particles.qmltypes" "4" +"qtmobility-connectivity.qmltypes" "4" +"qtmobility-contacts.qmltypes" "4" +"qtmobility-feedback.qmltypes" "4" +"qtmobility-gallery.qmltypes" "4" +"qtmobility-location.qmltypes" "4" +"qtmobility-messaging.qmltypes" "4" +"qtmobility-organizer.qmltypes" "4" +"qtmobility-publishsubscribe.qmltypes" "4" +"qtmobility-sensors.qmltypes" "4" +"qtmobility-serviceframework.qmltypes" "4" +"qtmobility-systeminfo.qmltypes" "4" +"qtmultimediakit.qmltypes" "4" +"qtwebkit.qmltypes" "4" +"qmldesigner/propertyeditor/Qt" "3" +"images" "4" +"alignmentbottom-h-icon.png" "5" +"alignmentbottom-icon.png" "5" +"alignmentcenterh-h-icon.png" "5" +"alignmentcenterh-icon.png" "5" +"alignmentleft-h-icon.png" "5" +"alignmentleft-icon.png" "5" +"alignmentmiddle-h-icon.png" "5" +"alignmentmiddle-icon.png" "5" +"alignmentright-h-icon.png" "5" +"alignmentright-icon.png" "5" +"alignmenttop-h-icon.png" "5" +"alignmenttop-icon.png" "5" +"apply.png" "5" +"behaivour.png" "5" +"blended-image-icon.png" "5" +"bold-h-icon.png" "5" +"bold-icon.png" "5" +"button.png" "5" +"cancel.png" "5" +"default-icon.png" "5" +"downArrow.png" "5" +"expression.png" "5" +"extended.png" "5" +"grid-icon.png" "5" +"icon_color_gradient.png" "5" +"icon_color_none.png" "5" +"icon_color_solid.png" "5" +"image-icon.png" "5" +"italic-h-icon.png" "5" +"italic-icon.png" "5" +"item-icon.png" "5" +"layout.png" "5" +"leftArrow.png" "5" +"list-icon.png" "5" +"mouse-area-icon.png" "5" +"placeholder.png" "5" +"rect-icon.png" "5" +"reset-button.png" "5" +"rightArrow.png" "5" +"standard.png" "5" +"strikeout-h-icon.png" "5" +"strikeout-icon.png" "5" +"submenu.png" "5" +"text-edit-icon.png" "5" +"text-icon.png" "5" +"underline-h-icon.png" "5" +"underline-icon.png" "5" +"upArrow.png" "5" +"anchorbottom.css" "4" +"anchorbox.css" "4" +"anchorfill.css" "4" +"anchorhorizontal.css" "4" +"anchorleft.css" "4" +"anchorright.css" "4" +"anchorspacer.css" "4" +"anchortop.css" "4" +"anchorvertical.css" "4" +"applybutton.css" "4" +"aspectlock.css" "4" +"cancelbutton.css" "4" +"checkbox_tr.css" "4" +"layoutWidget.css" "4" +"propertyEditor.css" "4" +"specialCheckBox.css" "4" +"styledbuttonleft.css" "4" +"styledbuttonmiddle.css" "4" +"styledbuttonright.css" "4" +"switch.css" "4" +"typeLabel.css" "4" +"qmlicons" "3" +"Qt/16x16" "4" +"BorderImage.png" "5" +"BusyIndicator.png" "5" +"Button.png" "5" +"ButtonColumn.png" "5" +"ButtonRow.png" "5" +"CheckBox.png" "5" +"ChoiceList.png" "5" +"ColorAnimation.png" "5" +"Component.png" "5" +"CountBubble.png" "5" +"DatePickerDialog.png" "5" +"Flickable.png" "5" +"Flipable.png" "5" +"FocusScope.png" "5" +"GridView.png" "5" +"Image.png" "5" +"InfoBanner.png" "5" +"item-icon16.png" "5" +"Item.png" "5" +"ListButton.png" "5" +"ListDelegate.png" "5" +"ListView.png" "5" +"MoreIndicator.png" "5" +"MouseArea.png" "5" +"PageIndicator.png" "5" +"ParallelAnimation.png" "5" +"PathView.png" "5" +"PauseAnimation.png" "5" +"ProgressBar.png" "5" +"PropertyChanges.png" "5" +"RadioButton.png" "5" +"RatingIndicator.png" "5" +"Rectangle.png" "5" +"SequentialAnimation.png" "5" +"Slider.png" "5" +"State.png" "5" +"Switch.png" "5" +"TabBar.png" "5" +"TabButton.png" "5" +"Text.png" "5" +"TextArea.png" "5" +"TextEdit.png" "5" +"TextField.png" "5" +"TextInput.png" "5" +"TimePickerDialog.png" "5" +"ToolBar.png" "5" +"Transition.png" "5" +"Tumbler.png" "5" +"TumblerButton.png" "5" +"TumblerColumn.png" "5" +"TumblerDialog.png" "5" +"Window.png" "5" +"QtWebkit/16x16" "4" +"WebView.png" "5" +"schemes" "3" +"MS_Visual_C++.kms" "4" +"Xcode.kms" "4" +"snippets" "3" +"cpp.xml" "4" +"qml.xml" "4" +"text.xml" "4" +"styles" "3" +"darkvim.xml" "4" +"default.xml" "4" +"grayscale.xml" "4" +"inkpot.xml" "4" +"intellij.xml" "4" +"templates" "3" +"html5app" "4" +"html" "5" +"index.html" "6" +"html5applicationviewer" "5" +"touchnavigation" "6" +"navigationcontroller.cpp" "7" +"navigationcontroller.h" "7" +"touchnavigation.pri" "7" +"webnavigation.cpp" "7" +"webnavigation.h" "7" +"webtouchevent.cpp" "7" +"webtouchevent.h" "7" +"webtouchnavigation.cpp" "7" +"webtouchnavigation.h" "7" +"webtouchphysics.cpp" "7" +"webtouchphysics.h" "7" +"webtouchphysicsinterface.cpp" "7" +"webtouchphysicsinterface.h" "7" +"webtouchscroller.cpp" "7" +"webtouchscroller.h" "7" +"html5applicationviewer.cpp" "6" +"html5applicationviewer.h" "6" +"html5applicationviewer.pri" "6" +"app.pro" "5" +"main.cpp" "5" +"mobileapp" "4" +"app.pro" "5" +"main.cpp" "5" +"mainwindow.cpp" "5" +"mainwindow.h" "5" +"mainwindow.ui" "5" +"qt4project" "4" +"customwidgetwizard" "5" +"tpl_collection.cpp" "6" +"tpl_collection.h" "6" +"tpl_plugin.pro" "6" +"tpl_resources.qrc" "6" +"tpl_single.cpp" "6" +"tpl_single.h" "6" +"tpl_widget.cpp" "6" +"tpl_widget.h" "6" +"tpl_widget_include.pri" "6" +"tpl_widget_lib.pro" "6" +"main.cpp" "5" +"mywidget.cpp" "5" +"mywidget.h" "5" +"mywidget_form.cpp" "5" +"mywidget_form.h" "5" +"widget.ui" "5" +"qtquickapp" "4" +"qmlapplicationviewer" "5" +"qmlapplicationviewer.cpp" "6" +"qmlapplicationviewer.h" "6" +"qmlapplicationviewer.pri" "6" +"app.pro" "5" +"main.cpp" "5" +"shared" "4" +"app.desktop" "5" +"deployment.pri" "5" +"icon64.png" "5" +"icon80.png" "5" +"manifest.aegis" "5" +"symbianicon.svg" "5" +"wizards" "4" +"helloworld" "5" +"console.png" "6" +"main.cpp" "6" +"project.pro" "6" +"wizard_sample.xml" "6" +"listmodel" "5" +"listmodel.cpp" "6" +"listmodel.h" "6" +"wizard_sample.xml" "6" +"plaincapp" "5" +"console.png" "6" +"main.c" "6" +"project.pro" "6" +"wizard.xml" "6" +"plaincapp-cmake" "5" +"CMakeLists.txt" "6" +"console.png" "6" +"main.c" "6" +"wizard.xml" "6" +"plaincppapp" "5" +"console.png" "6" +"main.cpp" "6" +"project.pro" "6" +"wizard.xml" "6" +"plaincppapp-cmake" "5" +"CMakeLists.txt" "6" +"console.png" "6" +"main.cpp" "6" +"wizard.xml" "6" +"qml-extension" "5" +"lib.png" "6" +"object.cpp" "6" +"object.h" "6" +"plugin.cpp" "6" +"plugin.h" "6" +"project.pro" "6" +"qmldir" "6" +"wizard.xml" "6" +"qtcreatorplugin" "5" +"myplugin.cpp" "6" +"myplugin.h" "6" +"MyPlugin.pluginspec.in" "6" +"myplugin.pro" "6" +"myplugin_global.h" "6" +"mypluginconstants.h" "6" +"qtcreator_logo_24.png" "6" +"wizard.xml" "6" +"scriptgeneratedproject" "5" +"generate.pl" "6" +"wizard_sample.xml" "6" +"README.txt" "5" +"welcomescreen" "3" +"qtcreator/externaltools" "4" +"lrelease.xml" "5" +"lupdate.xml" "5" +"qmlviewer.xml" "5" +"sort.xml" "5" +"qtcreator" "4" +"translations" "1" +"translations.pro" "2" +"qtcreator" "2" +"qtcreator.pri" "3" +"src" "0" +"src.pro" "1" +"aggregation" "1" +"aggregation.pro" "2" +"qtcreator" "2" +"qtcreator.pri" "3" +"qtcreatorlibrary" "2" +"qtcreatorlibrary.pri" "3" +"rpath" "2" +"rpath.pri" "3" +"Headers" "2" +"aggregate.h" "3" +"aggregation_global.h" "3" +"Sources" "2" +"aggregate.cpp" "3" +"app" "1" +"app.pro" "2" +"qtcreator" "2" +"qtcreator.pri" "3" +"qtlockedfile" "2" +"qtlockedfile.pri" "3" +"Headers" "3" +"qtlockedfile.h" "4" +"Sources" "3" +"qtlockedfile.cpp" "4" +"qtlockedfile_unix.cpp" "4" +"qtlockedfile_win.cpp" "4" +"qtsingleapplication" "2" +"qtsingleapplication.pri" "3" +"Headers" "3" +"qtlocalpeer.h" "4" +"qtsingleapplication.h" "4" +"Sources" "3" +"qtlocalpeer.cpp" "4" +"qtsingleapplication.cpp" "4" +"rpath" "2" +"rpath.pri" "3" +"Sources" "2" +"main.cpp" "3" +"Other files" "2" +"app_version.h.in" "3" +"Info.plist.in" "3" +"qtcreator.icns" "3" +"qtcreator.rc" "3" +"extensionsystem" "1" +"extensionsystem.pro" "2" +"aggregation" "2" +"aggregation.pri" "3" +"extensionsystem_dependencies" "2" +"extensionsystem_dependencies.pri" "3" +"qtcreator" "2" +"qtcreator.pri" "3" +"qtcreatorlibrary" "2" +"qtcreatorlibrary.pri" "3" +"rpath" "2" +"rpath.pri" "3" +"Headers" "2" +"extensionsystem_global.h" "3" +"invoker.h" "3" +"iplugin.h" "3" +"iplugin_p.h" "3" +"optionsparser.h" "3" +"plugincollection.h" "3" +"plugindetailsview.h" "3" +"pluginerroroverview.h" "3" +"pluginerrorview.h" "3" +"pluginmanager.h" "3" +"pluginmanager_p.h" "3" +"pluginspec.h" "3" +"pluginspec_p.h" "3" +"pluginview.h" "3" +"pluginview_p.h" "3" +"Sources" "2" +"invoker.cpp" "3" +"iplugin.cpp" "3" +"optionsparser.cpp" "3" +"plugincollection.cpp" "3" +"plugindetailsview.cpp" "3" +"pluginerroroverview.cpp" "3" +"pluginerrorview.cpp" "3" +"pluginmanager.cpp" "3" +"pluginspec.cpp" "3" +"pluginview.cpp" "3" +"Forms" "2" +"plugindetailsview.ui" "3" +"pluginerroroverview.ui" "3" +"pluginerrorview.ui" "3" +"pluginview.ui" "3" +"Resources" "2" +"pluginview.qrc" "3" +"libs" "1" +"libs.pro" "2" +"3rdparty" "2" +"3rdparty.pro" "3" +"botan" "3" +"botan.pro" "4" +"src" "4" +"src.pro" "5" +"qtcreator" "5" +"qtcreator.pri" "6" +"qtcreatorlibrary" "5" +"qtcreatorlibrary.pri" "6" +"rpath" "5" +"rpath.pri" "6" +"Headers" "5" +"algo_factory" "6" +"algo_cache.h" "7" +"algo_factory.h" "7" +"alloc" "6" +"alloc_mmap" "7" +"mmap_mem.h" "8" +"mem_pool" "7" +"mem_pool.h" "8" +"system_alloc" "7" +"defalloc.h" "8" +"allocate.h" "7" +"secmem.h" "7" +"asn1" "6" +"alg_id.h" "7" +"asn1_int.h" "7" +"asn1_obj.h" "7" +"asn1_oid.h" "7" +"ber_dec.h" "7" +"der_enc.h" "7" +"benchmark" "6" +"benchmark.h" "7" +"block" "6" +"aes" "7" +"aes.h" "8" +"blowfish" "7" +"blowfish.h" "8" +"cast" "7" +"cast128.h" "8" +"cast256.h" "8" +"des" "7" +"des.h" "8" +"desx.h" "8" +"gost_28147" "7" +"gost_28147.h" "8" +"idea" "7" +"idea.h" "8" +"kasumi" "7" +"kasumi.h" "8" +"lion" "7" +"lion.h" "8" +"lubyrack" "7" +"lubyrack.h" "8" +"mars" "7" +"mars.h" "8" +"misty1" "7" +"misty1.h" "8" +"noekeon" "7" +"noekeon.h" "8" +"rc2" "7" +"rc2.h" "8" +"rc5" "7" +"rc5.h" "8" +"rc6" "7" +"rc6.h" "8" +"safer" "7" +"safer_sk.h" "8" +"seed" "7" +"seed.h" "8" +"serpent" "7" +"serpent.h" "8" +"skipjack" "7" +"skipjack.h" "8" +"square" "7" +"square.h" "8" +"tea" "7" +"tea.h" "8" +"twofish" "7" +"twofish.h" "8" +"xtea" "7" +"xtea.h" "8" +"block_cipher.h" "7" +"cert" "6" +"cvc" "7" +"cvc_ado.h" "8" +"cvc_ca.h" "8" +"cvc_cert.h" "8" +"cvc_gen_cert.h" "8" +"cvc_key.h" "8" +"cvc_req.h" "8" +"cvc_self.h" "8" +"eac_asn_obj.h" "8" +"eac_obj.h" "8" +"ecdsa_sig.h" "8" +"freestore.h" "8" +"signed_obj.h" "8" +"x509" "7" +"certstor.h" "8" +"crl_ent.h" "8" +"pkcs10.h" "8" +"x509_ca.h" "8" +"x509_crl.h" "8" +"x509_ext.h" "8" +"x509_obj.h" "8" +"x509cert.h" "8" +"x509find.h" "8" +"x509self.h" "8" +"x509stor.h" "8" +"checksum" "6" +"adler32" "7" +"adler32.h" "8" +"crc24" "7" +"crc24.h" "8" +"crc32" "7" +"crc32.h" "8" +"cms" "6" +"cms_dec.h" "7" +"cms_enc.h" "7" +"codec" "6" +"base64" "7" +"base64.h" "8" +"hex" "7" +"hex.h" "8" +"openpgp" "7" +"openpgp.h" "8" +"pem" "7" +"pem.h" "8" +"cryptobox" "6" +"cryptobox.h" "7" +"engine" "6" +"def_engine" "7" +"def_eng.h" "8" +"engine.h" "7" +"entropy" "6" +"cryptoapi_rng" "7" +"es_capi.h" "8" +"dev_random" "7" +"es_dev.h" "8" +"egd" "7" +"es_egd.h" "8" +"proc_walk" "7" +"es_ftw.h" "8" +"unix_procs" "7" +"es_unix.h" "8" +"unix_cmd.h" "8" +"win32_stats" "7" +"es_win32.h" "8" +"entropy_src.h" "7" +"filters" "6" +"fd_unix" "7" +"fd_unix.h" "8" +"basefilt.h" "7" +"buf_filt.h" "7" +"data_snk.h" "7" +"data_src.h" "7" +"filter.h" "7" +"filters.h" "7" +"out_buf.h" "7" +"pbe.h" "7" +"pipe.h" "7" +"secqueue.h" "7" +"hash" "6" +"fork256" "7" +"fork256.h" "8" +"gost_3411" "7" +"gost_3411.h" "8" +"has160" "7" +"has160.h" "8" +"md2" "7" +"md2.h" "8" +"md4" "7" +"md4.h" "8" +"md5" "7" +"md5.h" "8" +"mdx_hash" "7" +"mdx_hash.h" "8" +"par_hash" "7" +"par_hash.h" "8" +"rmd128" "7" +"rmd128.h" "8" +"rmd160" "7" +"rmd160.h" "8" +"sha1" "7" +"sha160.h" "8" +"sha2" "7" +"sha2_32.h" "8" +"sha2_64.h" "8" +"skein" "7" +"skein_512.h" "8" +"tiger" "7" +"tiger.h" "8" +"whirlpool" "7" +"whrlpool.h" "8" +"hash.h" "7" +"kdf" "6" +"kdf1" "7" +"kdf1.h" "8" +"kdf2" "7" +"kdf2.h" "8" +"mgf1" "7" +"mgf1.h" "8" +"ssl_prf" "7" +"prf_ssl3.h" "8" +"tls_prf" "7" +"prf_tls.h" "8" +"x942_prf" "7" +"prf_x942.h" "8" +"kdf.h" "7" +"libstate" "6" +"oid_lookup" "7" +"oids.h" "8" +"botan.h" "7" +"init.h" "7" +"libstate.h" "7" +"look_pk.h" "7" +"lookup.h" "7" +"pk_engine.h" "7" +"scan_name.h" "7" +"mac" "6" +"cbc_mac" "7" +"cbc_mac.h" "8" +"cmac" "7" +"cmac.h" "8" +"hmac" "7" +"hmac.h" "8" +"ssl3mac" "7" +"ssl3_mac.h" "8" +"x919_mac" "7" +"x919_mac.h" "8" +"mac.h" "7" +"math" "6" +"bigint" "7" +"mp_generic" "8" +"mp_asm.h" "9" +"mp_asmi.h" "9" +"bigint.h" "8" +"divide.h" "8" +"mp_core.h" "8" +"mp_types.h" "8" +"gfpmath" "7" +"curve_gfp.h" "8" +"gfp_element.h" "8" +"gfp_modulus.h" "8" +"point_gfp.h" "8" +"numbertheory" "7" +"blinding.h" "8" +"def_powm.h" "8" +"numthry.h" "8" +"pow_mod.h" "8" +"reducer.h" "8" +"modes" "6" +"cbc" "7" +"cbc.h" "8" +"cfb" "7" +"cfb.h" "8" +"ctr" "7" +"ctr.h" "8" +"cts" "7" +"cts.h" "8" +"eax" "7" +"eax.h" "8" +"ecb" "7" +"ecb.h" "8" +"mode_pad" "7" +"mode_pad.h" "8" +"ofb" "7" +"ofb.h" "8" +"xts" "7" +"xts.h" "8" +"modebase.h" "7" +"mutex" "6" +"noop_mutex" "7" +"mux_noop.h" "8" +"pthreads" "7" +"mux_pthr.h" "8" +"win32_crit_section" "7" +"mux_win32.h" "8" +"mutex.h" "7" +"pbe" "6" +"pbes1" "7" +"pbes1.h" "8" +"pbes2" "7" +"pbes2.h" "8" +"get_pbe.h" "7" +"pk_pad" "6" +"eme1" "7" +"eme1.h" "8" +"eme_pkcs" "7" +"eme_pkcs.h" "8" +"emsa1" "7" +"emsa1.h" "8" +"emsa1_bsi" "7" +"emsa1_bsi.h" "8" +"emsa2" "7" +"emsa2.h" "8" +"emsa3" "7" +"emsa3.h" "8" +"emsa4" "7" +"emsa4.h" "8" +"emsa_raw" "7" +"emsa_raw.h" "8" +"hash_id" "7" +"hash_id.h" "8" +"eme.h" "7" +"emsa.h" "7" +"pubkey" "6" +"dh" "7" +"dh.h" "8" +"dh_core.h" "8" +"dh_op.h" "8" +"dl_algo" "7" +"dl_algo.h" "8" +"dl_group" "7" +"dl_group.h" "8" +"dlies" "7" +"dlies.h" "8" +"dsa" "7" +"dsa.h" "8" +"dsa_core.h" "8" +"dsa_op.h" "8" +"ec_dompar" "7" +"ec_dompar.h" "8" +"ecc_key" "7" +"ecc_key.h" "8" +"ecdsa" "7" +"ecdsa.h" "8" +"ecdsa_core.h" "8" +"ecdsa_op.h" "8" +"eckaeg" "7" +"eckaeg.h" "8" +"eckaeg_core.h" "8" +"eckaeg_op.h" "8" +"elgamal" "7" +"elg_core.h" "8" +"elg_op.h" "8" +"elgamal.h" "8" +"if_algo" "7" +"if_algo.h" "8" +"if_core.h" "8" +"if_op.h" "8" +"keypair" "7" +"keypair.h" "8" +"nr" "7" +"nr.h" "8" +"nr_core.h" "8" +"nr_op.h" "8" +"pk_codecs" "7" +"pkcs8.h" "8" +"x509_key.h" "8" +"rsa" "7" +"rsa.h" "8" +"rw" "7" +"rw.h" "8" +"pk_algs.h" "7" +"pk_filts.h" "7" +"pk_keys.h" "7" +"pubkey.h" "7" +"pubkey_enums.h" "7" +"rng" "6" +"auto_rng" "7" +"auto_rng.h" "8" +"hmac_rng" "7" +"hmac_rng.h" "8" +"randpool" "7" +"randpool.h" "8" +"x931_rng" "7" +"x931_rng.h" "8" +"rng.h" "7" +"s2k" "6" +"pbkdf1" "7" +"pbkdf1.h" "8" +"pbkdf2" "7" +"pbkdf2.h" "8" +"pgps2k" "7" +"pgp_s2k.h" "8" +"s2k.h" "7" +"selftest" "6" +"selftest.h" "7" +"stream" "6" +"arc4" "7" +"arc4.h" "8" +"salsa20" "7" +"salsa20.h" "8" +"turing" "7" +"turing.h" "8" +"wid_wake" "7" +"wid_wake.h" "8" +"stream_cipher.h" "7" +"sym_algo" "6" +"sym_algo.h" "7" +"symkey.h" "7" +"timer" "6" +"gettimeofday" "7" +"tm_unix.h" "8" +"posix_rt" "7" +"tm_posix.h" "8" +"win32_query_perf_ctr" "7" +"tm_win32.h" "8" +"timer.h" "7" +"utils" "6" +"buf_comp" "7" +"buf_comp.h" "8" +"datastor" "7" +"datastor.h" "8" +"bit_ops.h" "7" +"bswap.h" "7" +"charset.h" "7" +"exceptn.h" "7" +"loadstor.h" "7" +"mem_ops.h" "7" +"parsing.h" "7" +"rotate.h" "7" +"sharedpointer.h" "7" +"stl_util.h" "7" +"types.h" "7" +"ui.h" "7" +"util.h" "7" +"version.h" "7" +"xor_buf.h" "7" +"Sources" "5" +"algo_factory" "6" +"algo_factory.cpp" "7" +"prov_weight.cpp" "7" +"alloc" "6" +"alloc_mmap" "7" +"mmap_mem.cpp" "8" +"mem_pool" "7" +"mem_pool.cpp" "8" +"system_alloc" "7" +"defalloc.cpp" "8" +"asn1" "6" +"alg_id.cpp" "7" +"asn1_alt.cpp" "7" +"asn1_att.cpp" "7" +"asn1_dn.cpp" "7" +"asn1_int.cpp" "7" +"asn1_oid.cpp" "7" +"asn1_str.cpp" "7" +"asn1_tm.cpp" "7" +"ber_dec.cpp" "7" +"der_enc.cpp" "7" +"benchmark" "6" +"benchmark.cpp" "7" +"block" "6" +"aes" "7" +"aes.cpp" "8" +"aes_tab.cpp" "8" +"blowfish" "7" +"blfs_tab.cpp" "8" +"blowfish.cpp" "8" +"cast" "7" +"cast128.cpp" "8" +"cast256.cpp" "8" +"cast_tab.cpp" "8" +"des" "7" +"des.cpp" "8" +"des_tab.cpp" "8" +"desx.cpp" "8" +"gost_28147" "7" +"gost_28147.cpp" "8" +"idea" "7" +"idea.cpp" "8" +"kasumi" "7" +"kasumi.cpp" "8" +"lion" "7" +"lion.cpp" "8" +"lubyrack" "7" +"lubyrack.cpp" "8" +"mars" "7" +"mars.cpp" "8" +"mars_tab.cpp" "8" +"misty1" "7" +"misty1.cpp" "8" +"noekeon" "7" +"noekeon.cpp" "8" +"rc2" "7" +"rc2.cpp" "8" +"rc5" "7" +"rc5.cpp" "8" +"rc6" "7" +"rc6.cpp" "8" +"safer" "7" +"safe_tab.cpp" "8" +"safer_sk.cpp" "8" +"seed" "7" +"seed.cpp" "8" +"seed_tab.cpp" "8" +"serpent" "7" +"serpent.cpp" "8" +"skipjack" "7" +"skipjack.cpp" "8" +"square" "7" +"sqr_tab.cpp" "8" +"square.cpp" "8" +"tea" "7" +"tea.cpp" "8" +"twofish" "7" +"two_tab.cpp" "8" +"twofish.cpp" "8" +"xtea" "7" +"xtea.cpp" "8" +"cert" "6" +"cvc" "7" +"asn1_eac_str.cpp" "8" +"asn1_eac_tm.cpp" "8" +"cvc_ado.cpp" "8" +"cvc_ca.cpp" "8" +"cvc_cert.cpp" "8" +"cvc_req.cpp" "8" +"cvc_self.cpp" "8" +"ecdsa_sig.cpp" "8" +"signed_obj.cpp" "8" +"x509" "7" +"certstor.cpp" "8" +"crl_ent.cpp" "8" +"pkcs10.cpp" "8" +"x509_ca.cpp" "8" +"x509_crl.cpp" "8" +"x509_ext.cpp" "8" +"x509_obj.cpp" "8" +"x509cert.cpp" "8" +"x509find.cpp" "8" +"x509opt.cpp" "8" +"x509self.cpp" "8" +"x509stor.cpp" "8" +"checksum" "6" +"adler32" "7" +"adler32.cpp" "8" +"crc24" "7" +"crc24.cpp" "8" +"crc32" "7" +"crc32.cpp" "8" +"cms" "6" +"cms_algo.cpp" "7" +"cms_comp.cpp" "7" +"cms_dalg.cpp" "7" +"cms_dec.cpp" "7" +"cms_ealg.cpp" "7" +"cms_enc.cpp" "7" +"codec" "6" +"base64" "7" +"b64_char.cpp" "8" +"base64.cpp" "8" +"hex" "7" +"hex.cpp" "8" +"hex_char.cpp" "8" +"openpgp" "7" +"openpgp.cpp" "8" +"pem" "7" +"pem.cpp" "8" +"cryptobox" "6" +"cryptobox.cpp" "7" +"engine/def_engine" "6" +"def_mode.cpp" "7" +"def_pk_ops.cpp" "7" +"def_powm.cpp" "7" +"lookup_block.cpp" "7" +"lookup_hash.cpp" "7" +"lookup_mac.cpp" "7" +"lookup_stream.cpp" "7" +"entropy" "6" +"cryptoapi_rng" "7" +"es_capi.cpp" "8" +"dev_random" "7" +"es_dev.cpp" "8" +"egd" "7" +"es_egd.cpp" "8" +"proc_walk" "7" +"es_ftw.cpp" "8" +"unix_procs" "7" +"es_unix.cpp" "8" +"unix_cmd.cpp" "8" +"unix_src.cpp" "8" +"win32_stats" "7" +"es_win32.cpp" "8" +"filters" "6" +"fd_unix" "7" +"fd_unix.cpp" "8" +"algo_filt.cpp" "7" +"basefilt.cpp" "7" +"buf_filt.cpp" "7" +"data_snk.cpp" "7" +"data_src.cpp" "7" +"filter.cpp" "7" +"out_buf.cpp" "7" +"pipe.cpp" "7" +"pipe_io.cpp" "7" +"pipe_rw.cpp" "7" +"secqueue.cpp" "7" +"hash" "6" +"fork256" "7" +"fork256.cpp" "8" +"gost_3411" "7" +"gost_3411.cpp" "8" +"has160" "7" +"has160.cpp" "8" +"md2" "7" +"md2.cpp" "8" +"md4" "7" +"md4.cpp" "8" +"md5" "7" +"md5.cpp" "8" +"mdx_hash" "7" +"mdx_hash.cpp" "8" +"par_hash" "7" +"par_hash.cpp" "8" +"rmd128" "7" +"rmd128.cpp" "8" +"rmd160" "7" +"rmd160.cpp" "8" +"sha1" "7" +"sha160.cpp" "8" +"sha2" "7" +"sha2_32.cpp" "8" +"sha2_64.cpp" "8" +"skein" "7" +"skein_512.cpp" "8" +"tiger" "7" +"tig_tab.cpp" "8" +"tiger.cpp" "8" +"whirlpool" "7" +"whrl_tab.cpp" "8" +"whrlpool.cpp" "8" +"kdf" "6" +"kdf1" "7" +"kdf1.cpp" "8" +"kdf2" "7" +"kdf2.cpp" "8" +"mgf1" "7" +"mgf1.cpp" "8" +"ssl_prf" "7" +"prf_ssl3.cpp" "8" +"tls_prf" "7" +"prf_tls.cpp" "8" +"x942_prf" "7" +"prf_x942.cpp" "8" +"kdf.cpp" "7" +"libstate" "6" +"oid_lookup" "7" +"oids.cpp" "8" +"get_enc.cpp" "7" +"init.cpp" "7" +"libstate.cpp" "7" +"look_pk.cpp" "7" +"lookup.cpp" "7" +"pk_engine.cpp" "7" +"policy.cpp" "7" +"scan_name.cpp" "7" +"mac" "6" +"cbc_mac" "7" +"cbc_mac.cpp" "8" +"cmac" "7" +"cmac.cpp" "8" +"hmac" "7" +"hmac.cpp" "8" +"ssl3mac" "7" +"ssl3_mac.cpp" "8" +"x919_mac" "7" +"x919_mac.cpp" "8" +"mac.cpp" "7" +"math" "6" +"bigint" "7" +"monty_generic" "8" +"mp_monty.cpp" "9" +"mulop_generic" "8" +"mp_mulop.cpp" "9" +"big_code.cpp" "8" +"big_io.cpp" "8" +"big_ops2.cpp" "8" +"big_ops3.cpp" "8" +"big_rand.cpp" "8" +"bigint.cpp" "8" +"divide.cpp" "8" +"mp_asm.cpp" "8" +"mp_comba.cpp" "8" +"mp_karat.cpp" "8" +"mp_misc.cpp" "8" +"mp_shift.cpp" "8" +"gfpmath" "7" +"curve_gfp.cpp" "8" +"gfp_element.cpp" "8" +"point_gfp.cpp" "8" +"numbertheory" "7" +"blinding.cpp" "8" +"dsa_gen.cpp" "8" +"jacobi.cpp" "8" +"make_prm.cpp" "8" +"mp_numth.cpp" "8" +"numthry.cpp" "8" +"pow_mod.cpp" "8" +"powm_fw.cpp" "8" +"powm_mnt.cpp" "8" +"primes.cpp" "8" +"reducer.cpp" "8" +"ressol.cpp" "8" +"modes" "6" +"cbc" "7" +"cbc.cpp" "8" +"cfb" "7" +"cfb.cpp" "8" +"ctr" "7" +"ctr.cpp" "8" +"cts" "7" +"cts.cpp" "8" +"eax" "7" +"eax.cpp" "8" +"eax_dec.cpp" "8" +"ecb" "7" +"ecb.cpp" "8" +"mode_pad" "7" +"mode_pad.cpp" "8" +"ofb" "7" +"ofb.cpp" "8" +"xts" "7" +"xts.cpp" "8" +"modebase.cpp" "7" +"mutex" "6" +"noop_mutex" "7" +"mux_noop.cpp" "8" +"pthreads" "7" +"mux_pthr.cpp" "8" +"win32_crit_section" "7" +"mux_win32.cpp" "8" +"pbe" "6" +"pbes1" "7" +"pbes1.cpp" "8" +"pbes2" "7" +"pbes2.cpp" "8" +"get_pbe.cpp" "7" +"pk_pad" "6" +"eme1" "7" +"eme1.cpp" "8" +"eme_pkcs" "7" +"eme_pkcs.cpp" "8" +"emsa1" "7" +"emsa1.cpp" "8" +"emsa1_bsi" "7" +"emsa1_bsi.cpp" "8" +"emsa2" "7" +"emsa2.cpp" "8" +"emsa3" "7" +"emsa3.cpp" "8" +"emsa4" "7" +"emsa4.cpp" "8" +"emsa_raw" "7" +"emsa_raw.cpp" "8" +"hash_id" "7" +"hash_id.cpp" "8" +"eme.cpp" "7" +"pubkey" "6" +"dh" "7" +"dh.cpp" "8" +"dh_core.cpp" "8" +"dl_algo" "7" +"dl_algo.cpp" "8" +"dl_group" "7" +"dl_group.cpp" "8" +"dlies" "7" +"dlies.cpp" "8" +"dsa" "7" +"dsa.cpp" "8" +"dsa_core.cpp" "8" +"dsa_op.cpp" "8" +"ec_dompar" "7" +"ec_dompar.cpp" "8" +"ecc_key" "7" +"ecc_key.cpp" "8" +"ecdsa" "7" +"ecdsa.cpp" "8" +"ecdsa_core.cpp" "8" +"ecdsa_op.cpp" "8" +"eckaeg" "7" +"eckaeg.cpp" "8" +"eckaeg_core.cpp" "8" +"eckaeg_op.cpp" "8" +"elgamal" "7" +"elg_core.cpp" "8" +"elg_op.cpp" "8" +"elgamal.cpp" "8" +"if_algo" "7" +"if_algo.cpp" "8" +"if_core.cpp" "8" +"if_op.cpp" "8" +"keypair" "7" +"keypair.cpp" "8" +"nr" "7" +"nr.cpp" "8" +"nr_core.cpp" "8" +"nr_op.cpp" "8" +"pk_codecs" "7" +"pkcs8.cpp" "8" +"x509_key.cpp" "8" +"rsa" "7" +"rsa.cpp" "8" +"rw" "7" +"rw.cpp" "8" +"pk_algs.cpp" "7" +"pk_filts.cpp" "7" +"pk_keys.cpp" "7" +"pubkey.cpp" "7" +"pubkey_enums.cpp" "7" +"rng" "6" +"auto_rng" "7" +"auto_rng.cpp" "8" +"hmac_rng" "7" +"hmac_rng.cpp" "8" +"randpool" "7" +"randpool.cpp" "8" +"x931_rng" "7" +"x931_rng.cpp" "8" +"rng.cpp" "7" +"s2k" "6" +"pbkdf1" "7" +"pbkdf1.cpp" "8" +"pbkdf2" "7" +"pbkdf2.cpp" "8" +"pgps2k" "7" +"pgp_s2k.cpp" "8" +"s2k.cpp" "7" +"selftest" "6" +"selftest.cpp" "7" +"stream" "6" +"arc4" "7" +"arc4.cpp" "8" +"salsa20" "7" +"salsa20.cpp" "8" +"turing" "7" +"tur_tab.cpp" "8" +"turing.cpp" "8" +"wid_wake" "7" +"wid_wake.cpp" "8" +"stream_cipher.cpp" "7" +"sym_algo" "6" +"symkey.cpp" "7" +"timer" "6" +"gettimeofday" "7" +"tm_unix.cpp" "8" +"posix_rt" "7" +"tm_posix.cpp" "8" +"win32_query_perf_ctr" "7" +"tm_win32.cpp" "8" +"timer.cpp" "7" +"utils" "6" +"datastor" "7" +"datastor.cpp" "8" +"charset.cpp" "7" +"exceptn.cpp" "7" +"mlock.cpp" "7" +"parsing.cpp" "7" +"ui.cpp" "7" +"util.cpp" "7" +"version.cpp" "7" +"cdb_detect" "2" +"cdb_detect.pri" "3" +"cplusplus" "2" +"cplusplus.pro" "3" +"cplusplus" "3" +"cplusplus.pri" "4" +"Headers" "4" +"AST.h" "5" +"ASTfwd.h" "5" +"ASTMatcher.h" "5" +"ASTPatternBuilder.h" "5" +"ASTVisitor.h" "5" +"Bind.h" "5" +"Control.h" "5" +"CoreTypes.h" "5" +"CPlusPlus.h" "5" +"CPlusPlusForwardDeclarations.h" "5" +"DiagnosticClient.h" "5" +"FullySpecifiedType.h" "5" +"Lexer.h" "5" +"Literals.h" "5" +"LiteralTable.h" "5" +"MemoryPool.h" "5" +"Name.h" "5" +"Names.h" "5" +"NameVisitor.h" "5" +"ObjectiveCTypeQualifiers.h" "5" +"Parser.h" "5" +"QtContextKeywords.h" "5" +"Scope.h" "5" +"Symbol.h" "5" +"Symbols.h" "5" +"SymbolVisitor.h" "5" +"Templates.h" "5" +"Token.h" "5" +"TranslationUnit.h" "5" +"Type.h" "5" +"TypeMatcher.h" "5" +"TypeVisitor.h" "5" +"Sources" "4" +"AST.cpp" "5" +"ASTClone.cpp" "5" +"ASTMatch0.cpp" "5" +"ASTMatcher.cpp" "5" +"ASTPatternBuilder.cpp" "5" +"ASTVisit.cpp" "5" +"ASTVisitor.cpp" "5" +"Bind.cpp" "5" +"Control.cpp" "5" +"CoreTypes.cpp" "5" +"DiagnosticClient.cpp" "5" +"FullySpecifiedType.cpp" "5" +"Keywords.cpp" "5" +"Lexer.cpp" "5" +"Literals.cpp" "5" +"LiteralTable.cpp" "5" +"MemoryPool.cpp" "5" +"Name.cpp" "5" +"Names.cpp" "5" +"NameVisitor.cpp" "5" +"ObjectiveCAtKeywords.cpp" "5" +"ObjectiveCTypeQualifiers.cpp" "5" +"Parser.cpp" "5" +"QtContextKeywords.cpp" "5" +"Scope.cpp" "5" +"Symbol.cpp" "5" +"Symbols.cpp" "5" +"SymbolVisitor.cpp" "5" +"Templates.cpp" "5" +"Token.cpp" "5" +"TranslationUnit.cpp" "5" +"Type.cpp" "5" +"TypeMatcher.cpp" "5" +"TypeVisitor.cpp" "5" +"cplusplus-lib" "3" +"cplusplus-lib.pri" "4" +"Headers" "4" +"ASTParent.h" "5" +"ASTPath.h" "5" +"BackwardsScanner.h" "5" +"CppDocument.h" "5" +"CppRewriter.h" "5" +"DependencyTable.h" "5" +"DeprecatedGenTemplateInstance.h" "5" +"ExpressionUnderCursor.h" "5" +"FastPreprocessor.h" "5" +"findcdbbreakpoint.h" "5" +"FindUsages.h" "5" +"Icons.h" "5" +"LookupContext.h" "5" +"LookupItem.h" "5" +"Macro.h" "5" +"MatchingText.h" "5" +"NamePrettyPrinter.h" "5" +"Overview.h" "5" +"OverviewModel.h" "5" +"pp-cctype.h" "5" +"pp-engine.h" "5" +"pp-scanner.h" "5" +"pp.h" "5" +"PreprocessorClient.h" "5" +"PreprocessorEnvironment.h" "5" +"ResolveExpression.h" "5" +"SimpleLexer.h" "5" +"SnapshotSymbolVisitor.h" "5" +"SymbolNameVisitor.h" "5" +"TypeOfExpression.h" "5" +"TypePrettyPrinter.h" "5" +"Sources" "4" +"ASTParent.cpp" "5" +"ASTPath.cpp" "5" +"BackwardsScanner.cpp" "5" +"CppDocument.cpp" "5" +"CppRewriter.cpp" "5" +"DependencyTable.cpp" "5" +"DeprecatedGenTemplateInstance.cpp" "5" +"ExpressionUnderCursor.cpp" "5" +"FastPreprocessor.cpp" "5" +"findcdbbreakpoint.cpp" "5" +"FindUsages.cpp" "5" +"Icons.cpp" "5" +"LookupContext.cpp" "5" +"LookupItem.cpp" "5" +"Macro.cpp" "5" +"MatchingText.cpp" "5" +"NamePrettyPrinter.cpp" "5" +"Overview.cpp" "5" +"OverviewModel.cpp" "5" +"pp-engine.cpp" "5" +"pp-scanner.cpp" "5" +"PreprocessorClient.cpp" "5" +"PreprocessorEnvironment.cpp" "5" +"ResolveExpression.cpp" "5" +"SimpleLexer.cpp" "5" +"SnapshotSymbolVisitor.cpp" "5" +"SymbolNameVisitor.cpp" "5" +"TypeOfExpression.cpp" "5" +"TypePrettyPrinter.cpp" "5" +"Resources" "4" +"cplusplus.qrc" "5" +"languageutils" "3" +"languageutils.pri" "4" +"qtcreator" "3" +"qtcreator.pri" "4" +"qtcreatorlibrary" "3" +"qtcreatorlibrary.pri" "4" +"rpath" "3" +"rpath.pri" "4" +"glsl" "2" +"glsl.pro" "3" +"glsl-lib" "3" +"glsl-lib.pri" "4" +"Headers" "4" +"glsl.h" "5" +"glslast.h" "5" +"glslastdump.h" "5" +"glslastvisitor.h" "5" +"glslengine.h" "5" +"glsllexer.h" "5" +"glslmemorypool.h" "5" +"glslparser.h" "5" +"glslparsertable_p.h" "5" +"glslsemantic.h" "5" +"glslsymbol.h" "5" +"glslsymbols.h" "5" +"glsltype.h" "5" +"glsltypes.h" "5" +"Sources" "4" +"glslast.cpp" "5" +"glslastdump.cpp" "5" +"glslastvisitor.cpp" "5" +"glslengine.cpp" "5" +"glslkeywords.cpp" "5" +"glsllexer.cpp" "5" +"glslmemorypool.cpp" "5" +"glslparser.cpp" "5" +"glslparsertable.cpp" "5" +"glslsemantic.cpp" "5" +"glslsymbol.cpp" "5" +"glslsymbols.cpp" "5" +"glsltype.cpp" "5" +"glsltypes.cpp" "5" +"Other files" "4" +"glsl.g" "5" +"qtcreator" "3" +"qtcreator.pri" "4" +"qtcreatorlibrary" "3" +"qtcreatorlibrary.pri" "4" +"rpath" "3" +"rpath.pri" "4" +"utils" "3" +"utils.pri" "4" +"utils_dependencies" "3" +"utils_dependencies.pri" "4" +"languageutils" "2" +"languageutils.pro" "3" +"languageutils-lib" "3" +"languageutils-lib.pri" "4" +"Headers" "4" +"componentversion.h" "5" +"fakemetaobject.h" "5" +"languageutils_global.h" "5" +"Sources" "4" +"componentversion.cpp" "5" +"fakemetaobject.cpp" "5" +"qtcreator" "3" +"qtcreator.pri" "4" +"qtcreatorlibrary" "3" +"qtcreatorlibrary.pri" "4" +"rpath" "3" +"rpath.pri" "4" +"utils" "3" +"utils.pri" "4" +"utils_dependencies" "3" +"utils_dependencies.pri" "4" +"process_ctrlc_stub" "2" +"process_ctrlc_stub.pro" "3" +"qtcreator" "3" +"qtcreator.pri" "4" +"Sources" "3" +"process_ctrlc_stub.cpp" "4" +"process_stub" "2" +"process_stub.pro" "3" +"qtcreator" "3" +"qtcreator.pri" "4" +"Sources" "3" +"process_stub_unix.c" "4" +"process_stub_win.c" "4" +"qmleditorwidgets" "2" +"qmleditorwidgets.pro" "3" +"easingpane" "3" +"easingpane.pri" "4" +"Headers" "4" +"easingcontextpane.h" "5" +"easinggraph.h" "5" +"Sources" "4" +"easingcontextpane.cpp" "5" +"easinggraph.cpp" "5" +"Forms" "4" +"easingcontextpane.ui" "5" +"Resources" "4" +"easingpane.qrc" "5" +"qmleditorwidgets-lib" "3" +"qmleditorwidgets-lib.pri" "4" +"Headers" "4" +"colorbox.h" "5" +"colorbutton.h" "5" +"colorwidgets.h" "5" +"contextpanetextwidget.h" "5" +"contextpanewidget.h" "5" +"contextpanewidgetimage.h" "5" +"contextpanewidgetrectangle.h" "5" +"customcolordialog.h" "5" +"filewidget.h" "5" +"fontsizespinbox.h" "5" +"gradientline.h" "5" +"huecontrol.h" "5" +"qmleditorwidgets_global.h" "5" +"Sources" "4" +"colorbox.cpp" "5" +"colorbutton.cpp" "5" +"colorwidgets.cpp" "5" +"contextpanetextwidget.cpp" "5" +"contextpanewidget.cpp" "5" +"contextpanewidgetimage.cpp" "5" +"contextpanewidgetrectangle.cpp" "5" +"customcolordialog.cpp" "5" +"filewidget.cpp" "5" +"fontsizespinbox.cpp" "5" +"gradientline.cpp" "5" +"huecontrol.cpp" "5" +"Forms" "4" +"contextpanetext.ui" "5" +"contextpanewidgetborderimage.ui" "5" +"contextpanewidgetimage.ui" "5" +"contextpanewidgetrectangle.ui" "5" +"Resources" "4" +"resources.qrc" "5" +"Other files" "4" +"qmleditorwidgets.pri" "5" +"qmljs" "3" +"qmljs.pri" "4" +"qtcreator" "3" +"qtcreator.pri" "4" +"qtcreatorlibrary" "3" +"qtcreatorlibrary.pri" "4" +"rpath" "3" +"rpath.pri" "4" +"qmljs" "2" +"qmljs.pro" "3" +"languageutils" "3" +"languageutils.pri" "4" +"parser" "3" +"parser.pri" "4" +"Headers" "4" +"qmldirparser_p.h" "5" +"qmlerror.h" "5" +"qmljsast_p.h" "5" +"qmljsastfwd_p.h" "5" +"qmljsastvisitor_p.h" "5" +"qmljsengine_p.h" "5" +"qmljsglobal_p.h" "5" +"qmljsgrammar_p.h" "5" +"qmljskeywords_p.h" "5" +"qmljslexer_p.h" "5" +"qmljsmemorypool_p.h" "5" +"qmljsparser_p.h" "5" +"Sources" "4" +"qmldirparser.cpp" "5" +"qmlerror.cpp" "5" +"qmljsast.cpp" "5" +"qmljsastvisitor.cpp" "5" +"qmljsengine_p.cpp" "5" +"qmljsgrammar.cpp" "5" +"qmljslexer.cpp" "5" +"qmljsparser.cpp" "5" +"qmljs-lib" "3" +"qmljs-lib.pri" "4" +"Headers" "4" +"jsoncheck.h" "5" +"qmljs_global.h" "5" +"qmljsbind.h" "5" +"qmljscheck.h" "5" +"qmljscodeformatter.h" "5" +"qmljscompletioncontextfinder.h" "5" +"qmljscontext.h" "5" +"qmljsdelta.h" "5" +"qmljsdocument.h" "5" +"qmljsevaluate.h" "5" +"qmljsicons.h" "5" +"qmljsicontextpane.h" "5" +"qmljsindenter.h" "5" +"qmljsinterpreter.h" "5" +"qmljslineinfo.h" "5" +"qmljslink.h" "5" +"qmljsmodelmanagerinterface.h" "5" +"qmljspropertyreader.h" "5" +"qmljsreformatter.h" "5" +"qmljsrewriter.h" "5" +"qmljsscanner.h" "5" +"qmljsscopeastpath.h" "5" +"qmljsscopebuilder.h" "5" +"qmljsscopechain.h" "5" +"qmljsstaticanalysismessage.h" "5" +"qmljstypedescriptionreader.h" "5" +"qmljsutils.h" "5" +"qmljsvalueowner.h" "5" +"Sources" "4" +"jsoncheck.cpp" "5" +"qmljsbind.cpp" "5" +"qmljscheck.cpp" "5" +"qmljscodeformatter.cpp" "5" +"qmljscompletioncontextfinder.cpp" "5" +"qmljscontext.cpp" "5" +"qmljsdelta.cpp" "5" +"qmljsdocument.cpp" "5" +"qmljsevaluate.cpp" "5" +"qmljsicons.cpp" "5" +"qmljsindenter.cpp" "5" +"qmljsinterpreter.cpp" "5" +"qmljslineinfo.cpp" "5" +"qmljslink.cpp" "5" +"qmljsmodelmanagerinterface.cpp" "5" +"qmljspropertyreader.cpp" "5" +"qmljsreformatter.cpp" "5" +"qmljsrewriter.cpp" "5" +"qmljsscanner.cpp" "5" +"qmljsscopeastpath.cpp" "5" +"qmljsscopebuilder.cpp" "5" +"qmljsscopechain.cpp" "5" +"qmljsstaticanalysismessage.cpp" "5" +"qmljstypedescriptionreader.cpp" "5" +"qmljsutils.cpp" "5" +"qmljsvalueowner.cpp" "5" +"Resources" "4" +"qmljs.qrc" "5" +"Other files" "4" +"parser" "5" +"qmljs.g" "6" +"qtcreator" "3" +"qtcreator.pri" "4" +"qtcreatorlibrary" "3" +"qtcreatorlibrary.pri" "4" +"rpath" "3" +"rpath.pri" "4" +"utils" "3" +"utils.pri" "4" +"utils_dependencies" "3" +"utils_dependencies.pri" "4" +"qtcreatorcdbext" "2" +"qtcreatorcdbext.pro" "3" +"cdb_detect" "3" +"cdb_detect.pri" "4" +"qtcreator" "3" +"qtcreator.pri" "4" +"Headers" "3" +"base64.h" "4" +"common.h" "4" +"containers.h" "4" +"eventcallback.h" "4" +"extensioncontext.h" "4" +"gdbmihelpers.h" "4" +"iinterfacepointer.h" "4" +"knowntype.h" "4" +"outputcallback.h" "4" +"stringutils.h" "4" +"symbolgroup.h" "4" +"symbolgroupnode.h" "4" +"symbolgroupvalue.h" "4" +"Sources" "3" +"base64.cpp" "4" +"common.cpp" "4" +"containers.cpp" "4" +"eventcallback.cpp" "4" +"extensioncontext.cpp" "4" +"gdbmihelpers.cpp" "4" +"outputcallback.cpp" "4" +"qtcreatorcdbextension.cpp" "4" +"stringutils.cpp" "4" +"symbolgroup.cpp" "4" +"symbolgroupnode.cpp" "4" +"symbolgroupvalue.cpp" "4" +"styleitem" "2" +"styleitem.pro" "3" +"qtcreator" "3" +"qtcreator.pri" "4" +"Headers" "3" +"qdeclarativefolderlistmodel.h" "4" +"qrangemodel.h" "4" +"qrangemodel_p.h" "4" +"qstyleitem.h" "4" +"qstyleplugin.h" "4" +"qtmenu.h" "4" +"qtmenubar.h" "4" +"qtmenuitem.h" "4" +"qwheelarea.h" "4" +"Sources" "3" +"qdeclarativefolderlistmodel.cpp" "4" +"qrangemodel.cpp" "4" +"qstyleitem.cpp" "4" +"qstyleplugin.cpp" "4" +"qtmenu.cpp" "4" +"qtmenubar.cpp" "4" +"qtmenuitem.cpp" "4" +"qwheelarea.cpp" "4" +"symbianutils" "2" +"symbianutils.pro" "3" +"json" "3" +"json.pri" "4" +"Headers" "4" +"json.h" "5" +"json_global.h" "5" +"Sources" "4" +"json.cpp" "5" +"private_headers" "3" +"private_headers.pri" "4" +"qtcreator" "3" +"qtcreator.pri" "4" +"qtcreatorlibrary" "3" +"qtcreatorlibrary.pri" "4" +"rpath" "3" +"rpath.pri" "4" +"symbianutils" "3" +"symbianutils.pri" "4" +"Headers" "4" +"callback.h" "5" +"codadevice.h" "5" +"codamessage.h" "5" +"codautils.h" "5" +"codautils_p.h" "5" +"symbiandevicemanager.h" "5" +"symbianutils_global.h" "5" +"virtualserialdevice.h" "5" +"Sources" "4" +"codadevice.cpp" "5" +"codamessage.cpp" "5" +"codautils.cpp" "5" +"symbiandevicemanager.cpp" "5" +"virtualserialdevice.cpp" "5" +"virtualserialdevice_posix.cpp" "5" +"virtualserialdevice_win.cpp" "5" +"utils" "2" +"utils.pro" "3" +"private_headers" "3" +"private_headers.pri" "4" +"qtcreator" "3" +"qtcreator.pri" "4" +"qtcreatorlibrary" "3" +"qtcreatorlibrary.pri" "4" +"rpath" "3" +"rpath.pri" "4" +"utils-lib" "3" +"utils-lib.pri" "4" +"Headers" "4" +"annotateditemdelegate.h" "5" +"basevalidatinglineedit.h" "5" +"buildablehelperlibrary.h" "5" +"changeset.h" "5" +"checkablemessagebox.h" "5" +"classnamevalidatinglineedit.h" "5" +"codegeneration.h" "5" +"completingtextedit.h" "5" +"consoleprocess.h" "5" +"consoleprocess_p.h" "5" +"crumblepath.h" "5" +"detailsbutton.h" "5" +"detailswidget.h" "5" +"environment.h" "5" +"environmentmodel.h" "5" +"faketooltip.h" "5" +"fancylineedit.h" "5" +"fancymainwindow.h" "5" +"fileinprojectfinder.h" "5" +"filenamevalidatinglineedit.h" "5" +"filesearch.h" "5" +"filesystemwatcher.h" "5" +"fileutils.h" "5" +"filewizarddialog.h" "5" +"filewizardpage.h" "5" +"filterlineedit.h" "5" +"flowlayout.h" "5" +"historycompleter.h" "5" +"htmldocextractor.h" "5" +"ipaddresslineedit.h" "5" +"iwelcomepage.h" "5" +"json.h" "5" +"linecolumnlabel.h" "5" +"listutils.h" "5" +"multitask.h" "5" +"navigationtreeview.h" "5" +"networkaccessmanager.h" "5" +"newclasswidget.h" "5" +"outputformat.h" "5" +"outputformatter.h" "5" +"parameteraction.h" "5" +"pathchooser.h" "5" +"pathlisteditor.h" "5" +"persistentsettings.h" "5" +"projectintropage.h" "5" +"projectnamevalidatinglineedit.h" "5" +"qtcassert.h" "5" +"qtcolorbutton.h" "5" +"qtcprocess.h" "5" +"reloadpromptutils.h" "5" +"runextensions.h" "5" +"savedaction.h" "5" +"savefile.h" "5" +"settingsselector.h" "5" +"statuslabel.h" "5" +"stringutils.h" "5" +"styledbar.h" "5" +"stylehelper.h" "5" +"submiteditorwidget.h" "5" +"submitfieldwidget.h" "5" +"synchronousprocess.h" "5" +"textfileformat.h" "5" +"treewidgetcolumnstretcher.h" "5" +"uncommentselection.h" "5" +"unixutils.h" "5" +"utils_global.h" "5" +"winutils.h" "5" +"wizard.h" "5" +"Sources" "4" +"annotateditemdelegate.cpp" "5" +"basevalidatinglineedit.cpp" "5" +"buildablehelperlibrary.cpp" "5" +"changeset.cpp" "5" +"checkablemessagebox.cpp" "5" +"classnamevalidatinglineedit.cpp" "5" +"codegeneration.cpp" "5" +"completingtextedit.cpp" "5" +"consoleprocess.cpp" "5" +"consoleprocess_unix.cpp" "5" +"consoleprocess_win.cpp" "5" +"crumblepath.cpp" "5" +"detailsbutton.cpp" "5" +"detailswidget.cpp" "5" +"environment.cpp" "5" +"environmentmodel.cpp" "5" +"faketooltip.cpp" "5" +"fancylineedit.cpp" "5" +"fancymainwindow.cpp" "5" +"fileinprojectfinder.cpp" "5" +"filenamevalidatinglineedit.cpp" "5" +"filesearch.cpp" "5" +"filesystemwatcher.cpp" "5" +"fileutils.cpp" "5" +"filewizarddialog.cpp" "5" +"filewizardpage.cpp" "5" +"filterlineedit.cpp" "5" +"flowlayout.cpp" "5" +"historycompleter.cpp" "5" +"htmldocextractor.cpp" "5" +"ipaddresslineedit.cpp" "5" +"iwelcomepage.cpp" "5" +"json.cpp" "5" +"linecolumnlabel.cpp" "5" +"navigationtreeview.cpp" "5" +"networkaccessmanager.cpp" "5" +"newclasswidget.cpp" "5" +"outputformatter.cpp" "5" +"parameteraction.cpp" "5" +"pathchooser.cpp" "5" +"pathlisteditor.cpp" "5" +"persistentsettings.cpp" "5" +"projectintropage.cpp" "5" +"projectnamevalidatinglineedit.cpp" "5" +"qtcolorbutton.cpp" "5" +"qtcprocess.cpp" "5" +"reloadpromptutils.cpp" "5" +"savedaction.cpp" "5" +"savefile.cpp" "5" +"settingsselector.cpp" "5" +"statuslabel.cpp" "5" +"stringutils.cpp" "5" +"styledbar.cpp" "5" +"stylehelper.cpp" "5" +"submiteditorwidget.cpp" "5" +"submitfieldwidget.cpp" "5" +"synchronousprocess.cpp" "5" +"textfileformat.cpp" "5" +"treewidgetcolumnstretcher.cpp" "5" +"uncommentselection.cpp" "5" +"unixutils.cpp" "5" +"winutils.cpp" "5" +"wizard.cpp" "5" +"Forms" "4" +"filewizardpage.ui" "5" +"newclasswidget.ui" "5" +"projectintropage.ui" "5" +"submiteditorwidget.ui" "5" +"Resources" "4" +"utils.qrc" "5" +"utils_dependencies" "3" +"utils_dependencies.pri" "4" +"Headers" "3" +"proxyaction.h" "4" +"Sources" "3" +"proxyaction.cpp" "4" +"zeroconf" "2" +"zeroconf.pro" "3" +"qtcreator" "3" +"qtcreator.pri" "4" +"qtcreatorlibrary" "3" +"qtcreatorlibrary.pri" "4" +"rpath" "3" +"rpath.pri" "4" +"Headers" "3" +"dns_sd_types.h" "4" +"mdnsderived.h" "4" +"servicebrowser.h" "4" +"servicebrowser_p.h" "4" +"syssocket.h" "4" +"zeroconf_global.h" "4" +"Sources" "3" +"avahiLib.cpp" "4" +"dnsSdLib.cpp" "4" +"embeddedLib.cpp" "4" +"mdnsderived.cpp" "4" +"servicebrowser.cpp" "4" +"plugins" "1" +"plugins.pro" "2" +"analyzerbase" "2" +"analyzerbase.pro" "3" +"aggregation" "3" +"aggregation.pri" "4" +"analyzerbase_dependencies" "3" +"analyzerbase_dependencies.pri" "4" +"botan" "3" +"botan.pri" "4" +"coreplugin" "3" +"coreplugin.pri" "4" +"coreplugin_dependencies" "3" +"coreplugin_dependencies.pri" "4" +"cplusplus" "3" +"cplusplus.pri" "4" +"cpptools" "3" +"cpptools.pri" "4" +"cpptools_dependencies" "3" +"cpptools_dependencies.pri" "4" +"debugger" "3" +"debugger.pri" "4" +"debugger_dependencies" "3" +"debugger_dependencies.pri" "4" +"extensionsystem" "3" +"extensionsystem.pri" "4" +"extensionsystem_dependencies" "3" +"extensionsystem_dependencies.pri" "4" +"find" "3" +"find.pri" "4" +"find_dependencies" "3" +"find_dependencies.pri" "4" +"languageutils" "3" +"languageutils.pri" "4" +"locator" "3" +"locator.pri" "4" +"locator_dependencies" "3" +"locator_dependencies.pri" "4" +"projectexplorer" "3" +"projectexplorer.pri" "4" +"projectexplorer_dependencies" "3" +"projectexplorer_dependencies.pri" "4" +"qmljs" "3" +"qmljs.pri" "4" +"qt4projectmanager" "3" +"qt4projectmanager.pri" "4" +"qt4projectmanager_dependencies" "3" +"qt4projectmanager_dependencies.pri" "4" +"qtcreator" "3" +"qtcreator.pri" "4" +"qtcreatorplugin" "3" +"qtcreatorplugin.pri" "4" +"Other files" "4" +"plugins/analyzerbase" "5" +"AnalyzerBase.pluginspec.in" "6" +"qtsupport" "3" +"qtsupport.pri" "4" +"qtsupport_dependencies" "3" +"qtsupport_dependencies.pri" "4" +"remotelinux" "3" +"remotelinux.pri" "4" +"remotelinux_dependencies" "3" +"remotelinux_dependencies.pri" "4" +"symbianutils" "3" +"symbianutils.pri" "4" +"texteditor" "3" +"texteditor.pri" "4" +"texteditor_dependencies" "3" +"texteditor_dependencies.pri" "4" +"utils" "3" +"utils.pri" "4" +"utils_dependencies" "3" +"utils_dependencies.pri" "4" +"Headers" "3" +"analyzerbase_global.h" "4" +"analyzerconstants.h" "4" +"analyzermanager.h" "4" +"analyzeroptionspage.h" "4" +"analyzerplugin.h" "4" +"analyzerrunconfigwidget.h" "4" +"analyzerruncontrol.h" "4" +"analyzerruncontrolfactory.h" "4" +"analyzersettings.h" "4" +"analyzerstartparameters.h" "4" +"analyzerutils.h" "4" +"ianalyzerengine.h" "4" +"ianalyzertool.h" "4" +"startremotedialog.h" "4" +"Sources" "3" +"analyzermanager.cpp" "4" +"analyzeroptionspage.cpp" "4" +"analyzerplugin.cpp" "4" +"analyzerrunconfigwidget.cpp" "4" +"analyzerruncontrol.cpp" "4" +"analyzerruncontrolfactory.cpp" "4" +"analyzersettings.cpp" "4" +"analyzerutils.cpp" "4" +"ianalyzerengine.cpp" "4" +"ianalyzertool.cpp" "4" +"startremotedialog.cpp" "4" +"Forms" "3" +"startremotedialog.ui" "4" +"Resources" "3" +"analyzerbase.qrc" "4" +"autotoolsprojectmanager" "2" +"autotoolsprojectmanager.pro" "3" +"aggregation" "3" +"aggregation.pri" "4" +"autotoolsprojectmanager_dependencies" "3" +"autotoolsprojectmanager_dependencies.pri" "4" +"coreplugin" "3" +"coreplugin.pri" "4" +"coreplugin_dependencies" "3" +"coreplugin_dependencies.pri" "4" +"cplusplus" "3" +"cplusplus.pri" "4" +"cpptools" "3" +"cpptools.pri" "4" +"cpptools_dependencies" "3" +"cpptools_dependencies.pri" "4" +"extensionsystem" "3" +"extensionsystem.pri" "4" +"extensionsystem_dependencies" "3" +"extensionsystem_dependencies.pri" "4" +"find" "3" +"find.pri" "4" +"find_dependencies" "3" +"find_dependencies.pri" "4" +"languageutils" "3" +"languageutils.pri" "4" +"locator" "3" +"locator.pri" "4" +"locator_dependencies" "3" +"locator_dependencies.pri" "4" +"projectexplorer" "3" +"projectexplorer.pri" "4" +"projectexplorer_dependencies" "3" +"projectexplorer_dependencies.pri" "4" +"qtcreator" "3" +"qtcreator.pri" "4" +"qtcreatorplugin" "3" +"qtcreatorplugin.pri" "4" +"Other files" "4" +"plugins/autotoolsprojectmanager" "5" +"AutotoolsProjectManager.pluginspec.in" "6" +"texteditor" "3" +"texteditor.pri" "4" +"texteditor_dependencies" "3" +"texteditor_dependencies.pri" "4" +"utils" "3" +"utils.pri" "4" +"utils_dependencies" "3" +"utils_dependencies.pri" "4" +"Headers" "3" +"autogenstep.h" "4" +"autoreconfstep.h" "4" +"autotoolsbuildconfiguration.h" "4" +"autotoolsbuildsettingswidget.h" "4" +"autotoolsmanager.h" "4" +"autotoolsopenprojectwizard.h" "4" +"autotoolsproject.h" "4" +"autotoolsprojectconstants.h" "4" +"autotoolsprojectfile.h" "4" +"autotoolsprojectnode.h" "4" +"autotoolsprojectplugin.h" "4" +"autotoolstarget.h" "4" +"configurestep.h" "4" +"makefileparser.h" "4" +"makefileparserthread.h" "4" +"makestep.h" "4" +"Sources" "3" +"autogenstep.cpp" "4" +"autoreconfstep.cpp" "4" +"autotoolsbuildconfiguration.cpp" "4" +"autotoolsbuildsettingswidget.cpp" "4" +"autotoolsmanager.cpp" "4" +"autotoolsopenprojectwizard.cpp" "4" +"autotoolsproject.cpp" "4" +"autotoolsprojectfile.cpp" "4" +"autotoolsprojectnode.cpp" "4" +"autotoolsprojectplugin.cpp" "4" +"autotoolstarget.cpp" "4" +"configurestep.cpp" "4" +"makefileparser.cpp" "4" +"makefileparserthread.cpp" "4" +"makestep.cpp" "4" +"Resources" "3" +"autotoolsproject.qrc" "4" +"bazaar" "2" +"bazaar.pro" "3" +"aggregation" "3" +"aggregation.pri" "4" +"bazaar_dependencies" "3" +"bazaar_dependencies.pri" "4" +"coreplugin" "3" +"coreplugin.pri" "4" +"coreplugin_dependencies" "3" +"coreplugin_dependencies.pri" "4" +"cplusplus" "3" +"cplusplus.pri" "4" +"extensionsystem" "3" +"extensionsystem.pri" "4" +"extensionsystem_dependencies" "3" +"extensionsystem_dependencies.pri" "4" +"find" "3" +"find.pri" "4" +"find_dependencies" "3" +"find_dependencies.pri" "4" +"locator" "3" +"locator.pri" "4" +"locator_dependencies" "3" +"locator_dependencies.pri" "4" +"projectexplorer" "3" +"projectexplorer.pri" "4" +"projectexplorer_dependencies" "3" +"projectexplorer_dependencies.pri" "4" +"qtcreator" "3" +"qtcreator.pri" "4" +"qtcreatorplugin" "3" +"qtcreatorplugin.pri" "4" +"Other files" "4" +"plugins/bazaar" "5" +"Bazaar.pluginspec.in" "6" +"texteditor" "3" +"texteditor.pri" "4" +"texteditor_dependencies" "3" +"texteditor_dependencies.pri" "4" +"utils" "3" +"utils.pri" "4" +"utils_dependencies" "3" +"utils_dependencies.pri" "4" +"vcsbase" "3" +"vcsbase.pri" "4" +"vcsbase_dependencies" "3" +"vcsbase_dependencies.pri" "4" +"Headers" "3" +"annotationhighlighter.h" "4" +"bazaarclient.h" "4" +"bazaarcommitwidget.h" "4" +"bazaarcontrol.h" "4" +"bazaareditor.h" "4" +"bazaarplugin.h" "4" +"bazaarsettings.h" "4" +"branchinfo.h" "4" +"cloneoptionspanel.h" "4" +"clonewizard.h" "4" +"clonewizardpage.h" "4" +"commiteditor.h" "4" +"constants.h" "4" +"optionspage.h" "4" +"pullorpushdialog.h" "4" +"Sources" "3" +"annotationhighlighter.cpp" "4" +"bazaarclient.cpp" "4" +"bazaarcommitwidget.cpp" "4" +"bazaarcontrol.cpp" "4" +"bazaareditor.cpp" "4" +"bazaarplugin.cpp" "4" +"bazaarsettings.cpp" "4" +"branchinfo.cpp" "4" +"cloneoptionspanel.cpp" "4" +"clonewizard.cpp" "4" +"clonewizardpage.cpp" "4" +"commiteditor.cpp" "4" +"optionspage.cpp" "4" +"pullorpushdialog.cpp" "4" +"Forms" "3" +"bazaarcommitpanel.ui" "4" +"cloneoptionspanel.ui" "4" +"optionspage.ui" "4" +"pullorpushdialog.ui" "4" +"revertdialog.ui" "4" +"Resources" "3" +"bazaar.qrc" "4" +"bineditor" "2" +"bineditor.pro" "3" +"aggregation" "3" +"aggregation.pri" "4" +"bineditor_dependencies" "3" +"bineditor_dependencies.pri" "4" +"coreplugin" "3" +"coreplugin.pri" "4" +"coreplugin_dependencies" "3" +"coreplugin_dependencies.pri" "4" +"extensionsystem" "3" +"extensionsystem.pri" "4" +"extensionsystem_dependencies" "3" +"extensionsystem_dependencies.pri" "4" +"find" "3" +"find.pri" "4" +"find_dependencies" "3" +"find_dependencies.pri" "4" +"locator" "3" +"locator.pri" "4" +"locator_dependencies" "3" +"locator_dependencies.pri" "4" +"qtcreator" "3" +"qtcreator.pri" "4" +"qtcreatorplugin" "3" +"qtcreatorplugin.pri" "4" +"Other files" "4" +"plugins/bineditor" "5" +"BinEditor.pluginspec.in" "6" +"texteditor" "3" +"texteditor.pri" "4" +"texteditor_dependencies" "3" +"texteditor_dependencies.pri" "4" +"utils" "3" +"utils.pri" "4" +"utils_dependencies" "3" +"utils_dependencies.pri" "4" +"Headers" "3" +"bineditor.h" "4" +"bineditorconstants.h" "4" +"bineditorplugin.h" "4" +"markup.h" "4" +"Sources" "3" +"bineditor.cpp" "4" +"bineditorplugin.cpp" "4" +"bookmarks" "2" +"bookmarks.pro" "3" +"aggregation" "3" +"aggregation.pri" "4" +"coreplugin" "3" +"coreplugin.pri" "4" +"coreplugin_dependencies" "3" +"coreplugin_dependencies.pri" "4" +"extensionsystem" "3" +"extensionsystem.pri" "4" +"extensionsystem_dependencies" "3" +"extensionsystem_dependencies.pri" "4" +"find" "3" +"find.pri" "4" +"find_dependencies" "3" +"find_dependencies.pri" "4" +"locator" "3" +"locator.pri" "4" +"locator_dependencies" "3" +"locator_dependencies.pri" "4" +"projectexplorer" "3" +"projectexplorer.pri" "4" +"projectexplorer_dependencies" "3" +"projectexplorer_dependencies.pri" "4" +"qtcreator" "3" +"qtcreator.pri" "4" +"qtcreatorplugin" "3" +"qtcreatorplugin.pri" "4" +"Other files" "4" +"plugins/bookmarks" "5" +"Bookmarks.pluginspec.in" "6" +"texteditor" "3" +"texteditor.pri" "4" +"texteditor_dependencies" "3" +"texteditor_dependencies.pri" "4" +"utils" "3" +"utils.pri" "4" +"utils_dependencies" "3" +"utils_dependencies.pri" "4" +"Headers" "3" +"bookmark.h" "4" +"bookmarkmanager.h" "4" +"bookmarks_global.h" "4" +"bookmarksplugin.h" "4" +"Sources" "3" +"bookmark.cpp" "4" +"bookmarkmanager.cpp" "4" +"bookmarksplugin.cpp" "4" +"Resources" "3" +"bookmarks.qrc" "4" +"classview" "2" +"classview.pro" "3" +"aggregation" "3" +"aggregation.pri" "4" +"classview_dependencies" "3" +"classview_dependencies.pri" "4" +"coreplugin" "3" +"coreplugin.pri" "4" +"coreplugin_dependencies" "3" +"coreplugin_dependencies.pri" "4" +"cplusplus" "3" +"cplusplus.pri" "4" +"cpptools" "3" +"cpptools.pri" "4" +"cpptools_dependencies" "3" +"cpptools_dependencies.pri" "4" +"extensionsystem" "3" +"extensionsystem.pri" "4" +"extensionsystem_dependencies" "3" +"extensionsystem_dependencies.pri" "4" +"find" "3" +"find.pri" "4" +"find_dependencies" "3" +"find_dependencies.pri" "4" +"languageutils" "3" +"languageutils.pri" "4" +"locator" "3" +"locator.pri" "4" +"locator_dependencies" "3" +"locator_dependencies.pri" "4" +"projectexplorer" "3" +"projectexplorer.pri" "4" +"projectexplorer_dependencies" "3" +"projectexplorer_dependencies.pri" "4" +"qtcreator" "3" +"qtcreator.pri" "4" +"qtcreatorplugin" "3" +"qtcreatorplugin.pri" "4" +"Other files" "4" +"plugins/classview" "5" +"ClassView.pluginspec.in" "6" +"texteditor" "3" +"texteditor.pri" "4" +"texteditor_dependencies" "3" +"texteditor_dependencies.pri" "4" +"utils" "3" +"utils.pri" "4" +"utils_dependencies" "3" +"utils_dependencies.pri" "4" +"Headers" "3" +"classviewconstants.h" "4" +"classviewmanager.h" "4" +"classviewnavigationwidget.h" "4" +"classviewnavigationwidgetfactory.h" "4" +"classviewparser.h" "4" +"classviewparsertreeitem.h" "4" +"classviewplugin.h" "4" +"classviewsymbolinformation.h" "4" +"classviewsymbollocation.h" "4" +"classviewtreeitemmodel.h" "4" +"classviewutils.h" "4" +"Sources" "3" +"classviewmanager.cpp" "4" +"classviewnavigationwidget.cpp" "4" +"classviewnavigationwidgetfactory.cpp" "4" +"classviewparser.cpp" "4" +"classviewparsertreeitem.cpp" "4" +"classviewplugin.cpp" "4" +"classviewsymbolinformation.cpp" "4" +"classviewsymbollocation.cpp" "4" +"classviewtreeitemmodel.cpp" "4" +"classviewutils.cpp" "4" +"Forms" "3" +"classviewnavigationwidget.ui" "4" +"Resources" "3" +"classview.qrc" "4" +"cmakeprojectmanager" "2" +"cmakeprojectmanager.pro" "3" +"aggregation" "3" +"aggregation.pri" "4" +"cmakeprojectmanager_dependencies" "3" +"cmakeprojectmanager_dependencies.pri" "4" +"coreplugin" "3" +"coreplugin.pri" "4" +"coreplugin_dependencies" "3" +"coreplugin_dependencies.pri" "4" +"cplusplus" "3" +"cplusplus.pri" "4" +"cpptools" "3" +"cpptools.pri" "4" +"cpptools_dependencies" "3" +"cpptools_dependencies.pri" "4" +"extensionsystem" "3" +"extensionsystem.pri" "4" +"extensionsystem_dependencies" "3" +"extensionsystem_dependencies.pri" "4" +"find" "3" +"find.pri" "4" +"find_dependencies" "3" +"find_dependencies.pri" "4" +"languageutils" "3" +"languageutils.pri" "4" +"locator" "3" +"locator.pri" "4" +"locator_dependencies" "3" +"locator_dependencies.pri" "4" +"projectexplorer" "3" +"projectexplorer.pri" "4" +"projectexplorer_dependencies" "3" +"projectexplorer_dependencies.pri" "4" +"qtcreator" "3" +"qtcreator.pri" "4" +"qtcreatorplugin" "3" +"qtcreatorplugin.pri" "4" +"Other files" "4" +"plugins/cmakeprojectmanager" "5" +"CMakeProjectManager.pluginspec.in" "6" +"texteditor" "3" +"texteditor.pri" "4" +"texteditor_dependencies" "3" +"texteditor_dependencies.pri" "4" +"utils" "3" +"utils.pri" "4" +"utils_dependencies" "3" +"utils_dependencies.pri" "4" +"Headers" "3" +"cmakebuildconfiguration.h" "4" +"cmakeeditor.h" "4" +"cmakeeditorfactory.h" "4" +"cmakehighlighter.h" "4" +"cmakelocatorfilter.h" "4" +"cmakeopenprojectwizard.h" "4" +"cmakeproject.h" "4" +"cmakeprojectconstants.h" "4" +"cmakeprojectmanager.h" "4" +"cmakeprojectnodes.h" "4" +"cmakeprojectplugin.h" "4" +"cmakerunconfiguration.h" "4" +"cmaketarget.h" "4" +"cmakeuicodemodelsupport.h" "4" +"makestep.h" "4" +"Sources" "3" +"cmakebuildconfiguration.cpp" "4" +"cmakeeditor.cpp" "4" +"cmakeeditorfactory.cpp" "4" +"cmakehighlighter.cpp" "4" +"cmakelocatorfilter.cpp" "4" +"cmakeopenprojectwizard.cpp" "4" +"cmakeproject.cpp" "4" +"cmakeprojectmanager.cpp" "4" +"cmakeprojectnodes.cpp" "4" +"cmakeprojectplugin.cpp" "4" +"cmakerunconfiguration.cpp" "4" +"cmaketarget.cpp" "4" +"cmakeuicodemodelsupport.cpp" "4" +"makestep.cpp" "4" +"Resources" "3" +"cmakeproject.qrc" "4" +"Other files" "3" +"CMakeProject.mimetypes.xml" "4" +"coreplugin" "2" +"coreplugin.pro" "3" +"aggregation" "3" +"aggregation.pri" "4" +"coreplugin_dependencies" "3" +"coreplugin_dependencies.pri" "4" +"extensionsystem" "3" +"extensionsystem.pri" "4" +"extensionsystem_dependencies" "3" +"extensionsystem_dependencies.pri" "4" +"qtcreator" "3" +"qtcreator.pri" "4" +"qtcreatorplugin" "3" +"qtcreatorplugin.pri" "4" +"Other files" "4" +"plugins/coreplugin" "5" +"Core.pluginspec.in" "6" +"scriptwrapper" "3" +"scriptwrapper.pri" "4" +"Headers" "4" +"wrap_helpers.h" "5" +"utils" "3" +"utils.pri" "4" +"utils_dependencies" "3" +"utils_dependencies.pri" "4" +"Headers" "3" +"actionmanager" "4" +"actioncontainer.h" "5" +"actioncontainer_p.h" "5" +"actionmanager.h" "5" +"actionmanager_p.h" "5" +"command.h" "5" +"command_p.h" "5" +"commandmappings.h" "5" +"commandsfile.h" "5" +"dialogs" "4" +"externaltoolconfig.h" "5" +"ioptionspage.h" "5" +"iwizard.h" "5" +"newdialog.h" "5" +"openwithdialog.h" "5" +"promptoverwritedialog.h" "5" +"saveitemsdialog.h" "5" +"settingsdialog.h" "5" +"shortcutsettings.h" "5" +"editormanager" "4" +"editormanager.h" "5" +"editorview.h" "5" +"ieditor.h" "5" +"ieditorfactory.h" "5" +"iexternaleditor.h" "5" +"openeditorsmodel.h" "5" +"openeditorsview.h" "5" +"openeditorswindow.h" "5" +"systemeditor.h" "5" +"progressmanager" "4" +"futureprogress.h" "5" +"progressbar.h" "5" +"progressmanager.h" "5" +"progressmanager_p.h" "5" +"progressview.h" "5" +"scriptmanager" "4" +"metatypedeclarations.h" "5" +"scriptmanager.h" "5" +"scriptmanager_p.h" "5" +"basefilewizard.h" "4" +"core_global.h" "4" +"coreconstants.h" "4" +"coreplugin.h" "4" +"designmode.h" "4" +"documentmanager.h" "4" +"editmode.h" "4" +"editortoolbar.h" "4" +"externaltool.h" "4" +"externaltoolmanager.h" "4" +"fancyactionbar.h" "4" +"fancytabwidget.h" "4" +"featureprovider.h" "4" +"fileiconprovider.h" "4" +"fileutils.h" "4" +"findplaceholder.h" "4" +"generalsettings.h" "4" +"generatedfile.h" "4" +"helpmanager.h" "4" +"icontext.h" "4" +"icore.h" "4" +"icorelistener.h" "4" +"id.h" "4" +"idocument.h" "4" +"idocumentfactory.h" "4" +"ifilewizardextension.h" "4" +"imode.h" "4" +"inavigationwidgetfactory.h" "4" +"infobar.h" "4" +"ioutputpane.h" "4" +"iversioncontrol.h" "4" +"mainwindow.h" "4" +"manhattanstyle.h" "4" +"messagemanager.h" "4" +"messageoutputwindow.h" "4" +"mimedatabase.h" "4" +"mimetypemagicdialog.h" "4" +"mimetypesettings.h" "4" +"minisplitter.h" "4" +"modemanager.h" "4" +"navigationsubwidget.h" "4" +"navigationwidget.h" "4" +"outputpane.h" "4" +"outputpanemanager.h" "4" +"outputwindow.h" "4" +"plugindialog.h" "4" +"rightpane.h" "4" +"settingsdatabase.h" "4" +"sidebar.h" "4" +"sidebarwidget.h" "4" +"statusbarmanager.h" "4" +"statusbarwidget.h" "4" +"styleanimator.h" "4" +"tabpositionindicator.h" "4" +"textdocument.h" "4" +"toolsettings.h" "4" +"variablechooser.h" "4" +"variablemanager.h" "4" +"vcsmanager.h" "4" +"versiondialog.h" "4" +"Sources" "3" +"actionmanager" "4" +"actioncontainer.cpp" "5" +"actionmanager.cpp" "5" +"command.cpp" "5" +"commandmappings.cpp" "5" +"commandsfile.cpp" "5" +"dialogs" "4" +"externaltoolconfig.cpp" "5" +"ioptionspage.cpp" "5" +"iwizard.cpp" "5" +"newdialog.cpp" "5" +"openwithdialog.cpp" "5" +"promptoverwritedialog.cpp" "5" +"saveitemsdialog.cpp" "5" +"settingsdialog.cpp" "5" +"shortcutsettings.cpp" "5" +"editormanager" "4" +"editormanager.cpp" "5" +"editorview.cpp" "5" +"ieditor.cpp" "5" +"iexternaleditor.cpp" "5" +"openeditorsmodel.cpp" "5" +"openeditorsview.cpp" "5" +"openeditorswindow.cpp" "5" +"systemeditor.cpp" "5" +"progressmanager" "4" +"futureprogress.cpp" "5" +"progressbar.cpp" "5" +"progressmanager.cpp" "5" +"progressmanager_mac.mm" "5" +"progressmanager_win.cpp" "5" +"progressmanager_x11.cpp" "5" +"progressview.cpp" "5" +"scriptmanager" "4" +"scriptmanager.cpp" "5" +"basefilewizard.cpp" "4" +"coreplugin.cpp" "4" +"designmode.cpp" "4" +"documentmanager.cpp" "4" +"editmode.cpp" "4" +"editortoolbar.cpp" "4" +"externaltool.cpp" "4" +"fancyactionbar.cpp" "4" +"fancytabwidget.cpp" "4" +"featureprovider.cpp" "4" +"fileiconprovider.cpp" "4" +"fileutils.cpp" "4" +"findplaceholder.cpp" "4" +"generalsettings.cpp" "4" +"generatedfile.cpp" "4" +"helpmanager.cpp" "4" +"icore.cpp" "4" +"id.cpp" "4" +"idocument.cpp" "4" +"imode.cpp" "4" +"inavigationwidgetfactory.cpp" "4" +"infobar.cpp" "4" +"mainwindow.cpp" "4" +"manhattanstyle.cpp" "4" +"messagemanager.cpp" "4" +"messageoutputwindow.cpp" "4" +"mimedatabase.cpp" "4" +"mimetypemagicdialog.cpp" "4" +"mimetypesettings.cpp" "4" +"minisplitter.cpp" "4" +"modemanager.cpp" "4" +"navigationsubwidget.cpp" "4" +"navigationwidget.cpp" "4" +"outputpane.cpp" "4" +"outputpanemanager.cpp" "4" +"outputwindow.cpp" "4" +"plugindialog.cpp" "4" +"rightpane.cpp" "4" +"settingsdatabase.cpp" "4" +"sidebar.cpp" "4" +"sidebarwidget.cpp" "4" +"statusbarmanager.cpp" "4" +"statusbarwidget.cpp" "4" +"styleanimator.cpp" "4" +"tabpositionindicator.cpp" "4" +"textdocument.cpp" "4" +"toolsettings.cpp" "4" +"variablechooser.cpp" "4" +"variablemanager.cpp" "4" +"vcsmanager.cpp" "4" +"versiondialog.cpp" "4" +"Forms" "3" +"actionmanager" "4" +"commandmappings.ui" "5" +"dialogs" "4" +"externaltoolconfig.ui" "5" +"newdialog.ui" "5" +"openwithdialog.ui" "5" +"saveitemsdialog.ui" "5" +"editormanager" "4" +"openeditorsview.ui" "5" +"generalsettings.ui" "4" +"mimetypemagicdialog.ui" "4" +"mimetypesettingspage.ui" "4" +"variablechooser.ui" "4" +"Resources" "3" +"core.qrc" "4" +"fancyactionbar.qrc" "4" +"Other files" "3" +"editormanager" "4" +"BinFiles.mimetypes.xml" "5" +"cpaster" "2" +"cpaster.pro" "3" +"aggregation" "3" +"aggregation.pri" "4" +"coreplugin" "3" +"coreplugin.pri" "4" +"coreplugin_dependencies" "3" +"coreplugin_dependencies.pri" "4" +"cpaster" "3" +"cpaster.pri" "4" +"Headers" "4" +"cgi.h" "5" +"splitter.h" "5" +"Sources" "4" +"cgi.cpp" "5" +"splitter.cpp" "5" +"cpaster_dependencies" "3" +"cpaster_dependencies.pri" "4" +"extensionsystem" "3" +"extensionsystem.pri" "4" +"extensionsystem_dependencies" "3" +"extensionsystem_dependencies.pri" "4" +"find" "3" +"find.pri" "4" +"find_dependencies" "3" +"find_dependencies.pri" "4" +"locator" "3" +"locator.pri" "4" +"locator_dependencies" "3" +"locator_dependencies.pri" "4" +"qtcreator" "3" +"qtcreator.pri" "4" +"qtcreatorplugin" "3" +"qtcreatorplugin.pri" "4" +"Other files" "4" +"plugins/cpaster" "5" +"CodePaster.pluginspec.in" "6" +"texteditor" "3" +"texteditor.pri" "4" +"texteditor_dependencies" "3" +"texteditor_dependencies.pri" "4" +"utils" "3" +"utils.pri" "4" +"utils_dependencies" "3" +"utils_dependencies.pri" "4" +"Headers" "3" +"codepasterprotocol.h" "4" +"codepastersettings.h" "4" +"columnindicatortextedit.h" "4" +"cpasterconstants.h" "4" +"cpasterplugin.h" "4" +"fileshareprotocol.h" "4" +"fileshareprotocolsettingspage.h" "4" +"kdepasteprotocol.h" "4" +"pastebindotcaprotocol.h" "4" +"pastebindotcomprotocol.h" "4" +"pasteselectdialog.h" "4" +"pasteview.h" "4" +"protocol.h" "4" +"settings.h" "4" +"settingspage.h" "4" +"Sources" "3" +"codepasterprotocol.cpp" "4" +"codepastersettings.cpp" "4" +"columnindicatortextedit.cpp" "4" +"cpasterplugin.cpp" "4" +"fileshareprotocol.cpp" "4" +"fileshareprotocolsettingspage.cpp" "4" +"kdepasteprotocol.cpp" "4" +"pastebindotcaprotocol.cpp" "4" +"pastebindotcomprotocol.cpp" "4" +"pasteselectdialog.cpp" "4" +"pasteview.cpp" "4" +"protocol.cpp" "4" +"settings.cpp" "4" +"settingspage.cpp" "4" +"Forms" "3" +"fileshareprotocolsettingswidget.ui" "4" +"pastebindotcomsettings.ui" "4" +"pasteselect.ui" "4" +"pasteview.ui" "4" +"settingspage.ui" "4" +"cppeditor" "2" +"cppeditor.pro" "3" +"aggregation" "3" +"aggregation.pri" "4" +"coreplugin" "3" +"coreplugin.pri" "4" +"coreplugin_dependencies" "3" +"coreplugin_dependencies.pri" "4" +"cplusplus" "3" +"cplusplus.pri" "4" +"cppeditor_dependencies" "3" +"cppeditor_dependencies.pri" "4" +"cpptools" "3" +"cpptools.pri" "4" +"cpptools_dependencies" "3" +"cpptools_dependencies.pri" "4" +"extensionsystem" "3" +"extensionsystem.pri" "4" +"extensionsystem_dependencies" "3" +"extensionsystem_dependencies.pri" "4" +"find" "3" +"find.pri" "4" +"find_dependencies" "3" +"find_dependencies.pri" "4" +"languageutils" "3" +"languageutils.pri" "4" +"locator" "3" +"locator.pri" "4" +"locator_dependencies" "3" +"locator_dependencies.pri" "4" +"projectexplorer" "3" +"projectexplorer.pri" "4" +"projectexplorer_dependencies" "3" +"projectexplorer_dependencies.pri" "4" +"qtcreator" "3" +"qtcreator.pri" "4" +"qtcreatorplugin" "3" +"qtcreatorplugin.pri" "4" +"Other files" "4" +"plugins/cppeditor" "5" +"CppEditor.pluginspec.in" "6" +"texteditor" "3" +"texteditor.pri" "4" +"texteditor_dependencies" "3" +"texteditor_dependencies.pri" "4" +"utils" "3" +"utils.pri" "4" +"utils_dependencies" "3" +"utils_dependencies.pri" "4" +"Headers" "3" +"cppautocompleter.h" "4" +"cppclasswizard.h" "4" +"cppcompleteswitch.h" "4" +"cppeditor.h" "4" +"cppeditor_global.h" "4" +"cppeditorconstants.h" "4" +"cppeditorenums.h" "4" +"cppelementevaluator.h" "4" +"cppfilewizard.h" "4" +"cppfunctiondecldeflink.h" "4" +"cpphighlighter.h" "4" +"cpphoverhandler.h" "4" +"cppinsertdecldef.h" "4" +"cppinsertqtpropertymembers.h" "4" +"cppoutline.h" "4" +"cppplugin.h" "4" +"cppquickfix.h" "4" +"cppquickfixassistant.h" "4" +"cppsnippetprovider.h" "4" +"cpptypehierarchy.h" "4" +"Sources" "3" +"cppautocompleter.cpp" "4" +"cppclasswizard.cpp" "4" +"cppcompleteswitch.cpp" "4" +"cppeditor.cpp" "4" +"cppelementevaluator.cpp" "4" +"cppfilewizard.cpp" "4" +"cppfunctiondecldeflink.cpp" "4" +"cpphighlighter.cpp" "4" +"cpphoverhandler.cpp" "4" +"cppinsertdecldef.cpp" "4" +"cppinsertqtpropertymembers.cpp" "4" +"cppoutline.cpp" "4" +"cppplugin.cpp" "4" +"cppquickfix.cpp" "4" +"cppquickfixassistant.cpp" "4" +"cppquickfixes.cpp" "4" +"cppsnippetprovider.cpp" "4" +"cpptypehierarchy.cpp" "4" +"Resources" "3" +"cppeditor.qrc" "4" +"Other files" "3" +"CppEditor.mimetypes.xml" "4" +"cpptools" "2" +"cpptools.pro" "3" +"aggregation" "3" +"aggregation.pri" "4" +"coreplugin" "3" +"coreplugin.pri" "4" +"coreplugin_dependencies" "3" +"coreplugin_dependencies.pri" "4" +"cplusplus" "3" +"cplusplus.pri" "4" +"cpptools_dependencies" "3" +"cpptools_dependencies.pri" "4" +"extensionsystem" "3" +"extensionsystem.pri" "4" +"extensionsystem_dependencies" "3" +"extensionsystem_dependencies.pri" "4" +"find" "3" +"find.pri" "4" +"find_dependencies" "3" +"find_dependencies.pri" "4" +"languageutils" "3" +"languageutils.pri" "4" +"locator" "3" +"locator.pri" "4" +"locator_dependencies" "3" +"locator_dependencies.pri" "4" +"projectexplorer" "3" +"projectexplorer.pri" "4" +"projectexplorer_dependencies" "3" +"projectexplorer_dependencies.pri" "4" +"qtcreator" "3" +"qtcreator.pri" "4" +"qtcreatorplugin" "3" +"qtcreatorplugin.pri" "4" +"Other files" "4" +"plugins/cpptools" "5" +"CppTools.pluginspec.in" "6" +"texteditor" "3" +"texteditor.pri" "4" +"texteditor_dependencies" "3" +"texteditor_dependencies.pri" "4" +"utils" "3" +"utils.pri" "4" +"utils_dependencies" "3" +"utils_dependencies.pri" "4" +"Headers" "3" +"abstracteditorsupport.h" "4" +"commentssettings.h" "4" +"completionsettingspage.h" "4" +"cppchecksymbols.h" "4" +"cppclassesfilter.h" "4" +"cppcodeformatter.h" "4" +"cppcodestylepreferences.h" "4" +"cppcodestylepreferencesfactory.h" "4" +"cppcodestylesettings.h" "4" +"cppcodestylesettingspage.h" "4" +"cppcompletionassist.h" "4" +"cppcompletionassistprovider.h" "4" +"cppcompletionsupport.h" "4" +"cppcurrentdocumentfilter.h" "4" +"cppdoxygen.h" "4" +"cppfilesettingspage.h" "4" +"cppfindreferences.h" "4" +"cppfunctionsfilter.h" "4" +"cpphighlightingsupport.h" "4" +"cpphighlightingsupportinternal.h" "4" +"cpplocalsymbols.h" "4" +"cpplocatorfilter.h" "4" +"cppmodelmanager.h" "4" +"cppqtstyleindenter.h" "4" +"cpprefactoringchanges.h" "4" +"cppsemanticinfo.h" "4" +"cpptools_global.h" "4" +"cpptoolsconstants.h" "4" +"cpptoolseditorsupport.h" "4" +"cpptoolsplugin.h" "4" +"cpptoolsreuse.h" "4" +"cpptoolssettings.h" "4" +"doxygengenerator.h" "4" +"insertionpointlocator.h" "4" +"searchsymbols.h" "4" +"symbolfinder.h" "4" +"symbolsfindfilter.h" "4" +"uicodecompletionsupport.h" "4" +"Sources" "3" +"abstracteditorsupport.cpp" "4" +"commentssettings.cpp" "4" +"completionsettingspage.cpp" "4" +"cppchecksymbols.cpp" "4" +"cppclassesfilter.cpp" "4" +"cppcodeformatter.cpp" "4" +"cppcodegen_test.cpp" "4" +"cppcodestylepreferences.cpp" "4" +"cppcodestylepreferencesfactory.cpp" "4" +"cppcodestylesettings.cpp" "4" +"cppcodestylesettingspage.cpp" "4" +"cppcompletionassist.cpp" "4" +"cppcompletionassistprovider.cpp" "4" +"cppcompletionsupport.cpp" "4" +"cppcurrentdocumentfilter.cpp" "4" +"cppdoxygen.cpp" "4" +"cppfilesettingspage.cpp" "4" +"cppfindreferences.cpp" "4" +"cppfunctionsfilter.cpp" "4" +"cpphighlightingsupport.cpp" "4" +"cpphighlightingsupportinternal.cpp" "4" +"cpplocalsymbols.cpp" "4" +"cpplocatorfilter.cpp" "4" +"cppmodelmanager.cpp" "4" +"cppqtstyleindenter.cpp" "4" +"cpprefactoringchanges.cpp" "4" +"cppsemanticinfo.cpp" "4" +"cpptoolseditorsupport.cpp" "4" +"cpptoolsplugin.cpp" "4" +"cpptoolsreuse.cpp" "4" +"cpptoolssettings.cpp" "4" +"doxygengenerator.cpp" "4" +"insertionpointlocator.cpp" "4" +"searchsymbols.cpp" "4" +"symbolfinder.cpp" "4" +"symbolsfindfilter.cpp" "4" +"uicodecompletionsupport.cpp" "4" +"Forms" "3" +"completionsettingspage.ui" "4" +"cppcodestylesettingspage.ui" "4" +"cppfilesettingspage.ui" "4" +"cvs" "2" +"cvs.pro" "3" +"aggregation" "3" +"aggregation.pri" "4" +"coreplugin" "3" +"coreplugin.pri" "4" +"coreplugin_dependencies" "3" +"coreplugin_dependencies.pri" "4" +"cplusplus" "3" +"cplusplus.pri" "4" +"extensionsystem" "3" +"extensionsystem.pri" "4" +"extensionsystem_dependencies" "3" +"extensionsystem_dependencies.pri" "4" +"find" "3" +"find.pri" "4" +"find_dependencies" "3" +"find_dependencies.pri" "4" +"locator" "3" +"locator.pri" "4" +"locator_dependencies" "3" +"locator_dependencies.pri" "4" +"projectexplorer" "3" +"projectexplorer.pri" "4" +"projectexplorer_dependencies" "3" +"projectexplorer_dependencies.pri" "4" +"qtcreator" "3" +"qtcreator.pri" "4" +"qtcreatorplugin" "3" +"qtcreatorplugin.pri" "4" +"Other files" "4" +"plugins/cvs" "5" +"CVS.pluginspec.in" "6" +"texteditor" "3" +"texteditor.pri" "4" +"texteditor_dependencies" "3" +"texteditor_dependencies.pri" "4" +"utils" "3" +"utils.pri" "4" +"utils_dependencies" "3" +"utils_dependencies.pri" "4" +"vcsbase" "3" +"vcsbase.pri" "4" +"vcsbase_dependencies" "3" +"vcsbase_dependencies.pri" "4" +"Headers" "3" +"annotationhighlighter.h" "4" +"checkoutwizard.h" "4" +"checkoutwizardpage.h" "4" +"cvsconstants.h" "4" +"cvscontrol.h" "4" +"cvseditor.h" "4" +"cvsplugin.h" "4" +"cvssettings.h" "4" +"cvssubmiteditor.h" "4" +"cvsutils.h" "4" +"settingspage.h" "4" +"Sources" "3" +"annotationhighlighter.cpp" "4" +"checkoutwizard.cpp" "4" +"checkoutwizardpage.cpp" "4" +"cvscontrol.cpp" "4" +"cvseditor.cpp" "4" +"cvsplugin.cpp" "4" +"cvssettings.cpp" "4" +"cvssubmiteditor.cpp" "4" +"cvsutils.cpp" "4" +"settingspage.cpp" "4" +"Forms" "3" +"settingspage.ui" "4" +"Resources" "3" +"cvs.qrc" "4" +"debugger" "2" +"debugger.pro" "3" +"aggregation" "3" +"aggregation.pri" "4" +"botan" "3" +"botan.pri" "4" +"cdb" "3" +"cdb.pri" "4" +"Headers" "4" +"bytearrayinputstream.h" "5" +"cdbengine.h" "5" +"cdboptions.h" "5" +"cdboptionspage.h" "5" +"cdbparsehelpers.h" "5" +"Sources" "4" +"bytearrayinputstream.cpp" "5" +"cdbengine.cpp" "5" +"cdboptions.cpp" "5" +"cdboptionspage.cpp" "5" +"cdbparsehelpers.cpp" "5" +"Forms" "4" +"cdboptionspagewidget.ui" "5" +"coreplugin" "3" +"coreplugin.pri" "4" +"coreplugin_dependencies" "3" +"coreplugin_dependencies.pri" "4" +"cplusplus" "3" +"cplusplus.pri" "4" +"cpptools" "3" +"cpptools.pri" "4" +"cpptools_dependencies" "3" +"cpptools_dependencies.pri" "4" +"debugger_dependencies" "3" +"debugger_dependencies.pri" "4" +"extensionsystem" "3" +"extensionsystem.pri" "4" +"extensionsystem_dependencies" "3" +"extensionsystem_dependencies.pri" "4" +"find" "3" +"find.pri" "4" +"find_dependencies" "3" +"find_dependencies.pri" "4" +"gdb" "3" +"gdb.pri" "4" +"Headers" "4" +"abstractgdbadapter.h" "5" +"abstractgdbprocess.h" "5" +"abstractplaingdbadapter.h" "5" +"attachgdbadapter.h" "5" +"codagdbadapter.h" "5" +"coregdbadapter.h" "5" +"gdbengine.h" "5" +"gdbmi.h" "5" +"gdboptionspage.h" "5" +"localgdbprocess.h" "5" +"localplaingdbadapter.h" "5" +"remotegdbprocess.h" "5" +"remotegdbserveradapter.h" "5" +"remoteplaingdbadapter.h" "5" +"symbian.h" "5" +"termgdbadapter.h" "5" +"Sources" "4" +"abstractgdbadapter.cpp" "5" +"abstractgdbprocess.cpp" "5" +"abstractplaingdbadapter.cpp" "5" +"attachgdbadapter.cpp" "5" +"classicgdbengine.cpp" "5" +"codagdbadapter.cpp" "5" +"coregdbadapter.cpp" "5" +"gdbengine.cpp" "5" +"gdbmi.cpp" "5" +"gdboptionspage.cpp" "5" +"localgdbprocess.cpp" "5" +"localplaingdbadapter.cpp" "5" +"pythongdbengine.cpp" "5" +"remotegdbprocess.cpp" "5" +"remotegdbserveradapter.cpp" "5" +"remoteplaingdbadapter.cpp" "5" +"symbian.cpp" "5" +"termgdbadapter.cpp" "5" +"Resources" "4" +"gdb.qrc" "5" +"languageutils" "3" +"languageutils.pri" "4" +"lldbhost" "3" +"lldbhost.pri" "4" +"Headers" "4" +"ipcenginehost.h" "5" +"lldbenginehost.h" "5" +"lldboptionspage.h" "5" +"Sources" "4" +"ipcenginehost.cpp" "5" +"lldbenginehost.cpp" "5" +"lldboptionspage.cpp" "5" +"Forms" "4" +"lldboptionspagewidget.ui" "5" +"locator" "3" +"locator.pri" "4" +"locator_dependencies" "3" +"locator_dependencies.pri" "4" +"modeltest" "3" +"modeltest.pri" "4" +"Headers" "4" +"modeltest.h" "5" +"Sources" "4" +"modeltest.cpp" "5" +"pdb" "3" +"pdb.pri" "4" +"Headers" "4" +"pdbengine.h" "5" +"Sources" "4" +"pdbengine.cpp" "5" +"projectexplorer" "3" +"projectexplorer.pri" "4" +"projectexplorer_dependencies" "3" +"projectexplorer_dependencies.pri" "4" +"qml" "3" +"qml.pri" "4" +"Headers" "4" +"interactiveinterpreter.h" "5" +"qmladapter.h" "5" +"qmlcppengine.h" "5" +"qmlengine.h" "5" +"qmlv8debuggerclient.h" "5" +"qmlv8debuggerclientconstants.h" "5" +"qscriptdebuggerclient.h" "5" +"Sources" "4" +"interactiveinterpreter.cpp" "5" +"qmladapter.cpp" "5" +"qmlcppengine.cpp" "5" +"qmlengine.cpp" "5" +"qmlv8debuggerclient.cpp" "5" +"qscriptdebuggerclient.cpp" "5" +"qmljs" "3" +"qmljs.pri" "4" +"qtcreator" "3" +"qtcreator.pri" "4" +"qtcreatorplugin" "3" +"qtcreatorplugin.pri" "4" +"Other files" "4" +"plugins/debugger" "5" +"Debugger.pluginspec.in" "6" +"registryaccess" "3" +"registryaccess.pri" "4" +"Headers" "4" +"registryaccess.h" "5" +"Sources" "4" +"registryaccess.cpp" "5" +"script" "3" +"script.pri" "4" +"Headers" "4" +"scriptengine.h" "5" +"Sources" "4" +"scriptengine.cpp" "5" +"shared" "3" +"shared.pri" "4" +"Headers" "4" +"backtrace.h" "5" +"cdbsymbolpathlisteditor.h" "5" +"hostutils.h" "5" +"peutils.h" "5" +"Sources" "4" +"backtrace.cpp" "5" +"cdbsymbolpathlisteditor.cpp" "5" +"hostutils.cpp" "5" +"peutils.cpp" "5" +"symbianutils" "3" +"symbianutils.pri" "4" +"texteditor" "3" +"texteditor.pri" "4" +"texteditor_dependencies" "3" +"texteditor_dependencies.pri" "4" +"utils" "3" +"utils.pri" "4" +"utils_dependencies" "3" +"utils_dependencies.pri" "4" +"Headers" "3" +"basewindow.h" "4" +"breakhandler.h" "4" +"breakpoint.h" "4" +"breakpointmarker.h" "4" +"breakwindow.h" "4" +"commonoptionspage.h" "4" +"debugger_global.h" "4" +"debuggeractions.h" "4" +"debuggerconstants.h" "4" +"debuggercore.h" "4" +"debuggerdialogs.h" "4" +"debuggerengine.h" "4" +"debuggerinternalconstants.h" "4" +"debuggermainwindow.h" "4" +"debuggerplugin.h" "4" +"debuggerruncontrolfactory.h" "4" +"debuggerrunner.h" "4" +"debuggersourcepathmappingwidget.h" "4" +"debuggerstartparameters.h" "4" +"debuggerstreamops.h" "4" +"debuggerstringutils.h" "4" +"debuggertoolchaincombobox.h" "4" +"debuggertooltipmanager.h" "4" +"disassembleragent.h" "4" +"disassemblerlines.h" "4" +"logwindow.h" "4" +"memoryagent.h" "4" +"memoryview.h" "4" +"moduleshandler.h" "4" +"moduleswindow.h" "4" +"name_demangler.h" "4" +"outputcollector.h" "4" +"procinterrupt.h" "4" +"qtmessagelogeditor.h" "4" +"qtmessageloghandler.h" "4" +"qtmessagelogitemdelegate.h" "4" +"qtmessagelogproxymodel.h" "4" +"qtmessagelogview.h" "4" +"qtmessagelogwindow.h" "4" +"registerhandler.h" "4" +"registerpostmortemaction.h" "4" +"registerwindow.h" "4" +"snapshothandler.h" "4" +"snapshotwindow.h" "4" +"sourceagent.h" "4" +"sourcefileshandler.h" "4" +"sourcefileswindow.h" "4" +"stackframe.h" "4" +"stackhandler.h" "4" +"stackwindow.h" "4" +"threaddata.h" "4" +"threadshandler.h" "4" +"threadswindow.h" "4" +"watchdelegatewidgets.h" "4" +"watchhandler.h" "4" +"watchutils.h" "4" +"watchwindow.h" "4" +"Sources" "3" +"basewindow.cpp" "4" +"breakhandler.cpp" "4" +"breakpoint.cpp" "4" +"breakpointmarker.cpp" "4" +"breakwindow.cpp" "4" +"commonoptionspage.cpp" "4" +"debuggeractions.cpp" "4" +"debuggerdialogs.cpp" "4" +"debuggerengine.cpp" "4" +"debuggermainwindow.cpp" "4" +"debuggerplugin.cpp" "4" +"debuggerrunner.cpp" "4" +"debuggersourcepathmappingwidget.cpp" "4" +"debuggerstreamops.cpp" "4" +"debuggertoolchaincombobox.cpp" "4" +"debuggertooltipmanager.cpp" "4" +"disassembleragent.cpp" "4" +"disassemblerlines.cpp" "4" +"logwindow.cpp" "4" +"memoryagent.cpp" "4" +"memoryview.cpp" "4" +"moduleshandler.cpp" "4" +"moduleswindow.cpp" "4" +"name_demangler.cpp" "4" +"outputcollector.cpp" "4" +"procinterrupt.cpp" "4" +"qtmessagelogeditor.cpp" "4" +"qtmessageloghandler.cpp" "4" +"qtmessagelogitemdelegate.cpp" "4" +"qtmessagelogproxymodel.cpp" "4" +"qtmessagelogview.cpp" "4" +"qtmessagelogwindow.cpp" "4" +"registerhandler.cpp" "4" +"registerpostmortemaction.cpp" "4" +"registerwindow.cpp" "4" +"snapshothandler.cpp" "4" +"snapshotwindow.cpp" "4" +"sourceagent.cpp" "4" +"sourcefileshandler.cpp" "4" +"sourcefileswindow.cpp" "4" +"stackframe.cpp" "4" +"stackhandler.cpp" "4" +"stackwindow.cpp" "4" +"threadshandler.cpp" "4" +"threadswindow.cpp" "4" +"watchdata.cpp" "4" +"watchdelegatewidgets.cpp" "4" +"watchhandler.cpp" "4" +"watchutils.cpp" "4" +"watchwindow.cpp" "4" +"Forms" "3" +"attachcoredialog.ui" "4" +"attachexternaldialog.ui" "4" +"attachtoqmlportdialog.ui" "4" +"breakcondition.ui" "4" +"breakpoint.ui" "4" +"commonoptionspage.ui" "4" +"startexternaldialog.ui" "4" +"startremotedialog.ui" "4" +"startremoteenginedialog.ui" "4" +"Resources" "3" +"debugger.qrc" "4" +"designer" "2" +"designer.pro" "3" +"aggregation" "3" +"aggregation.pri" "4" +"coreplugin" "3" +"coreplugin.pri" "4" +"coreplugin_dependencies" "3" +"coreplugin_dependencies.pri" "4" +"cplusplus" "3" +"cplusplus.pri" "4" +"cpp" "3" +"cpp.pri" "4" +"Headers" "4" +"cppsettingspage.h" "5" +"formclasswizard.h" "5" +"formclasswizarddialog.h" "5" +"formclasswizardpage.h" "5" +"formclasswizardparameters.h" "5" +"Sources" "4" +"cppsettingspage.cpp" "5" +"formclasswizard.cpp" "5" +"formclasswizarddialog.cpp" "5" +"formclasswizardpage.cpp" "5" +"formclasswizardparameters.cpp" "5" +"Forms" "4" +"cppsettingspagewidget.ui" "5" +"formclasswizardpage.ui" "5" +"cpptools" "3" +"cpptools.pri" "4" +"cpptools_dependencies" "3" +"cpptools_dependencies.pri" "4" +"designer_dependencies" "3" +"designer_dependencies.pri" "4" +"designerintegration" "3" +"designerintegration.pri" "4" +"Headers" "4" +"formresizer.h" "5" +"sizehandlerect.h" "5" +"widgethost.h" "5" +"widgethostconstants.h" "5" +"Sources" "4" +"formresizer.cpp" "5" +"sizehandlerect.cpp" "5" +"widgethost.cpp" "5" +"extensionsystem" "3" +"extensionsystem.pri" "4" +"extensionsystem_dependencies" "3" +"extensionsystem_dependencies.pri" "4" +"find" "3" +"find.pri" "4" +"find_dependencies" "3" +"find_dependencies.pri" "4" +"languageutils" "3" +"languageutils.pri" "4" +"locator" "3" +"locator.pri" "4" +"locator_dependencies" "3" +"locator_dependencies.pri" "4" +"projectexplorer" "3" +"projectexplorer.pri" "4" +"projectexplorer_dependencies" "3" +"projectexplorer_dependencies.pri" "4" +"qtcreator" "3" +"qtcreator.pri" "4" +"qtcreatorplugin" "3" +"qtcreatorplugin.pri" "4" +"Other files" "4" +"plugins/designer" "5" +"Designer.pluginspec.in" "6" +"texteditor" "3" +"texteditor.pri" "4" +"texteditor_dependencies" "3" +"texteditor_dependencies.pri" "4" +"utils" "3" +"utils.pri" "4" +"utils_dependencies" "3" +"utils_dependencies.pri" "4" +"Headers" "3" +"codemodelhelpers.h" "4" +"designer_export.h" "4" +"designerconstants.h" "4" +"designercontext.h" "4" +"designerxmleditor.h" "4" +"editordata.h" "4" +"editorwidget.h" "4" +"formeditorfactory.h" "4" +"formeditorplugin.h" "4" +"formeditorstack.h" "4" +"formeditorw.h" "4" +"formtemplatewizardpage.h" "4" +"formwindoweditor.h" "4" +"formwindowfile.h" "4" +"formwizard.h" "4" +"formwizarddialog.h" "4" +"qtcreatorintegration.h" "4" +"qtdesignerformclasscodegenerator.h" "4" +"resourcehandler.h" "4" +"settingsmanager.h" "4" +"settingspage.h" "4" +"Sources" "3" +"codemodelhelpers.cpp" "4" +"designercontext.cpp" "4" +"designerxmleditor.cpp" "4" +"editorwidget.cpp" "4" +"formeditorfactory.cpp" "4" +"formeditorplugin.cpp" "4" +"formeditorstack.cpp" "4" +"formeditorw.cpp" "4" +"formtemplatewizardpage.cpp" "4" +"formwindoweditor.cpp" "4" +"formwindowfile.cpp" "4" +"formwizard.cpp" "4" +"formwizarddialog.cpp" "4" +"qtcreatorintegration.cpp" "4" +"qtdesignerformclasscodegenerator.cpp" "4" +"resourcehandler.cpp" "4" +"settingsmanager.cpp" "4" +"settingspage.cpp" "4" +"Resources" "3" +"designer.qrc" "4" +"Other files" "3" +"Designer.mimetypes.xml" "4" +"README.txt" "4" +"dumper" "2" +"dumper.pro" "3" +"qtcreator" "3" +"qtcreator.pri" "4" +"qtcreatorlibrary" "3" +"qtcreatorlibrary.pri" "4" +"rpath" "3" +"rpath.pri" "4" +"Sources" "3" +"dumper.cpp" "5" +"fakevim" "2" +"fakevim.pro" "3" +"aggregation" "3" +"aggregation.pri" "4" +"coreplugin" "3" +"coreplugin.pri" "4" +"coreplugin_dependencies" "3" +"coreplugin_dependencies.pri" "4" +"extensionsystem" "3" +"extensionsystem.pri" "4" +"extensionsystem_dependencies" "3" +"extensionsystem_dependencies.pri" "4" +"find" "3" +"find.pri" "4" +"find_dependencies" "3" +"find_dependencies.pri" "4" +"locator" "3" +"locator.pri" "4" +"locator_dependencies" "3" +"locator_dependencies.pri" "4" +"qtcreator" "3" +"qtcreator.pri" "4" +"qtcreatorplugin" "3" +"qtcreatorplugin.pri" "4" +"Other files" "4" +"plugins/fakevim" "5" +"FakeVim.pluginspec.in" "6" +"texteditor" "3" +"texteditor.pri" "4" +"texteditor_dependencies" "3" +"texteditor_dependencies.pri" "4" +"utils" "3" +"utils.pri" "4" +"utils_dependencies" "3" +"utils_dependencies.pri" "4" +"Headers" "3" +"fakevimactions.h" "4" +"fakevimhandler.h" "4" +"fakevimplugin.h" "4" +"Sources" "3" +"fakevimactions.cpp" "4" +"fakevimhandler.cpp" "4" +"fakevimplugin.cpp" "4" +"Forms" "3" +"fakevimoptions.ui" "4" +"find" "2" +"find.pro" "3" +"aggregation" "3" +"aggregation.pri" "4" +"coreplugin" "3" +"coreplugin.pri" "4" +"coreplugin_dependencies" "3" +"coreplugin_dependencies.pri" "4" +"extensionsystem" "3" +"extensionsystem.pri" "4" +"extensionsystem_dependencies" "3" +"extensionsystem_dependencies.pri" "4" +"find_dependencies" "3" +"find_dependencies.pri" "4" +"qtcreator" "3" +"qtcreator.pri" "4" +"qtcreatorplugin" "3" +"qtcreatorplugin.pri" "4" +"Other files" "4" +"plugins/find" "5" +"Find.pluginspec.in" "6" +"utils" "3" +"utils.pri" "4" +"utils_dependencies" "3" +"utils_dependencies.pri" "4" +"Headers" "3" +"basetextfind.h" "4" +"currentdocumentfind.h" "4" +"find_global.h" "4" +"findplugin.h" "4" +"findtoolbar.h" "4" +"findtoolwindow.h" "4" +"ifindfilter.h" "4" +"ifindsupport.h" "4" +"searchresulttreeitemdelegate.h" "4" +"searchresulttreeitemroles.h" "4" +"searchresulttreeitems.h" "4" +"searchresulttreemodel.h" "4" +"searchresulttreeview.h" "4" +"searchresultwidget.h" "4" +"searchresultwindow.h" "4" +"textfindconstants.h" "4" +"treeviewfind.h" "4" +"Sources" "3" +"basetextfind.cpp" "4" +"currentdocumentfind.cpp" "4" +"findplugin.cpp" "4" +"findtoolbar.cpp" "4" +"findtoolwindow.cpp" "4" +"ifindfilter.cpp" "4" +"ifindsupport.cpp" "4" +"searchresulttreeitemdelegate.cpp" "4" +"searchresulttreeitems.cpp" "4" +"searchresulttreemodel.cpp" "4" +"searchresulttreeview.cpp" "4" +"searchresultwidget.cpp" "4" +"searchresultwindow.cpp" "4" +"treeviewfind.cpp" "4" +"Forms" "3" +"finddialog.ui" "4" +"findwidget.ui" "4" +"Resources" "3" +"find.qrc" "4" +"genericprojectmanager" "2" +"genericprojectmanager.pro" "3" +"aggregation" "3" +"aggregation.pri" "4" +"coreplugin" "3" +"coreplugin.pri" "4" +"coreplugin_dependencies" "3" +"coreplugin_dependencies.pri" "4" +"cplusplus" "3" +"cplusplus.pri" "4" +"cpptools" "3" +"cpptools.pri" "4" +"cpptools_dependencies" "3" +"cpptools_dependencies.pri" "4" +"extensionsystem" "3" +"extensionsystem.pri" "4" +"extensionsystem_dependencies" "3" +"extensionsystem_dependencies.pri" "4" +"find" "3" +"find.pri" "4" +"find_dependencies" "3" +"find_dependencies.pri" "4" +"genericprojectmanager_dependencies" "3" +"genericprojectmanager_dependencies.pri" "4" +"languageutils" "3" +"languageutils.pri" "4" +"locator" "3" +"locator.pri" "4" +"locator_dependencies" "3" +"locator_dependencies.pri" "4" +"projectexplorer" "3" +"projectexplorer.pri" "4" +"projectexplorer_dependencies" "3" +"projectexplorer_dependencies.pri" "4" +"qtcreator" "3" +"qtcreator.pri" "4" +"qtcreatorplugin" "3" +"qtcreatorplugin.pri" "4" +"Other files" "4" +"plugins/genericprojectmanager" "5" +"GenericProjectManager.pluginspec.in" "6" +"texteditor" "3" +"texteditor.pri" "4" +"texteditor_dependencies" "3" +"texteditor_dependencies.pri" "4" +"utils" "3" +"utils.pri" "4" +"utils_dependencies" "3" +"utils_dependencies.pri" "4" +"Headers" "3" +"filesselectionwizardpage.h" "4" +"genericbuildconfiguration.h" "4" +"genericmakestep.h" "4" +"genericproject.h" "4" +"genericprojectconstants.h" "4" +"genericprojectfileseditor.h" "4" +"genericprojectmanager.h" "4" +"genericprojectnodes.h" "4" +"genericprojectplugin.h" "4" +"genericprojectwizard.h" "4" +"generictarget.h" "4" +"pkgconfigtool.h" "4" +"selectablefilesmodel.h" "4" +"Sources" "3" +"filesselectionwizardpage.cpp" "4" +"genericbuildconfiguration.cpp" "4" +"genericmakestep.cpp" "4" +"genericproject.cpp" "4" +"genericprojectfileseditor.cpp" "4" +"genericprojectmanager.cpp" "4" +"genericprojectnodes.cpp" "4" +"genericprojectplugin.cpp" "4" +"genericprojectwizard.cpp" "4" +"generictarget.cpp" "4" +"pkgconfigtool.cpp" "4" +"selectablefilesmodel.cpp" "4" +"Forms" "3" +"genericmakestep.ui" "4" +"Resources" "3" +"genericproject.qrc" "4" +"git" "2" +"git.pro" "3" +"aggregation" "3" +"aggregation.pri" "4" +"coreplugin" "3" +"coreplugin.pri" "4" +"coreplugin_dependencies" "3" +"coreplugin_dependencies.pri" "4" +"cplusplus" "3" +"cplusplus.pri" "4" +"extensionsystem" "3" +"extensionsystem.pri" "4" +"extensionsystem_dependencies" "3" +"extensionsystem_dependencies.pri" "4" +"find" "3" +"find.pri" "4" +"find_dependencies" "3" +"find_dependencies.pri" "4" +"gitorious" "3" +"gitorious.pri" "4" +"Headers" "4" +"gitorious.h" "5" +"gitoriousclonewizard.h" "5" +"gitorioushostwidget.h" "5" +"gitorioushostwizardpage.h" "5" +"gitoriousprojectwidget.h" "5" +"gitoriousprojectwizardpage.h" "5" +"gitoriousrepositorywizardpage.h" "5" +"Sources" "4" +"gitorious.cpp" "5" +"gitoriousclonewizard.cpp" "5" +"gitorioushostwidget.cpp" "5" +"gitorioushostwizardpage.cpp" "5" +"gitoriousprojectwidget.cpp" "5" +"gitoriousprojectwizardpage.cpp" "5" +"gitoriousrepositorywizardpage.cpp" "5" +"Forms" "4" +"gitorioushostwidget.ui" "5" +"gitoriousprojectwidget.ui" "5" +"gitoriousrepositorywizardpage.ui" "5" +"locator" "3" +"locator.pri" "4" +"locator_dependencies" "3" +"locator_dependencies.pri" "4" +"projectexplorer" "3" +"projectexplorer.pri" "4" +"projectexplorer_dependencies" "3" +"projectexplorer_dependencies.pri" "4" +"qtcreator" "3" +"qtcreator.pri" "4" +"qtcreatorplugin" "3" +"qtcreatorplugin.pri" "4" +"Other files" "4" +"plugins/git" "5" +"Git.pluginspec.in" "6" +"texteditor" "3" +"texteditor.pri" "4" +"texteditor_dependencies" "3" +"texteditor_dependencies.pri" "4" +"utils" "3" +"utils.pri" "4" +"utils_dependencies" "3" +"utils_dependencies.pri" "4" +"vcsbase" "3" +"vcsbase.pri" "4" +"vcsbase_dependencies" "3" +"vcsbase_dependencies.pri" "4" +"Headers" "3" +"annotationhighlighter.h" "4" +"branchadddialog.h" "4" +"branchdialog.h" "4" +"branchmodel.h" "4" +"changeselectiondialog.h" "4" +"clonewizard.h" "4" +"clonewizardpage.h" "4" +"commitdata.h" "4" +"gitclient.h" "4" +"gitconstants.h" "4" +"giteditor.h" "4" +"gitplugin.h" "4" +"gitsettings.h" "4" +"gitsubmiteditor.h" "4" +"gitsubmiteditorwidget.h" "4" +"gitutils.h" "4" +"gitversioncontrol.h" "4" +"remotedialog.h" "4" +"remotemodel.h" "4" +"settingspage.h" "4" +"stashdialog.h" "4" +"Sources" "3" +"annotationhighlighter.cpp" "4" +"branchadddialog.cpp" "4" +"branchdialog.cpp" "4" +"branchmodel.cpp" "4" +"changeselectiondialog.cpp" "4" +"clonewizard.cpp" "4" +"clonewizardpage.cpp" "4" +"commitdata.cpp" "4" +"gitclient.cpp" "4" +"giteditor.cpp" "4" +"gitplugin.cpp" "4" +"gitsettings.cpp" "4" +"gitsubmiteditor.cpp" "4" +"gitsubmiteditorwidget.cpp" "4" +"gitutils.cpp" "4" +"gitversioncontrol.cpp" "4" +"remotedialog.cpp" "4" +"remotemodel.cpp" "4" +"settingspage.cpp" "4" +"stashdialog.cpp" "4" +"Forms" "3" +"branchadddialog.ui" "4" +"branchdialog.ui" "4" +"changeselectiondialog.ui" "4" +"gitsubmitpanel.ui" "4" +"remoteadditiondialog.ui" "4" +"remotedialog.ui" "4" +"settingspage.ui" "4" +"stashdialog.ui" "4" +"Resources" "3" +"git.qrc" "4" +"glsleditor" "2" +"glsleditor.pro" "3" +"aggregation" "3" +"aggregation.pri" "4" +"coreplugin" "3" +"coreplugin.pri" "4" +"coreplugin_dependencies" "3" +"coreplugin_dependencies.pri" "4" +"cplusplus" "3" +"cplusplus.pri" "4" +"cpptools" "3" +"cpptools.pri" "4" +"cpptools_dependencies" "3" +"cpptools_dependencies.pri" "4" +"extensionsystem" "3" +"extensionsystem.pri" "4" +"extensionsystem_dependencies" "3" +"extensionsystem_dependencies.pri" "4" +"find" "3" +"find.pri" "4" +"find_dependencies" "3" +"find_dependencies.pri" "4" +"glsl" "3" +"glsl.pri" "4" +"glsleditor_dependencies" "3" +"glsleditor_dependencies.pri" "4" +"languageutils" "3" +"languageutils.pri" "4" +"locator" "3" +"locator.pri" "4" +"locator_dependencies" "3" +"locator_dependencies.pri" "4" +"projectexplorer" "3" +"projectexplorer.pri" "4" +"projectexplorer_dependencies" "3" +"projectexplorer_dependencies.pri" "4" +"qtcreator" "3" +"qtcreator.pri" "4" +"qtcreatorplugin" "3" +"qtcreatorplugin.pri" "4" +"Other files" "4" +"plugins/glsleditor" "5" +"GLSLEditor.pluginspec.in" "6" +"texteditor" "3" +"texteditor.pri" "4" +"texteditor_dependencies" "3" +"texteditor_dependencies.pri" "4" +"utils" "3" +"utils.pri" "4" +"utils_dependencies" "3" +"utils_dependencies.pri" "4" +"Headers" "3" +"glslautocompleter.h" "4" +"glslcompletionassist.h" "4" +"glsleditor.h" "4" +"glsleditor_global.h" "4" +"glsleditoractionhandler.h" "4" +"glsleditorconstants.h" "4" +"glsleditoreditable.h" "4" +"glsleditorfactory.h" "4" +"glsleditorplugin.h" "4" +"glslfilewizard.h" "4" +"glslhighlighter.h" "4" +"glslhoverhandler.h" "4" +"glslindenter.h" "4" +"reuse.h" "4" +"Sources" "3" +"glslautocompleter.cpp" "4" +"glslcompletionassist.cpp" "4" +"glsleditor.cpp" "4" +"glsleditoractionhandler.cpp" "4" +"glsleditoreditable.cpp" "4" +"glsleditorfactory.cpp" "4" +"glsleditorplugin.cpp" "4" +"glslfilewizard.cpp" "4" +"glslhighlighter.cpp" "4" +"glslhoverhandler.cpp" "4" +"glslindenter.cpp" "4" +"reuse.cpp" "4" +"Resources" "3" +"glsleditor.qrc" "4" +"Other files" "3" +"GLSLEditor.mimetypes.xml" "4" +"helloworld" "2" +"helloworld.pro" "3" +"aggregation" "3" +"aggregation.pri" "4" +"coreplugin" "3" +"coreplugin.pri" "4" +"coreplugin_dependencies" "3" +"coreplugin_dependencies.pri" "4" +"extensionsystem" "3" +"extensionsystem.pri" "4" +"extensionsystem_dependencies" "3" +"extensionsystem_dependencies.pri" "4" +"qtcreator" "3" +"qtcreator.pri" "4" +"qtcreatorplugin" "3" +"qtcreatorplugin.pri" "4" +"Other files" "4" +"plugins/helloworld" "5" +"HelloWorld.pluginspec.in" "6" +"utils" "3" +"utils.pri" "4" +"utils_dependencies" "3" +"utils_dependencies.pri" "4" +"Headers" "3" +"helloworldplugin.h" "4" +"helloworldwindow.h" "4" +"Sources" "3" +"helloworldplugin.cpp" "4" +"helloworldwindow.cpp" "4" +"help" "2" +"help.pro" "3" +"aggregation" "3" +"aggregation.pri" "4" +"coreplugin" "3" +"coreplugin.pri" "4" +"coreplugin_dependencies" "3" +"coreplugin_dependencies.pri" "4" +"extensionsystem" "3" +"extensionsystem.pri" "4" +"extensionsystem_dependencies" "3" +"extensionsystem_dependencies.pri" "4" +"find" "3" +"find.pri" "4" +"find_dependencies" "3" +"find_dependencies.pri" "4" +"help" "3" +"help.pri" "4" +"Headers" "4" +"bookmarkmanager.h" "5" +"contentwindow.h" "5" +"filternamedialog.h" "5" +"indexwindow.h" "5" +"topicchooser.h" "5" +"Sources" "4" +"bookmarkmanager.cpp" "5" +"contentwindow.cpp" "5" +"filternamedialog.cpp" "5" +"indexwindow.cpp" "5" +"topicchooser.cpp" "5" +"Forms" "4" +"bookmarkdialog.ui" "5" +"filternamedialog.ui" "5" +"topicchooser.ui" "5" +"help_dependencies" "3" +"help_dependencies.pri" "4" +"locator" "3" +"locator.pri" "4" +"locator_dependencies" "3" +"locator_dependencies.pri" "4" +"qtcreator" "3" +"qtcreator.pri" "4" +"qtcreatorplugin" "3" +"qtcreatorplugin.pri" "4" +"Other files" "4" +"plugins/help" "5" +"Help.pluginspec.in" "6" +"utils" "3" +"utils.pri" "4" +"utils_dependencies" "3" +"utils_dependencies.pri" "4" +"Headers" "3" +"centralwidget.h" "4" +"docsettingspage.h" "4" +"externalhelpwindow.h" "4" +"filtersettingspage.h" "4" +"generalsettingspage.h" "4" +"help_global.h" "4" +"helpconstants.h" "4" +"helpfindsupport.h" "4" +"helpindexfilter.h" "4" +"helpmode.h" "4" +"helpplugin.h" "4" +"helpviewer.h" "4" +"helpviewer_p.h" "4" +"localhelpmanager.h" "4" +"openpagesmanager.h" "4" +"openpagesmodel.h" "4" +"openpagesswitcher.h" "4" +"openpageswidget.h" "4" +"remotehelpfilter.h" "4" +"searchwidget.h" "4" +"xbelsupport.h" "4" +"Sources" "3" +"centralwidget.cpp" "4" +"docsettingspage.cpp" "4" +"externalhelpwindow.cpp" "4" +"filtersettingspage.cpp" "4" +"generalsettingspage.cpp" "4" +"helpfindsupport.cpp" "4" +"helpindexfilter.cpp" "4" +"helpmode.cpp" "4" +"helpplugin.cpp" "4" +"helpviewer.cpp" "4" +"helpviewer_qtb.cpp" "4" +"helpviewer_qwv.cpp" "4" +"localhelpmanager.cpp" "4" +"openpagesmanager.cpp" "4" +"openpagesmodel.cpp" "4" +"openpagesswitcher.cpp" "4" +"openpageswidget.cpp" "4" +"remotehelpfilter.cpp" "4" +"searchwidget.cpp" "4" +"xbelsupport.cpp" "4" +"Forms" "3" +"docsettingspage.ui" "4" +"filtersettingspage.ui" "4" +"generalsettingspage.ui" "4" +"remotehelpfilter.ui" "4" +"Resources" "3" +"help.qrc" "4" +"imageviewer" "2" +"imageviewer.pro" "3" +"aggregation" "3" +"aggregation.pri" "4" +"coreplugin" "3" +"coreplugin.pri" "4" +"coreplugin_dependencies" "3" +"coreplugin_dependencies.pri" "4" +"extensionsystem" "3" +"extensionsystem.pri" "4" +"extensionsystem_dependencies" "3" +"extensionsystem_dependencies.pri" "4" +"imageviewer_dependencies" "3" +"imageviewer_dependencies.pri" "4" +"qtcreator" "3" +"qtcreator.pri" "4" +"qtcreatorplugin" "3" +"qtcreatorplugin.pri" "4" +"Other files" "4" +"plugins/imageviewer" "5" +"ImageViewer.pluginspec.in" "6" +"utils" "3" +"utils.pri" "4" +"utils_dependencies" "3" +"utils_dependencies.pri" "4" +"Headers" "3" +"imageview.h" "4" +"imageviewer.h" "4" +"imagevieweractionhandler.h" "4" +"imageviewerconstants.h" "4" +"imageviewerfactory.h" "4" +"imageviewerfile.h" "4" +"imageviewerplugin.h" "4" +"Sources" "3" +"imageview.cpp" "4" +"imageviewer.cpp" "4" +"imagevieweractionhandler.cpp" "4" +"imageviewerfactory.cpp" "4" +"imageviewerfile.cpp" "4" +"imageviewerplugin.cpp" "4" +"Forms" "3" +"imageviewertoolbar.ui" "4" +"Resources" "3" +"imageviewer.qrc" "4" +"Other files" "3" +"ImageViewer.mimetypes.xml" "4" +"locator" "2" +"locator.pro" "3" +"aggregation" "3" +"aggregation.pri" "4" +"coreplugin" "3" +"coreplugin.pri" "4" +"coreplugin_dependencies" "3" +"coreplugin_dependencies.pri" "4" +"extensionsystem" "3" +"extensionsystem.pri" "4" +"extensionsystem_dependencies" "3" +"extensionsystem_dependencies.pri" "4" +"locator_dependencies" "3" +"locator_dependencies.pri" "4" +"qtcreator" "3" +"qtcreator.pri" "4" +"qtcreatorplugin" "3" +"qtcreatorplugin.pri" "4" +"Other files" "4" +"plugins/locator" "5" +"Locator.pluginspec.in" "6" +"utils" "3" +"utils.pri" "4" +"utils_dependencies" "3" +"utils_dependencies.pri" "4" +"Headers" "3" +"basefilefilter.h" "4" +"commandlocator.h" "4" +"directoryfilter.h" "4" +"executefilter.h" "4" +"filesystemfilter.h" "4" +"ilocatorfilter.h" "4" +"locator_global.h" "4" +"locatorconstants.h" "4" +"locatorfiltersfilter.h" "4" +"locatormanager.h" "4" +"locatorplugin.h" "4" +"locatorwidget.h" "4" +"opendocumentsfilter.h" "4" +"settingspage.h" "4" +"Sources" "3" +"basefilefilter.cpp" "4" +"commandlocator.cpp" "4" +"directoryfilter.cpp" "4" +"executefilter.cpp" "4" +"filesystemfilter.cpp" "4" +"ilocatorfilter.cpp" "4" +"locatorfiltersfilter.cpp" "4" +"locatormanager.cpp" "4" +"locatorplugin.cpp" "4" +"locatorwidget.cpp" "4" +"opendocumentsfilter.cpp" "4" +"settingspage.cpp" "4" +"Forms" "3" +"directoryfilter.ui" "4" +"filesystemfilter.ui" "4" +"settingspage.ui" "4" +"Resources" "3" +"locator.qrc" "4" +"macros" "2" +"macros.pro" "3" +"aggregation" "3" +"aggregation.pri" "4" +"coreplugin" "3" +"coreplugin.pri" "4" +"coreplugin_dependencies" "3" +"coreplugin_dependencies.pri" "4" +"extensionsystem" "3" +"extensionsystem.pri" "4" +"extensionsystem_dependencies" "3" +"extensionsystem_dependencies.pri" "4" +"find" "3" +"find.pri" "4" +"find_dependencies" "3" +"find_dependencies.pri" "4" +"locator" "3" +"locator.pri" "4" +"locator_dependencies" "3" +"locator_dependencies.pri" "4" +"macros_dependencies" "3" +"macros_dependencies.pri" "4" +"qtcreator" "3" +"qtcreator.pri" "4" +"qtcreatorplugin" "3" +"qtcreatorplugin.pri" "4" +"Other files" "4" +"plugins/macros" "5" +"Macros.pluginspec.in" "6" +"texteditor" "3" +"texteditor.pri" "4" +"texteditor_dependencies" "3" +"texteditor_dependencies.pri" "4" +"utils" "3" +"utils.pri" "4" +"utils_dependencies" "3" +"utils_dependencies.pri" "4" +"Headers" "3" +"actionmacrohandler.h" "4" +"findmacrohandler.h" "4" +"imacrohandler.h" "4" +"macro.h" "4" +"macroevent.h" "4" +"macrolocatorfilter.h" "4" +"macromanager.h" "4" +"macrooptionspage.h" "4" +"macrooptionswidget.h" "4" +"macros_global.h" "4" +"macrosconstants.h" "4" +"macrosplugin.h" "4" +"macrotextfind.h" "4" +"savedialog.h" "4" +"texteditormacrohandler.h" "4" +"Sources" "3" +"actionmacrohandler.cpp" "4" +"findmacrohandler.cpp" "4" +"imacrohandler.cpp" "4" +"macro.cpp" "4" +"macroevent.cpp" "4" +"macrolocatorfilter.cpp" "4" +"macromanager.cpp" "4" +"macrooptionspage.cpp" "4" +"macrooptionswidget.cpp" "4" +"macrosplugin.cpp" "4" +"macrotextfind.cpp" "4" +"savedialog.cpp" "4" +"texteditormacrohandler.cpp" "4" +"Forms" "3" +"macrooptionswidget.ui" "4" +"savedialog.ui" "4" +"Resources" "3" +"macros.qrc" "4" +"madde" "2" +"madde.pro" "3" +"aggregation" "3" +"aggregation.pri" "4" +"botan" "3" +"botan.pri" "4" +"coreplugin" "3" +"coreplugin.pri" "4" +"coreplugin_dependencies" "3" +"coreplugin_dependencies.pri" "4" +"cplusplus" "3" +"cplusplus.pri" "4" +"cpptools" "3" +"cpptools.pri" "4" +"cpptools_dependencies" "3" +"cpptools_dependencies.pri" "4" +"debugger" "3" +"debugger.pri" "4" +"debugger_dependencies" "3" +"debugger_dependencies.pri" "4" +"extensionsystem" "3" +"extensionsystem.pri" "4" +"extensionsystem_dependencies" "3" +"extensionsystem_dependencies.pri" "4" +"find" "3" +"find.pri" "4" +"find_dependencies" "3" +"find_dependencies.pri" "4" +"languageutils" "3" +"languageutils.pri" "4" +"locator" "3" +"locator.pri" "4" +"locator_dependencies" "3" +"locator_dependencies.pri" "4" +"madde_dependencies" "3" +"madde_dependencies.pri" "4" +"projectexplorer" "3" +"projectexplorer.pri" "4" +"projectexplorer_dependencies" "3" +"projectexplorer_dependencies.pri" "4" +"qmljs" "3" +"qmljs.pri" "4" +"qt4projectmanager" "3" +"qt4projectmanager.pri" "4" +"qt4projectmanager_dependencies" "3" +"qt4projectmanager_dependencies.pri" "4" +"qtcreator" "3" +"qtcreator.pri" "4" +"qtcreatorplugin" "3" +"qtcreatorplugin.pri" "4" +"Other files" "4" +"plugins/madde" "5" +"Madde.pluginspec.in" "6" +"qtsupport" "3" +"qtsupport.pri" "4" +"qtsupport_dependencies" "3" +"qtsupport_dependencies.pri" "4" +"remotelinux" "3" +"remotelinux.pri" "4" +"remotelinux_dependencies" "3" +"remotelinux_dependencies.pri" "4" +"symbianutils" "3" +"symbianutils.pri" "4" +"texteditor" "3" +"texteditor.pri" "4" +"texteditor_dependencies" "3" +"texteditor_dependencies.pri" "4" +"utils" "3" +"utils.pri" "4" +"utils_dependencies" "3" +"utils_dependencies.pri" "4" +"Headers" "3" +"madde_exports.h" "4" +"maddedeviceconfigurationfactory.h" "4" +"maddedevicetester.h" "4" +"maddeplugin.h" "4" +"maddeuploadandinstallpackagesteps.h" "4" +"maemoconstants.h" "4" +"maemodebugsupport.h" "4" +"maemodeploybymountsteps.h" "4" +"maemodeployconfigurationwidget.h" "4" +"maemodeploymentmounter.h" "4" +"maemodeploystepfactory.h" "4" +"maemodeviceconfigwizard.h" "4" +"maemoglobal.h" "4" +"maemoinstalltosysrootstep.h" "4" +"maemomountspecification.h" "4" +"maemopackagecreationfactory.h" "4" +"maemopackagecreationstep.h" "4" +"maemopackagecreationwidget.h" "4" +"maemopackageinstaller.h" "4" +"maemopublishedprojectmodel.h" "4" +"maemopublisherfremantlefree.h" "4" +"maemopublishingbuildsettingspagefremantlefree.h" "4" +"maemopublishingfileselectiondialog.h" "4" +"maemopublishingresultpagefremantlefree.h" "4" +"maemopublishinguploadsettingspagefremantlefree.h" "4" +"maemopublishingwizardfactories.h" "4" +"maemopublishingwizardfremantlefree.h" "4" +"maemoqemumanager.h" "4" +"maemoqemuruntime.h" "4" +"maemoqemuruntimeparser.h" "4" +"maemoqemusettings.h" "4" +"maemoqemusettingswidget.h" "4" +"maemoqtversion.h" "4" +"maemoqtversionfactory.h" "4" +"maemoremotecopyfacility.h" "4" +"maemoremotemounter.h" "4" +"maemoremotemountsmodel.h" "4" +"maemorunconfiguration.h" "4" +"maemorunconfigurationwidget.h" "4" +"maemoruncontrol.h" "4" +"maemorunfactories.h" "4" +"maemosettingspages.h" "4" +"maemosshrunner.h" "4" +"maemotoolchain.h" "4" +"qt4maemodeployconfiguration.h" "4" +"qt4maemotarget.h" "4" +"qt4maemotargetfactory.h" "4" +"Sources" "3" +"maddedeviceconfigurationfactory.cpp" "4" +"maddedevicetester.cpp" "4" +"maddeplugin.cpp" "4" +"maddeuploadandinstallpackagesteps.cpp" "4" +"maemodebugsupport.cpp" "4" +"maemodeploybymountsteps.cpp" "4" +"maemodeployconfigurationwidget.cpp" "4" +"maemodeploymentmounter.cpp" "4" +"maemodeploystepfactory.cpp" "4" +"maemodeviceconfigwizard.cpp" "4" +"maemoglobal.cpp" "4" +"maemoinstalltosysrootstep.cpp" "4" +"maemomountspecification.cpp" "4" +"maemopackagecreationfactory.cpp" "4" +"maemopackagecreationstep.cpp" "4" +"maemopackagecreationwidget.cpp" "4" +"maemopackageinstaller.cpp" "4" +"maemopublishedprojectmodel.cpp" "4" +"maemopublisherfremantlefree.cpp" "4" +"maemopublishingbuildsettingspagefremantlefree.cpp" "4" +"maemopublishingfileselectiondialog.cpp" "4" +"maemopublishingresultpagefremantlefree.cpp" "4" +"maemopublishinguploadsettingspagefremantlefree.cpp" "4" +"maemopublishingwizardfactories.cpp" "4" +"maemopublishingwizardfremantlefree.cpp" "4" +"maemoqemumanager.cpp" "4" +"maemoqemuruntimeparser.cpp" "4" +"maemoqemusettings.cpp" "4" +"maemoqemusettingswidget.cpp" "4" +"maemoqtversion.cpp" "4" +"maemoqtversionfactory.cpp" "4" +"maemoremotecopyfacility.cpp" "4" +"maemoremotemounter.cpp" "4" +"maemoremotemountsmodel.cpp" "4" +"maemorunconfiguration.cpp" "4" +"maemorunconfigurationwidget.cpp" "4" +"maemoruncontrol.cpp" "4" +"maemorunfactories.cpp" "4" +"maemosettingspages.cpp" "4" +"maemosshrunner.cpp" "4" +"maemotoolchain.cpp" "4" +"qt4maemodeployconfiguration.cpp" "4" +"qt4maemotarget.cpp" "4" +"qt4maemotargetfactory.cpp" "4" +"Forms" "3" +"maemodeployconfigurationwidget.ui" "4" +"maemodeviceconfigwizardkeycreationpage.ui" "4" +"maemodeviceconfigwizardkeydeploymentpage.ui" "4" +"maemodeviceconfigwizardpreviouskeysetupcheckpage.ui" "4" +"maemodeviceconfigwizardreusekeyscheckpage.ui" "4" +"maemodeviceconfigwizardstartpage.ui" "4" +"maemopackagecreationwidget.ui" "4" +"maemopublishingbuildsettingspagefremantlefree.ui" "4" +"maemopublishingfileselectiondialog.ui" "4" +"maemopublishingresultpagefremantlefree.ui" "4" +"maemopublishinguploadsettingspagefremantlefree.ui" "4" +"maemoqemusettingswidget.ui" "4" +"Resources" "3" +"qt-maemo.qrc" "4" +"mercurial" "2" +"mercurial.pro" "3" +"aggregation" "3" +"aggregation.pri" "4" +"coreplugin" "3" +"coreplugin.pri" "4" +"coreplugin_dependencies" "3" +"coreplugin_dependencies.pri" "4" +"cplusplus" "3" +"cplusplus.pri" "4" +"extensionsystem" "3" +"extensionsystem.pri" "4" +"extensionsystem_dependencies" "3" +"extensionsystem_dependencies.pri" "4" +"find" "3" +"find.pri" "4" +"find_dependencies" "3" +"find_dependencies.pri" "4" +"locator" "3" +"locator.pri" "4" +"locator_dependencies" "3" +"locator_dependencies.pri" "4" +"mercurial_dependencies" "3" +"mercurial_dependencies.pri" "4" +"projectexplorer" "3" +"projectexplorer.pri" "4" +"projectexplorer_dependencies" "3" +"projectexplorer_dependencies.pri" "4" +"qtcreator" "3" +"qtcreator.pri" "4" +"qtcreatorplugin" "3" +"qtcreatorplugin.pri" "4" +"Other files" "4" +"plugins/mercurial" "5" +"Mercurial.pluginspec.in" "6" +"texteditor" "3" +"texteditor.pri" "4" +"texteditor_dependencies" "3" +"texteditor_dependencies.pri" "4" +"utils" "3" +"utils.pri" "4" +"utils_dependencies" "3" +"utils_dependencies.pri" "4" +"vcsbase" "3" +"vcsbase.pri" "4" +"vcsbase_dependencies" "3" +"vcsbase_dependencies.pri" "4" +"Headers" "3" +"annotationhighlighter.h" "4" +"clonewizard.h" "4" +"clonewizardpage.h" "4" +"commiteditor.h" "4" +"constants.h" "4" +"mercurialclient.h" "4" +"mercurialcommitwidget.h" "4" +"mercurialcontrol.h" "4" +"mercurialeditor.h" "4" +"mercurialplugin.h" "4" +"mercurialsettings.h" "4" +"optionspage.h" "4" +"revertdialog.h" "4" +"srcdestdialog.h" "4" +"Sources" "3" +"annotationhighlighter.cpp" "4" +"clonewizard.cpp" "4" +"clonewizardpage.cpp" "4" +"commiteditor.cpp" "4" +"mercurialclient.cpp" "4" +"mercurialcommitwidget.cpp" "4" +"mercurialcontrol.cpp" "4" +"mercurialeditor.cpp" "4" +"mercurialplugin.cpp" "4" +"mercurialsettings.cpp" "4" +"optionspage.cpp" "4" +"revertdialog.cpp" "4" +"srcdestdialog.cpp" "4" +"Forms" "3" +"mercurialcommitpanel.ui" "4" +"optionspage.ui" "4" +"revertdialog.ui" "4" +"srcdestdialog.ui" "4" +"Resources" "3" +"mercurial.qrc" "4" +"perforce" "2" +"perforce.pro" "3" +"aggregation" "3" +"aggregation.pri" "4" +"coreplugin" "3" +"coreplugin.pri" "4" +"coreplugin_dependencies" "3" +"coreplugin_dependencies.pri" "4" +"cplusplus" "3" +"cplusplus.pri" "4" +"extensionsystem" "3" +"extensionsystem.pri" "4" +"extensionsystem_dependencies" "3" +"extensionsystem_dependencies.pri" "4" +"find" "3" +"find.pri" "4" +"find_dependencies" "3" +"find_dependencies.pri" "4" +"locator" "3" +"locator.pri" "4" +"locator_dependencies" "3" +"locator_dependencies.pri" "4" +"perforce_dependencies" "3" +"perforce_dependencies.pri" "4" +"projectexplorer" "3" +"projectexplorer.pri" "4" +"projectexplorer_dependencies" "3" +"projectexplorer_dependencies.pri" "4" +"qtcreator" "3" +"qtcreator.pri" "4" +"qtcreatorplugin" "3" +"qtcreatorplugin.pri" "4" +"Other files" "4" +"plugins/perforce" "5" +"Perforce.pluginspec.in" "6" +"texteditor" "3" +"texteditor.pri" "4" +"texteditor_dependencies" "3" +"texteditor_dependencies.pri" "4" +"utils" "3" +"utils.pri" "4" +"utils_dependencies" "3" +"utils_dependencies.pri" "4" +"vcsbase" "3" +"vcsbase.pri" "4" +"vcsbase_dependencies" "3" +"vcsbase_dependencies.pri" "4" +"Headers" "3" +"annotationhighlighter.h" "4" +"changenumberdialog.h" "4" +"pendingchangesdialog.h" "4" +"perforcechecker.h" "4" +"perforceconstants.h" "4" +"perforceeditor.h" "4" +"perforceplugin.h" "4" +"perforcesettings.h" "4" +"perforcesubmiteditor.h" "4" +"perforcesubmiteditorwidget.h" "4" +"perforceversioncontrol.h" "4" +"settingspage.h" "4" +"Sources" "3" +"annotationhighlighter.cpp" "4" +"changenumberdialog.cpp" "4" +"pendingchangesdialog.cpp" "4" +"perforcechecker.cpp" "4" +"perforceeditor.cpp" "4" +"perforceplugin.cpp" "4" +"perforcesettings.cpp" "4" +"perforcesubmiteditor.cpp" "4" +"perforcesubmiteditorwidget.cpp" "4" +"perforceversioncontrol.cpp" "4" +"settingspage.cpp" "4" +"Forms" "3" +"changenumberdialog.ui" "4" +"pendingchangesdialog.ui" "4" +"settingspage.ui" "4" +"submitpanel.ui" "4" +"Resources" "3" +"perforce.qrc" "4" +"private_headers" "2" +"private_headers.pri" "3" +"projectexplorer" "2" +"projectexplorer.pro" "3" +"aggregation" "3" +"aggregation.pri" "4" +"botan" "3" +"botan.pri" "4" +"coreplugin" "3" +"coreplugin.pri" "4" +"coreplugin_dependencies" "3" +"coreplugin_dependencies.pri" "4" +"customwizard" "3" +"customwizard.pri" "4" +"Headers" "4" +"customwizard.h" "5" +"customwizardpage.h" "5" +"customwizardparameters.h" "5" +"customwizardpreprocessor.h" "5" +"customwizardscriptgenerator.h" "5" +"Sources" "4" +"customwizard.cpp" "5" +"customwizardpage.cpp" "5" +"customwizardparameters.cpp" "5" +"customwizardpreprocessor.cpp" "5" +"customwizardscriptgenerator.cpp" "5" +"extensionsystem" "3" +"extensionsystem.pri" "4" +"extensionsystem_dependencies" "3" +"extensionsystem_dependencies.pri" "4" +"find" "3" +"find.pri" "4" +"find_dependencies" "3" +"find_dependencies.pri" "4" +"locator" "3" +"locator.pri" "4" +"locator_dependencies" "3" +"locator_dependencies.pri" "4" +"projectexplorer_dependencies" "3" +"projectexplorer_dependencies.pri" "4" +"qtcreator" "3" +"qtcreator.pri" "4" +"qtcreatorplugin" "3" +"qtcreatorplugin.pri" "4" +"Other files" "4" +"plugins/projectexplorer" "5" +"ProjectExplorer.pluginspec.in" "6" +"texteditor" "3" +"texteditor.pri" "4" +"texteditor_dependencies" "3" +"texteditor_dependencies.pri" "4" +"utils" "3" +"utils.pri" "4" +"utils_dependencies" "3" +"utils_dependencies.pri" "4" +"Headers" "3" +"publishing" "4" +"ipublishingwizardfactory.h" "5" +"publishingwizardselectiondialog.h" "5" +"abi.h" "4" +"abiwidget.h" "4" +"abstractmsvctoolchain.h" "4" +"abstractprocessstep.h" "4" +"allprojectsfilter.h" "4" +"allprojectsfind.h" "4" +"applicationlauncher.h" "4" +"applicationrunconfiguration.h" "4" +"appoutputpane.h" "4" +"baseprojectwizarddialog.h" "4" +"buildconfiguration.h" "4" +"buildconfigurationmodel.h" "4" +"buildenvironmentwidget.h" "4" +"buildmanager.h" "4" +"buildprogress.h" "4" +"buildsettingspropertiespage.h" "4" +"buildstep.h" "4" +"buildsteplist.h" "4" +"buildstepspage.h" "4" +"cesdkhandler.h" "4" +"clangparser.h" "4" +"codestylesettingspropertiespage.h" "4" +"compileoutputwindow.h" "4" +"copytaskhandler.h" "4" +"corelistenercheckingforrunningbuild.h" "4" +"currentprojectfilter.h" "4" +"currentprojectfind.h" "4" +"dependenciespanel.h" "4" +"deployconfiguration.h" "4" +"deployconfigurationmodel.h" "4" +"doubletabwidget.h" "4" +"editorconfiguration.h" "4" +"editorsettingspropertiespage.h" "4" +"environmentitemswidget.h" "4" +"environmentwidget.h" "4" +"foldernavigationwidget.h" "4" +"gccparser.h" "4" +"gcctoolchain.h" "4" +"gcctoolchainfactories.h" "4" +"gnumakeparser.h" "4" +"headerpath.h" "4" +"ioutputparser.h" "4" +"iprojectmanager.h" "4" +"iprojectproperties.h" "4" +"itaskhandler.h" "4" +"ldparser.h" "4" +"linuxiccparser.h" "4" +"localapplicationruncontrol.h" "4" +"metatypedeclarations.h" "4" +"miniprojecttargetselector.h" "4" +"msvcparser.h" "4" +"msvctoolchain.h" "4" +"namedwidget.h" "4" +"nodesvisitor.h" "4" +"outputparser_test.h" "4" +"pluginfilefactory.h" "4" +"processparameters.h" "4" +"processstep.h" "4" +"project.h" "4" +"projectconfiguration.h" "4" +"projectexplorer.h" "4" +"projectexplorer_export.h" "4" +"projectexplorerconstants.h" "4" +"projectexplorersettings.h" "4" +"projectexplorersettingspage.h" "4" +"projectfilewizardextension.h" "4" +"projectmodels.h" "4" +"projectnodes.h" "4" +"projecttreewidget.h" "4" +"projectwelcomepage.h" "4" +"projectwindow.h" "4" +"projectwizardpage.h" "4" +"removefiledialog.h" "4" +"runconfiguration.h" "4" +"runconfigurationmodel.h" "4" +"runsettingspropertiespage.h" "4" +"session.h" "4" +"sessiondialog.h" "4" +"settingsaccessor.h" "4" +"showineditortaskhandler.h" "4" +"showoutputtaskhandler.h" "4" +"target.h" "4" +"targetselector.h" "4" +"targetsettingspanel.h" "4" +"targetsettingswidget.h" "4" +"task.h" "4" +"taskhub.h" "4" +"taskmodel.h" "4" +"taskwindow.h" "4" +"toolchain.h" "4" +"toolchainconfigwidget.h" "4" +"toolchainmanager.h" "4" +"toolchainoptionspage.h" "4" +"vcsannotatetaskhandler.h" "4" +"wincetoolchain.h" "4" +"windebuginterface.h" "4" +"Sources" "3" +"publishing" "4" +"publishingwizardselectiondialog.cpp" "5" +"abi.cpp" "4" +"abiwidget.cpp" "4" +"abstractmsvctoolchain.cpp" "4" +"abstractprocessstep.cpp" "4" +"allprojectsfilter.cpp" "4" +"allprojectsfind.cpp" "4" +"applicationlauncher.cpp" "4" +"applicationrunconfiguration.cpp" "4" +"appoutputpane.cpp" "4" +"baseprojectwizarddialog.cpp" "4" +"buildconfiguration.cpp" "4" +"buildconfigurationmodel.cpp" "4" +"buildenvironmentwidget.cpp" "4" +"buildmanager.cpp" "4" +"buildprogress.cpp" "4" +"buildsettingspropertiespage.cpp" "4" +"buildstep.cpp" "4" +"buildsteplist.cpp" "4" +"buildstepspage.cpp" "4" +"cesdkhandler.cpp" "4" +"clangparser.cpp" "4" +"codestylesettingspropertiespage.cpp" "4" +"compileoutputwindow.cpp" "4" +"copytaskhandler.cpp" "4" +"corelistenercheckingforrunningbuild.cpp" "4" +"currentprojectfilter.cpp" "4" +"currentprojectfind.cpp" "4" +"dependenciespanel.cpp" "4" +"deployconfiguration.cpp" "4" +"deployconfigurationmodel.cpp" "4" +"doubletabwidget.cpp" "4" +"editorconfiguration.cpp" "4" +"editorsettingspropertiespage.cpp" "4" +"environmentitemswidget.cpp" "4" +"environmentwidget.cpp" "4" +"foldernavigationwidget.cpp" "4" +"gccparser.cpp" "4" +"gcctoolchain.cpp" "4" +"gnumakeparser.cpp" "4" +"ioutputparser.cpp" "4" +"ldparser.cpp" "4" +"linuxiccparser.cpp" "4" +"localapplicationruncontrol.cpp" "4" +"miniprojecttargetselector.cpp" "4" +"msvcparser.cpp" "4" +"msvctoolchain.cpp" "4" +"namedwidget.cpp" "4" +"nodesvisitor.cpp" "4" +"outputparser_test.cpp" "4" +"pluginfilefactory.cpp" "4" +"processparameters.cpp" "4" +"processstep.cpp" "4" +"project.cpp" "4" +"projectconfiguration.cpp" "4" +"projectexplorer.cpp" "4" +"projectexplorersettingspage.cpp" "4" +"projectfilewizardextension.cpp" "4" +"projectmodels.cpp" "4" +"projectnodes.cpp" "4" +"projecttreewidget.cpp" "4" +"projectwelcomepage.cpp" "4" +"projectwindow.cpp" "4" +"projectwizardpage.cpp" "4" +"removefiledialog.cpp" "4" +"runconfiguration.cpp" "4" +"runconfigurationmodel.cpp" "4" +"runsettingspropertiespage.cpp" "4" +"session.cpp" "4" +"sessiondialog.cpp" "4" +"settingsaccessor.cpp" "4" +"showineditortaskhandler.cpp" "4" +"showoutputtaskhandler.cpp" "4" +"target.cpp" "4" +"targetselector.cpp" "4" +"targetsettingspanel.cpp" "4" +"targetsettingswidget.cpp" "4" +"task.cpp" "4" +"taskhub.cpp" "4" +"taskmodel.cpp" "4" +"taskwindow.cpp" "4" +"toolchain.cpp" "4" +"toolchainconfigwidget.cpp" "4" +"toolchainmanager.cpp" "4" +"toolchainoptionspage.cpp" "4" +"vcsannotatetaskhandler.cpp" "4" +"wincetoolchain.cpp" "4" +"windebuginterface.cpp" "4" +"Forms" "3" +"publishing" "4" +"publishingwizardselectiondialog.ui" "5" +"codestylesettingspropertiespage.ui" "4" +"doubletabwidget.ui" "4" +"editorsettingspropertiespage.ui" "4" +"processstep.ui" "4" +"projectexplorersettingspage.ui" "4" +"projectwizardpage.ui" "4" +"removefiledialog.ui" "4" +"sessiondialog.ui" "4" +"targetsettingswidget.ui" "4" +"toolchainoptionspage.ui" "4" +"Resources" "3" +"projectexplorer.qrc" "4" +"ptracepreload" "2" +"ptracepreload.pro" "3" +"qtcreator" "3" +"qtcreator.pri" "4" +"qtcreatorlibrary" "3" +"qtcreatorlibrary.pri" "4" +"rpath" "3" +"rpath.pri" "4" +"Sources" "3" +"ptracepreload.c" "4" +"qmldesigner" "2" +"qmldesigner.pro" "3" +"customstyleplugin" "3" +"customstyleplugin.pro" "4" +"customstyleplugin" "4" +"customstyleplugin.pri" "5" +"Headers" "5" +"iwidgetplugin.h" "7" +"customstyleplugin.h" "6" +"Sources" "5" +"customstyleplugin.cpp" "6" +"Resources" "5" +"customstyleplugin.qrc" "6" +"Other files" "5" +"customstyle.metainfo" "6" +"iwidgetplugin" "4" +"iwidgetplugin.pri" "5" +"plugindestdir" "4" +"plugindestdir.pri" "5" +"private_headers" "4" +"private_headers.pri" "5" +"qtcreator" "4" +"qtcreator.pri" "5" +"desktopplugin" "3" +"desktopplugin.pro" "4" +"desktopplugin" "4" +"desktopplugin.pri" "5" +"Headers" "5" +"iwidgetplugin.h" "7" +"desktopplugin.h" "6" +"Sources" "5" +"desktopplugin.cpp" "6" +"Resources" "5" +"desktopplugin.qrc" "6" +"Other files" "5" +"desktop.metainfo" "6" +"iwidgetplugin" "4" +"iwidgetplugin.pri" "5" +"plugindestdir" "4" +"plugindestdir.pri" "5" +"private_headers" "4" +"private_headers.pri" "5" +"qtcreator" "4" +"qtcreator.pri" "5" +"extrasplugin" "3" +"extrasplugin.pro" "4" +"extrasplugin" "4" +"extrasplugin.pri" "5" +"Headers" "5" +"iwidgetplugin.h" "7" +"extrasplugin.h" "6" +"Sources" "5" +"extrasplugin.cpp" "6" +"Resources" "5" +"extrasplugin.qrc" "6" +"Other files" "5" +"extras.metainfo" "6" +"iwidgetplugin" "4" +"iwidgetplugin.pri" "5" +"plugindestdir" "4" +"plugindestdir.pri" "5" +"private_headers" "4" +"private_headers.pri" "5" +"qtcreator" "4" +"qtcreator.pri" "5" +"meegoplugin" "3" +"meegoplugin.pro" "4" +"iwidgetplugin" "4" +"iwidgetplugin.pri" "5" +"meegoplugin" "4" +"meegoplugin.pri" "5" +"Headers" "5" +"iwidgetplugin.h" "7" +"meegoplugin.h" "6" +"Sources" "5" +"meegoplugin.cpp" "6" +"Resources" "5" +"meegoplugin.qrc" "6" +"Other files" "5" +"meego.metainfo" "6" +"plugindestdir" "4" +"plugindestdir.pri" "5" +"private_headers" "4" +"private_headers.pri" "5" +"qtcreator" "4" +"qtcreator.pri" "5" +"qmldesignerplugin" "3" +"qmldesignerplugin.pro" "4" +"aggregation" "4" +"aggregation.pri" "5" +"botan" "4" +"botan.pri" "5" +"commands" "4" +"commands.pri" "5" +"Headers" "5" +"changeauxiliarycommand.h" "6" +"changebindingscommand.h" "6" +"changefileurlcommand.h" "6" +"changeidscommand.h" "6" +"changenodesourcecommand.h" "6" +"changestatecommand.h" "6" +"changevaluescommand.h" "6" +"childrenchangedcommand.h" "6" +"clearscenecommand.h" "6" +"completecomponentcommand.h" "6" +"componentcompletedcommand.h" "6" +"createinstancescommand.h" "6" +"createscenecommand.h" "6" +"informationchangedcommand.h" "6" +"pixmapchangedcommand.h" "6" +"removeinstancescommand.h" "6" +"removepropertiescommand.h" "6" +"reparentinstancescommand.h" "6" +"statepreviewimagechangedcommand.h" "6" +"synchronizecommand.h" "6" +"tokencommand.h" "6" +"valueschangedcommand.h" "6" +"Sources" "5" +"changeauxiliarycommand.cpp" "6" +"changebindingscommand.cpp" "6" +"changefileurlcommand.cpp" "6" +"changeidscommand.cpp" "6" +"changenodesourcecommand.cpp" "6" +"changestatecommand.cpp" "6" +"changevaluescommand.cpp" "6" +"childrenchangedcommand.cpp" "6" +"clearscenecommand.cpp" "6" +"completecomponentcommand.cpp" "6" +"componentcompletedcommand.cpp" "6" +"createinstancescommand.cpp" "6" +"createscenecommand.cpp" "6" +"informationchangedcommand.cpp" "6" +"pixmapchangedcommand.cpp" "6" +"removeinstancescommand.cpp" "6" +"removepropertiescommand.cpp" "6" +"reparentinstancescommand.cpp" "6" +"statepreviewimagechangedcommand.cpp" "6" +"synchronizecommand.cpp" "6" +"tokencommand.cpp" "6" +"valueschangedcommand.cpp" "6" +"config" "4" +"config.pri" "5" +"container" "4" +"container.pri" "5" +"Headers" "5" +"addimportcontainer.h" "6" +"idcontainer.h" "6" +"imagecontainer.h" "6" +"informationcontainer.h" "6" +"instancecontainer.h" "6" +"propertyabstractcontainer.h" "6" +"propertybindingcontainer.h" "6" +"propertyvaluecontainer.h" "6" +"reparentcontainer.h" "6" +"Sources" "5" +"addimportcontainer.cpp" "6" +"idcontainer.cpp" "6" +"imagecontainer.cpp" "6" +"informationcontainer.cpp" "6" +"instancecontainer.cpp" "6" +"propertyabstractcontainer.cpp" "6" +"propertybindingcontainer.cpp" "6" +"propertyvaluecontainer.cpp" "6" +"reparentcontainer.cpp" "6" +"coreplugin" "4" +"coreplugin.pri" "5" +"coreplugin_dependencies" "4" +"coreplugin_dependencies.pri" "5" +"cplusplus" "4" +"cplusplus.pri" "5" +"cpptools" "4" +"cpptools.pri" "5" +"cpptools_dependencies" "4" +"cpptools_dependencies.pri" "5" +"debugger" "4" +"debugger.pri" "5" +"debugger_dependencies" "4" +"debugger_dependencies.pri" "5" +"designercore" "4" +"designercore.pri" "5" +"Headers" "5" +"include" "6" +"abstractproperty.h" "7" +"abstractview.h" "7" +"basetexteditmodifier.h" "7" +"bindingproperty.h" "7" +"componenttextmodifier.h" "7" +"corelib_global.h" "7" +"customnotifications.h" "7" +"exception.h" "7" +"forwardview.h" "7" +"import.h" "7" +"invalidargumentexception.h" "7" +"invalididexception.h" "7" +"invalidmetainfoexception.h" "7" +"invalidmodelstateexception.h" "7" +"invalidpropertyexception.h" "7" +"invalidqmlsourceexception.h" "7" +"invalidreparentingexception.h" "7" +"invalidslideindexexception.h" "7" +"itemlibraryinfo.h" "7" +"mathutils.h" "7" +"metainfo.h" "7" +"metainfoparser.h" "7" +"model.h" "7" +"modelmerger.h" "7" +"modelnode.h" "7" +"modelnodepositionstorage.h" "7" +"nodeabstractproperty.h" "7" +"nodeinstanceview.h" "7" +"nodelistproperty.h" "7" +"nodemetainfo.h" "7" +"nodeproperty.h" "7" +"notimplementedexception.h" "7" +"plaintexteditmodifier.h" "7" +"propertycontainer.h" "7" +"propertynode.h" "7" +"qmlanchors.h" "7" +"qmlchangeset.h" "7" +"qmlitemnode.h" "7" +"qmlmodelnodefacade.h" "7" +"qmlmodelview.h" "7" +"qmlobjectnode.h" "7" +"qmlstate.h" "7" +"removebasestateexception.h" "7" +"rewriterview.h" "7" +"rewritingexception.h" "7" +"subcomponentmanager.h" "7" +"textmodifier.h" "7" +"variantproperty.h" "7" +"model" "6" +"internalbindingproperty.h" "7" +"internalnode_p.h" "7" +"internalnodeabstractproperty.h" "7" +"internalnodelistproperty.h" "7" +"internalnodeproperty.h" "7" +"internalproperty.h" "7" +"internalvariantproperty.h" "7" +"model_p.h" "7" +"modelnodecontextmenu.h" "7" +"modelnodepositionrecalculator.h" "7" +"modeltotextmerger.h" "7" +"painteventfilter_p.h" "7" +"propertyparser.h" "7" +"qmltextgenerator.h" "7" +"rewriteaction.h" "7" +"rewriteactioncompressor.h" "7" +"texttomodelmerger.h" "7" +"variantparser.h" "7" +"viewlogger.h" "7" +"pluginmanager" "6" +"widgetpluginmanager.h" "7" +"widgetpluginpath.h" "7" +"rewritertransaction.h" "6" +"Sources" "5" +"exceptions" "6" +"exception.cpp" "7" +"invalidargumentexception.cpp" "7" +"invalididexception.cpp" "7" +"invalidmetainfoexception.cpp" "7" +"invalidmodelnodeexception.cpp" "7" +"invalidmodelstateexception.cpp" "7" +"invalidpropertyexception.cpp" "7" +"invalidqmlsourceexception.cpp" "7" +"invalidreparentingexception.cpp" "7" +"invalidslideindexexception.cpp" "7" +"notimplementedexception.cpp" "7" +"removebasestateexception.cpp" "7" +"rewritingexception.cpp" "7" +"metainfo" "6" +"itemlibraryinfo.cpp" "7" +"metainfo.cpp" "7" +"metainfoparser.cpp" "7" +"nodemetainfo.cpp" "7" +"subcomponentmanager.cpp" "7" +"model" "6" +"abstractproperty.cpp" "7" +"abstractview.cpp" "7" +"basetexteditmodifier.cpp" "7" +"bindingproperty.cpp" "7" +"componenttextmodifier.cpp" "7" +"import.cpp" "7" +"internalbindingproperty.cpp" "7" +"internalnode.cpp" "7" +"internalnodeabstractproperty.cpp" "7" +"internalnodelistproperty.cpp" "7" +"internalnodeproperty.cpp" "7" +"internalproperty.cpp" "7" +"internalvariantproperty.cpp" "7" +"model.cpp" "7" +"modelmerger.cpp" "7" +"modelnode.cpp" "7" +"modelnodecontextmenu.cpp" "7" +"modelnodepositionrecalculator.cpp" "7" +"modelnodepositionstorage.cpp" "7" +"modeltotextmerger.cpp" "7" +"nodeabstractproperty.cpp" "7" +"nodelistproperty.cpp" "7" +"nodeproperty.cpp" "7" +"painteventfilter.cpp" "7" +"plaintexteditmodifier.cpp" "7" +"propertycontainer.cpp" "7" +"propertynode.cpp" "7" +"propertyparser.cpp" "7" +"qmlanchors.cpp" "7" +"qmlchangeset.cpp" "7" +"qmlitemnode.cpp" "7" +"qmlmodelnodefacade.cpp" "7" +"qmlmodelview.cpp" "7" +"qmlobjectnode.cpp" "7" +"qmlstate.cpp" "7" +"qmltextgenerator.cpp" "7" +"rewriteaction.cpp" "7" +"rewriteactioncompressor.cpp" "7" +"rewriterview.cpp" "7" +"textmodifier.cpp" "7" +"texttomodelmerger.cpp" "7" +"variantparser.cpp" "7" +"variantproperty.cpp" "7" +"viewlogger.cpp" "7" +"pluginmanager" "6" +"widgetpluginmanager.cpp" "7" +"widgetpluginpath.cpp" "7" +"rewritertransaction.cpp" "6" +"extensionsystem" "4" +"extensionsystem.pri" "5" +"extensionsystem_dependencies" "4" +"extensionsystem_dependencies.pri" "5" +"filemanager" "4" +"filemanager.pri" "5" +"Headers" "5" +"addarraymembervisitor.h" "6" +"addobjectvisitor.h" "6" +"addpropertyvisitor.h" "6" +"astobjecttextextractor.h" "6" +"changeimportsvisitor.h" "6" +"changeobjecttypevisitor.h" "6" +"changepropertyvisitor.h" "6" +"firstdefinitionfinder.h" "6" +"moveobjectbeforeobjectvisitor.h" "6" +"moveobjectvisitor.h" "6" +"objectlengthcalculator.h" "6" +"qmlrefactoring.h" "6" +"qmlrewriter.h" "6" +"removepropertyvisitor.h" "6" +"removeuiobjectmembervisitor.h" "6" +"Sources" "5" +"addarraymembervisitor.cpp" "6" +"addobjectvisitor.cpp" "6" +"addpropertyvisitor.cpp" "6" +"astobjecttextextractor.cpp" "6" +"changeimportsvisitor.cpp" "6" +"changeobjecttypevisitor.cpp" "6" +"changepropertyvisitor.cpp" "6" +"firstdefinitionfinder.cpp" "6" +"moveobjectbeforeobjectvisitor.cpp" "6" +"moveobjectvisitor.cpp" "6" +"objectlengthcalculator.cpp" "6" +"qmlrefactoring.cpp" "6" +"qmlrewriter.cpp" "6" +"removepropertyvisitor.cpp" "6" +"removeuiobjectmembervisitor.cpp" "6" +"find" "4" +"find.pri" "5" +"find_dependencies" "4" +"find_dependencies.pri" "5" +"formeditor" "4" +"formeditor.pri" "5" +"Headers" "5" +"abstractformeditortool.h" "6" +"anchorcontroller.h" "6" +"anchorhandleitem.h" "6" +"anchorindicator.h" "6" +"anchorlinecontroller.h" "6" +"anchorlinehandleitem.h" "6" +"anchorlineindicator.h" "6" +"anchormanipulator.h" "6" +"anchortool.h" "6" +"controlelement.h" "6" +"dragtool.h" "6" +"formeditorgraphicsview.h" "6" +"formeditoritem.h" "6" +"formeditorscene.h" "6" +"formeditorview.h" "6" +"formeditorwidget.h" "6" +"itemutilfunctions.h" "6" +"layeritem.h" "6" +"lineeditaction.h" "6" +"movemanipulator.h" "6" +"movetool.h" "6" +"numberseriesaction.h" "6" +"onedimensionalcluster.h" "6" +"resizecontroller.h" "6" +"resizehandleitem.h" "6" +"resizeindicator.h" "6" +"resizemanipulator.h" "6" +"resizetool.h" "6" +"rubberbandselectionmanipulator.h" "6" +"scaleitem.h" "6" +"scalemanipulator.h" "6" +"selectionindicator.h" "6" +"selectionrectangle.h" "6" +"selectiontool.h" "6" +"singleselectionmanipulator.h" "6" +"snapper.h" "6" +"snappinglinecreator.h" "6" +"toolbox.h" "6" +"zoomaction.h" "6" +"Sources" "5" +"abstractformeditortool.cpp" "6" +"anchorcontroller.cpp" "6" +"anchorhandleitem.cpp" "6" +"anchorindicator.cpp" "6" +"anchorlinecontroller.cpp" "6" +"anchorlinehandleitem.cpp" "6" +"anchorlineindicator.cpp" "6" +"anchormanipulator.cpp" "6" +"anchortool.cpp" "6" +"controlelement.cpp" "6" +"dragtool.cpp" "6" +"formeditorgraphicsview.cpp" "6" +"formeditoritem.cpp" "6" +"formeditorscene.cpp" "6" +"formeditorview.cpp" "6" +"formeditorwidget.cpp" "6" +"itemutilfunctions.cpp" "6" +"layeritem.cpp" "6" +"lineeditaction.cpp" "6" +"movemanipulator.cpp" "6" +"movetool.cpp" "6" +"numberseriesaction.cpp" "6" +"onedimensionalcluster.cpp" "6" +"resizecontroller.cpp" "6" +"resizehandleitem.cpp" "6" +"resizeindicator.cpp" "6" +"resizemanipulator.cpp" "6" +"resizetool.cpp" "6" +"rubberbandselectionmanipulator.cpp" "6" +"scaleitem.cpp" "6" +"scalemanipulator.cpp" "6" +"selectionindicator.cpp" "6" +"selectionrectangle.cpp" "6" +"selectiontool.cpp" "6" +"singleselectionmanipulator.cpp" "6" +"snapper.cpp" "6" +"snappinglinecreator.cpp" "6" +"toolbox.cpp" "6" +"zoomaction.cpp" "6" +"Resources" "5" +"formeditor.qrc" "6" +"instances" "4" +"instances.pri" "5" +"Headers" "5" +"nodeinstance.h" "7" +"nodeinstanceserverproxy.h" "6" +"Sources" "5" +"nodeinstance.cpp" "6" +"nodeinstanceserverproxy.cpp" "6" +"nodeinstanceview.cpp" "6" +"integration" "4" +"integration.pri" "5" +"Headers" "5" +"componentaction.h" "6" +"componentview.h" "6" +"designdocumentcontroller.h" "6" +"designdocumentcontrollerview.h" "6" +"integrationcore.h" "6" +"stackedutilitypanelcontroller.h" "6" +"utilitypanelcontroller.h" "6" +"xuifiledialog.h" "6" +"Sources" "5" +"componentaction.cpp" "6" +"componentview.cpp" "6" +"designdocumentcontroller.cpp" "6" +"designdocumentcontrollerview.cpp" "6" +"integrationcore.cpp" "6" +"stackedutilitypanelcontroller.cpp" "6" +"utilitypanelcontroller.cpp" "6" +"xuifiledialog.cpp" "6" +"interfaces" "4" +"interfaces.pri" "5" +"Headers" "5" +"commondefines.h" "6" +"nodeinstanceclientinterface.h" "6" +"nodeinstanceserverinterface.h" "6" +"Sources" "5" +"nodeinstanceserverinterface.cpp" "6" +"itemlibrary" "4" +"itemlibrary.pri" "5" +"Headers" "5" +"customdraganddrop.h" "6" +"itemlibrarycomponents.h" "6" +"itemlibraryimageprovider.h" "6" +"itemlibrarymodel.h" "6" +"itemlibraryview.h" "6" +"itemlibrarywidget.h" "6" +"Sources" "5" +"customdraganddrop.cpp" "6" +"itemlibrarycomponents.cpp" "6" +"itemlibraryimageprovider.cpp" "6" +"itemlibrarymodel.cpp" "6" +"itemlibraryview.cpp" "6" +"itemlibrarywidget.cpp" "6" +"Resources" "5" +"itemlibrary.qrc" "6" +"QML" "5" +"qml" "6" +"ItemsView.qml" "7" +"ItemsViewStyle.qml" "7" +"ItemView.qml" "7" +"Scrollbar.qml" "7" +"SectionView.qml" "7" +"Selector.qml" "7" +"languageutils" "4" +"languageutils.pri" "5" +"locator" "4" +"locator.pri" "5" +"locator_dependencies" "4" +"locator_dependencies.pri" "5" +"navigator" "4" +"navigator.pri" "5" +"Headers" "5" +"navigatortreemodel.h" "6" +"navigatortreeview.h" "6" +"navigatorview.h" "6" +"navigatorwidget.h" "6" +"Sources" "5" +"navigatortreemodel.cpp" "6" +"navigatortreeview.cpp" "6" +"navigatorview.cpp" "6" +"navigatorwidget.cpp" "6" +"Resources" "5" +"navigator.qrc" "6" +"pluginmanager" "4" +"pluginmanager.pri" "5" +"Headers" "5" +"iplugin.h" "6" +"pluginmanager.h" "6" +"pluginpath.h" "6" +"Sources" "5" +"iplugin.cpp" "6" +"pluginmanager.cpp" "6" +"pluginpath.cpp" "6" +"private_headers" "4" +"private_headers.pri" "5" +"projectexplorer" "4" +"projectexplorer.pri" "5" +"projectexplorer_dependencies" "4" +"projectexplorer_dependencies.pri" "5" +"propertyeditor" "4" +"propertyeditor.pri" "5" +"Headers" "5" +"basiclayouts.h" "6" +"basicwidgets.h" "6" +"behaviordialog.h" "6" +"declarativewidgetview.h" "6" +"designerpropertymap.h" "6" +"filewidget.h" "6" +"fontwidget.h" "6" +"gradientlineqmladaptor.h" "6" +"layoutwidget.h" "6" +"originwidget.h" "6" +"propertyeditor.h" "6" +"propertyeditorcontextobject.h" "6" +"propertyeditortransaction.h" "6" +"propertyeditorvalue.h" "6" +"qlayoutobject.h" "6" +"qmlanchorbindingproxy.h" "6" +"qproxylayoutitem.h" "6" +"resetwidget.h" "6" +"siblingcombobox.h" "6" +"Sources" "5" +"basiclayouts.cpp" "6" +"basicwidgets.cpp" "6" +"behaviordialog.cpp" "6" +"declarativewidgetview.cpp" "6" +"filewidget.cpp" "6" +"fontwidget.cpp" "6" +"gradientlineqmladaptor.cpp" "6" +"layoutwidget.cpp" "6" +"originwidget.cpp" "6" +"propertyeditor.cpp" "6" +"propertyeditorcontextobject.cpp" "6" +"propertyeditortransaction.cpp" "6" +"propertyeditorvalue.cpp" "6" +"qlayoutobject.cpp" "6" +"qmlanchorbindingproxy.cpp" "6" +"qproxylayoutitem.cpp" "6" +"resetwidget.cpp" "6" +"siblingcombobox.cpp" "6" +"Forms" "5" +"behaviordialog.ui" "6" +"Resources" "5" +"propertyeditor.qrc" "6" +"qmldesigner_dependencies" "4" +"qmldesigner_dependencies.pri" "5" +"qmleditorwidgets" "4" +"qmleditorwidgets.pri" "5" +"qmljs" "4" +"qmljs.pri" "5" +"qmljseditor" "4" +"qmljseditor.pri" "5" +"qmljseditor_dependencies" "4" +"qmljseditor_dependencies.pri" "5" +"qmljstools" "4" +"qmljstools.pri" "5" +"qmljstools_dependencies" "4" +"qmljstools_dependencies.pri" "5" +"qmlprojectmanager" "4" +"qmlprojectmanager.pri" "5" +"qmlprojectmanager_dependencies" "4" +"qmlprojectmanager_dependencies.pri" "5" +"qt4projectmanager" "4" +"qt4projectmanager.pri" "5" +"qt4projectmanager_dependencies" "4" +"qt4projectmanager_dependencies.pri" "5" +"qtcreator" "4" +"qtcreator.pri" "5" +"qtcreatorplugin" "4" +"qtcreatorplugin.pri" "5" +"Other files" "5" +"plugins/qmldesigner" "6" +"QmlDesigner.pluginspec.in" "7" +"qts60stylethemeio" "4" +"qts60stylethemeio.pri" "5" +"qtsupport" "4" +"qtsupport.pri" "5" +"qtsupport_dependencies" "4" +"qtsupport_dependencies.pri" "5" +"resources" "4" +"resources.pri" "5" +"Resources" "5" +"resources.qrc" "6" +"stateseditor" "4" +"stateseditor.pri" "5" +"Headers" "5" +"stateseditorimageprovider.cpp" "6" +"stateseditormodel.h" "6" +"stateseditorview.h" "6" +"stateseditorwidget.h" "6" +"Sources" "5" +"stateseditorimageprovider.cpp" "6" +"stateseditormodel.cpp" "6" +"stateseditorview.cpp" "6" +"stateseditorwidget.cpp" "6" +"Resources" "5" +"stateseditor.qrc" "6" +"QML" "5" +"HorizontalScrollBar.qml" "6" +"stateslist.qml" "6" +"symbianutils" "4" +"symbianutils.pri" "5" +"texteditor" "4" +"texteditor.pri" "5" +"texteditor_dependencies" "4" +"texteditor_dependencies.pri" "5" +"utils" "4" +"utils.pri" "5" +"utils_dependencies" "4" +"utils_dependencies.pri" "5" +"Headers" "4" +"designersettings.h" "5" +"designmodecontext.h" "5" +"designmodewidget.h" "5" +"qmldesignerconstants.h" "5" +"qmldesignerplugin.h" "5" +"settingspage.h" "5" +"styledoutputpaneplaceholder.h" "5" +"Sources" "4" +"designersettings.cpp" "5" +"designmodecontext.cpp" "5" +"designmodewidget.cpp" "5" +"qmldesignerplugin.cpp" "5" +"settingspage.cpp" "5" +"styledoutputpaneplaceholder.cpp" "5" +"Forms" "4" +"settingspage.ui" "5" +"qtquickplugin" "3" +"qtquickplugin.pro" "4" +"iwidgetplugin" "4" +"iwidgetplugin.pri" "5" +"plugindestdir" "4" +"plugindestdir.pri" "5" +"private_headers" "4" +"private_headers.pri" "5" +"qtcreator" "4" +"qtcreator.pri" "5" +"qtquickplugin" "4" +"qtquickplugin.pri" "5" +"Headers" "5" +"iwidgetplugin.h" "7" +"qtquickplugin.h" "6" +"Sources" "5" +"qtquickplugin.cpp" "6" +"Resources" "5" +"qtquickplugin.qrc" "6" +"Other files" "5" +"quick.metainfo" "6" +"symbianplugin" "3" +"symbianplugin.pro" "4" +"iwidgetplugin" "4" +"iwidgetplugin.pri" "5" +"plugindestdir" "4" +"plugindestdir.pri" "5" +"private_headers" "4" +"private_headers.pri" "5" +"qtcreator" "4" +"qtcreator.pri" "5" +"symbianplugin" "4" +"symbianplugin.pri" "5" +"Headers" "5" +"iwidgetplugin.h" "7" +"symbianplugin.h" "6" +"Sources" "5" +"symbianplugin.cpp" "6" +"Resources" "5" +"symbianplugin.qrc" "6" +"Other files" "5" +"symbian.metainfo" "6" +"qmljseditor" "2" +"qmljseditor.pro" "3" +"aggregation" "3" +"aggregation.pri" "4" +"coreplugin" "3" +"coreplugin.pri" "4" +"coreplugin_dependencies" "3" +"coreplugin_dependencies.pri" "4" +"cplusplus" "3" +"cplusplus.pri" "4" +"extensionsystem" "3" +"extensionsystem.pri" "4" +"extensionsystem_dependencies" "3" +"extensionsystem_dependencies.pri" "4" +"find" "3" +"find.pri" "4" +"find_dependencies" "3" +"find_dependencies.pri" "4" +"languageutils" "3" +"languageutils.pri" "4" +"locator" "3" +"locator.pri" "4" +"locator_dependencies" "3" +"locator_dependencies.pri" "4" +"projectexplorer" "3" +"projectexplorer.pri" "4" +"projectexplorer_dependencies" "3" +"projectexplorer_dependencies.pri" "4" +"qmleditorwidgets" "3" +"qmleditorwidgets.pri" "4" +"qmljs" "3" +"qmljs.pri" "4" +"qmljseditor_dependencies" "3" +"qmljseditor_dependencies.pri" "4" +"qmljstools" "3" +"qmljstools.pri" "4" +"qmljstools_dependencies" "3" +"qmljstools_dependencies.pri" "4" +"qtcreator" "3" +"qtcreator.pri" "4" +"qtcreatorplugin" "3" +"qtcreatorplugin.pri" "4" +"Other files" "4" +"plugins/qmljseditor" "5" +"QmlJSEditor.pluginspec.in" "6" +"qtsupport" "3" +"qtsupport.pri" "4" +"qtsupport_dependencies" "3" +"qtsupport_dependencies.pri" "4" +"texteditor" "3" +"texteditor.pri" "4" +"texteditor_dependencies" "3" +"texteditor_dependencies.pri" "4" +"utils" "3" +"utils.pri" "4" +"utils_dependencies" "3" +"utils_dependencies.pri" "4" +"Headers" "3" +"jsfilewizard.h" "4" +"qmlexpressionundercursor.h" "4" +"qmlfilewizard.h" "4" +"qmljsautocompleter.h" "4" +"qmljscompletionassist.h" "4" +"qmljscomponentfromobjectdef.h" "4" +"qmljscomponentnamedialog.h" "4" +"qmljseditor.h" "4" +"qmljseditor_global.h" "4" +"qmljseditoractionhandler.h" "4" +"qmljseditorconstants.h" "4" +"qmljseditoreditable.h" "4" +"qmljseditorfactory.h" "4" +"qmljseditorplugin.h" "4" +"qmljsfindreferences.h" "4" +"qmljshighlighter.h" "4" +"qmljshoverhandler.h" "4" +"qmljsoutline.h" "4" +"qmljsoutlinetreeview.h" "4" +"qmljspreviewrunner.h" "4" +"qmljsquickfix.h" "4" +"qmljsquickfixassist.h" "4" +"qmljsreuse.h" "4" +"qmljssemantichighlighter.h" "4" +"qmljssemanticinfoupdater.h" "4" +"qmljssnippetprovider.h" "4" +"qmljswrapinloader.h" "4" +"qmloutlinemodel.h" "4" +"qmltaskmanager.h" "4" +"quicktoolbar.h" "4" +"quicktoolbarsettingspage.h" "4" +"Sources" "3" +"jsfilewizard.cpp" "4" +"qmlexpressionundercursor.cpp" "4" +"qmlfilewizard.cpp" "4" +"qmljsautocompleter.cpp" "4" +"qmljscompletionassist.cpp" "4" +"qmljscomponentfromobjectdef.cpp" "4" +"qmljscomponentnamedialog.cpp" "4" +"qmljseditor.cpp" "4" +"qmljseditoractionhandler.cpp" "4" +"qmljseditoreditable.cpp" "4" +"qmljseditorfactory.cpp" "4" +"qmljseditorplugin.cpp" "4" +"qmljsfindreferences.cpp" "4" +"qmljshighlighter.cpp" "4" +"qmljshoverhandler.cpp" "4" +"qmljsoutline.cpp" "4" +"qmljsoutlinetreeview.cpp" "4" +"qmljspreviewrunner.cpp" "4" +"qmljsquickfix.cpp" "4" +"qmljsquickfixassist.cpp" "4" +"qmljsquickfixes.cpp" "4" +"qmljsreuse.cpp" "4" +"qmljssemantichighlighter.cpp" "4" +"qmljssemanticinfoupdater.cpp" "4" +"qmljssnippetprovider.cpp" "4" +"qmljswrapinloader.cpp" "4" +"qmloutlinemodel.cpp" "4" +"qmltaskmanager.cpp" "4" +"quicktoolbar.cpp" "4" +"quicktoolbarsettingspage.cpp" "4" +"Forms" "3" +"qmljscomponentnamedialog.ui" "4" +"quicktoolbarsettingspage.ui" "4" +"Resources" "3" +"qmljseditor.qrc" "4" +"Other files" "3" +"QmlJSEditor.mimetypes.xml" "4" +"qmljstools" "2" +"qmljstools.pro" "3" +"aggregation" "3" +"aggregation.pri" "4" +"coreplugin" "3" +"coreplugin.pri" "4" +"coreplugin_dependencies" "3" +"coreplugin_dependencies.pri" "4" +"cplusplus" "3" +"cplusplus.pri" "4" +"extensionsystem" "3" +"extensionsystem.pri" "4" +"extensionsystem_dependencies" "3" +"extensionsystem_dependencies.pri" "4" +"find" "3" +"find.pri" "4" +"find_dependencies" "3" +"find_dependencies.pri" "4" +"languageutils" "3" +"languageutils.pri" "4" +"locator" "3" +"locator.pri" "4" +"locator_dependencies" "3" +"locator_dependencies.pri" "4" +"projectexplorer" "3" +"projectexplorer.pri" "4" +"projectexplorer_dependencies" "3" +"projectexplorer_dependencies.pri" "4" +"qmljs" "3" +"qmljs.pri" "4" +"qmljstools-lib" "3" +"qmljstools-lib.pri" "4" +"Headers" "4" +"qmljscodestylepreferencesfactory.h" "5" +"qmljscodestylesettingspage.h" "5" +"qmljsfindexportedcpptypes.h" "5" +"qmljsfunctionfilter.h" "5" +"qmljsindenter.h" "5" +"qmljslocatordata.h" "5" +"qmljsmodelmanager.h" "5" +"qmljsplugindumper.h" "5" +"qmljsqtstylecodeformatter.h" "5" +"qmljsrefactoringchanges.h" "5" +"qmljstools_global.h" "5" +"qmljstoolsconstants.h" "5" +"qmljstoolsplugin.h" "5" +"qmljstoolssettings.h" "5" +"Sources" "4" +"qmljscodestylepreferencesfactory.cpp" "5" +"qmljscodestylesettingspage.cpp" "5" +"qmljsfindexportedcpptypes.cpp" "5" +"qmljsfunctionfilter.cpp" "5" +"qmljsindenter.cpp" "5" +"qmljslocatordata.cpp" "5" +"qmljsmodelmanager.cpp" "5" +"qmljsplugindumper.cpp" "5" +"qmljsqtstylecodeformatter.cpp" "5" +"qmljsrefactoringchanges.cpp" "5" +"qmljstools_test.cpp" "5" +"qmljstoolsplugin.cpp" "5" +"qmljstoolssettings.cpp" "5" +"Forms" "4" +"qmljscodestylesettingspage.ui" "5" +"qmljstools_dependencies" "3" +"qmljstools_dependencies.pri" "4" +"qtcreator" "3" +"qtcreator.pri" "4" +"qtcreatorplugin" "3" +"qtcreatorplugin.pri" "4" +"Other files" "4" +"plugins/qmljstools" "5" +"QmlJSTools.pluginspec.in" "6" +"qtsupport" "3" +"qtsupport.pri" "4" +"qtsupport_dependencies" "3" +"qtsupport_dependencies.pri" "4" +"texteditor" "3" +"texteditor.pri" "4" +"texteditor_dependencies" "3" +"texteditor_dependencies.pri" "4" +"utils" "3" +"utils.pri" "4" +"utils_dependencies" "3" +"utils_dependencies.pri" "4" +"qmlprofiler" "2" +"qmlprofiler.pro" "3" +"aggregation" "3" +"aggregation.pri" "4" +"analyzerbase" "3" +"analyzerbase.pri" "4" +"analyzerbase_dependencies" "3" +"analyzerbase_dependencies.pri" "4" +"botan" "3" +"botan.pri" "4" +"canvas" "3" +"canvas.pri" "4" +"Headers" "4" +"qdeclarativecanvas_p.h" "5" +"qdeclarativecanvastimer_p.h" "5" +"qdeclarativecontext2d_p.h" "5" +"qmlprofilercanvas.h" "5" +"Sources" "4" +"qdeclarativecanvas.cpp" "5" +"qdeclarativecanvastimer.cpp" "5" +"qdeclarativecontext2d.cpp" "5" +"qmlprofilercanvas.cpp" "5" +"coreplugin" "3" +"coreplugin.pri" "4" +"coreplugin_dependencies" "3" +"coreplugin_dependencies.pri" "4" +"cplusplus" "3" +"cplusplus.pri" "4" +"cpptools" "3" +"cpptools.pri" "4" +"cpptools_dependencies" "3" +"cpptools_dependencies.pri" "4" +"debugger" "3" +"debugger.pri" "4" +"debugger_dependencies" "3" +"debugger_dependencies.pri" "4" +"extensionsystem" "3" +"extensionsystem.pri" "4" +"extensionsystem_dependencies" "3" +"extensionsystem_dependencies.pri" "4" +"find" "3" +"find.pri" "4" +"find_dependencies" "3" +"find_dependencies.pri" "4" +"languageutils" "3" +"languageutils.pri" "4" +"locator" "3" +"locator.pri" "4" +"locator_dependencies" "3" +"locator_dependencies.pri" "4" +"projectexplorer" "3" +"projectexplorer.pri" "4" +"projectexplorer_dependencies" "3" +"projectexplorer_dependencies.pri" "4" +"qmleditorwidgets" "3" +"qmleditorwidgets.pri" "4" +"qmljs" "3" +"qmljs.pri" "4" +"qmljseditor" "3" +"qmljseditor.pri" "4" +"qmljseditor_dependencies" "3" +"qmljseditor_dependencies.pri" "4" +"qmljstools" "3" +"qmljstools.pri" "4" +"qmljstools_dependencies" "3" +"qmljstools_dependencies.pri" "4" +"qmlprojectmanager" "3" +"qmlprojectmanager.pri" "4" +"qmlprojectmanager_dependencies" "3" +"qmlprojectmanager_dependencies.pri" "4" +"qt4projectmanager" "3" +"qt4projectmanager.pri" "4" +"qt4projectmanager_dependencies" "3" +"qt4projectmanager_dependencies.pri" "4" +"qtcreator" "3" +"qtcreator.pri" "4" +"qtcreatorplugin" "3" +"qtcreatorplugin.pri" "4" +"Other files" "4" +"plugins/qmlprofiler" "5" +"QmlProfiler.pluginspec.in" "6" +"qtsupport" "3" +"qtsupport.pri" "4" +"qtsupport_dependencies" "3" +"qtsupport_dependencies.pri" "4" +"remotelinux" "3" +"remotelinux.pri" "4" +"remotelinux_dependencies" "3" +"remotelinux_dependencies.pri" "4" +"symbianutils" "3" +"symbianutils.pri" "4" +"texteditor" "3" +"texteditor.pri" "4" +"texteditor_dependencies" "3" +"texteditor_dependencies.pri" "4" +"utils" "3" +"utils.pri" "4" +"utils_dependencies" "3" +"utils_dependencies.pri" "4" +"Headers" "3" +"abstractqmlprofilerrunner.h" "4" +"codaqmlprofilerrunner.h" "4" +"localqmlprofilerrunner.h" "4" +"qmlprofiler_global.h" "4" +"qmlprofilerattachdialog.h" "4" +"qmlprofilerconstants.h" "4" +"qmlprofilerdetailsrewriter.h" "4" +"qmlprofilerengine.h" "4" +"qmlprofilereventview.h" "4" +"qmlprofilerplugin.h" "4" +"qmlprofilertool.h" "4" +"remotelinuxqmlprofilerrunner.h" "4" +"Sources" "3" +"codaqmlprofilerrunner.cpp" "4" +"localqmlprofilerrunner.cpp" "4" +"qmlprofilerattachdialog.cpp" "4" +"qmlprofilerdetailsrewriter.cpp" "4" +"qmlprofilerengine.cpp" "4" +"qmlprofilereventview.cpp" "4" +"qmlprofilerplugin.cpp" "4" +"qmlprofilertool.cpp" "4" +"remotelinuxqmlprofilerrunner.cpp" "4" +"Forms" "3" +"qmlprofilerattachdialog.ui" "4" +"Resources" "3" +"qml" "4" +"qmlprofiler.qrc" "5" +"QML" "3" +"qml" "4" +"Detail.qml" "5" +"Label.qml" "5" +"MainView.qml" "5" +"Overview.qml" "5" +"RangeDetails.qml" "5" +"RangeMover.qml" "5" +"SelectionRange.qml" "5" +"SelectionRangeDetails.qml" "5" +"TimeDisplay.qml" "5" +"TimeMarks.qml" "5" +"qmlprojectmanager" "2" +"qmlprojectmanager.pro" "3" +"aggregation" "3" +"aggregation.pri" "4" +"botan" "3" +"botan.pri" "4" +"coreplugin" "3" +"coreplugin.pri" "4" +"coreplugin_dependencies" "3" +"coreplugin_dependencies.pri" "4" +"cplusplus" "3" +"cplusplus.pri" "4" +"cpptools" "3" +"cpptools.pri" "4" +"cpptools_dependencies" "3" +"cpptools_dependencies.pri" "4" +"debugger" "3" +"debugger.pri" "4" +"debugger_dependencies" "3" +"debugger_dependencies.pri" "4" +"extensionsystem" "3" +"extensionsystem.pri" "4" +"extensionsystem_dependencies" "3" +"extensionsystem_dependencies.pri" "4" +"fileformat" "3" +"fileformat.pri" "4" +"Headers" "4" +"filefilteritems.h" "5" +"qmlprojectfileformat.h" "5" +"qmlprojectitem.h" "5" +"Sources" "4" +"filefilteritems.cpp" "5" +"qmlprojectfileformat.cpp" "5" +"qmlprojectitem.cpp" "5" +"find" "3" +"find.pri" "4" +"find_dependencies" "3" +"find_dependencies.pri" "4" +"languageutils" "3" +"languageutils.pri" "4" +"locator" "3" +"locator.pri" "4" +"locator_dependencies" "3" +"locator_dependencies.pri" "4" +"projectexplorer" "3" +"projectexplorer.pri" "4" +"projectexplorer_dependencies" "3" +"projectexplorer_dependencies.pri" "4" +"qmleditorwidgets" "3" +"qmleditorwidgets.pri" "4" +"qmljs" "3" +"qmljs.pri" "4" +"qmljseditor" "3" +"qmljseditor.pri" "4" +"qmljseditor_dependencies" "3" +"qmljseditor_dependencies.pri" "4" +"qmljstools" "3" +"qmljstools.pri" "4" +"qmljstools_dependencies" "3" +"qmljstools_dependencies.pri" "4" +"qmlprojectmanager_dependencies" "3" +"qmlprojectmanager_dependencies.pri" "4" +"qtcreator" "3" +"qtcreator.pri" "4" +"qtcreatorplugin" "3" +"qtcreatorplugin.pri" "4" +"Other files" "4" +"plugins/qmlprojectmanager" "5" +"QmlProjectManager.pluginspec.in" "6" +"qtsupport" "3" +"qtsupport.pri" "4" +"qtsupport_dependencies" "3" +"qtsupport_dependencies.pri" "4" +"symbianutils" "3" +"symbianutils.pri" "4" +"texteditor" "3" +"texteditor.pri" "4" +"texteditor_dependencies" "3" +"texteditor_dependencies.pri" "4" +"utils" "3" +"utils.pri" "4" +"utils_dependencies" "3" +"utils_dependencies.pri" "4" +"Headers" "3" +"qmlproject.h" "4" +"qmlprojectapplicationwizard.h" "4" +"qmlprojectconstants.h" "4" +"qmlprojectfile.h" "4" +"qmlprojectmanager.h" "4" +"qmlprojectmanager_global.h" "4" +"qmlprojectmanagerconstants.h" "4" +"qmlprojectnodes.h" "4" +"qmlprojectplugin.h" "4" +"qmlprojectrunconfiguration.h" "4" +"qmlprojectrunconfigurationfactory.h" "4" +"qmlprojectrunconfigurationwidget.h" "4" +"qmlprojectruncontrol.h" "4" +"qmlprojecttarget.h" "4" +"Sources" "3" +"qmlproject.cpp" "4" +"qmlprojectapplicationwizard.cpp" "4" +"qmlprojectfile.cpp" "4" +"qmlprojectmanager.cpp" "4" +"qmlprojectnodes.cpp" "4" +"qmlprojectplugin.cpp" "4" +"qmlprojectrunconfiguration.cpp" "4" +"qmlprojectrunconfigurationfactory.cpp" "4" +"qmlprojectrunconfigurationwidget.cpp" "4" +"qmlprojectruncontrol.cpp" "4" +"qmlprojecttarget.cpp" "4" +"Resources" "3" +"qmlproject.qrc" "4" +"Other files" "3" +"QmlProject.mimetypes.xml" "4" +"qt4projectmanager" "2" +"qt4projectmanager.pro" "3" +"aggregation" "3" +"aggregation.pri" "4" +"botan" "3" +"botan.pri" "4" +"coreplugin" "3" +"coreplugin.pri" "4" +"coreplugin_dependencies" "3" +"coreplugin_dependencies.pri" "4" +"cplusplus" "3" +"cplusplus.pri" "4" +"cpptools" "3" +"cpptools.pri" "4" +"cpptools_dependencies" "3" +"cpptools_dependencies.pri" "4" +"customwidgetwizard" "3" +"customwidgetwizard.pri" "4" +"Headers" "4" +"classdefinition.h" "5" +"classlist.h" "5" +"customwidgetpluginwizardpage.h" "5" +"customwidgetwidgetswizardpage.h" "5" +"customwidgetwizard.h" "5" +"customwidgetwizarddialog.h" "5" +"filenamingparameters.h" "5" +"plugingenerator.h" "5" +"pluginoptions.h" "5" +"Sources" "4" +"classdefinition.cpp" "5" +"classlist.cpp" "5" +"customwidgetpluginwizardpage.cpp" "5" +"customwidgetwidgetswizardpage.cpp" "5" +"customwidgetwizard.cpp" "5" +"customwidgetwizarddialog.cpp" "5" +"plugingenerator.cpp" "5" +"Forms" "4" +"classdefinition.ui" "5" +"customwidgetpluginwizardpage.ui" "5" +"customwidgetwidgetswizardpage.ui" "5" +"debugger" "3" +"debugger.pri" "4" +"debugger_dependencies" "3" +"debugger_dependencies.pri" "4" +"extensionsystem" "3" +"extensionsystem.pri" "4" +"extensionsystem_dependencies" "3" +"extensionsystem_dependencies.pri" "4" +"find" "3" +"find.pri" "4" +"find_dependencies" "3" +"find_dependencies.pri" "4" +"json" "3" +"json.pri" "4" +"Headers" "4" +"json.h" "5" +"json_global.h" "5" +"Sources" "4" +"json.cpp" "5" +"languageutils" "3" +"languageutils.pri" "4" +"locator" "3" +"locator.pri" "4" +"locator_dependencies" "3" +"locator_dependencies.pri" "4" +"projectexplorer" "3" +"projectexplorer.pri" "4" +"projectexplorer_dependencies" "3" +"projectexplorer_dependencies.pri" "4" +"qmljs" "3" +"qmljs.pri" "4" +"qt-desktop" "3" +"qt-desktop.pri" "4" +"Headers" "4" +"desktopqtversion.h" "5" +"desktopqtversionfactory.h" "5" +"qt4desktoptarget.h" "5" +"qt4desktoptargetfactory.h" "5" +"qt4runconfiguration.h" "5" +"qt4simulatortarget.h" "5" +"qt4simulatortargetfactory.h" "5" +"simulatorqtversion.h" "5" +"simulatorqtversionfactory.h" "5" +"Sources" "4" +"desktopqtversion.cpp" "5" +"desktopqtversionfactory.cpp" "5" +"qt4desktoptarget.cpp" "5" +"qt4desktoptargetfactory.cpp" "5" +"qt4runconfiguration.cpp" "5" +"qt4simulatortarget.cpp" "5" +"qt4simulatortargetfactory.cpp" "5" +"simulatorqtversion.cpp" "5" +"simulatorqtversionfactory.cpp" "5" +"qt-s60" "3" +"qt-s60.pri" "4" +"Headers" "4" +"abldparser.h" "5" +"certificatepathchooser.h" "5" +"codaruncontrol.h" "5" +"gccetoolchain.h" "5" +"passphraseforkeydialog.h" "5" +"qt4symbiantarget.h" "5" +"qt4symbiantargetfactory.h" "5" +"rvctparser.h" "5" +"rvcttoolchain.h" "5" +"s60certificatedetailsdialog.h" "5" +"s60certificateinfo.h" "5" +"s60createpackageparser.h" "5" +"s60createpackagestep.h" "5" +"s60deployconfiguration.h" "5" +"s60deployconfigurationwidget.h" "5" +"s60deploystep.h" "5" +"s60devicedebugruncontrol.h" "5" +"s60devicerunconfiguration.h" "5" +"s60devicerunconfigurationwidget.h" "5" +"s60manager.h" "5" +"s60publisherovi.h" "5" +"s60publishingbuildsettingspageovi.h" "5" +"s60publishingresultspageovi.h" "5" +"s60publishingsissettingspageovi.h" "5" +"s60publishingwizardfactories.h" "5" +"s60publishingwizardovi.h" "5" +"s60runcontrolbase.h" "5" +"s60runcontrolfactory.h" "5" +"s60symbiancertificate.h" "5" +"sbsv2parser.h" "5" +"symbianqtversion.h" "5" +"symbianqtversionfactory.h" "5" +"Sources" "4" +"abldparser.cpp" "5" +"certificatepathchooser.cpp" "5" +"codaruncontrol.cpp" "5" +"gccetoolchain.cpp" "5" +"passphraseforkeydialog.cpp" "5" +"qt4symbiantarget.cpp" "5" +"qt4symbiantargetfactory.cpp" "5" +"rvctparser.cpp" "5" +"rvcttoolchain.cpp" "5" +"s60certificatedetailsdialog.cpp" "5" +"s60certificateinfo.cpp" "5" +"s60createpackageparser.cpp" "5" +"s60createpackagestep.cpp" "5" +"s60deployconfiguration.cpp" "5" +"s60deployconfigurationwidget.cpp" "5" +"s60deploystep.cpp" "5" +"s60devicedebugruncontrol.cpp" "5" +"s60devicerunconfiguration.cpp" "5" +"s60devicerunconfigurationwidget.cpp" "5" +"s60manager.cpp" "5" +"s60publisherovi.cpp" "5" +"s60publishingbuildsettingspageovi.cpp" "5" +"s60publishingresultspageovi.cpp" "5" +"s60publishingsissettingspageovi.cpp" "5" +"s60publishingwizardfactories.cpp" "5" +"s60publishingwizardovi.cpp" "5" +"s60runcontrolbase.cpp" "5" +"s60runcontrolfactory.cpp" "5" +"s60symbiancertificate.cpp" "5" +"sbsv2parser.cpp" "5" +"symbianqtversion.cpp" "5" +"symbianqtversionfactory.cpp" "5" +"Forms" "4" +"rvcttoolchainconfigwidget.ui" "5" +"s60certificatedetailsdialog.ui" "5" +"s60createpackagestep.ui" "5" +"s60publishingbuildsettingspageovi.ui" "5" +"s60publishingresultspageovi.ui" "5" +"s60publishingsissettingspageovi.ui" "5" +"qt4projectmanager_dependencies" "3" +"qt4projectmanager_dependencies.pri" "4" +"qtcreator" "3" +"qtcreator.pri" "4" +"qtcreatorplugin" "3" +"qtcreatorplugin.pri" "4" +"Other files" "4" +"plugins/qt4projectmanager" "5" +"Qt4ProjectManager.pluginspec.in" "6" +"qtsupport" "3" +"qtsupport.pri" "4" +"qtsupport_dependencies" "3" +"qtsupport_dependencies.pri" "4" +"symbianutils" "3" +"symbianutils.pri" "4" +"texteditor" "3" +"texteditor.pri" "4" +"texteditor_dependencies" "3" +"texteditor_dependencies.pri" "4" +"utils" "3" +"utils.pri" "4" +"utils_dependencies" "3" +"utils_dependencies.pri" "4" +"Headers" "3" +"wizards" "4" +"abstractmobileapp.h" "5" +"abstractmobileappwizard.h" "5" +"consoleappwizard.h" "5" +"consoleappwizarddialog.h" "5" +"emptyprojectwizard.h" "5" +"emptyprojectwizarddialog.h" "5" +"filespage.h" "5" +"guiappwizard.h" "5" +"guiappwizarddialog.h" "5" +"html5app.h" "5" +"html5appwizard.h" "5" +"html5appwizardpages.h" "5" +"libraryparameters.h" "5" +"librarywizard.h" "5" +"librarywizarddialog.h" "5" +"mobileapp.h" "5" +"mobileappwizard.h" "5" +"mobileappwizardpages.h" "5" +"mobilelibraryparameters.h" "5" +"mobilelibrarywizardoptionpage.h" "5" +"modulespage.h" "5" +"qtprojectparameters.h" "5" +"qtquickapp.h" "5" +"qtquickappwizard.h" "5" +"qtquickappwizardpages.h" "5" +"qtwizard.h" "5" +"subdirsprojectwizard.h" "5" +"subdirsprojectwizarddialog.h" "5" +"targetsetuppage.h" "5" +"testwizard.h" "5" +"testwizarddialog.h" "5" +"testwizardpage.h" "5" +"addlibrarywizard.h" "4" +"buildconfigurationinfo.h" "4" +"externaleditors.h" "4" +"findqt4profiles.h" "4" +"librarydetailscontroller.h" "4" +"makestep.h" "4" +"profilecompletionassist.h" "4" +"profileeditor.h" "4" +"profileeditorfactory.h" "4" +"profilehighlighter.h" "4" +"profilehoverhandler.h" "4" +"profilekeywords.h" "4" +"qmakeparser.h" "4" +"qmakestep.h" "4" +"qt4basetargetfactory.h" "4" +"qt4buildconfiguration.h" "4" +"qt4nodes.h" "4" +"qt4project.h" "4" +"qt4projectconfigwidget.h" "4" +"qt4projectmanager.h" "4" +"qt4projectmanager_global.h" "4" +"qt4projectmanagerconstants.h" "4" +"qt4projectmanagerplugin.h" "4" +"qt4target.h" "4" +"qt4targetsetupwidget.h" "4" +"qtmodulesinfo.h" "4" +"qtuicodemodelsupport.h" "4" +"unconfiguredprojectpanel.h" "4" +"unconfiguredsettingsoptionpage.h" "4" +"winceqtversion.h" "4" +"winceqtversionfactory.h" "4" +"Sources" "3" +"wizards" "4" +"abstractmobileapp.cpp" "5" +"abstractmobileappwizard.cpp" "5" +"consoleappwizard.cpp" "5" +"consoleappwizarddialog.cpp" "5" +"emptyprojectwizard.cpp" "5" +"emptyprojectwizarddialog.cpp" "5" +"filespage.cpp" "5" +"guiappwizard.cpp" "5" +"guiappwizarddialog.cpp" "5" +"html5app.cpp" "5" +"html5appwizard.cpp" "5" +"html5appwizardpages.cpp" "5" +"libraryparameters.cpp" "5" +"librarywizard.cpp" "5" +"librarywizarddialog.cpp" "5" +"mobileapp.cpp" "5" +"mobileappwizard.cpp" "5" +"mobileappwizardpages.cpp" "5" +"mobilelibraryparameters.cpp" "5" +"mobilelibrarywizardoptionpage.cpp" "5" +"modulespage.cpp" "5" +"qtprojectparameters.cpp" "5" +"qtquickapp.cpp" "5" +"qtquickappwizard.cpp" "5" +"qtquickappwizardpages.cpp" "5" +"qtwizard.cpp" "5" +"subdirsprojectwizard.cpp" "5" +"subdirsprojectwizarddialog.cpp" "5" +"targetsetuppage.cpp" "5" +"testwizard.cpp" "5" +"testwizarddialog.cpp" "5" +"testwizardpage.cpp" "5" +"addlibrarywizard.cpp" "4" +"externaleditors.cpp" "4" +"findqt4profiles.cpp" "4" +"librarydetailscontroller.cpp" "4" +"makestep.cpp" "4" +"profilecompletionassist.cpp" "4" +"profileeditor.cpp" "4" +"profileeditorfactory.cpp" "4" +"profilehighlighter.cpp" "4" +"profilehoverhandler.cpp" "4" +"profilekeywords.cpp" "4" +"qmakeparser.cpp" "4" +"qmakestep.cpp" "4" +"qt4buildconfiguration.cpp" "4" +"qt4nodes.cpp" "4" +"qt4project.cpp" "4" +"qt4projectconfigwidget.cpp" "4" +"qt4projectmanager.cpp" "4" +"qt4projectmanagerplugin.cpp" "4" +"qt4target.cpp" "4" +"qtmodulesinfo.cpp" "4" +"qtuicodemodelsupport.cpp" "4" +"unconfiguredprojectpanel.cpp" "4" +"unconfiguredsettingsoptionpage.cpp" "4" +"winceqtversion.cpp" "4" +"winceqtversionfactory.cpp" "4" +"Forms" "3" +"wizards" "4" +"html5appwizardsourcespage.ui" "5" +"mobileappwizardgenericoptionspage.ui" "5" +"mobileappwizardharmattanoptionspage.ui" "5" +"mobileappwizardmaemooptionspage.ui" "5" +"mobileappwizardsymbianoptionspage.ui" "5" +"mobilelibrarywizardoptionpage.ui" "5" +"qtquickcomponentsetoptionspage.ui" "5" +"targetsetuppage.ui" "5" +"testwizardpage.ui" "5" +"librarydetailswidget.ui" "4" +"makestep.ui" "4" +"qmakestep.ui" "4" +"qt4projectconfigwidget.ui" "4" +"Resources" "3" +"wizards" "4" +"wizards.qrc" "5" +"qt4projectmanager.qrc" "4" +"Other files" "3" +"Qt4ProjectManager.mimetypes.xml" "4" +"qtcreator" "2" +"qtcreator.pri" "3" +"qtcreator-lldb" "2" +"qtcreator-lldb.pro" "3" +"qtcreator" "3" +"qtcreator.pri" "4" +"Headers" "3" +"lldb" "5" +"ipcengineguest.h" "6" +"breakpoint.h" "5" +"debuggerstreamops.h" "5" +"disassemblerlines.h" "5" +"stackframe.h" "5" +"watchdata.h" "5" +"lldbengineguest.h" "4" +"Sources" "3" +"lldb" "5" +"ipcengineguest.cpp" "6" +"breakpoint.cpp" "5" +"debuggerstreamops.cpp" "5" +"disassemblerlines.cpp" "5" +"stackframe.cpp" "5" +"watchdata.cpp" "5" +"lldbengineguest.cpp" "4" +"main.cpp" "4" +"qtsupport" "2" +"qtsupport.pro" "3" +"aggregation" "3" +"aggregation.pri" "4" +"coreplugin" "3" +"coreplugin.pri" "4" +"coreplugin_dependencies" "3" +"coreplugin_dependencies.pri" "4" +"extensionsystem" "3" +"extensionsystem.pri" "4" +"extensionsystem_dependencies" "3" +"extensionsystem_dependencies.pri" "4" +"find" "3" +"find.pri" "4" +"find_dependencies" "3" +"find_dependencies.pri" "4" +"locator" "3" +"locator.pri" "4" +"locator_dependencies" "3" +"locator_dependencies.pri" "4" +"projectexplorer" "3" +"projectexplorer.pri" "4" +"projectexplorer_dependencies" "3" +"projectexplorer_dependencies.pri" "4" +"proparser" "3" +"proparser.pri" "4" +"Headers" "4" +"ioutils.h" "5" +"profileevaluator.h" "5" +"profileparser.h" "5" +"proitems.h" "5" +"proparser_global.h" "5" +"prowriter.h" "5" +"Sources" "4" +"ioutils.cpp" "5" +"profileevaluator.cpp" "5" +"profileparser.cpp" "5" +"proitems.cpp" "5" +"prowriter.cpp" "5" +"Resources" "4" +"proparser.qrc" "5" +"qmljs" "3" +"qmljs.pri" "4" +"qtcreator" "3" +"qtcreator.pri" "4" +"qtcreatorplugin" "3" +"qtcreatorplugin.pri" "4" +"Other files" "4" +"plugins/qtsupport" "5" +"QtSupport.pluginspec.in" "6" +"qtsupport_dependencies" "3" +"qtsupport_dependencies.pri" "4" +"texteditor" "3" +"texteditor.pri" "4" +"texteditor_dependencies" "3" +"texteditor_dependencies.pri" "4" +"utils" "3" +"utils.pri" "4" +"utils_dependencies" "3" +"utils_dependencies.pri" "4" +"Headers" "3" +"baseqtversion.h" "4" +"debugginghelperbuildtask.h" "4" +"exampleslistmodel.h" "4" +"gettingstartedwelcomepage.h" "4" +"profilereader.h" "4" +"qmldebugginglibrary.h" "4" +"qmldumptool.h" "4" +"qmlobservertool.h" "4" +"qtoptionspage.h" "4" +"qtoutputformatter.h" "4" +"qtparser.h" "4" +"qtsupport_global.h" "4" +"qtsupportconstants.h" "4" +"qtsupportplugin.h" "4" +"qtversionfactory.h" "4" +"qtversionmanager.h" "4" +"screenshotcropper.h" "4" +"Sources" "3" +"baseqtversion.cpp" "4" +"debugginghelperbuildtask.cpp" "4" +"exampleslistmodel.cpp" "4" +"gettingstartedwelcomepage.cpp" "4" +"profilereader.cpp" "4" +"qmldebugginglibrary.cpp" "4" +"qmldumptool.cpp" "4" +"qmlobservertool.cpp" "4" +"qtoptionspage.cpp" "4" +"qtoutputformatter.cpp" "4" +"qtparser.cpp" "4" +"qtsupportplugin.cpp" "4" +"qtversionfactory.cpp" "4" +"qtversionmanager.cpp" "4" +"screenshotcropper.cpp" "4" +"Forms" "3" +"debugginghelper.ui" "4" +"qtversioninfo.ui" "4" +"qtversionmanager.ui" "4" +"showbuildlog.ui" "4" +"remotelinux" "2" +"remotelinux.pro" "3" +"aggregation" "3" +"aggregation.pri" "4" +"botan" "3" +"botan.pri" "4" +"coreplugin" "3" +"coreplugin.pri" "4" +"coreplugin_dependencies" "3" +"coreplugin_dependencies.pri" "4" +"cplusplus" "3" +"cplusplus.pri" "4" +"cpptools" "3" +"cpptools.pri" "4" +"cpptools_dependencies" "3" +"cpptools_dependencies.pri" "4" +"debugger" "3" +"debugger.pri" "4" +"debugger_dependencies" "3" +"debugger_dependencies.pri" "4" +"extensionsystem" "3" +"extensionsystem.pri" "4" +"extensionsystem_dependencies" "3" +"extensionsystem_dependencies.pri" "4" +"find" "3" +"find.pri" "4" +"find_dependencies" "3" +"find_dependencies.pri" "4" +"languageutils" "3" +"languageutils.pri" "4" +"locator" "3" +"locator.pri" "4" +"locator_dependencies" "3" +"locator_dependencies.pri" "4" +"projectexplorer" "3" +"projectexplorer.pri" "4" +"projectexplorer_dependencies" "3" +"projectexplorer_dependencies.pri" "4" +"qmljs" "3" +"qmljs.pri" "4" +"qt4projectmanager" "3" +"qt4projectmanager.pri" "4" +"qt4projectmanager_dependencies" "3" +"qt4projectmanager_dependencies.pri" "4" +"qtcreator" "3" +"qtcreator.pri" "4" +"qtcreatorplugin" "3" +"qtcreatorplugin.pri" "4" +"Other files" "4" +"plugins/remotelinux" "5" +"RemoteLinux.pluginspec.in" "6" +"qtsupport" "3" +"qtsupport.pri" "4" +"qtsupport_dependencies" "3" +"qtsupport_dependencies.pri" "4" +"remotelinux_dependencies" "3" +"remotelinux_dependencies.pri" "4" +"symbianutils" "3" +"symbianutils.pri" "4" +"texteditor" "3" +"texteditor.pri" "4" +"texteditor_dependencies" "3" +"texteditor_dependencies.pri" "4" +"utils" "3" +"utils.pri" "4" +"utils_dependencies" "3" +"utils_dependencies.pri" "4" +"Headers" "3" +"abstractembeddedlinuxtarget.h" "4" +"abstractpackagingstep.h" "4" +"abstractremotelinuxdeployservice.h" "4" +"abstractremotelinuxdeploystep.h" "4" +"abstractuploadandinstallpackageservice.h" "4" +"deployablefile.h" "4" +"deployablefilesperprofile.h" "4" +"deploymentinfo.h" "4" +"deploymentsettingsassistant.h" "4" +"embeddedlinuxqtversion.h" "4" +"embeddedlinuxqtversionfactory.h" "4" +"embeddedlinuxtargetfactory.h" "4" +"genericdirectuploadservice.h" "4" +"genericdirectuploadstep.h" "4" +"genericembeddedlinuxtarget.h" "4" +"genericlinuxdeviceconfigurationfactory.h" "4" +"genericlinuxdeviceconfigurationwidget.h" "4" +"genericlinuxdeviceconfigurationwizard.h" "4" +"genericlinuxdeviceconfigurationwizardpages.h" "4" +"genericremotelinuxdeploystepfactory.h" "4" +"linuxdeviceconfiguration.h" "4" +"linuxdevicetestdialog.h" "4" +"linuxdevicetester.h" "4" +"packageuploader.h" "4" +"profilesupdatedialog.h" "4" +"publickeydeploymentdialog.h" "4" +"remotelinux_constants.h" "4" +"remotelinux_export.h" "4" +"remotelinuxapplicationrunner.h" "4" +"remotelinuxcustomcommanddeploymentstep.h" "4" +"remotelinuxcustomcommanddeployservice.h" "4" +"remotelinuxdebugsupport.h" "4" +"remotelinuxdeployconfiguration.h" "4" +"remotelinuxdeployconfigurationfactory.h" "4" +"remotelinuxdeployconfigurationwidget.h" "4" +"remotelinuxenvironmentreader.h" "4" +"remotelinuxpackageinstaller.h" "4" +"remotelinuxplugin.h" "4" +"remotelinuxprocessesdialog.h" "4" +"remotelinuxprocesslist.h" "4" +"remotelinuxrunconfiguration.h" "4" +"remotelinuxrunconfigurationfactory.h" "4" +"remotelinuxrunconfigurationwidget.h" "4" +"remotelinuxruncontrol.h" "4" +"remotelinuxruncontrolfactory.h" "4" +"remotelinuxusedportsgatherer.h" "4" +"remotelinuxutils.h" "4" +"sshkeydeployer.h" "4" +"startgdbserverdialog.h" "4" +"tarpackagecreationstep.h" "4" +"typespecificdeviceconfigurationlistmodel.h" "4" +"uploadandinstalltarpackagestep.h" "4" +"Sources" "3" +"abstractembeddedlinuxtarget.cpp" "4" +"abstractpackagingstep.cpp" "4" +"abstractremotelinuxdeployservice.cpp" "4" +"abstractremotelinuxdeploystep.cpp" "4" +"abstractuploadandinstallpackageservice.cpp" "4" +"deployablefilesperprofile.cpp" "4" +"deploymentinfo.cpp" "4" +"deploymentsettingsassistant.cpp" "4" +"embeddedlinuxqtversion.cpp" "4" +"embeddedlinuxqtversionfactory.cpp" "4" +"embeddedlinuxtargetfactory.cpp" "4" +"genericdirectuploadservice.cpp" "4" +"genericdirectuploadstep.cpp" "4" +"genericembeddedlinuxtarget.cpp" "4" +"genericlinuxdeviceconfigurationfactory.cpp" "4" +"genericlinuxdeviceconfigurationwidget.cpp" "4" +"genericlinuxdeviceconfigurationwizard.cpp" "4" +"genericlinuxdeviceconfigurationwizardpages.cpp" "4" +"genericremotelinuxdeploystepfactory.cpp" "4" +"linuxdeviceconfiguration.cpp" "4" +"linuxdevicetestdialog.cpp" "4" +"linuxdevicetester.cpp" "4" +"packageuploader.cpp" "4" +"profilesupdatedialog.cpp" "4" +"publickeydeploymentdialog.cpp" "4" +"remotelinuxapplicationrunner.cpp" "4" +"remotelinuxcustomcommanddeploymentstep.cpp" "4" +"remotelinuxcustomcommanddeployservice.cpp" "4" +"remotelinuxdebugsupport.cpp" "4" +"remotelinuxdeployconfiguration.cpp" "4" +"remotelinuxdeployconfigurationfactory.cpp" "4" +"remotelinuxdeployconfigurationwidget.cpp" "4" +"remotelinuxenvironmentreader.cpp" "4" +"remotelinuxpackageinstaller.cpp" "4" +"remotelinuxplugin.cpp" "4" +"remotelinuxprocessesdialog.cpp" "4" +"remotelinuxprocesslist.cpp" "4" +"remotelinuxrunconfiguration.cpp" "4" +"remotelinuxrunconfigurationfactory.cpp" "4" +"remotelinuxrunconfigurationwidget.cpp" "4" +"remotelinuxruncontrol.cpp" "4" +"remotelinuxruncontrolfactory.cpp" "4" +"remotelinuxusedportsgatherer.cpp" "4" +"remotelinuxutils.cpp" "4" +"sshkeydeployer.cpp" "4" +"startgdbserverdialog.cpp" "4" +"tarpackagecreationstep.cpp" "4" +"typespecificdeviceconfigurationlistmodel.cpp" "4" +"uploadandinstalltarpackagestep.cpp" "4" +"Forms" "3" +"genericlinuxdeviceconfigurationwidget.ui" "4" +"genericlinuxdeviceconfigurationwizardsetuppage.ui" "4" +"linuxdevicetestdialog.ui" "4" +"profilesupdatedialog.ui" "4" +"remotelinuxdeployconfigurationwidget.ui" "4" +"remotelinuxprocessesdialog.ui" "4" +"Resources" "3" +"remotelinux.qrc" "4" +"resourceeditor" "2" +"resourceeditor.pro" "3" +"aggregation" "3" +"aggregation.pri" "4" +"coreplugin" "3" +"coreplugin.pri" "4" +"coreplugin_dependencies" "3" +"coreplugin_dependencies.pri" "4" +"extensionsystem" "3" +"extensionsystem.pri" "4" +"extensionsystem_dependencies" "3" +"extensionsystem_dependencies.pri" "4" +"qrceditor" "3" +"qrceditor.pri" "4" +"Headers" "4" +"qrceditor.h" "5" +"resourcefile_p.h" "5" +"resourceview.h" "5" +"undocommands_p.h" "5" +"Sources" "4" +"qrceditor.cpp" "5" +"resourcefile.cpp" "5" +"resourceview.cpp" "5" +"undocommands.cpp" "5" +"Forms" "4" +"qrceditor.ui" "5" +"qtcreator" "3" +"qtcreator.pri" "4" +"qtcreatorplugin" "3" +"qtcreatorplugin.pri" "4" +"Other files" "4" +"plugins/resourceeditor" "5" +"ResourceEditor.pluginspec.in" "6" +"utils" "3" +"utils.pri" "4" +"utils_dependencies" "3" +"utils_dependencies.pri" "4" +"Headers" "3" +"resourceeditorconstants.h" "4" +"resourceeditorfactory.h" "4" +"resourceeditorplugin.h" "4" +"resourceeditorw.h" "4" +"resourcewizard.h" "4" +"Sources" "3" +"resourceeditorfactory.cpp" "4" +"resourceeditorplugin.cpp" "4" +"resourceeditorw.cpp" "4" +"resourcewizard.cpp" "4" +"Resources" "3" +"resourceeditor.qrc" "4" +"subversion" "2" +"subversion.pro" "3" +"aggregation" "3" +"aggregation.pri" "4" +"coreplugin" "3" +"coreplugin.pri" "4" +"coreplugin_dependencies" "3" +"coreplugin_dependencies.pri" "4" +"cplusplus" "3" +"cplusplus.pri" "4" +"extensionsystem" "3" +"extensionsystem.pri" "4" +"extensionsystem_dependencies" "3" +"extensionsystem_dependencies.pri" "4" +"find" "3" +"find.pri" "4" +"find_dependencies" "3" +"find_dependencies.pri" "4" +"locator" "3" +"locator.pri" "4" +"locator_dependencies" "3" +"locator_dependencies.pri" "4" +"projectexplorer" "3" +"projectexplorer.pri" "4" +"projectexplorer_dependencies" "3" +"projectexplorer_dependencies.pri" "4" +"qtcreator" "3" +"qtcreator.pri" "4" +"qtcreatorplugin" "3" +"qtcreatorplugin.pri" "4" +"Other files" "4" +"plugins/subversion" "5" +"Subversion.pluginspec.in" "6" +"texteditor" "3" +"texteditor.pri" "4" +"texteditor_dependencies" "3" +"texteditor_dependencies.pri" "4" +"utils" "3" +"utils.pri" "4" +"utils_dependencies" "3" +"utils_dependencies.pri" "4" +"vcsbase" "3" +"vcsbase.pri" "4" +"vcsbase_dependencies" "3" +"vcsbase_dependencies.pri" "4" +"Headers" "3" +"annotationhighlighter.h" "4" +"checkoutwizard.h" "4" +"checkoutwizardpage.h" "4" +"settingspage.h" "4" +"subversionconstants.h" "4" +"subversioncontrol.h" "4" +"subversioneditor.h" "4" +"subversionplugin.h" "4" +"subversionsettings.h" "4" +"subversionsubmiteditor.h" "4" +"Sources" "3" +"annotationhighlighter.cpp" "4" +"checkoutwizard.cpp" "4" +"checkoutwizardpage.cpp" "4" +"settingspage.cpp" "4" +"subversioncontrol.cpp" "4" +"subversioneditor.cpp" "4" +"subversionplugin.cpp" "4" +"subversionsettings.cpp" "4" +"subversionsubmiteditor.cpp" "4" +"Forms" "3" +"settingspage.ui" "4" +"Resources" "3" +"subversion.qrc" "4" +"tasklist" "2" +"tasklist.pro" "3" +"aggregation" "3" +"aggregation.pri" "4" +"coreplugin" "3" +"coreplugin.pri" "4" +"coreplugin_dependencies" "3" +"coreplugin_dependencies.pri" "4" +"extensionsystem" "3" +"extensionsystem.pri" "4" +"extensionsystem_dependencies" "3" +"extensionsystem_dependencies.pri" "4" +"find" "3" +"find.pri" "4" +"find_dependencies" "3" +"find_dependencies.pri" "4" +"locator" "3" +"locator.pri" "4" +"locator_dependencies" "3" +"locator_dependencies.pri" "4" +"projectexplorer" "3" +"projectexplorer.pri" "4" +"projectexplorer_dependencies" "3" +"projectexplorer_dependencies.pri" "4" +"qtcreator" "3" +"qtcreator.pri" "4" +"qtcreatorplugin" "3" +"qtcreatorplugin.pri" "4" +"Other files" "4" +"plugins/tasklist" "5" +"TaskList.pluginspec.in" "6" +"texteditor" "3" +"texteditor.pri" "4" +"texteditor_dependencies" "3" +"texteditor_dependencies.pri" "4" +"utils" "3" +"utils.pri" "4" +"utils_dependencies" "3" +"utils_dependencies.pri" "4" +"Headers" "3" +"stopmonitoringhandler.h" "4" +"taskfile.h" "4" +"taskfilefactory.h" "4" +"tasklist_export.h" "4" +"tasklistconstants.h" "4" +"tasklistplugin.h" "4" +"Sources" "3" +"stopmonitoringhandler.cpp" "4" +"taskfile.cpp" "4" +"taskfilefactory.cpp" "4" +"tasklistplugin.cpp" "4" +"Resources" "3" +"tasklist.qrc" "4" +"Other files" "3" +"TaskList.mimetypes.xml" "4" +"texteditor" "2" +"texteditor.pro" "3" +"aggregation" "3" +"aggregation.pri" "4" +"coreplugin" "3" +"coreplugin.pri" "4" +"coreplugin_dependencies" "3" +"coreplugin_dependencies.pri" "4" +"extensionsystem" "3" +"extensionsystem.pri" "4" +"extensionsystem_dependencies" "3" +"extensionsystem_dependencies.pri" "4" +"find" "3" +"find.pri" "4" +"find_dependencies" "3" +"find_dependencies.pri" "4" +"locator" "3" +"locator.pri" "4" +"locator_dependencies" "3" +"locator_dependencies.pri" "4" +"qtcreator" "3" +"qtcreator.pri" "4" +"qtcreatorplugin" "3" +"qtcreatorplugin.pri" "4" +"Other files" "4" +"plugins/texteditor" "5" +"TextEditor.pluginspec.in" "6" +"texteditor_dependencies" "3" +"texteditor_dependencies.pri" "4" +"utils" "3" +"utils.pri" "4" +"utils_dependencies" "3" +"utils_dependencies.pri" "4" +"Headers" "3" +"codeassist" "4" +"assistenums.h" "5" +"basicproposalitem.h" "5" +"basicproposalitemlistmodel.h" "5" +"codeassistant.h" "5" +"completionassistprovider.h" "5" +"defaultassistinterface.h" "5" +"functionhintproposal.h" "5" +"functionhintproposalwidget.h" "5" +"genericproposal.h" "5" +"genericproposalwidget.h" "5" +"iassistinterface.h" "5" +"iassistprocessor.h" "5" +"iassistproposal.h" "5" +"iassistproposalitem.h" "5" +"iassistproposalmodel.h" "5" +"iassistproposalwidget.h" "5" +"iassistprovider.h" "5" +"ifunctionhintproposalmodel.h" "5" +"igenericproposalmodel.h" "5" +"quickfixassistprocessor.h" "5" +"quickfixassistprovider.h" "5" +"runner.h" "5" +"generichighlighter" "4" +"context.h" "5" +"definitiondownloader.h" "5" +"dynamicrule.h" "5" +"highlightdefinition.h" "5" +"highlightdefinitionhandler.h" "5" +"highlightdefinitionmetadata.h" "5" +"highlighter.h" "5" +"highlighterexception.h" "5" +"highlightersettings.h" "5" +"highlightersettingspage.h" "5" +"includerulesinstruction.h" "5" +"itemdata.h" "5" +"keywordlist.h" "5" +"managedefinitionsdialog.h" "5" +"manager.h" "5" +"progressdata.h" "5" +"reuse.h" "5" +"rule.h" "5" +"specificrules.h" "5" +"snippets" "4" +"isnippetprovider.h" "5" +"plaintextsnippetprovider.h" "5" +"reuse.h" "5" +"snippet.h" "5" +"snippetassistcollector.h" "5" +"snippeteditor.h" "5" +"snippetscollection.h" "5" +"snippetssettings.h" "5" +"snippetssettingspage.h" "5" +"tooltip" "4" +"effects.h" "5" +"reuse.h" "5" +"tipcontents.h" "5" +"tipfactory.h" "5" +"tips.h" "5" +"tooltip.h" "5" +"autocompleter.h" "4" +"basefilefind.h" "4" +"basefilefind_p.h" "4" +"basehoverhandler.h" "4" +"basetextdocument.h" "4" +"basetextdocumentlayout.h" "4" +"basetexteditor.h" "4" +"basetexteditor_p.h" "4" +"basetextmark.h" "4" +"behaviorsettings.h" "4" +"behaviorsettingspage.h" "4" +"behaviorsettingswidget.h" "4" +"circularclipboard.h" "4" +"codecselector.h" "4" +"codestyleeditor.h" "4" +"codestylepool.h" "4" +"codestyleselectorwidget.h" "4" +"colorscheme.h" "4" +"colorschemeedit.h" "4" +"completionsettings.h" "4" +"convenience.h" "4" +"displaysettings.h" "4" +"displaysettingspage.h" "4" +"extraencodingsettings.h" "4" +"findincurrentfile.h" "4" +"findinfiles.h" "4" +"findinopenfiles.h" "4" +"fontsettings.h" "4" +"fontsettingspage.h" "4" +"helpitem.h" "4" +"icodestylepreferences.h" "4" +"icodestylepreferencesfactory.h" "4" +"indenter.h" "4" +"ioutlinewidget.h" "4" +"itexteditor.h" "4" +"itextmark.h" "4" +"linenumberfilter.h" "4" +"normalindenter.h" "4" +"outlinefactory.h" "4" +"plaintexteditor.h" "4" +"plaintexteditorfactory.h" "4" +"quickfix.h" "4" +"refactoringchanges.h" "4" +"refactoroverlay.h" "4" +"semantichighlighter.h" "4" +"simplecodestylepreferences.h" "4" +"simplecodestylepreferenceswidget.h" "4" +"storagesettings.h" "4" +"syntaxhighlighter.h" "4" +"tabsettings.h" "4" +"tabsettingswidget.h" "4" +"texteditor_global.h" "4" +"texteditoractionhandler.h" "4" +"texteditorconstants.h" "4" +"texteditoroptionspage.h" "4" +"texteditoroverlay.h" "4" +"texteditorplugin.h" "4" +"texteditorsettings.h" "4" +"textfilewizard.h" "4" +"typingsettings.h" "4" +"Sources" "3" +"codeassist" "4" +"basicproposalitem.cpp" "5" +"basicproposalitemlistmodel.cpp" "5" +"codeassistant.cpp" "5" +"completionassistprovider.cpp" "5" +"defaultassistinterface.cpp" "5" +"functionhintproposal.cpp" "5" +"functionhintproposalwidget.cpp" "5" +"genericproposal.cpp" "5" +"genericproposalwidget.cpp" "5" +"iassistinterface.cpp" "5" +"iassistprocessor.cpp" "5" +"iassistproposal.cpp" "5" +"iassistproposalitem.cpp" "5" +"iassistproposalmodel.cpp" "5" +"iassistproposalwidget.cpp" "5" +"iassistprovider.cpp" "5" +"ifunctionhintproposalmodel.cpp" "5" +"igenericproposalmodel.cpp" "5" +"quickfixassistprocessor.cpp" "5" +"quickfixassistprovider.cpp" "5" +"runner.cpp" "5" +"generichighlighter" "4" +"context.cpp" "5" +"definitiondownloader.cpp" "5" +"dynamicrule.cpp" "5" +"highlightdefinition.cpp" "5" +"highlightdefinitionhandler.cpp" "5" +"highlightdefinitionmetadata.cpp" "5" +"highlighter.cpp" "5" +"highlightersettings.cpp" "5" +"highlightersettingspage.cpp" "5" +"includerulesinstruction.cpp" "5" +"itemdata.cpp" "5" +"keywordlist.cpp" "5" +"managedefinitionsdialog.cpp" "5" +"manager.cpp" "5" +"progressdata.cpp" "5" +"rule.cpp" "5" +"specificrules.cpp" "5" +"snippets" "4" +"isnippetprovider.cpp" "5" +"plaintextsnippetprovider.cpp" "5" +"snippet.cpp" "5" +"snippetassistcollector.cpp" "5" +"snippeteditor.cpp" "5" +"snippetscollection.cpp" "5" +"snippetssettings.cpp" "5" +"snippetssettingspage.cpp" "5" +"tooltip" "4" +"tipcontents.cpp" "5" +"tipfactory.cpp" "5" +"tips.cpp" "5" +"tooltip.cpp" "5" +"autocompleter.cpp" "4" +"basefilefind.cpp" "4" +"basehoverhandler.cpp" "4" +"basetextdocument.cpp" "4" +"basetextdocumentlayout.cpp" "4" +"basetexteditor.cpp" "4" +"basetextmark.cpp" "4" +"behaviorsettings.cpp" "4" +"behaviorsettingspage.cpp" "4" +"behaviorsettingswidget.cpp" "4" +"circularclipboard.cpp" "4" +"codecselector.cpp" "4" +"codestyleeditor.cpp" "4" +"codestylepool.cpp" "4" +"codestyleselectorwidget.cpp" "4" +"colorscheme.cpp" "4" +"colorschemeedit.cpp" "4" +"completionsettings.cpp" "4" +"convenience.cpp" "4" +"displaysettings.cpp" "4" +"displaysettingspage.cpp" "4" +"extraencodingsettings.cpp" "4" +"findincurrentfile.cpp" "4" +"findinfiles.cpp" "4" +"findinopenfiles.cpp" "4" +"fontsettings.cpp" "4" +"fontsettingspage.cpp" "4" +"helpitem.cpp" "4" +"icodestylepreferences.cpp" "4" +"icodestylepreferencesfactory.cpp" "4" +"indenter.cpp" "4" +"itexteditor.cpp" "4" +"itextmark.cpp" "4" +"linenumberfilter.cpp" "4" +"normalindenter.cpp" "4" +"outlinefactory.cpp" "4" +"plaintexteditor.cpp" "4" +"plaintexteditorfactory.cpp" "4" +"quickfix.cpp" "4" +"refactoringchanges.cpp" "4" +"refactoroverlay.cpp" "4" +"semantichighlighter.cpp" "4" +"simplecodestylepreferences.cpp" "4" +"simplecodestylepreferenceswidget.cpp" "4" +"storagesettings.cpp" "4" +"syntaxhighlighter.cpp" "4" +"tabsettings.cpp" "4" +"tabsettingswidget.cpp" "4" +"texteditoractionhandler.cpp" "4" +"texteditoroptionspage.cpp" "4" +"texteditoroverlay.cpp" "4" +"texteditorplugin.cpp" "4" +"texteditorsettings.cpp" "4" +"textfilewizard.cpp" "4" +"typingsettings.cpp" "4" +"Forms" "3" +"generichighlighter" "4" +"highlightersettingspage.ui" "5" +"managedefinitionsdialog.ui" "5" +"snippets" "4" +"snippetssettingspage.ui" "5" +"behaviorsettingspage.ui" "4" +"behaviorsettingswidget.ui" "4" +"codestyleselectorwidget.ui" "4" +"colorschemeedit.ui" "4" +"displaysettingspage.ui" "4" +"fontsettingspage.ui" "4" +"tabsettingswidget.ui" "4" +"Resources" "3" +"texteditor.qrc" "4" +"Other files" "3" +"TextEditor.mimetypes.xml" "4" +"todo" "2" +"todo.pro" "3" +"aggregation" "3" +"aggregation.pri" "4" +"coreplugin" "3" +"coreplugin.pri" "4" +"coreplugin_dependencies" "3" +"coreplugin_dependencies.pri" "4" +"cplusplus" "3" +"cplusplus.pri" "4" +"cpptools" "3" +"cpptools.pri" "4" +"cpptools_dependencies" "3" +"cpptools_dependencies.pri" "4" +"extensionsystem" "3" +"extensionsystem.pri" "4" +"extensionsystem_dependencies" "3" +"extensionsystem_dependencies.pri" "4" +"find" "3" +"find.pri" "4" +"find_dependencies" "3" +"find_dependencies.pri" "4" +"languageutils" "3" +"languageutils.pri" "4" +"locator" "3" +"locator.pri" "4" +"locator_dependencies" "3" +"locator_dependencies.pri" "4" +"projectexplorer" "3" +"projectexplorer.pri" "4" +"projectexplorer_dependencies" "3" +"projectexplorer_dependencies.pri" "4" +"qmljs" "3" +"qmljs.pri" "4" +"qtcreator" "3" +"qtcreator.pri" "4" +"qtcreatorplugin" "3" +"qtcreatorplugin.pri" "4" +"Other files" "4" +"plugins/todo" "5" +"Todo.pluginspec.in" "6" +"texteditor" "3" +"texteditor.pri" "4" +"texteditor_dependencies" "3" +"texteditor_dependencies.pri" "4" +"todo_dependencies" "3" +"todo_dependencies.pri" "4" +"utils" "3" +"utils.pri" "4" +"utils_dependencies" "3" +"utils_dependencies.pri" "4" +"Headers" "3" +"constants.h" "4" +"cpptodoitemsscanner.h" "4" +"keyword.h" "4" +"keyworddialog.h" "4" +"lineparser.h" "4" +"optionsdialog.h" "4" +"optionspage.h" "4" +"qmljstodoitemsscanner.h" "4" +"settings.h" "4" +"todoitem.h" "4" +"todoitemsmodel.h" "4" +"todoitemsprovider.h" "4" +"todoitemsscanner.h" "4" +"todooutputpane.h" "4" +"todoplugin.h" "4" +"Sources" "3" +"cpptodoitemsscanner.cpp" "4" +"keyword.cpp" "4" +"keyworddialog.cpp" "4" +"lineparser.cpp" "4" +"optionsdialog.cpp" "4" +"optionspage.cpp" "4" +"qmljstodoitemsscanner.cpp" "4" +"settings.cpp" "4" +"todoitemsmodel.cpp" "4" +"todoitemsprovider.cpp" "4" +"todoitemsscanner.cpp" "4" +"todooutputpane.cpp" "4" +"todoplugin.cpp" "4" +"Forms" "3" +"keyworddialog.ui" "4" +"optionsdialog.ui" "4" +"Resources" "3" +"todoplugin.qrc" "4" +"Other files" "3" +"Todo.pluginspec.in" "4" +"updateinfo" "2" +"updateinfo.pro" "3" +"aggregation" "3" +"aggregation.pri" "4" +"coreplugin" "3" +"coreplugin.pri" "4" +"coreplugin_dependencies" "3" +"coreplugin_dependencies.pri" "4" +"extensionsystem" "3" +"extensionsystem.pri" "4" +"extensionsystem_dependencies" "3" +"extensionsystem_dependencies.pri" "4" +"qtcreator" "3" +"qtcreator.pri" "4" +"qtcreatorplugin" "3" +"qtcreatorplugin.pri" "4" +"Other files" "4" +"plugins/updateinfo" "5" +"UpdateInfo.pluginspec.in" "6" +"updateinfo_dependencies" "3" +"updateinfo_dependencies.pri" "4" +"utils" "3" +"utils.pri" "4" +"utils_dependencies" "3" +"utils_dependencies.pri" "4" +"Headers" "3" +"updateinfobutton.h" "4" +"updateinfoplugin.h" "4" +"Sources" "3" +"updateinfobutton.cpp" "4" +"updateinfoplugin.cpp" "4" +"Resources" "3" +"updateinfo.qrc" "4" +"valgrind" "2" +"valgrind.pro" "3" +"aggregation" "3" +"aggregation.pri" "4" +"analyzerbase" "3" +"analyzerbase.pri" "4" +"analyzerbase_dependencies" "3" +"analyzerbase_dependencies.pri" "4" +"botan" "3" +"botan.pri" "4" +"callgrind" "3" +"callgrind.pri" "4" +"Headers" "4" +"callgrindabstractmodel.h" "5" +"callgrindcallmodel.h" "5" +"callgrindcontroller.h" "5" +"callgrindcostitem.h" "5" +"callgrindcycledetection.h" "5" +"callgrinddatamodel.h" "5" +"callgrindfunction.h" "5" +"callgrindfunction_p.h" "5" +"callgrindfunctioncall.h" "5" +"callgrindfunctioncycle.h" "5" +"callgrindparsedata.h" "5" +"callgrindparser.h" "5" +"callgrindproxymodel.h" "5" +"callgrindrunner.h" "5" +"callgrindstackbrowser.h" "5" +"Sources" "4" +"callgrindcallmodel.cpp" "5" +"callgrindcontroller.cpp" "5" +"callgrindcostitem.cpp" "5" +"callgrindcycledetection.cpp" "5" +"callgrinddatamodel.cpp" "5" +"callgrindfunction.cpp" "5" +"callgrindfunctioncall.cpp" "5" +"callgrindfunctioncycle.cpp" "5" +"callgrindparsedata.cpp" "5" +"callgrindparser.cpp" "5" +"callgrindproxymodel.cpp" "5" +"callgrindrunner.cpp" "5" +"callgrindstackbrowser.cpp" "5" +"coreplugin" "3" +"coreplugin.pri" "4" +"coreplugin_dependencies" "3" +"coreplugin_dependencies.pri" "4" +"cplusplus" "3" +"cplusplus.pri" "4" +"cpptools" "3" +"cpptools.pri" "4" +"cpptools_dependencies" "3" +"cpptools_dependencies.pri" "4" +"debugger" "3" +"debugger.pri" "4" +"debugger_dependencies" "3" +"debugger_dependencies.pri" "4" +"extensionsystem" "3" +"extensionsystem.pri" "4" +"extensionsystem_dependencies" "3" +"extensionsystem_dependencies.pri" "4" +"find" "3" +"find.pri" "4" +"find_dependencies" "3" +"find_dependencies.pri" "4" +"languageutils" "3" +"languageutils.pri" "4" +"locator" "3" +"locator.pri" "4" +"locator_dependencies" "3" +"locator_dependencies.pri" "4" +"memcheck" "3" +"memcheck.pri" "4" +"Headers" "4" +"memcheckrunner.h" "5" +"Sources" "4" +"memcheckrunner.cpp" "5" +"projectexplorer" "3" +"projectexplorer.pri" "4" +"projectexplorer_dependencies" "3" +"projectexplorer_dependencies.pri" "4" +"qmljs" "3" +"qmljs.pri" "4" +"qt4projectmanager" "3" +"qt4projectmanager.pri" "4" +"qt4projectmanager_dependencies" "3" +"qt4projectmanager_dependencies.pri" "4" +"qtcreator" "3" +"qtcreator.pri" "4" +"qtcreatorplugin" "3" +"qtcreatorplugin.pri" "4" +"Other files" "4" +"plugins/valgrind" "5" +"Valgrind.pluginspec.in" "6" +"qtsupport" "3" +"qtsupport.pri" "4" +"qtsupport_dependencies" "3" +"qtsupport_dependencies.pri" "4" +"remotelinux" "3" +"remotelinux.pri" "4" +"remotelinux_dependencies" "3" +"remotelinux_dependencies.pri" "4" +"symbianutils" "3" +"symbianutils.pri" "4" +"texteditor" "3" +"texteditor.pri" "4" +"texteditor_dependencies" "3" +"texteditor_dependencies.pri" "4" +"utils" "3" +"utils.pri" "4" +"utils_dependencies" "3" +"utils_dependencies.pri" "4" +"valgrind_dependencies" "3" +"valgrind_dependencies.pri" "4" +"xmlprotocol" "3" +"xmlprotocol.pri" "4" +"Headers" "4" +"announcethread.h" "5" +"error.h" "5" +"errorlistmodel.h" "5" +"frame.h" "5" +"modelhelpers.h" "5" +"parser.h" "5" +"stack.h" "5" +"stackmodel.h" "5" +"status.h" "5" +"suppression.h" "5" +"threadedparser.h" "5" +"Sources" "4" +"announcethread.cpp" "5" +"error.cpp" "5" +"errorlistmodel.cpp" "5" +"frame.cpp" "5" +"modelhelpers.cpp" "5" +"parser.cpp" "5" +"stack.cpp" "5" +"stackmodel.cpp" "5" +"status.cpp" "5" +"suppression.cpp" "5" +"threadedparser.cpp" "5" +"Headers" "3" +"callgrindcostdelegate.h" "4" +"callgrindcostview.h" "4" +"callgrindengine.h" "4" +"callgrindhelper.h" "4" +"callgrindnamedelegate.h" "4" +"callgrindtextmark.h" "4" +"callgrindtool.h" "4" +"callgrindvisualisation.h" "4" +"memcheckengine.h" "4" +"memcheckerrorview.h" "4" +"memchecktool.h" "4" +"suppressiondialog.h" "4" +"valgrindconfigwidget.h" "4" +"valgrindengine.h" "4" +"valgrindplugin.h" "4" +"valgrindprocess.h" "4" +"valgrindrunner.h" "4" +"valgrindsettings.h" "4" +"valgrindtool.h" "4" +"workarounds.h" "4" +"Sources" "3" +"callgrindcostdelegate.cpp" "4" +"callgrindcostview.cpp" "4" +"callgrindengine.cpp" "4" +"callgrindhelper.cpp" "4" +"callgrindnamedelegate.cpp" "4" +"callgrindtextmark.cpp" "4" +"callgrindtool.cpp" "4" +"callgrindvisualisation.cpp" "4" +"memcheckengine.cpp" "4" +"memcheckerrorview.cpp" "4" +"memchecktool.cpp" "4" +"suppressiondialog.cpp" "4" +"valgrindconfigwidget.cpp" "4" +"valgrindengine.cpp" "4" +"valgrindplugin.cpp" "4" +"valgrindprocess.cpp" "4" +"valgrindrunner.cpp" "4" +"valgrindsettings.cpp" "4" +"valgrindtool.cpp" "4" +"workarounds.cpp" "4" +"Forms" "3" +"valgrindconfigwidget.ui" "4" +"vcsbase" "2" +"vcsbase.pro" "3" +"aggregation" "3" +"aggregation.pri" "4" +"coreplugin" "3" +"coreplugin.pri" "4" +"coreplugin_dependencies" "3" +"coreplugin_dependencies.pri" "4" +"cplusplus" "3" +"cplusplus.pri" "4" +"extensionsystem" "3" +"extensionsystem.pri" "4" +"extensionsystem_dependencies" "3" +"extensionsystem_dependencies.pri" "4" +"find" "3" +"find.pri" "4" +"find_dependencies" "3" +"find_dependencies.pri" "4" +"locator" "3" +"locator.pri" "4" +"locator_dependencies" "3" +"locator_dependencies.pri" "4" +"projectexplorer" "3" +"projectexplorer.pri" "4" +"projectexplorer_dependencies" "3" +"projectexplorer_dependencies.pri" "4" +"qtcreator" "3" +"qtcreator.pri" "4" +"qtcreatorplugin" "3" +"qtcreatorplugin.pri" "4" +"Other files" "4" +"plugins/vcsbase" "5" +"VcsBase.pluginspec.in" "6" +"texteditor" "3" +"texteditor.pri" "4" +"texteditor_dependencies" "3" +"texteditor_dependencies.pri" "4" +"utils" "3" +"utils.pri" "4" +"utils_dependencies" "3" +"utils_dependencies.pri" "4" +"vcsbase_dependencies" "3" +"vcsbase_dependencies.pri" "4" +"Headers" "3" +"baseannotationhighlighter.h" "4" +"basecheckoutwizard.h" "4" +"basecheckoutwizardpage.h" "4" +"basevcseditorfactory.h" "4" +"basevcssubmiteditorfactory.h" "4" +"checkoutjobs.h" "4" +"checkoutprogresswizardpage.h" "4" +"checkoutwizarddialog.h" "4" +"cleandialog.h" "4" +"command.h" "4" +"commonsettingspage.h" "4" +"commonvcssettings.h" "4" +"corelistener.h" "4" +"diffhighlighter.h" "4" +"nicknamedialog.h" "4" +"submiteditorfile.h" "4" +"submitfilemodel.h" "4" +"vcsbase_global.h" "4" +"vcsbaseclient.h" "4" +"vcsbaseclientsettings.h" "4" +"vcsbaseconstants.h" "4" +"vcsbaseeditor.h" "4" +"vcsbaseeditorparameterwidget.h" "4" +"vcsbaseoptionspage.h" "4" +"vcsbaseoutputwindow.h" "4" +"vcsbaseplugin.h" "4" +"vcsbasesubmiteditor.h" "4" +"vcsconfigurationpage.h" "4" +"vcsplugin.h" "4" +"Sources" "3" +"baseannotationhighlighter.cpp" "4" +"basecheckoutwizard.cpp" "4" +"basecheckoutwizardpage.cpp" "4" +"basevcseditorfactory.cpp" "4" +"basevcssubmiteditorfactory.cpp" "4" +"checkoutjobs.cpp" "4" +"checkoutprogresswizardpage.cpp" "4" +"checkoutwizarddialog.cpp" "4" +"cleandialog.cpp" "4" +"command.cpp" "4" +"commonsettingspage.cpp" "4" +"commonvcssettings.cpp" "4" +"corelistener.cpp" "4" +"diffhighlighter.cpp" "4" +"nicknamedialog.cpp" "4" +"submiteditorfile.cpp" "4" +"submitfilemodel.cpp" "4" +"vcsbaseclient.cpp" "4" +"vcsbaseclientsettings.cpp" "4" +"vcsbaseeditor.cpp" "4" +"vcsbaseeditorparameterwidget.cpp" "4" +"vcsbaseoptionspage.cpp" "4" +"vcsbaseoutputwindow.cpp" "4" +"vcsbaseplugin.cpp" "4" +"vcsbasesubmiteditor.cpp" "4" +"vcsconfigurationpage.cpp" "4" +"vcsplugin.cpp" "4" +"Forms" "3" +"basecheckoutwizardpage.ui" "4" +"checkoutprogresswizardpage.ui" "4" +"cleandialog.ui" "4" +"commonsettingspage.ui" "4" +"nicknamedialog.ui" "4" +"vcsconfigurationpage.ui" "4" +"Resources" "3" +"vcsbase.qrc" "4" +"welcome" "2" +"welcome.pro" "3" +"aggregation" "3" +"aggregation.pri" "4" +"coreplugin" "3" +"coreplugin.pri" "4" +"coreplugin_dependencies" "3" +"coreplugin_dependencies.pri" "4" +"extensionsystem" "3" +"extensionsystem.pri" "4" +"extensionsystem_dependencies" "3" +"extensionsystem_dependencies.pri" "4" +"find" "3" +"find.pri" "4" +"find_dependencies" "3" +"find_dependencies.pri" "4" +"locator" "3" +"locator.pri" "4" +"locator_dependencies" "3" +"locator_dependencies.pri" "4" +"projectexplorer" "3" +"projectexplorer.pri" "4" +"projectexplorer_dependencies" "3" +"projectexplorer_dependencies.pri" "4" +"qtcreator" "3" +"qtcreator.pri" "4" +"qtcreatorplugin" "3" +"qtcreatorplugin.pri" "4" +"Other files" "4" +"plugins/welcome" "5" +"Welcome.pluginspec.in" "6" +"texteditor" "3" +"texteditor.pri" "4" +"texteditor_dependencies" "3" +"texteditor_dependencies.pri" "4" +"utils" "3" +"utils.pri" "4" +"utils_dependencies" "3" +"utils_dependencies.pri" "4" +"welcome_dependencies" "3" +"welcome_dependencies.pri" "4" +"Headers" "3" +"welcome_global.h" "4" +"welcomeplugin.h" "4" +"Sources" "3" +"welcomeplugin.cpp" "4" +"tools" "1" +"tools.pro" "2" +"mdnssd" "2" +"mdnssd.pro" "3" +"qtcreator" "3" +"qtcreator.pri" "4" +"Headers" "3" +"DebugServices.h" "4" +"dns_sd.h" "4" +"DNSCommon.h" "4" +"dnssd_ipc.h" "4" +"GenLinkedList.h" "4" +"mDNSDebug.h" "4" +"mDNSEmbeddedAPI.h" "4" +"mDNSPosix.h" "4" +"mDNSUNP.h" "4" +"PlatformCommon.h" "4" +"uDNS.h" "4" +"uds_daemon.h" "4" +"Sources" "3" +"DNSCommon.c" "4" +"DNSDigest.c" "4" +"dnssd_ipc.c" "4" +"GenLinkedList.c" "4" +"mDNS.c" "4" +"mDNSDebug.c" "4" +"mDNSPosix.c" "4" +"mDNSUNP.c" "4" +"PlatformCommon.c" "4" +"PosixDaemon.c" "4" +"uDNS.c" "4" +"uds_daemon.c" "4" +"qmlpuppet" "2" +"qmlpuppet.pro" "3" +"private_headers" "3" +"private_headers.pri" "4" +"qmlpuppet" "3" +"qmlpuppet.pro" "4" +"commands" "4" +"commands.pri" "5" +"Headers" "5" +"changeauxiliarycommand.h" "6" +"changebindingscommand.h" "6" +"changefileurlcommand.h" "6" +"changeidscommand.h" "6" +"changenodesourcecommand.h" "6" +"changestatecommand.h" "6" +"changevaluescommand.h" "6" +"childrenchangedcommand.h" "6" +"clearscenecommand.h" "6" +"completecomponentcommand.h" "6" +"componentcompletedcommand.h" "6" +"createinstancescommand.h" "6" +"createscenecommand.h" "6" +"informationchangedcommand.h" "6" +"pixmapchangedcommand.h" "6" +"removeinstancescommand.h" "6" +"removepropertiescommand.h" "6" +"reparentinstancescommand.h" "6" +"statepreviewimagechangedcommand.h" "6" +"synchronizecommand.h" "6" +"tokencommand.h" "6" +"valueschangedcommand.h" "6" +"Sources" "5" +"changeauxiliarycommand.cpp" "6" +"changebindingscommand.cpp" "6" +"changefileurlcommand.cpp" "6" +"changeidscommand.cpp" "6" +"changenodesourcecommand.cpp" "6" +"changestatecommand.cpp" "6" +"changevaluescommand.cpp" "6" +"childrenchangedcommand.cpp" "6" +"clearscenecommand.cpp" "6" +"completecomponentcommand.cpp" "6" +"componentcompletedcommand.cpp" "6" +"createinstancescommand.cpp" "6" +"createscenecommand.cpp" "6" +"informationchangedcommand.cpp" "6" +"pixmapchangedcommand.cpp" "6" +"removeinstancescommand.cpp" "6" +"removepropertiescommand.cpp" "6" +"reparentinstancescommand.cpp" "6" +"statepreviewimagechangedcommand.cpp" "6" +"synchronizecommand.cpp" "6" +"tokencommand.cpp" "6" +"valueschangedcommand.cpp" "6" +"container" "4" +"container.pri" "5" +"Headers" "5" +"addimportcontainer.h" "6" +"idcontainer.h" "6" +"imagecontainer.h" "6" +"informationcontainer.h" "6" +"instancecontainer.h" "6" +"propertyabstractcontainer.h" "6" +"propertybindingcontainer.h" "6" +"propertyvaluecontainer.h" "6" +"reparentcontainer.h" "6" +"Sources" "5" +"addimportcontainer.cpp" "6" +"idcontainer.cpp" "6" +"imagecontainer.cpp" "6" +"informationcontainer.cpp" "6" +"instancecontainer.cpp" "6" +"propertyabstractcontainer.cpp" "6" +"propertybindingcontainer.cpp" "6" +"propertyvaluecontainer.cpp" "6" +"reparentcontainer.cpp" "6" +"instances" "4" +"instances.pri" "5" +"Headers" "5" +"anchorchangesnodeinstance.h" "6" +"behaviornodeinstance.h" "6" +"childrenchangeeventfilter.h" "6" +"componentnodeinstance.h" "6" +"dummycontextobject.h" "6" +"dummynodeinstance.h" "6" +"nodeinstanceclientproxy.h" "6" +"nodeinstancemetaobject.h" "6" +"nodeinstanceserver.h" "6" +"nodeinstancesignalspy.h" "6" +"objectnodeinstance.h" "6" +"qmlpropertychangesnodeinstance.h" "6" +"qmlstatenodeinstance.h" "6" +"qmltransitionnodeinstance.h" "6" +"servernodeinstance.h" "6" +"Sources" "5" +"anchorchangesnodeinstance.cpp" "6" +"behaviornodeinstance.cpp" "6" +"childrenchangeeventfilter.cpp" "6" +"componentnodeinstance.cpp" "6" +"dummycontextobject.cpp" "6" +"dummynodeinstance.cpp" "6" +"nodeinstanceclientproxy.cpp" "6" +"nodeinstancemetaobject.cpp" "6" +"nodeinstanceserver.cpp" "6" +"nodeinstancesignalspy.cpp" "6" +"objectnodeinstance.cpp" "6" +"qmlpropertychangesnodeinstance.cpp" "6" +"qmlstatenodeinstance.cpp" "6" +"qmltransitionnodeinstance.cpp" "6" +"servernodeinstance.cpp" "6" +"instances" "4" +"instances.pri" "5" +"Headers" "5" +"graphicsobjectnodeinstance.h" "6" +"positionernodeinstance.h" "6" +"qmlgraphicsitemnodeinstance.h" "6" +"qt4informationnodeinstanceserver.h" "6" +"qt4nodeinstanceclientproxy.h" "6" +"qt4nodeinstanceserver.h" "6" +"qt4previewnodeinstanceserver.h" "6" +"qt4rendernodeinstanceserver.h" "6" +"Sources" "5" +"graphicsobjectnodeinstance.cpp" "6" +"positionernodeinstance.cpp" "6" +"qmlgraphicsitemnodeinstance.cpp" "6" +"qt4informationnodeinstanceserver.cpp" "6" +"qt4nodeinstanceclientproxy.cpp" "6" +"qt4nodeinstanceserver.cpp" "6" +"qt4previewnodeinstanceserver.cpp" "6" +"qt4rendernodeinstanceserver.cpp" "6" +"interfaces" "4" +"interfaces.pri" "5" +"Headers" "5" +"commondefines.h" "6" +"nodeinstanceclientinterface.h" "6" +"nodeinstanceserverinterface.h" "6" +"Sources" "5" +"nodeinstanceserverinterface.cpp" "6" +"private_headers" "4" +"private_headers.pri" "5" +"qmlpuppet" "4" +"qmlpuppet.pri" "5" +"Sources" "5" +"main.cpp" "6" +"Resources" "5" +"qmlpuppet.qrc" "7" +"qtcreator" "4" +"qtcreator.pri" "5" +"rpath" "4" +"rpath.pri" "5" +"qtcreator" "3" +"qtcreator.pri" "4" +"qtcdebugger" "2" +"qtcdebugger.pro" "3" +"registryaccess" "3" +"registryaccess.pri" "4" +"Headers" "4" +"registryaccess.h" "5" +"Sources" "4" +"registryaccess.cpp" "5" +"Sources" "3" +"main.cpp" "4" +"qtcrashhandler" "2" +"qtcrashhandler.pro" "3" +"private_headers" "3" +"private_headers.pri" "4" +"qtcreator" "3" +"qtcreator.pri" "4" +"rpath" "3" +"rpath.pri" "4" +"qtpromaker" "2" +"qtpromaker.pro" "3" +"qtcreator" "3" +"qtcreator.pri" "4" +"Sources" "3" +"main.cpp" "4" +"valgrindfake" "2" +"valgrindfake.pro" "3" +"Headers" "3" +"outputgenerator.h" "4" +"Sources" "3" +"main.cpp" "4" +"outputgenerator.cpp" "4" +"win64interrupt" "2" +"win64interrupt.pro" "3" +"qtcreator" "3" +"qtcreator.pri" "4" +"Sources" "3" +"win64interrupt.c" "4" +"Other files" "0" +"dist" "1" +"changes-1.1.0" "2" +"changes-1.1.1" "2" +"changes-1.2.0" "2" +"changes-1.2.1" "2" +"changes-1.3.0" "2" +"changes-1.3.1" "2" +"changes-2.0.0" "2" +"changes-2.0.1" "2" +"changes-2.1.0" "2" +"changes-2.2.0" "2" +"changes-2.3.0" "2" +"changes-2.3.1" "2" +"changes-2.4.0" "2" +"changes-2.4.1" "2" +"changes-2.5.0" "2" +"copyright_template.txt" "2" diff -Nru qtcreator-2.5.0/tests/system/suite_general/tst_openqt_creator/testdata/projecttree_speedcrunch.tsv qtcreator-2.5.2/tests/system/suite_general/tst_openqt_creator/testdata/projecttree_speedcrunch.tsv --- qtcreator-2.5.0/tests/system/suite_general/tst_openqt_creator/testdata/projecttree_speedcrunch.tsv 1970-01-01 00:00:00.000000000 +0000 +++ qtcreator-2.5.2/tests/system/suite_general/tst_openqt_creator/testdata/projecttree_speedcrunch.tsv 2012-08-08 13:47:06.000000000 +0000 @@ -0,0 +1,83 @@ +"text" "nestinglevel" +"speedcrunch.pro" "0" +"Headers" "0" +"core" "1" +"constants.h" "2" +"evaluator.h" "2" +"functions.h" "2" +"gui" "1" +"aboutbox.h" "2" +"application.h" "2" +"autohidelabel.h" "2" +"bookdock.h" "2" +"constantsdock.h" "2" +"constantswidget.h" "2" +"deletevardlg.h" "2" +"editor.h" "2" +"functionsdialog.h" "2" +"functionsdock.h" "2" +"functionswidget.h" "2" +"historydock.h" "2" +"historywidget.h" "2" +"insertvardlg.h" "2" +"keypad.h" "2" +"mainwindow.h" "2" +"resultdisplay.h" "2" +"tipwidget.h" "2" +"variablesdock.h" "2" +"variableswidget.h" "2" +"Sources" "0" +"core" "1" +"constants.cpp" "2" +"evaluator.cpp" "2" +"functions.cpp" "2" +"settings.cpp" "2" +"gui" "1" +"aboutbox.cpp" "2" +"application.cpp" "2" +"autohidelabel.cpp" "2" +"bookdock.cpp" "2" +"constantsdock.cpp" "2" +"constantswidget.cpp" "2" +"deletevardlg.cpp" "2" +"editor.cpp" "2" +"functionsdialog.cpp" "2" +"functionsdock.cpp" "2" +"functionswidget.cpp" "2" +"historydock.cpp" "2" +"historywidget.cpp" "2" +"insertvardlg.cpp" "2" +"keypad.cpp" "2" +"mainwindow.cpp" "2" +"resultdisplay.cpp" "2" +"syntaxhighlighter.cpp" "2" +"textedit.cpp" "2" +"tipwidget.cpp" "2" +"variablesdock.cpp" "2" +"variableswidget.cpp" "2" +"main" "1" +"main.cpp" "2" +"math" "1" +"floatcommon.c" "2" +"floatconst.c" "2" +"floatconvert.c" "2" +"floaterf.c" "2" +"floatexp.c" "2" +"floatgamma.c" "2" +"floathmath.c" "2" +"floatio.c" "2" +"floatipower.c" "2" +"floatlog.c" "2" +"floatlogic.c" "2" +"floatlong.c" "2" +"floatnum.c" "2" +"floatpower.c" "2" +"floatseries.c" "2" +"floattrig.c" "2" +"hmath.cpp" "2" +"number.c" "2" +"thirdparty/binreloc" "1" +"binreloc.c" "2" +"Resources" "0" +"resources" "1" +"speedcrunch.qrc" "2" diff -Nru qtcreator-2.5.0/tests/system/suite_general/tst_select_all/test.py qtcreator-2.5.2/tests/system/suite_general/tst_select_all/test.py --- qtcreator-2.5.0/tests/system/suite_general/tst_select_all/test.py 2012-05-09 12:13:19.000000000 +0000 +++ qtcreator-2.5.2/tests/system/suite_general/tst_select_all/test.py 2012-08-08 13:47:06.000000000 +0000 @@ -11,7 +11,8 @@ def main(): files = [srcPath + "/creator/README", srcPath + "/creator/qtcreator.pri", - srcPath + "/creator/doc/snippets/qml/list-of-transitions.qml"] + srcPath + "/creator/doc/snippets/qml/list-of-transitions.qml", + srcPath + "/creator/share/qtcreator/glsl/glsl_120.frag"] for currentFile in files: if not neededFilePresent(currentFile): return @@ -35,7 +36,7 @@ test.compare(editor.textCursor().selectionStart(), 0) test.compare(editor.textCursor().selectionEnd(), size) test.compare(editor.textCursor().position(), size) - test.log("Pressing key %s" % key) + test.log("Pressing key: %s" % key.replace("<", "").replace(">", "")) type(editor, key) if key == "": test.compare(editor.textCursor().selectionStart(), editor.textCursor().selectionEnd()) @@ -43,7 +44,7 @@ pos = size if key == "": pos -= 1 - if platform.system() != 'Darwin' and JIRA.isBugStillOpen(7215, JIRA.Bug.CREATOR): + if JIRA.isBugStillOpen(7215, JIRA.Bug.CREATOR): test.warning("Using workaround for %s-%d" % (JIRA.Bug.CREATOR, 7215)) pos = 0 test.compare(editor.textCursor().selectionStart(), pos) diff -Nru qtcreator-2.5.0/tests/system/suite_qtquick/tst_qml_editor/test.py qtcreator-2.5.2/tests/system/suite_qtquick/tst_qml_editor/test.py --- qtcreator-2.5.0/tests/system/suite_qtquick/tst_qml_editor/test.py 2012-05-09 12:13:19.000000000 +0000 +++ qtcreator-2.5.2/tests/system/suite_qtquick/tst_qml_editor/test.py 2012-08-08 13:47:06.000000000 +0000 @@ -16,7 +16,7 @@ workingDir = tempDir() templateDir = prepareTemplate(sourceExample) prepareForSignal("{type='CppTools::Internal::CppModelManager' unnamed='1'}", "sourceFilesRefreshed(QStringList)") - installLazySignalHandler("{type='Core::FutureProgress' unnamed='1''}", "finished()", "__handleFutureProgress__") + installLazySignalHandler("{type='Core::FutureProgress' unnamed='1'}", "finished()", "__handleFutureProgress__") createNewQtQuickApplication(workingDir, "untitled", templateDir + "/qml/focus.qml") # wait for parsing to complete waitForSignal("{type='CppTools::Internal::CppModelManager' unnamed='1'}", "sourceFilesRefreshed(QStringList)") @@ -91,8 +91,7 @@ for ty in additionalKeyPresses: type(editor, ty) searchFinished = False - ctxtMenu = openContextMenuOnTextCursorPosition(editor) - activateItem(waitForObjectItem(objectMap.realName(ctxtMenu), "Find Usages")) + invokeContextMenuItem(editor, "Find Usages") waitFor("searchFinished") validateSearchResult(expectedCount) @@ -109,32 +108,6 @@ home = "" __invokeFindUsage__(navTree, "focus\\.qml", "id: window", ["", "", home], 26) -def validateSearchResult(expectedCount): - searchResult = waitForObject(":Qt Creator_SearchResult_Core::Internal::OutputPaneToggleButton") - ensureChecked(searchResult) - resultTreeView = waitForObject("{type='Find::Internal::SearchResultTreeView' unnamed='1' " - "visible='1' window=':Qt Creator_Core::Internal::MainWindow'}") - counterLabel = waitForObject("{type='QLabel' unnamed='1' visible='1' text?='*matches found.' " - "window=':Qt Creator_Core::Internal::MainWindow'}") - matches = cast((str(counterLabel.text)).split(" ", 1)[0], "int") - test.verify(matches==expectedCount, "Verfified match count.") - model = resultTreeView.model() - for row in range(model.rowCount()): - index = model.index(row, 0) - itemText = str(model.data(index).toString()) - doubleClickItem(resultTreeView, maskSpecialCharsForSearchResult(itemText), 5, 5, 0, Qt.LeftButton) - test.log("%d occurrences in %s" % (model.rowCount(index), itemText)) - for chRow in range(model.rowCount(index)): - chIndex = model.index(chRow, 0, index) - resultTreeView.scrollTo(chIndex) - text = str(chIndex.data()) - rect = resultTreeView.visualRect(chIndex) - doubleClick(resultTreeView, rect.x+5, rect.y+5, 0, Qt.LeftButton) - editor = waitForObject("{type='QmlJSEditor::QmlJSTextEditorWidget' unnamed='1' visible='1' " - "window=':Qt Creator_Core::Internal::MainWindow'}", 20000) - waitFor("lineUnderCursor(editor) == text", 2000) - test.compare(lineUnderCursor(editor), text) - def testHovering(): navTree = waitForObject("{type='Utils::NavigationTreeView' unnamed='1' visible='1' " "window=':Qt Creator_Core::Internal::MainWindow'}", 20000) @@ -220,10 +193,6 @@ filename = filename.replace("/?","\\?").replace("/*","\\*") return filename -def maskSpecialCharsForSearchResult(filename): - filename = filename.replace("_", "\\_").replace(".","\\.") - return filename - def cleanup(): global workingDir, templateDir waitForCleanShutdown() diff -Nru qtcreator-2.5.0/tests/system/suite_qtquick/tst_qml_indent/test.py qtcreator-2.5.2/tests/system/suite_qtquick/tst_qml_indent/test.py --- qtcreator-2.5.0/tests/system/suite_qtquick/tst_qml_indent/test.py 2012-05-09 12:13:19.000000000 +0000 +++ qtcreator-2.5.2/tests/system/suite_qtquick/tst_qml_indent/test.py 2012-08-08 13:47:06.000000000 +0000 @@ -53,7 +53,8 @@ def testReIndent(): global originalText,textHasChanged - installLazySignalHandler("QmlJSEditor::QmlJSTextEditorWidget", "textChanged()", "handleTextChanged") + installLazySignalHandler("{type='QmlJSEditor::QmlJSTextEditorWidget' unnamed='1' visible='1'}", + "textChanged()", "handleTextChanged") textHasChanged = False editor = waitForObject("{type='QmlJSEditor::QmlJSTextEditorWidget' unnamed='1' visible='1' " "window=':Qt Creator_Core::Internal::MainWindow'}")