Received: by 2002:a25:7ec1:0:0:0:0:0 with SMTP id z184csp2529109ybc; Wed, 20 Nov 2019 16:05:12 -0800 (PST) X-Google-Smtp-Source: APXvYqx68mYsQ2DvnsT+Kvmca2BCbFmnypEV/0Whtq4BaRefgxWAW7MYxmuWOZh2kB7JHWQqrbD6 X-Received: by 2002:a17:907:206d:: with SMTP id qp13mr8829056ejb.92.1574294712318; Wed, 20 Nov 2019 16:05:12 -0800 (PST) ARC-Seal: i=1; a=rsa-sha256; t=1574294712; cv=none; d=google.com; s=arc-20160816; b=RDPcdI0cVv7hiHecFow4apkTnVNl2D2ks54vsyQ9Qawu2kAdtMJZh/qvwYYVmVSJ2W JYEPkXYAwa/ALBnGiW/n/xqEY/yn6Yag+iD9gRzXMl9zyvZ9YTuoNULyNeQZx22d4twg K2/aK9JpwwM3C/OxgY35kWlPxm2JEjoh8uRVMoVjwfpRUB7LRvqLQEgWvlU+bhY9Ocbh 0Hvv5iDmUlhGR9tSByRXImb95xp+ApZDJzSP42wI+fvR6l6NucrQdrtwXc6L38semxgc sVoNp3RTFInhjYeCQGGffcriVNT8GnJ073HHVmrGlTmuP0dVGY3oAU95HGL3WzdaUBSP DSqA== ARC-Message-Signature: i=1; a=rsa-sha256; c=relaxed/relaxed; d=google.com; s=arc-20160816; h=list-id:precedence:sender:references:in-reply-to:message-id:date :subject:cc:to:from:dkim-signature; bh=6psxPlZl9k9hVxAcY5nJfgBFA/h6KZ9Pi6P6/8PS9Bs=; b=SbCOO0LhnTDWqnNawcxVaw5byfWSs2VaDeQEqP58lQWXzZdK/cDOMhuMxEUAaMn8GI OzdOg0fS1MS8d+msWJHzkB4qQFhJRwKtpD41YqpEX1kAy1lNhdEpJlwRya3hYdi9+xco aWxmYTaw+2UefTzDHtPtR54ADNsvdMK6w9T1Gyxru17+2YMHcR8r6VmYolPpOsEprRF4 ur75z3rPCHEB+vU4GGL5IHN4Nc49PO+9Qs/lpgmKlH8y8wcP7oj/tTUAcS0VkR9aKImP pT65oNbZNrqX2EB9uPnUIkK2LBvm/TGqvP+qN1EkgIsT8F4YUrUyON74IvPT1wDFhvb/ 7+6A== ARC-Authentication-Results: i=1; mx.google.com; dkim=pass header.i=@chromium.org header.s=google header.b=CRFILFHf; spf=pass (google.com: best guess record for domain of linux-kernel-owner@vger.kernel.org designates 209.132.180.67 as permitted sender) smtp.mailfrom=linux-kernel-owner@vger.kernel.org; dmarc=pass (p=NONE sp=NONE dis=NONE) header.from=chromium.org Return-Path: Received: from vger.kernel.org (vger.kernel.org. [209.132.180.67]) by mx.google.com with ESMTP id mf15si559520ejb.46.2019.11.20.16.04.48; Wed, 20 Nov 2019 16:05:12 -0800 (PST) Received-SPF: pass (google.com: best guess record for domain of linux-kernel-owner@vger.kernel.org designates 209.132.180.67 as permitted sender) client-ip=209.132.180.67; Authentication-Results: mx.google.com; dkim=pass header.i=@chromium.org header.s=google header.b=CRFILFHf; spf=pass (google.com: best guess record for domain of linux-kernel-owner@vger.kernel.org designates 209.132.180.67 as permitted sender) smtp.mailfrom=linux-kernel-owner@vger.kernel.org; dmarc=pass (p=NONE sp=NONE dis=NONE) header.from=chromium.org Received: (majordomo@vger.kernel.org) by vger.kernel.org via listexpand id S1726840AbfKUADX (ORCPT + 99 others); Wed, 20 Nov 2019 19:03:23 -0500 Received: from mail-pl1-f194.google.com ([209.85.214.194]:46269 "EHLO mail-pl1-f194.google.com" rhost-flags-OK-OK-OK-OK) by vger.kernel.org with ESMTP id S1726634AbfKUADM (ORCPT ); Wed, 20 Nov 2019 19:03:12 -0500 Received: by mail-pl1-f194.google.com with SMTP id l4so602440plt.13 for ; Wed, 20 Nov 2019 16:03:10 -0800 (PST) DKIM-Signature: v=1; a=rsa-sha256; c=relaxed/relaxed; d=chromium.org; s=google; h=from:to:cc:subject:date:message-id:in-reply-to:references; bh=6psxPlZl9k9hVxAcY5nJfgBFA/h6KZ9Pi6P6/8PS9Bs=; b=CRFILFHfiKc0hbSgbiktgDPENvDtSf8an3+vKm10BZnsX8g0nra/XcnzdM6nqNBJWu 39/vTFFigXtiz0ZeqeZF+RNZMC5+AtvcRrXQ9pm6AwjAu10Jbgc6r6P8L0uZr5RU5L6+ retbqw+fEVqIUjM3FmrMu9Pp1/m6Qjansn9RY= X-Google-DKIM-Signature: v=1; a=rsa-sha256; c=relaxed/relaxed; d=1e100.net; s=20161025; h=x-gm-message-state:from:to:cc:subject:date:message-id:in-reply-to :references; bh=6psxPlZl9k9hVxAcY5nJfgBFA/h6KZ9Pi6P6/8PS9Bs=; b=KGUrhyxiM0tLnAixF8Utf+DTAiW3T1a7WFMl054Swkks7RQduTWNZG3IPUEWUC0Sb4 r8WvNU3vMuIFmLUPZKMaLF9nEo+SYqmHf0oJRpSMpRExP4WqpzehD0q+ys+qqkQBdNxk tjnnG9MmYRagihwoXswHafVQ5GvgCYad3fKXWxObQLZBnJy/kcQErq7bkJZIKl2bWkMD ZH9FJYFr7AexvIsGs0XI8NCrIyIIwXPWagIL3+Q45OCdDwbcxqURxJgXxdMMp0fUOCf9 d3FiRBeb2bpIbJbY/9sBqMo3l2R0HhaAj8c1TKxO44sGKP/1R1ZcAkbS9MI0JBcgvTQC l3Xw== X-Gm-Message-State: APjAAAUHbb3DGByzN+rFnqcEFm5+S7Sx7ncJ9MOvSoVa5hB6X6jnxPoO nqUJlFWqMJhcGlJu/EpRowlOww== X-Received: by 2002:a17:90a:bf81:: with SMTP id d1mr7373559pjs.125.1574294590377; Wed, 20 Nov 2019 16:03:10 -0800 (PST) Received: from www.outflux.net (smtp.outflux.net. [198.145.64.163]) by smtp.gmail.com with ESMTPSA id s18sm573022pfm.27.2019.11.20.16.03.09 (version=TLS1_3 cipher=TLS_AES_256_GCM_SHA384 bits=256/256); Wed, 20 Nov 2019 16:03:09 -0800 (PST) From: Kees Cook To: Jonathan Corbet Cc: Kees Cook , Rasmus Villemoes , linux-doc@vger.kernel.org, linux-kernel@vger.kernel.org Subject: [PATCH 3/3] docs, parallelism: Rearrange how jobserver reservations are made Date: Wed, 20 Nov 2019 16:03:04 -0800 Message-Id: <20191121000304.48829-4-keescook@chromium.org> X-Mailer: git-send-email 2.17.1 In-Reply-To: <20191121000304.48829-1-keescook@chromium.org> References: <20191121000304.48829-1-keescook@chromium.org> Sender: linux-kernel-owner@vger.kernel.org Precedence: bulk List-ID: X-Mailing-List: linux-kernel@vger.kernel.org Rasmus correctly observed that the existing jobserver reservation only worked if no other build targets were specified. The correct approach is to hold the jobserver slots until sphinx has finished. To fix this, the following changes are made: - refactor (and rename) scripts/jobserver-exec to set an environment variable for the maximally reserved jobserver slots and exec a child, to release the slots on exit. - create Documentation/scripts/parallel-wrapper.sh which examines both $PARALLELISM and the detected "-jauto" logic from Documentation/Makefile to decide sphinx's final -j argument. - chain these together in Documentation/Makefile Suggested-by: Rasmus Villemoes Link: https://lore.kernel.org/lkml/eb25959a-9ec4-3530-2031-d9d716b40b20@rasmusvillemoes.dk Signed-off-by: Kees Cook --- Documentation/Makefile | 5 +- Documentation/sphinx/parallel-wrapper.sh | 25 +++++++ scripts/{jobserver-count => jobserver-exec} | 73 ++++++++++++--------- 3 files changed, 68 insertions(+), 35 deletions(-) create mode 100644 Documentation/sphinx/parallel-wrapper.sh rename scripts/{jobserver-count => jobserver-exec} (50%) mode change 100755 => 100644 diff --git a/Documentation/Makefile b/Documentation/Makefile index ce8eb63b523a..30554a2fbdd7 100644 --- a/Documentation/Makefile +++ b/Documentation/Makefile @@ -33,8 +33,6 @@ ifeq ($(HAVE_SPHINX),0) else # HAVE_SPHINX -export SPHINX_PARALLEL = $(shell perl -e 'open IN,"sphinx-build --version 2>&1 |"; while () { if (m/([\d\.]+)/) { print "auto" if ($$1 >= "1.7") } ;} close IN') - # User-friendly check for pdflatex and latexmk HAVE_PDFLATEX := $(shell if which $(PDFLATEX) >/dev/null 2>&1; then echo 1; else echo 0; fi) HAVE_LATEXMK := $(shell if which latexmk >/dev/null 2>&1; then echo 1; else echo 0; fi) @@ -67,8 +65,9 @@ quiet_cmd_sphinx = SPHINX $@ --> file://$(abspath $(BUILDDIR)/$3/$4) cmd_sphinx = $(MAKE) BUILDDIR=$(abspath $(BUILDDIR)) $(build)=Documentation/media $2 && \ PYTHONDONTWRITEBYTECODE=1 \ BUILDDIR=$(abspath $(BUILDDIR)) SPHINX_CONF=$(abspath $(srctree)/$(src)/$5/$(SPHINX_CONF)) \ + $(PYTHON) $(srctree)/scripts/jobserver-exec \ + $(SHELL) $(srctree)/Documentation/sphinx/parallel-wrapper.sh \ $(SPHINXBUILD) \ - -j $(shell python $(srctree)/scripts/jobserver-count $(SPHINX_PARALLEL)) \ -b $2 \ -c $(abspath $(srctree)/$(src)) \ -d $(abspath $(BUILDDIR)/.doctrees/$3) \ diff --git a/Documentation/sphinx/parallel-wrapper.sh b/Documentation/sphinx/parallel-wrapper.sh new file mode 100644 index 000000000000..a416dbfd2025 --- /dev/null +++ b/Documentation/sphinx/parallel-wrapper.sh @@ -0,0 +1,25 @@ +#!/bin/sh +# SPDX-License-Identifier: GPL-2.0+ +# +# Figure out if we should follow a specific parallelism from the make +# environment (as exported by scripts/jobserver-exec), or fall back to +# the "auto" parallelism when "-jN" is not specified at the top-level +# "make" invocation. + +sphinx="$1" +shift || true + +parallel="${PARALLELISM:-1}" +if [ ${parallel} -eq 1 ] ; then + auto=$(perl -e 'open IN,"'"$sphinx"' --version 2>&1 |"; + while () { + if (m/([\d\.]+)/) { + print "auto" if ($1 >= "1.7") + } + } + close IN') + if [ -n "$auto" ] ; then + parallel="$auto" + fi +fi +exec "$sphinx" "-j$parallel" "$@" diff --git a/scripts/jobserver-count b/scripts/jobserver-exec old mode 100755 new mode 100644 similarity index 50% rename from scripts/jobserver-count rename to scripts/jobserver-exec index a68a04ad304f..4593b2a1e36d --- a/scripts/jobserver-count +++ b/scripts/jobserver-exec @@ -2,17 +2,16 @@ # SPDX-License-Identifier: GPL-2.0+ # # This determines how many parallel tasks "make" is expecting, as it is -# not exposed via an special variables. +# not exposed via an special variables, reserves them all, runs a subprocess +# with PARALLELISM environment variable set, and releases the jobs back again. +# # https://www.gnu.org/software/make/manual/html_node/POSIX-Jobserver.html#POSIX-Jobserver from __future__ import print_function import os, sys, fcntl, errno - -# Default parallelism is "1" unless overridden on the command-line. -default="1" -if len(sys.argv) > 1: - default=sys.argv[1] +import subprocess # Extract and prepare jobserver file descriptors from envirnoment. +jobs = b"" try: # Fetch the make environment options. flags = os.environ['MAKEFLAGS'] @@ -30,31 +29,41 @@ try: reader = os.open("/proc/self/fd/%d" % (reader), os.O_RDONLY) flags = fcntl.fcntl(reader, fcntl.F_GETFL) fcntl.fcntl(reader, fcntl.F_SETFL, flags | os.O_NONBLOCK) -except (KeyError, IndexError, ValueError, IOError, OSError) as e: - print(e, file=sys.stderr) + + # Read out as many jobserver slots as possible. + while True: + try: + slot = os.read(reader, 1) + jobs += slot + except (OSError, IOError) as e: + if e.errno == errno.EWOULDBLOCK: + # Stop at the end of the jobserver queue. + break + # If something went wrong, give back the jobs. + if len(jobs): + os.write(writer, jobs) + raise e +except (KeyError, IndexError, ValueError, OSError, IOError) as e: # Any missing environment strings or bad fds should result in just - # using the default specified parallelism. - print(default) - sys.exit(0) + # not being parallel. + pass -# Read out as many jobserver slots as possible. -jobs = b"" -while True: - try: - slot = os.read(reader, 1) - jobs += slot - except (OSError, IOError) as e: - if e.errno == errno.EWOULDBLOCK: - # Stop when reach the end of the jobserver queue. - break - raise e -# Return all the reserved slots. -os.write(writer, jobs) - -# If the jobserver was (impossibly) full or communication failed, use default. -if len(jobs) < 1: - print(default) - sys.exit(0) - -# Report available slots (with a bump for our caller's reserveration). -print(len(jobs) + 1) +claim = len(jobs) +if claim < 1: + # If the jobserver was (impossibly) full or communication failed + # in some way do not use parallelism. + claim = 0 + +# Launch command with a bump for our caller's reserveration, +# since we're just going to sit here blocked on our child. +claim += 1 + +os.unsetenv('MAKEFLAGS') +os.environ['PARALLELISM'] = '%d' % (claim) +rc = subprocess.call(sys.argv[1:]) + +# Return all the actually reserved slots. +if len(jobs): + os.write(writer, jobs) + +sys.exit(rc) -- 2.17.1