mirror of
https://github.com/ruby/ruby.git
synced 2022-11-09 12:17:21 -05:00
9e5cd30bc1
https://github.com/ruby/dbm/pull/3 Co-authored-by: Yasuhiro KIMURA <> git-svn-id: svn+ssh://ci.ruby-lang.org/ruby/trunk@64210 b2dd03c8-39d4-4d8f-98ff-823fe69b080e
292 lines
8.1 KiB
Ruby
292 lines
8.1 KiB
Ruby
# frozen_string_literal: true
|
|
# configure option:
|
|
# --with-dbm-type=COMMA-SEPARATED-NDBM-TYPES
|
|
#
|
|
# ndbm type:
|
|
# libc ndbm compatible library in libc.
|
|
# db Berkeley DB (libdb)
|
|
# db2 Berkeley DB (libdb2)
|
|
# db1 Berkeley DB (libdb1)
|
|
# db6 Berkeley DB (libdb6)
|
|
# db5 Berkeley DB (libdb5)
|
|
# db4 Berkeley DB (libdb4)
|
|
# db3 Berkeley DB (libdb3)
|
|
# gdbm_compat GDBM since 1.8.1 (libgdbm_compat)
|
|
# gdbm GDBM until 1.8.0 (libgdbm)
|
|
# qdbm QDBM (libqdbm)
|
|
# ndbm Some legacy OS may have libndbm.
|
|
|
|
# :stopdoc:
|
|
require 'mkmf'
|
|
|
|
dir_config("dbm")
|
|
|
|
if dblib = with_config("dbm-type", nil)
|
|
dblib = dblib.split(/[ ,]+/)
|
|
else
|
|
dblib = %w(libc db db2 db1 db6 db5 db4 db3 gdbm_compat gdbm qdbm)
|
|
end
|
|
|
|
headers = {
|
|
"libc" => ["ndbm.h"], # 4.3BSD original ndbm, Berkeley DB 1 in 4.4BSD libc.
|
|
"db" => ["db.h"],
|
|
"db1" => ["db1/ndbm.h", "db1.h", "ndbm.h"],
|
|
"db2" => ["db2/db.h", "db2.h", "db.h"],
|
|
"db3" => ["db3/db.h", "db3.h", "db.h"],
|
|
"db4" => ["db4/db.h", "db4.h", "db.h"],
|
|
"db5" => ["db5/db.h", "db5.h", "db.h"],
|
|
"db6" => ["db6/db.h", "db6.h", "db.h"],
|
|
"gdbm_compat" => ["gdbm-ndbm.h", "gdbm/ndbm.h", "ndbm.h"], # GDBM since 1.8.1
|
|
"gdbm" => ["gdbm-ndbm.h", "gdbm/ndbm.h", "ndbm.h"], # GDBM until 1.8.0
|
|
"qdbm" => ["qdbm/relic.h", "relic.h"],
|
|
}
|
|
|
|
class << headers
|
|
attr_accessor :found
|
|
attr_accessor :defs
|
|
end
|
|
headers.found = []
|
|
headers.defs = nil
|
|
|
|
def headers.db_check(db, hdr)
|
|
old_libs = $libs.dup
|
|
old_defs = $defs.dup
|
|
result = db_check2(db, hdr)
|
|
if !result
|
|
$libs = old_libs
|
|
$defs = old_defs
|
|
end
|
|
result
|
|
end
|
|
|
|
def have_declared_libvar(var, headers = nil, opt = "", &b)
|
|
checking_for checking_message([*var].compact.join(' '), headers, opt) do
|
|
try_declared_libvar(var, headers, opt, &b)
|
|
end
|
|
end
|
|
|
|
def try_declared_libvar(var, headers = nil, opt = "", &b)
|
|
if try_link(<<"SRC", opt, &b)
|
|
#{cpp_include(headers)}
|
|
/*top*/
|
|
int main(int argc, char *argv[]) {
|
|
void *conftest_var = &#{var};
|
|
return 0;
|
|
}
|
|
SRC
|
|
$defs.push(format("-DHAVE_DECLARED_LIBVAR_%s", var.tr_cpp))
|
|
true
|
|
else
|
|
false
|
|
end
|
|
end
|
|
|
|
def have_undeclared_libvar(var, headers = nil, opt = "", &b)
|
|
checking_for checking_message([*var].compact.join(' '), headers, opt) do
|
|
try_undeclared_libvar(var, headers, opt, &b)
|
|
end
|
|
end
|
|
|
|
def try_undeclared_libvar(var, headers = nil, opt = "", &b)
|
|
var, type = *var
|
|
if try_link(<<"SRC", opt, &b)
|
|
#{cpp_include(headers)}
|
|
/*top*/
|
|
int main(int argc, char *argv[]) {
|
|
typedef #{type || 'int'} conftest_type;
|
|
extern conftest_type #{var};
|
|
conftest_type *conftest_var = &#{var};
|
|
return 0;
|
|
}
|
|
SRC
|
|
$defs.push(format("-DHAVE_UNDECLARED_LIBVAR_%s", var.tr_cpp))
|
|
true
|
|
else
|
|
false
|
|
end
|
|
end
|
|
|
|
def have_empty_macro_dbm_clearerr(headers = nil, opt = "", &b)
|
|
checking_for checking_message('empty macro of dbm_clearerr(foobarbaz)',
|
|
headers, opt) do
|
|
try_toplevel('dbm_clearerr(foobarbaz)', headers, opt, &b)
|
|
end
|
|
end
|
|
|
|
def try_toplevel(src, headers = nil, opt = "", &b)
|
|
if try_compile(<<"SRC", opt, &b)
|
|
#{cpp_include(headers)}
|
|
/*top*/
|
|
#{src}
|
|
SRC
|
|
true
|
|
else
|
|
false
|
|
end
|
|
end
|
|
|
|
|
|
def headers.db_check2(db, hdr)
|
|
$defs.push(%{-DRUBYDBM_DBM_HEADER='"#{hdr}"'})
|
|
$defs.push(%{-DRUBYDBM_DBM_TYPE='"#{db}"'})
|
|
|
|
hsearch = nil
|
|
|
|
case db
|
|
when /^db[2-6]?$/
|
|
hsearch = "-DDB_DBM_HSEARCH"
|
|
when "gdbm_compat"
|
|
have_library("gdbm") or return false
|
|
end
|
|
|
|
if !have_type("DBM", hdr, hsearch)
|
|
return false
|
|
end
|
|
|
|
# 'libc' means ndbm is provided by libc.
|
|
# 4.3BSD original ndbm is contained in libc.
|
|
# 4.4BSD (and its derivatives such as NetBSD) contains Berkeley DB 1 in libc.
|
|
if !(db == 'libc' ? have_func('dbm_open("", 0, 0)', hdr, hsearch) :
|
|
have_library(db, 'dbm_open("", 0, 0)', hdr, hsearch))
|
|
return false
|
|
end
|
|
|
|
# Skip a mismatch of Berkeley DB's ndbm.h and old GDBM library.
|
|
#
|
|
# dbm_clearerr() should be available for any ndbm implementation.
|
|
# It is available since the original (4.3BSD) ndbm and standardized by POSIX.
|
|
#
|
|
# However "can't resolve symbol 'dbm_clearerr'" problem may be caused by
|
|
# header/library mismatch: Berkeley DB ndbm.h and GDBM library until 1.8.3.
|
|
# GDBM (until 1.8.3) provides dbm_clearerr() as a empty macro in the header
|
|
# and the library don't provide dbm_clearerr().
|
|
# Berkeley DB provides dbm_clearerr() as a usual function.
|
|
# So Berkeley DB header with GDBM library causes the problem.
|
|
#
|
|
if !have_func('dbm_clearerr((DBM *)0)', hdr, hsearch)
|
|
return false
|
|
end
|
|
|
|
# Berkeley DB's ndbm.h (since 1.85 at least) defines DBM_SUFFIX.
|
|
# Note that _DB_H_ is not defined on Mac OS X because
|
|
# it uses Berkeley DB 1 but ndbm.h doesn't include db.h.
|
|
have_db_header = have_macro('DBM_SUFFIX', hdr, hsearch)
|
|
|
|
# Old GDBM's ndbm.h, until 1.8.3, defines dbm_clearerr as a macro which
|
|
# expands to no tokens.
|
|
have_gdbm_header1 = have_empty_macro_dbm_clearerr(hdr, hsearch)
|
|
|
|
# Recent GDBM's ndbm.h, since 1.9, includes gdbm.h and it defines _GDBM_H_.
|
|
# ndbm compatibility layer of GDBM is provided by libgdbm (until 1.8.0)
|
|
# and libgdbm_compat (since 1.8.1).
|
|
have_gdbm_header2 = have_macro('_GDBM_H_', hdr, hsearch)
|
|
|
|
# 4.3BSD's ndbm.h defines _DBM_IOERR.
|
|
# The original ndbm is provided by libc in 4.3BSD.
|
|
have_ndbm_header = have_macro('_DBM_IOERR', hdr, hsearch)
|
|
|
|
# GDBM provides ndbm functions in libgdbm_compat since GDBM 1.8.1.
|
|
# GDBM's ndbm.h defines _GDBM_H_ since GDBM 1.9.
|
|
# If _GDBM_H_ is defined, 'gdbm_compat' is required and reject 'gdbm'.
|
|
if have_gdbm_header2 && db == 'gdbm'
|
|
return false
|
|
end
|
|
|
|
if have_db_header
|
|
$defs.push('-DRUBYDBM_DB_HEADER')
|
|
end
|
|
|
|
have_gdbm_header = have_gdbm_header1 | have_gdbm_header2
|
|
if have_gdbm_header
|
|
$defs.push('-DRUBYDBM_GDBM_HEADER')
|
|
end
|
|
|
|
# ndbm.h is provided by the original (4.3BSD) ndbm,
|
|
# Berkeley DB 1 in libc of 4.4BSD and
|
|
# ndbm compatibility layer of GDBM.
|
|
# So, try to check header/library mismatch.
|
|
#
|
|
# Several (possibly historical) distributions provides libndbm.
|
|
# It may be Berkeley DB, GDBM or 4.3BSD ndbm.
|
|
# So mismatch check is not performed for that.
|
|
# Note that libndbm is searched only when --with-dbm-type=ndbm is
|
|
# given for configure.
|
|
#
|
|
if hdr == 'ndbm.h' && db != 'libc' && db != 'ndbm'
|
|
if /\Adb\d?\z/ !~ db && have_db_header
|
|
return false
|
|
end
|
|
|
|
if /\Agdbm/ !~ db && have_gdbm_header
|
|
return false
|
|
end
|
|
|
|
if have_ndbm_header
|
|
return false
|
|
end
|
|
end
|
|
|
|
# Berkeley DB
|
|
have_func('db_version((int *)0, (int *)0, (int *)0)', hdr, hsearch)
|
|
|
|
# GDBM
|
|
have_gdbm_version = have_declared_libvar("gdbm_version", hdr, hsearch)
|
|
# gdbm_version is available since very old version (GDBM 1.5 at least).
|
|
# However it is not declared by ndbm.h until GDBM 1.8.3.
|
|
# We can't include both ndbm.h and gdbm.h because they both define datum type.
|
|
# ndbm.h includes gdbm.h and gdbm_version is declared since GDBM 1.9.
|
|
have_gdbm_version |= have_undeclared_libvar(["gdbm_version", "char *"], hdr, hsearch)
|
|
|
|
# QDBM
|
|
have_var("dpversion", hdr, hsearch)
|
|
|
|
# detect mismatch between GDBM header and other library.
|
|
# If GDBM header is included, GDBM library should be linked.
|
|
if have_gdbm_header && !have_gdbm_version
|
|
return false
|
|
end
|
|
|
|
# DBC type is required to disable error messages by Berkeley DB 2 or later.
|
|
if have_db_header
|
|
have_type("DBC", hdr, hsearch)
|
|
end
|
|
|
|
if hsearch
|
|
$defs << hsearch
|
|
@defs = hsearch
|
|
end
|
|
$defs << '-DDBM_HDR="<'+hdr+'>"'
|
|
@found << hdr
|
|
|
|
puts "header: #{hdr}"
|
|
puts "library: #{db}"
|
|
|
|
true
|
|
end
|
|
|
|
if dblib.any? {|db| headers.fetch(db, ["ndbm.h"]).any? {|hdr| headers.db_check(db, hdr) } }
|
|
have_header("cdefs.h")
|
|
have_header("sys/cdefs.h")
|
|
have_func("dbm_pagfno((DBM *)0)", headers.found, headers.defs)
|
|
have_func("dbm_dirfno((DBM *)0)", headers.found, headers.defs)
|
|
convertible_int("datum.dsize", headers.found, headers.defs)
|
|
checking_for("sizeof(DBM) is available") {
|
|
if try_compile(<<SRC)
|
|
#ifdef HAVE_CDEFS_H
|
|
# include <cdefs.h>
|
|
#endif
|
|
#ifdef HAVE_SYS_CDEFS_H
|
|
# include <sys/cdefs.h>
|
|
#endif
|
|
#include DBM_HDR
|
|
|
|
const int sizeof_DBM = (int)sizeof(DBM);
|
|
SRC
|
|
$defs << '-DDBM_SIZEOF_DBM=sizeof(DBM)'
|
|
else
|
|
$defs << '-DDBM_SIZEOF_DBM=0'
|
|
end
|
|
}
|
|
create_makefile("dbm")
|
|
end
|
|
# :startdoc:
|