From dffca8bb0eac54a29248e5cbb69c35f85be333bf Mon Sep 17 00:00:00 2001 From: Marc Mutz <marc.mutz@qt.io> Date: Wed, 27 Apr 2022 20:57:32 +0200 Subject: [PATCH] Add a simple Qt 6/CMake-compatible script a la includemocs.py MIME-Version: 1.0 Content-Type: text/plain; charset=UTF-8 Content-Transfer-Encoding: 8bit Perl is dead and I don't speak Python, so if someone wants to port it, be my guest. I needed something that I can use to roll out includemocs across all Qt modules, and this is it. It works for me™, so I don't expect to do much development with it. This is an automated committer script I've been using so far: # SRCDIR=~~~ # BUILDDIR=~~~~ cd $BUILDDIR find */src -name mocs_compilation.cpp | while read FILE; do if grep -qsE '^#include' "$FILE"; then DIR="$(dirname "$FILE")" LIB="$(basename "$DIR")" case "$LIB" in Q*) LIB="${LIB%%_autogen}" ;; *) LIB="Qt${LIB%%_autogen}" ;; esac DIR="${DIR%/*}" path/to/includemocs6.sh "$SRCROOT/$DIR" "$DIR" "$FILE" (cd "$SRCROOT/$DIR" && git commit -am "$LIB: includemocs $(cat "$SRCROOT/commit-msg.txt")" --no-edit) fi done If the script cannot associate a moc file with a cpp file, it will print a warning and continue. The script tries to include the moc-file right after the QT_END_NAMESPACE to work around many TUs ending in an #endif from some #if QT_CONFIG or other. If there's no QT_END_NAMESPACE, it appends the include and prints a warning. Fixes: QTBUG-102886 Pick-to: 6.3 6.2 Change-Id: I16c5a9f845777ea2e82f15611b4fdd32f98ce0bb Reviewed-by: Thiago Macieira <thiago.macieira@intel.com> --- util/includemocs/includemocs6.sh | 119 +++++++++++++++++++++++++++++++ 1 file changed, 119 insertions(+) create mode 100755 util/includemocs/includemocs6.sh diff --git a/util/includemocs/includemocs6.sh b/util/includemocs/includemocs6.sh new file mode 100755 index 0000000000..09144f4c52 --- /dev/null +++ b/util/includemocs/includemocs6.sh @@ -0,0 +1,119 @@ +#!/bin/bash +############################################################################# +## +## Copyright (C) 2022 The Qt Company Ltd. +## Contact: https://www.qt.io/licensing/ +## +## This file is the build configuration utility of the Qt Toolkit. +## +## $QT_BEGIN_LICENSE:GPL-EXCEPT$ +## Commercial License Usage +## Licensees holding valid commercial Qt licenses may use this file in +## accordance with the commercial license agreement provided with the +## Software or, alternatively, in accordance with the terms contained in +## a written agreement between you and The Qt Company. For licensing terms +## and conditions see https://www.qt.io/terms-conditions. For further +## information use the contact form at https://www.qt.io/contact-us. +## +## GNU General Public License Usage +## Alternatively, this file may be used under the terms of the GNU +## General Public License version 3 as published by the Free Software +## Foundation with exceptions as appearing in the file LICENSE.GPL3-EXCEPT +## included in the packaging of this file. Please review the following +## information to ensure the GNU General Public License requirements will +## be met: https://www.gnu.org/licenses/gpl-3.0.html. +## +## $QT_END_LICENSE$ +## +############################################################################# + +VERBOSE=0 +CONTINUE_ON_FAILURE=1 + +function warn() { + echo "$@" >&2 +} + +function log() { + [[ "$VERBOSE" -eq 1 ]] && warn "$@" +} + +function die() { + warn "$@" + exit 1 +} + +function continue_or_die() { + if [[ "$CONTINUE_ON_FAILURE" -eq 1 ]] ; then + warn "$@" + else + die "$@" + fi +} + +function usage() { + die "usage: includemocs6.sh <srcdir> <builddir> <mocs_compilation_file>" +} + +# +# sanity checks: +# + +[[ ${#@} -eq 3 ]] || usage + +SRCDIR="$1" +BUILDDIR="$2" +MOCS_COMPILATION_FILE="$3" + +[[ -d "$BUILDDIR" ]] || die "Build dir \"$BUILDDIR\" doesn't exist or isn't a directory." +[[ -d "$SRCDIR" ]] || die "Source dir \"SRCDIR\" doesn't exist or isn't a directory." +[[ -f "$MOCS_COMPILATION_FILE" ]] || die "MOCs-compilation file \"$MOCS_COMPILATION_FILE\" doesn't exist or isn't a file." + +# +# extract the files included by the mocs_compilation.cpp +# + +# lines look like this: +# #include "SOFJASFD/moc_foo.cpp" +# #include "JFEJGKKS/moc_foo_p.cpp" +# extracting moc_foo(_p).cpp: +grep -E '^#include' "$MOCS_COMPILATION_FILE" | cut -d\" -f2 | cut -d/ -f2 | while read MOCFILE; do + log "MOCFILE=$MOCFILE" + C1="${MOCFILE##moc_}" # foo.cpp or foo_p.cpp or foo_p_p.cpp + case "$MOCFILE" in + moc_*_p_p.cpp) + C2="${C1/_p.cpp/.cpp}" # foo_p.cpp + C3="${C2/_p.cpp/.cpp}" # foo.cpp + CANDIDATES=("$C1" "$C2" "$C3") + ;; + moc_*_p.cpp) + C2="${C1/_p.cpp/.cpp}" # foo.cpp + CANDIDATES=("$C1" "$C2") + ;; + moc_*.cpp) + CANDIDATES=("$C1") + ;; + *) + die "Don't know how to handle moc-file \"$MOCFILE\"..." + ;; + esac + log "CANDIDATES=(${CANDIDATES[@]})" + for CANDIDATE in "${CANDIDATES[@]}"; do + CPPFILE="$(find "$SRCDIR" -name "$CANDIDATE")" + log "CPPFILE=$CPPFILE" + [[ -f "$CPPFILE" ]] && break + done + if [[ -f "$CPPFILE" ]] ; then + log "going to include $MOCFILE into $CPPFILE" + if ! grep -qE '^QT_END_NAMESPACE$' "$CPPFILE"; then + warn "Can't find QT_END_NAMESPACE in \"$CPPFILE\", simply appending the #include. Please check placement manually." + echo "#include \"$MOCFILE\"" >> "$CPPFILE" || die "Failed to write to \"$CPPFILE\"" + else + sed -i -e "1,/QT_END_NAMESPACE/ s/QT_END_NAMESPACE/QT_END_NAMESPACE\n\n#include \"$MOCFILE\"/" "$CPPFILE" || die "Failed to includemoc \"$MOCFILE\" into \"$CPPFILE\"." + fi + else + continue_or_die "Can't find a cpp file for $MOCFILE (not looking for .cxx .cc etc)." + fi +done + +exit 0