1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45 46 47 48 49 50 51 52 53 54 55 56 57 58 59 60 61 62 63 64 65 66 67 68 69 70 71 72 73 74 75 76 77 78 79 80 81 82 83 84 85 86 87 88 89 90 91 92 93 94 95 96 97 98 99 100 101 102 103 104 105 106 107 108 109 110 111 112 113 114 115 116 117 118 119 120 121 122 123
|
# -*- cperl -*-
# Copyright (c) 2004, 2025, Oracle and/or its affiliates.
#
# This program is free software; you can redistribute it and/or modify
# it under the terms of the GNU General Public License, version 2.0,
# as published by the Free Software Foundation.
#
# This program is designed to work with certain software (including
# but not limited to OpenSSL) that is licensed under separate terms,
# as designated in a particular file or component or in included license
# documentation. The authors of MySQL hereby grant you an additional
# permission to link the program and your derivative works with the
# separately licensed software that they have either included with
# the program or referenced in the documentation.
#
# This program is distributed in the hope that it will be useful,
# but WITHOUT ANY WARRANTY; without even the implied warranty of
# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
# GNU General Public License, version 2.0, for more details.
#
# You should have received a copy of the GNU General Public License
# along with this program; if not, write to the Free Software
# Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA
package mtr_unique;
use strict;
use base qw(Exporter);
our @EXPORT = qw(mtr_get_unique_id mtr_release_unique_id);
use Fcntl ':flock';
use My::Platform;
my @mtr_unique_fh;
my @mtr_unique_ids;
END {
mtr_release_unique_id();
}
# Get a unique, numerical ID in a specified range.
#
# If no unique ID within the specified parameters can be
# obtained, return undef.
sub mtr_get_unique_id($$$$$) {
my ($min, $max, $build_threads_per_thread, $min_exclude, $max_exclude)= @_;
if (scalar @mtr_unique_fh == $build_threads_per_thread) {
die "Can only get $build_threads_per_thread unique id(s) per process!";
}
my $build_thread = 0;
while ($build_thread < $build_threads_per_thread) {
for (my $id = $min ; $id <= $max ; $id++) {
# Determine if we hit the exclusion range, if so, skip it
if ($min_exclude <= $id && $id < $max_exclude) {
$id = $max_exclude;
next;
}
my $fh;
open($fh, ">$::build_thread_id_dir/$id");
chmod 0666, "$::build_thread_id_dir/$id";
# Try to lock the file exclusively. If lock succeeds, we're done.
if (flock($fh, LOCK_EX | LOCK_NB)) {
# Store file handle - we would need it to release the
# ID (i.e to unlock the file)
$mtr_unique_fh[$build_thread] = $fh;
$mtr_unique_ids[$build_thread] = "$::build_thread_id_dir/$id";
$build_thread = $build_thread + 1;
} else {
# Not able to get a lock on the file, start the search from
# next id(i.e min+1).
$min = $min + 1;
for (; $build_thread > 0 ; $build_thread--) {
if (defined $mtr_unique_fh[ $build_thread - 1 ]) {
close $mtr_unique_fh[ $build_thread - 1 ];
# This fails sometimes, but does not prevent MTR from working
# correctly. This applies to this instance, any instances running in
# parallel and future runs. We don't want to warn user about it.
unlink $mtr_unique_ids[ $build_thread - 1 ];
}
}
# Close the file opened in the current iteration.
close $fh;
last;
}
if ($build_thread == $build_threads_per_thread) {
open(FH, ">>", $::build_thread_id_file) or
die "Can't open file $::build_thread_id_file: $!";
for (my $i = 0 ; $i <= $#mtr_unique_ids ; $i++) {
# Write the build thread id file path to 'unique_ids.log' file
print FH $mtr_unique_ids[$i] . "\n";
}
close(FH);
return $id - $build_thread + 1;
}
}
return undef if ($min > $max);
}
return undef;
}
# Release a unique ID.
sub mtr_release_unique_id() {
for (my $i = 0 ; $i <= $#mtr_unique_fh ; $i++) {
if (defined $mtr_unique_fh[$i]) {
close $mtr_unique_fh[$i];
}
}
@mtr_unique_fh = ();
}
1;
|