Training courses

Kernel and Embedded Linux

Bootlin training courses

Embedded Linux, kernel,
Yocto Project, Buildroot, real-time,
graphics, boot time, debugging...

Bootlin logo

Elixir Cross Referencer

# Copyright 2021 The OpenSSL Project Authors. All Rights Reserved.
#
# Licensed under the Apache License 2.0 (the "License").  You may not use
# this file except in compliance with the License.  You can obtain a copy
# in the file LICENSE in the source distribution or at
# https://www.openssl.org/source/license.html

package OpenSSL::Config::Query;

use 5.10.0;
use strict;
use warnings;
use Carp;

=head1 NAME

OpenSSL::Config::Query - Query OpenSSL configuration info

=head1 SYNOPSIS

    use OpenSSL::Config::Info;

    my $query = OpenSSL::Config::Query->new(info => \%unified_info);

    # Query for something that's expected to give a scalar back
    my $variable = $query->method(... args ...);

    # Query for something that's expected to give a list back
    my @variable = $query->method(... args ...);

=head1 DESCRIPTION

The unified info structure, commonly known as the %unified_info table, has
become quite complex, and a bit overwhelming to look through directly.  This
module makes querying this structure simpler, through diverse methods.

=head2 Constructor

=over 4

=item B<new> I<%options>

Creates an instance of the B<OpenSSL::Config::Query> class.  It takes options
in keyed pair form, i.e. a series of C<< key => value >> pairs.  Available
options are:

=over 4

=item B<info> =E<gt> I<HASHREF>

A reference to a unified information hash table, most commonly known as
%unified_info.

=item B<config> =E<gt> I<HASHREF>

A reference to a config information hash table, most commonly known as
%config.

=back

Example:

    my $info = OpenSSL::Config::Info->new(info => \%unified_info);

=back

=cut

sub new {
    my $class = shift;
    my %opts = @_;

    my @messages = _check_accepted_options(\%opts,
                                           info => 'HASH',
                                           config => 'HASH');
    croak $messages[0] if @messages;

    # We make a shallow copy of the input structure.  We might make
    # a different choice in the future...
    my $instance = { info => $opts{info} // {},
                     config => $opts{config} // {} };
    bless $instance, $class;

    return $instance;
}

=head2 Query methods

=over 4

=item B<get_sources> I<LIST>

LIST is expected to be the collection of names of end products, such as
programs, modules, libraries.

The returned result is a hash table reference, with each key being one of
these end product names, and its value being a reference to an array of
source file names that constitutes everything that will or may become part
of that end product.

=cut

sub get_sources {
    my $self = shift;

    my $result = {};
    foreach (@_) {
        my @sources = @{$self->{info}->{sources}->{$_} // []};
        my @staticlibs =
            grep { $_ =~ m|\.a$| } @{$self->{info}->{depends}->{$_} // []};

        my %parts = ( %{$self->get_sources(@sources)},
                      %{$self->get_sources(@staticlibs)} );
        my @parts = map { @{$_} } values %parts;

        my @generator =
            ( ( $self->{info}->{generate}->{$_} // [] ) -> [0] // () );
        my %generator_parts = %{$self->get_sources(@generator)};
        # if there are any generator parts, we ignore it, because that means
        # it's a compiled program and thus NOT part of the source that's
        # queried.
        @generator = () if %generator_parts;

        my @partial_result =
            ( ( map { @{$_} } values %parts ),
              ( grep { !defined($parts{$_}) } @sources, @generator ) );

        # Push conditionally, to avoid creating $result->{$_} with an empty
        # value
        push @{$result->{$_}}, @partial_result if @partial_result;
    }

    return $result;
}

=item B<get_config> I<LIST>

LIST is expected to be the collection of names of configuration data, such
as build_infos, sourcedir, ...

The returned result is a hash table reference, with each key being one of
these configuration data names, and its value being a reference to the value
corresponding to that name.

=cut

sub get_config {
    my $self = shift;

    return { map { $_ => $self->{config}->{$_} } @_ };
}

########
#
#  Helper functions
#

sub _check_accepted_options {
    my $opts = shift;           # HASH reference (hopefully)
    my %conds = @_;             # key => type

    my @messages;
    my %optnames = map { $_ => 1 } keys %$opts;
    foreach (keys %conds) {
        delete $optnames{$_};
    }
    push @messages, "Unknown options: " . join(', ', sort keys %optnames)
        if keys %optnames;
    foreach (sort keys %conds) {
        push @messages, "'$_' value not a $conds{$_} reference"
            if (defined $conds{$_} && defined $opts->{$_}
                && ref $opts->{$_} ne $conds{$_});
    }
    return @messages;
}

1;