mirror of
https://gitlab.com/mbunkus/mkvtoolnix.git
synced 2024-12-31 23:38:34 +00:00
c7e004efa1
If the locale is set to non-UTF-8 when (d)rake is invoked then Ruby will set the encoding of all strings read from files to US-ASCII. As several source files do use non-ASCII characters this results in warnings from Ruby about "invalid byte sequence in US-ASCII". As all of my source code files are encoded in UTF-8 we can simply enforce this. This happens when building the RPMs which sets the locale to C.
668 lines
18 KiB
Ruby
668 lines
18 KiB
Ruby
# coding: utf-8
|
|
#
|
|
# mkvtoolnix - programs for manipulating Matroska files
|
|
# Copyright © 2003…2016 Moritz Bunkus
|
|
#
|
|
# This file is distributed as part of mkvtoolnix and is distributed under the
|
|
# GNU General Public License Version 2, or (at your option) any later version.
|
|
# See accompanying file COPYING for details or visit
|
|
# http://www.gnu.org/licenses/gpl.html .
|
|
#
|
|
# precompiled header support
|
|
#
|
|
# Authors: KonaBlend <kona8lend@gmail.com>
|
|
#
|
|
|
|
require_relative 'helpers'
|
|
|
|
require 'erb'
|
|
require 'json'
|
|
require 'pathname'
|
|
require 'pty'
|
|
|
|
module PCH
|
|
extend Rake::DSL
|
|
|
|
#############################################################################
|
|
#
|
|
# Parser for customized variable number of task[options...] in white-space
|
|
# separated, name=value pairs, with quoted strings permitted.
|
|
#
|
|
# name --> toggle action (must be a boolean)
|
|
# name= --> set to default value
|
|
# name=value --> set to value
|
|
#
|
|
class Options
|
|
def initialize(t, spec)
|
|
@task = t
|
|
@spec = spec
|
|
@pairs = {}
|
|
end
|
|
|
|
def handler(key, &block)
|
|
key = key.to_s
|
|
if block.arity == 0 # reader has 0 args
|
|
define_singleton_method(key.to_sym) { block.call }
|
|
else
|
|
@pairs.store(key, block)
|
|
end
|
|
end
|
|
|
|
def parse
|
|
return unless @spec
|
|
parse_root do |k,v|
|
|
fail "unknown name '#{k}' specified for task '#{@task.name}'" unless @pairs.has_key?(k)
|
|
@pairs.fetch(k).call(k, v)
|
|
end
|
|
end
|
|
|
|
def parse_root
|
|
@index = 0
|
|
@name = ''
|
|
@value = nil
|
|
|
|
while @index < @spec.length
|
|
c = @spec[@index]
|
|
if c == " " || c == "\t"
|
|
@index += 1
|
|
else
|
|
parse_name
|
|
yield @name,@value
|
|
@name = ''
|
|
@value = nil
|
|
end
|
|
end
|
|
end
|
|
|
|
def parse_name
|
|
while @index < @spec.length
|
|
c = @spec[@index]
|
|
case
|
|
when c == " "
|
|
@index += 1
|
|
break
|
|
when c == "="
|
|
fail "unexpected '#{c}' while parsing name" if @name.empty?
|
|
@index += 1
|
|
parse_value
|
|
break
|
|
else
|
|
@name += c
|
|
@index += 1
|
|
end
|
|
end
|
|
end
|
|
|
|
def parse_value
|
|
@value = ''
|
|
while @index < @spec.length
|
|
c = @spec[@index]
|
|
case
|
|
when c == " "
|
|
@index += 1
|
|
break
|
|
when c == '"' && (@index+1) < @spec.length
|
|
@index += 1
|
|
parse_quote
|
|
else
|
|
@value += c
|
|
@index += 1
|
|
end
|
|
end
|
|
end
|
|
|
|
def parse_quote
|
|
open = true
|
|
while @index < @spec.length
|
|
c = @spec[@index]
|
|
case
|
|
when c == '"'
|
|
open = false
|
|
@index += 1
|
|
break
|
|
when c == '\\' && (@index+1) < @spec.length
|
|
@value += @spec[@index+1]
|
|
@index += 2
|
|
else
|
|
@value += c
|
|
@index += 1
|
|
end
|
|
end
|
|
fail "unterminated quote while parsing value" if open
|
|
end
|
|
end
|
|
|
|
#############################################################################
|
|
|
|
def self.engage(&cxx_compiler)
|
|
return unless c?(:USE_PRECOMPILED_HEADERS)
|
|
load_config
|
|
@users.clear
|
|
$all_sources.each { |source| @users.store(source, nil) }
|
|
@moc_users.each_key { |moc| @users.store(moc, Pathname.new(moc).sub_ext(".h").to_s) }
|
|
namespace :pch do
|
|
t = file(@config_file.to_s => @config_deps) { scan_users }
|
|
t.invoke
|
|
end
|
|
add_tasks(&cxx_compiler)
|
|
add_prerequisites
|
|
end
|
|
|
|
#############################################################################
|
|
|
|
def self.add_tasks(&compiler)
|
|
pchs = []
|
|
headers = {}
|
|
@db_scan.each_value { |h| headers.store(h, nil) }
|
|
headers.keys.each do |header|
|
|
pch = "#{header}#{@extension}"
|
|
pchs.push(file pch => "#{header}", &compiler)
|
|
end
|
|
|
|
desc "Set pch related options (persistent)"
|
|
task :pch, :options do |t,args|
|
|
o = Options.new(t, args.options)
|
|
|
|
o.handler(:htrace) { @htrace }
|
|
o.handler(:htrace) do |k,v|
|
|
case v
|
|
when nil
|
|
@htrace = !@htrace
|
|
when ''
|
|
@htrace = false
|
|
when /^(true|yes|1)$/i
|
|
@htrace = true
|
|
when /^(false|no|0)$/i
|
|
@htrace = false
|
|
else
|
|
fail "invalid value for pch[htrace]: #{v}"
|
|
end
|
|
puts "pch[htrace] = #{@htrace}"
|
|
end
|
|
|
|
o.handler(:pretty) { @pretty }
|
|
o.handler(:pretty) do |k,v|
|
|
case v
|
|
when nil
|
|
@pretty = !@pretty
|
|
when ''
|
|
@pretty = false
|
|
when /^(true|yes|1)$/i
|
|
@pretty = true
|
|
when /^(false|no|0)$/i
|
|
@pretty = false
|
|
else
|
|
fail "invalid value for pch[pretty]: #{v}"
|
|
end
|
|
puts "pch[pretty] = #{@pretty}"
|
|
end
|
|
|
|
o.parse
|
|
end
|
|
|
|
namespace :pch do
|
|
desc "Overview"
|
|
rself = Pathname.new(__FILE__)
|
|
rself = rself.relative_path_from(Pathname.new(Dir.pwd)) unless rself.relative?
|
|
task :overview do
|
|
text = <<-ENDTEXT
|
|
:
|
|
: PCH - precompiled header support
|
|
:
|
|
: Precompiled headers are active when the host operating system and tools
|
|
: are capable (as detected and enabled by configure). A PCH configuration
|
|
: is saved in '#{@config_file.to_s}' and is automatically managed and
|
|
: also participates with task 'clean:dist'.
|
|
:
|
|
: The config is a task resolved at rake load-time. That is to say, this
|
|
: task effectively behaves like a prerequisite to all other tasks.
|
|
:
|
|
: PCH system will bootstrap by scanning all known user files (.cpp, .moc)
|
|
: for idiomatic precompiled header usage. The pattern scanned for is:
|
|
:
|
|
: #{@scan_include_re.to_s}
|
|
:
|
|
: PCH will re-bootstrap if '#{@config_file.to_s} is out of date to any of:
|
|
:
|
|
<% @config_deps.each do |x| %>
|
|
: <%= x %>
|
|
<% end %>
|
|
:
|
|
: During source file compilation, logic has been added to probe give PCH
|
|
: opportunity to supply additional compiler flags (-include) and thus use
|
|
: a precompiled header.
|
|
:
|
|
: The pch binary is added as a prerequisite for all of its users, so
|
|
: pch generation is automatic, but tasks 'pch:all' and 'pch:clean' may be
|
|
: used manually.
|
|
:
|
|
: 'pch:status' generates a short report on PCH state of affairs. Verbosity
|
|
: may be increased like so; note shell quotes placed around target to
|
|
: escape shell-special treatment of square-brackets.
|
|
:
|
|
: #{Rake.application.name} "pch:status[verbose]"
|
|
:
|
|
: 'pch[htrace]' is persistent flag which (when true) causes all subsequent
|
|
: source file compilation to record which header files were opened. The
|
|
: output is saved to .htrace files and .htrace files are clean as part
|
|
: of task 'clean'.
|
|
:
|
|
: 'pch[pretty]' is persistent flag which (when true) adds more information
|
|
: about files as they are compiled.
|
|
:
|
|
ENDTEXT
|
|
ERB.new(text, nil, trim_mode="<>").run(binding)
|
|
end
|
|
|
|
desc "Generate all precompiled headers"
|
|
task :all => pchs
|
|
|
|
desc "Remove all precompiled headers"
|
|
task :clean do
|
|
pats = []
|
|
pchs.each { |x| pats << x.name << (x.name + "-????????") }
|
|
remove_files_by_patterns(pats)
|
|
end
|
|
|
|
desc "Scan candidate user files for precompiled header pattern."
|
|
task :scan do
|
|
scan_users
|
|
end
|
|
|
|
desc "Status of database"
|
|
task :status, :options do |t,args|
|
|
o = Options.new(t, args.options)
|
|
|
|
o.handler(:verbose) { o.instance_exec { @verbose }}
|
|
o.handler(:verbose) do |k,v|
|
|
case v
|
|
when nil
|
|
o.instance_exec { @verbose = !@verbose }
|
|
when ""
|
|
o.instance_exec { @verbose = false }
|
|
when /^(true|yes|1)$/i
|
|
o.instance_exec { @verbose = true }
|
|
when /^(false|no|0)$/i
|
|
o.instance_exec { @verbose = false }
|
|
else
|
|
fail "invalid value for pch[verbose]: #{v}"
|
|
end
|
|
end
|
|
o.parse
|
|
status(o)
|
|
end
|
|
end
|
|
|
|
Rake::Task['clean:dist'].enhance { @endblock_skip_save = true }
|
|
end
|
|
|
|
#############################################################################
|
|
|
|
def self.add_prerequisites
|
|
@db_scan.each_pair do |user, header|
|
|
case File.extname(user)
|
|
when '.moc'
|
|
object = user + 'o'
|
|
file object => user
|
|
else
|
|
object = Pathname.new(user).sub_ext('.o')
|
|
end
|
|
file object => "#{header}#{@extension}"
|
|
end
|
|
end
|
|
|
|
#############################################################################
|
|
|
|
# return hash of file types (extension) => count
|
|
def self.file_types(files)
|
|
r = {}
|
|
files.each do |name|
|
|
key = File.extname(name)
|
|
i = r.fetch(key, 0)
|
|
r.store(key, i+1)
|
|
end
|
|
r.sort_by { |k,v| k }.collect { |x| "#{x[0]}=#{x[1]}" }.join(", ")
|
|
end
|
|
|
|
def self.status(options)
|
|
headers = {}
|
|
@db_scan.each_value do |h|
|
|
i = headers.fetch(h, 0)
|
|
headers.store(h, i+1)
|
|
end
|
|
unmarked = @users.keys.clone
|
|
unmarked.delete_if { |k,v| @db_scan.has_key?(k) }
|
|
|
|
text = <<-ENDTEXT
|
|
PCH status: <%= c?(:USE_PRECOMPILED_HEADERS) ? "enabled" : "disabled" %>
|
|
:
|
|
: htrace = #{@htrace}
|
|
: pretty = #{@pretty}
|
|
:
|
|
: <%= "%8d %-40s (%s)" % [@users.size, "total users", file_types(@users.keys)] %>
|
|
: <%= "%8d %-40s (%s)" % [headers.size, "unique pch headers", file_types(headers.keys)] %>
|
|
: <%= "%8d %-40s (%s)" % [@db_scan.size, "users marked for pch use", file_types(@db_scan.keys)] %>
|
|
: <%= "%8d %-40s (%s)" % [unmarked.size, "users not marked", file_types(unmarked)] %>
|
|
<% headers.each_pair do |k,v| %>
|
|
: <%= "%8d %-40s (%s)" % [v,"users of %s" % [k], file_types(@db_scan.select { |k1,v1| v1 == k }.keys)] %>
|
|
<% end %>
|
|
ENDTEXT
|
|
|
|
text.concat( <<-ENDTEXT
|
|
<% lines = @db_scan.each_pair.collect { |k,v| ": marked %s -> %s" % [k,v] } %>
|
|
<% unless lines.empty? %>
|
|
|
|
<% end %>
|
|
<%= lines.join("\n") %>
|
|
<% lines = unmarked.collect { |u| ": unmarked %s" % [u] } %>
|
|
<% unless lines.empty? %>
|
|
|
|
<% end %>
|
|
<%= lines.join("\n") %>
|
|
ENDTEXT
|
|
) if options.verbose
|
|
|
|
text << ":\n"
|
|
ERB.new(text, nil, trim_mode="<>").run(binding)
|
|
end
|
|
|
|
#############################################################################
|
|
|
|
def self.scan_users
|
|
if @verbose
|
|
verb = @config_file.exist? ? "rescan" : "scan"
|
|
else
|
|
verb = "%*s" % [$action_width, (@config_file.exist? ? "rescan" : "scan").upcase]
|
|
end
|
|
puts "#{verb} pch candidates (total=#{@users.size}, #{file_types(@users.keys)})"
|
|
@users.each_pair { |k,v| scan_user(k,v) }
|
|
end
|
|
|
|
def self.scan_user(user, indirect)
|
|
found = nil
|
|
input = indirect ? indirect : user
|
|
File.open(input) do |f|
|
|
f.each_line do |line|
|
|
line.force_encoding("UTF-8")
|
|
next if !@scan_include_re.match(line)
|
|
@scan_candidates.each do |pair|
|
|
(dir,header) = *pair
|
|
next unless $1 == header
|
|
found = dir ? "#{dir}/#{$1}" : $1
|
|
break
|
|
end
|
|
break if found
|
|
end
|
|
end
|
|
return unless found
|
|
@db_scan.store(user, found)
|
|
rescue StandardError => e
|
|
puts "WARNING: unable to read #{input}: #{e}"
|
|
end
|
|
|
|
#############################################################################
|
|
|
|
class Info
|
|
attr_accessor :language
|
|
attr_accessor :use_flags
|
|
attr_accessor :extra_flags
|
|
attr_accessor :pretty_flags
|
|
end
|
|
|
|
def self.info_for_user(user, ofile)
|
|
f = Info.new
|
|
if c?(:USE_PRECOMPILED_HEADERS)
|
|
user = Pathname.new(user).cleanpath.to_s
|
|
f.language = "c++-header" if user.end_with?(".h")
|
|
header = @db_scan.fetch(user, nil)
|
|
f.use_flags = header ? " -include #{header}" : nil
|
|
f.extra_flags = @htrace ? "-H" : nil
|
|
f.pretty_flags = (@pretty || @htrace) ? {
|
|
htrace: @htrace ? (ofile + ".htrace") : nil,
|
|
user: @users.has_key?(user),
|
|
precompile: f.use_flags != nil,
|
|
} : nil
|
|
end
|
|
f
|
|
end
|
|
|
|
#############################################################################
|
|
|
|
def self.moc_users(others)
|
|
others.each { |other| @moc_users.store(other, nil) }
|
|
end
|
|
|
|
#############################################################################
|
|
|
|
def self.make_task_filter(name)
|
|
return nil unless @htrace
|
|
|
|
lambda do |code,lines|
|
|
rx_stat = /^([.]*[!x]?)\s(.*)/
|
|
rx_ignore_begin = /^Multiple include guards may be useful for/i
|
|
rx_ignore_more = /\//
|
|
|
|
ignore = false
|
|
|
|
File.open(name + ".htrace", "w") do |io|
|
|
lines.each do |line|
|
|
case
|
|
when line =~ rx_stat
|
|
;
|
|
when !ignore && line =~ rx_ignore_begin
|
|
ignore = true
|
|
next
|
|
when ignore && line =~ rx_ignore_more
|
|
next
|
|
else
|
|
puts line
|
|
next
|
|
end
|
|
io.write("#{line}")
|
|
end
|
|
end
|
|
end
|
|
end
|
|
|
|
#############################################################################
|
|
|
|
def self.load_config
|
|
# puts "load #{@config_file}" if @verbose
|
|
config = {}
|
|
config = @config_file.open { |f| JSON.load(f) } if @config_file.exist?
|
|
@htrace = config.fetch('htrace', @htrace)
|
|
@pretty = config.fetch('pretty', @pretty)
|
|
@db_scan = config.fetch('scan', @db_scan)
|
|
end
|
|
|
|
def self.save_config
|
|
# puts "save #{@config_file}" if @verbose
|
|
root = {
|
|
htrace: @htrace,
|
|
pretty: @pretty,
|
|
scan: @db_scan,
|
|
}
|
|
@config_file.open("w") do |out|
|
|
out.write(JSON.generate(root, space:" ", indent:" ", object_nl:"\n"))
|
|
out.write("\n")
|
|
end
|
|
end
|
|
|
|
#############################################################################
|
|
|
|
def self.clean_patterns
|
|
%W[
|
|
**/*.[gp]ch **/*.pch-????????
|
|
**/*.htrace
|
|
#{@config_file}
|
|
]
|
|
end
|
|
|
|
#############################################################################
|
|
#
|
|
# Execute a system command, similar to helpers.rb/runq but differs mainly
|
|
# in offering generator-style filter model, delayed, and in some cases
|
|
# more information on normal rake stdout.
|
|
#
|
|
# All output is accumulated into a buffer until the command completes.
|
|
# Advantage of keeping command-output together in a parallel build.
|
|
# Disadvantage of delaying output until command is complete.
|
|
#
|
|
def self.runq(action, subject, command, options={})
|
|
command = command.gsub(/\n/, ' ').gsub(/^\s+/, '').gsub(/\s+$/, '').gsub(/\s+/, ' ')
|
|
if @verbose
|
|
puts command
|
|
else
|
|
h = options.fetch(:htrace, nil) ? 't' : '-'
|
|
u = options.fetch(:user, nil) ? 'u' : '-'
|
|
p = options.fetch(:precompile, nil) ? 'p' : '-'
|
|
puts "%c%c%c %*s %s" % [h, u, p, $action_width - 4, action.gsub(/ +/, '_').upcase, subject]
|
|
end
|
|
htrace = options.fetch(:htrace, nil)
|
|
return execute(command, options) unless htrace
|
|
|
|
# log -H output, return other
|
|
File.open(htrace, "w") do |io|
|
|
rx_stat = /^([.]*[!x]?)\s(.*)/
|
|
rx_ignore_begin = /^Multiple include guards may be useful for/i
|
|
rx_ignore_more = /\//
|
|
ignore = false
|
|
# generator records (sends to rake output) only lines we return
|
|
execute(command, options) do |line|
|
|
case
|
|
when line =~ rx_stat
|
|
io.write("#{line}")
|
|
next nil
|
|
when !ignore && line =~ rx_ignore_begin
|
|
ignore = true
|
|
next nil
|
|
when ignore && line =~ rx_ignore_more
|
|
next nil
|
|
else
|
|
next line
|
|
end
|
|
end
|
|
end
|
|
end
|
|
|
|
def self.execute(command, options={}, &block)
|
|
if STDOUT.tty?
|
|
execute_tty(command, options, &block)
|
|
else
|
|
execute_command(command, options, &block)
|
|
end
|
|
end
|
|
|
|
def self.execute_command(command, options={})
|
|
lines = []
|
|
IO.popen(command, :err => [:child, :out]) do |io|
|
|
if block_given?
|
|
io.each_line { |s| lines.push(s) if s = yield(s) }
|
|
else
|
|
io.each_line { |s| lines << s }
|
|
end
|
|
end
|
|
end_execute(lines, options)
|
|
end
|
|
|
|
def self.execute_tty(command, options={})
|
|
lines = []
|
|
IO.pipe do |read,write|
|
|
master,slave = PTY.open
|
|
begin
|
|
pid = spawn(command, :in => read, [:out, :err] => slave)
|
|
read.close
|
|
slave.close
|
|
begin
|
|
if block_given?
|
|
master.each_line { |s| lines.push(s) if s = yield(s) }
|
|
else
|
|
master.each_line { |s| lines << s }
|
|
end
|
|
rescue
|
|
;
|
|
ensure
|
|
Process.wait(pid)
|
|
end
|
|
ensure
|
|
master.close unless master.closed?
|
|
slave.close unless slave.closed?
|
|
end
|
|
end
|
|
end_execute(lines, options)
|
|
end
|
|
|
|
def self.end_execute(lines, options)
|
|
if options.fetch(:allow_failure, nil) != true
|
|
exit(1) if $?.exitstatus == nil
|
|
exit($?.exitstatus) if $?.exitstatus != 0
|
|
end
|
|
ps = $?.clone
|
|
puts lines
|
|
[ps, lines]
|
|
end
|
|
|
|
#############################################################################
|
|
|
|
# True when tracing headers during compilation. This controls when a filter
|
|
# in CXX compilation action engages, and also causes adds -H compiler flag.
|
|
@htrace = false
|
|
|
|
# True when PCH enhances typical rake output with extra information.
|
|
@pretty = false
|
|
|
|
# GCC and clang differ in their extension preferences for precompiled
|
|
# headers. clang has some knowledge of gch and at the time of this writing,
|
|
# actually does search for both { .pch, .gch } but it's not documented.
|
|
@extension = c?(:USE_CLANG) ? ".pch" : ".gch"
|
|
|
|
# Cache global verbose flag.
|
|
@verbose = $verbose
|
|
|
|
# Each pch file must be unambiguous for simple scanning techniques.
|
|
# The scanner needs to know the string of a pch file as it is used in
|
|
# CPP include directives. It also needs to know the prefix pathname
|
|
# as it exists in source tree, relative to Rakefile.
|
|
@scan_candidates = [ ["src", "common/common_pch.h"] ]
|
|
|
|
# All usage of project pch files are expected to be between quotes,
|
|
# and not angle brackets.
|
|
@scan_include_re = /^\s*#\s*include\s+"([^"]+)"/
|
|
|
|
# Location of pch configuration.
|
|
@config_file = Pathname.new("config.pch.json")
|
|
|
|
# Populated by project rakefile.
|
|
@moc_users = {}
|
|
|
|
# All candidate user files (.cpp, .moc)
|
|
# user => intermediary
|
|
# where user is the actual source file being compiled
|
|
# and intermediary is nil or alternate file to scan (used by .moc)
|
|
@users = {}
|
|
|
|
# Resident scan DB. Has mapping of user files (.cpp, .moc) to a pch file
|
|
# (as used in #include directive).
|
|
@db_scan = {}
|
|
|
|
rself = Pathname.new(__FILE__)
|
|
rself = rself.relative_path_from(Pathname.new(Dir.pwd)) unless rself.relative?
|
|
@config_deps = [Rake.application.rakefile, 'build-config', rself.to_s]
|
|
|
|
#############################################################################
|
|
|
|
@endblock_skip_save = false
|
|
|
|
END {
|
|
next if @endblock_skip_save
|
|
begin
|
|
save_config
|
|
rescue StandardError => e
|
|
puts "WARNING: failed to save config: #{e}"
|
|
end
|
|
}
|
|
|
|
#############################################################################
|
|
|
|
end # module PCH
|