summaryrefslogtreecommitdiffstats
path: root/Tests/RunCMake/FindPkgConfig/FindPkgConfig_IMPORTED_TARGET.cmake
blob: 04286ccb3f6e7caccf2ba50cc36998c2cfbacfb8 (plain)
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
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
cmake_minimum_required(VERSION 3.17)

project(FindPkgConfig_IMPORTED_TARGET C)

set(shared_lib_prefix "")
set(shared_lib_suffix ".lib")
set(static_lib_prefix "lib")
set(static_lib_suffix ".a")

set(CMAKE_SHARED_LIBRARY_PREFIX ${shared_lib_prefix})
set(CMAKE_SHARED_LIBRARY_SUFFIX ${shared_lib_suffix})
set(CMAKE_STATIC_LIBRARY_PREFIX ${static_lib_prefix})
set(CMAKE_STATIC_LIBRARY_SUFFIX ${static_lib_suffix})

find_package(PkgConfig REQUIRED)

# to test multiple variations, we must pick unique prefix names (same-named targets are cached for re-use)
set(prefix_uniquifiers 0 1)
# whether to apply STATIC_TARGET argument
set(static_target_args "" STATIC_TARGET)
foreach (prefix_uniquifier static_target_arg IN ZIP_LISTS prefix_uniquifiers static_target_args)
  set(prefix "NCURSES${prefix_uniquifier}")
  message(STATUS "static_target_arg: ${static_target_arg}")
  pkg_check_modules(${prefix} IMPORTED_TARGET QUIET ${static_target_arg} ncurses)

  message(STATUS "source: ${CMAKE_CURRENT_SOURCE_DIR} bin ${CMAKE_CURRENT_BINARY_DIR}")

  if (${prefix}_FOUND)
    set(tgt PkgConfig::${prefix})
    message(STATUS "Verifying target \"${tgt}\"")
    if (NOT TARGET ${tgt})
      message(FATAL_ERROR "FindPkgConfig found ncurses, but did not create an imported target for it")
    endif ()
    set(prop_found FALSE)
    foreach (prop IN ITEMS INTERFACE_INCLUDE_DIRECTORIES INTERFACE_LINK_LIBRARIES INTERFACE_COMPILE_OPTIONS)
      get_target_property(value ${tgt} ${prop})
      if (value)
        message(STATUS "Found property ${prop} on target: ${value}")
        set(prop_found TRUE)
      endif ()
    endforeach ()
    if (NOT prop_found)
      message(FATAL_ERROR "target ${tgt} found, but it has no properties")
    endif ()
  else ()
    message(STATUS "skipping test; ncurses not found")
  endif ()
endforeach ()

# Setup for the remaining package tests below
set(PKG_CONFIG_USE_CMAKE_PREFIX_PATH)
set(fakePkgDir ${CMAKE_CURRENT_BINARY_DIR}/pc-fakepackage)
foreach(i 1 2)
  set(pname cmakeinternalfakepackage${i})
  file(WRITE ${fakePkgDir}/lib/${static_lib_prefix}${pname}${static_lib_suffix} "")
  file(WRITE ${fakePkgDir}/lib/${shared_lib_prefix}${pname}${shared_lib_suffix} "")
  file(WRITE ${fakePkgDir}/lib/pkgconfig/${pname}.pc
"Name: CMakeInternalFakePackage${i}
Description: Dummy package (${i}) for FindPkgConfig IMPORTED_TARGET test
Version: 1.2.3
Libs: -l${pname} -l${pname}-doesnotexist
")
endforeach()

# Always find the .pc file in the calls further below so that we can test that
# the import target find_library() calls handle the NO...PATH options correctly
set(ENV{PKG_CONFIG_PATH} ${fakePkgDir}/lib/pkgconfig)

# find targets in subdir and check their visibility
add_subdirectory(target_subdir)

set(tgt PkgConfig::FakePackage1_dir)
if (TARGET ${tgt})
  message(FATAL_ERROR "imported target \"${tgt}\" is visible outside its directory")
endif()

set(tgt PkgConfig::FakePackage1_global)
if (NOT TARGET ${tgt})
  message(FATAL_ERROR "imported target \"${tgt}\" is not visible outside its directory")
endif()

# And now do the same for the NO_CMAKE_ENVIRONMENT_PATH - ENV{CMAKE_PREFIX_PATH}
# combination
unset(CMAKE_PREFIX_PATH)
unset(ENV{CMAKE_PREFIX_PATH})
set(ENV{CMAKE_PREFIX_PATH} ${fakePkgDir})


# to test multiple variations, we must pick unique prefix names (same-named targets are cached for re-use)
set(prefix_uniquifiers 0 1)
# whether to apply STATIC_TARGET argument
set(static_target_args "" STATIC_TARGET)
# whether target properties are populated from the unqualified (i.e. shared library) series of vars, or the STATIC_ series of vars
set(target_var_qualifiers "" STATIC_)
set(lib_types shared static)
foreach (prefix_uniquifier static_target_arg target_var_qualifier lib_type IN ZIP_LISTS prefix_uniquifiers static_target_args target_var_qualifiers lib_types)
  set(prefix "FakePackage2${prefix_uniquifier}")
  set(tgt "PkgConfig::${prefix}")
  pkg_check_modules(${prefix} REQUIRED QUIET IMPORTED_TARGET ${static_target_arg} cmakeinternalfakepackage2)

  message(STATUS "Verifying library path resolution for lib type \"${lib_type}\"")
  if (NOT TARGET ${tgt})
    message(FATAL_ERROR "No import target for fake package 2 with prefix path")
  endif()

  set(link_libraries_var ${prefix}_${target_var_qualifier}LINK_LIBRARIES)
  # check that 2 library entries exist
  list(LENGTH ${link_libraries_var} fp2_nlibs)
  if (NOT fp2_nlibs EQUAL 2)
    message(FATAL_ERROR "${link_libraries_var} has ${fp2_nlibs} entries but should have exactly 2")
  endif()

  set(lib_leafname ${${lib_type}_lib_prefix}cmakeinternalfakepackage2${${lib_type}_lib_suffix})
  message(STATUS "Expecting library leafname \"${lib_leafname}\"")
  # check that the full library path is also returned
  list(GET ${link_libraries_var} 0 fp2_lib0)
  if (NOT fp2_lib0 STREQUAL "${fakePkgDir}/lib/${lib_leafname}")
    message(FATAL_ERROR "${link_libraries_var} has bad content on first run: ${${link_libraries_var}}")
  endif()

  # check that the library that couldn't be found still shows up
  list(GET ${link_libraries_var} 1 fp2_lib1)
  if (NOT fp2_lib1 STREQUAL "cmakeinternalfakepackage2-doesnotexist")
    message(FATAL_ERROR "${link_libraries_var} has bad content on first run: ${${link_libraries_var}}")
  endif()

  # the information in *_LINK_LIBRARIES is not cached, so ensure is also is present on second run
  unset(${link_libraries_var})
  pkg_check_modules(${prefix} REQUIRED QUIET IMPORTED_TARGET ${static_target_arg} cmakeinternalfakepackage2)
  if (NOT ${link_libraries_var} STREQUAL "${fakePkgDir}/lib/${lib_leafname};cmakeinternalfakepackage2-doesnotexist")
    message(FATAL_ERROR "${link_libraries_var} has bad content on second run: ${${link_libraries_var}}")
  endif()
endforeach()

set(pname fakelinkoptionspackage)
file(WRITE ${fakePkgDir}/lib/pkgconfig/${pname}.pc
"Name: FakeLinkOptionsPackage
Description: Dummy package for FindPkgConfig IMPORTED_TARGET INTERFACE_LINK_OPTIONS test
Version: 1.2.3
Libs: -e dummy_main
Cflags: -I/special -isystem /other -isystem/more -DA-isystem/foo
")

set(expected_link_options -e dummy_main)
pkg_check_modules(FakeLinkOptionsPackage REQUIRED QUIET IMPORTED_TARGET fakelinkoptionspackage)

set(tgt PkgConfig::FakeLinkOptionsPackage)
message(STATUS "Verifying target \"${tgt}\"")
if (NOT TARGET ${tgt})
  message(FATAL_ERROR "No import target for fake link options package")
endif()
get_target_property(link_options ${tgt} INTERFACE_LINK_OPTIONS)
if (NOT link_options STREQUAL expected_link_options)
  message(FATAL_ERROR
    "Additional link options not present in INTERFACE_LINK_OPTIONS property\n"
    "expected: \"${expected_link_options}\", but got \"${link_options}\""
  )
endif()

get_target_property(inc_dirs ${tgt} INTERFACE_INCLUDE_DIRECTORIES)
set(expected_inc_dirs "/special" "/other" "/more")

if (NOT inc_dirs STREQUAL expected_inc_dirs)
  message(FATAL_ERROR
    "Additional include directories not correctly present in INTERFACE_INCLUDE_DIRECTORIES property\n"
    "expected: \"${expected_inc_dirs}\", got \"${inc_dirs}\""
  )
endif ()

get_target_property(c_opts ${tgt} INTERFACE_COMPILE_OPTIONS)
set(expected_c_opts "-DA-isystem/foo") # this is an invalid option, but a good testcase
if (NOT c_opts STREQUAL expected_c_opts)
    message(FATAL_ERROR
      "Additional compile options not present in INTERFACE_COMPILE_OPTIONS property\n"
      "expected: \"${expected_c_opts}\", got \"${c_opts}\""
    )
endif ()