diff options
Diffstat (limited to 'indra')
| -rw-r--r-- | indra/cmake/GetPrerequisites_2_8.cmake | 1572 | ||||
| -rw-r--r-- | indra/cmake/LLAddBuildTest.cmake | 552 | ||||
| -rw-r--r-- | indra/newview/llassetuploadresponders.cpp | 1 | ||||
| -rw-r--r-- | indra/newview/llfloaterwebcontent.cpp | 804 | ||||
| -rw-r--r-- | indra/newview/llfloaterwebcontent.h | 162 | ||||
| -rw-r--r-- | indra/newview/llimview.cpp | 26 | ||||
| -rw-r--r-- | indra/newview/llimview.h | 2 | ||||
| -rw-r--r-- | indra/newview/lllogchat.cpp | 18 | ||||
| -rw-r--r-- | indra/newview/skins/default/xui/en/floater_web_content.xml | 380 | ||||
| -rw-r--r-- | indra/newview/tests/llremoteparcelrequest_test.cpp | 268 | ||||
| -rw-r--r-- | indra/viewer_components/updater/tests/llupdaterservice_test.cpp | 400 | 
11 files changed, 2093 insertions, 2092 deletions
| diff --git a/indra/cmake/GetPrerequisites_2_8.cmake b/indra/cmake/GetPrerequisites_2_8.cmake index 5a24842c89..05ec1539ba 100644 --- a/indra/cmake/GetPrerequisites_2_8.cmake +++ b/indra/cmake/GetPrerequisites_2_8.cmake @@ -1,786 +1,786 @@ -# GetPrerequisites.cmake
 -#
 -# This script provides functions to list the .dll, .dylib or .so files that an
 -# executable or shared library file depends on. (Its prerequisites.)
 -#
 -# It uses various tools to obtain the list of required shared library files:
 -#   dumpbin (Windows)
 -#   ldd (Linux/Unix)
 -#   otool (Mac OSX)
 -#
 -# The following functions are provided by this script:
 -#   gp_append_unique
 -#   is_file_executable
 -#   gp_item_default_embedded_path
 -#     (projects can override with gp_item_default_embedded_path_override)
 -#   gp_resolve_item
 -#     (projects can override with gp_resolve_item_override)
 -#   gp_resolved_file_type
 -#   gp_file_type
 -#   get_prerequisites
 -#   list_prerequisites
 -#   list_prerequisites_by_glob
 -#
 -# Requires CMake 2.6 or greater because it uses function, break, return and
 -# PARENT_SCOPE.
 -
 -#=============================================================================
 -# Copyright 2008-2009 Kitware, Inc.
 -#
 -# Distributed under the OSI-approved BSD License (the "License");
 -# see accompanying file Copyright.txt for details.
 -#
 -# This software is distributed WITHOUT ANY WARRANTY; without even the
 -# implied warranty of MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.
 -# See the License for more information.
 -#=============================================================================
 -# (To distributed this file outside of CMake, substitute the full
 -#  License text for the above reference.)
 -
 -# gp_append_unique list_var value
 -#
 -# Append value to the list variable ${list_var} only if the value is not
 -# already in the list.
 -#
 -function(gp_append_unique list_var value)
 -  set(contains 0)
 -
 -  foreach(item ${${list_var}})
 -    if("${item}" STREQUAL "${value}")
 -      set(contains 1)
 -      break()
 -    endif("${item}" STREQUAL "${value}")
 -  endforeach(item)
 -
 -  if(NOT contains)
 -    set(${list_var} ${${list_var}} "${value}" PARENT_SCOPE)
 -  endif(NOT contains)
 -endfunction(gp_append_unique)
 -
 -
 -# is_file_executable file result_var
 -#
 -# Return 1 in ${result_var} if ${file} is a binary executable.
 -#
 -# Return 0 in ${result_var} otherwise.
 -#
 -function(is_file_executable file result_var)
 -  #
 -  # A file is not executable until proven otherwise:
 -  #
 -  set(${result_var} 0 PARENT_SCOPE)
 -
 -  get_filename_component(file_full "${file}" ABSOLUTE)
 -  string(TOLOWER "${file_full}" file_full_lower)
 -
 -  # If file name ends in .exe on Windows, *assume* executable:
 -  #
 -  if(WIN32)
 -    if("${file_full_lower}" MATCHES "\\.exe$")
 -      set(${result_var} 1 PARENT_SCOPE)
 -      return()
 -    endif("${file_full_lower}" MATCHES "\\.exe$")
 -
 -    # A clause could be added here that uses output or return value of dumpbin
 -    # to determine ${result_var}. In 99%+? practical cases, the exe name
 -    # match will be sufficient...
 -    #
 -  endif(WIN32)
 -
 -  # Use the information returned from the Unix shell command "file" to
 -  # determine if ${file_full} should be considered an executable file...
 -  #
 -  # If the file command's output contains "executable" and does *not* contain
 -  # "text" then it is likely an executable suitable for prerequisite analysis
 -  # via the get_prerequisites macro.
 -  #
 -  if(UNIX)
 -    if(NOT file_cmd)
 -      find_program(file_cmd "file")
 -    endif(NOT file_cmd)
 -
 -    if(file_cmd)
 -      execute_process(COMMAND "${file_cmd}" "${file_full}"
 -        OUTPUT_VARIABLE file_ov
 -        OUTPUT_STRIP_TRAILING_WHITESPACE
 -        )
 -
 -      # Replace the name of the file in the output with a placeholder token
 -      # (the string " _file_full_ ") so that just in case the path name of
 -      # the file contains the word "text" or "executable" we are not fooled
 -      # into thinking "the wrong thing" because the file name matches the
 -      # other 'file' command output we are looking for...
 -      #
 -      string(REPLACE "${file_full}" " _file_full_ " file_ov "${file_ov}")
 -      string(TOLOWER "${file_ov}" file_ov)
 -
 -      #message(STATUS "file_ov='${file_ov}'")
 -      if("${file_ov}" MATCHES "executable")
 -        #message(STATUS "executable!")
 -        if("${file_ov}" MATCHES "text")
 -          #message(STATUS "but text, so *not* a binary executable!")
 -        else("${file_ov}" MATCHES "text")
 -          set(${result_var} 1 PARENT_SCOPE)
 -          return()
 -        endif("${file_ov}" MATCHES "text")
 -      endif("${file_ov}" MATCHES "executable")
 -    else(file_cmd)
 -      message(STATUS "warning: No 'file' command, skipping execute_process...")
 -    endif(file_cmd)
 -  endif(UNIX)
 -endfunction(is_file_executable)
 -
 -
 -# gp_item_default_embedded_path item default_embedded_path_var
 -#
 -# Return the path that others should refer to the item by when the item
 -# is embedded inside a bundle.
 -#
 -# Override on a per-project basis by providing a project-specific
 -# gp_item_default_embedded_path_override function.
 -#
 -function(gp_item_default_embedded_path item default_embedded_path_var)
 -
 -  # On Windows and Linux, "embed" prerequisites in the same directory
 -  # as the executable by default:
 -  #
 -  set(path "@executable_path")
 -  set(overridden 0)
 -
 -  # On the Mac, relative to the executable depending on the type
 -  # of the thing we are embedding:
 -  #
 -  if(APPLE)
 -    #
 -    # The assumption here is that all executables in the bundle will be
 -    # in same-level-directories inside the bundle. The parent directory
 -    # of an executable inside the bundle should be MacOS or a sibling of
 -    # MacOS and all embedded paths returned from here will begin with
 -    # "@executable_path/../" and will work from all executables in all
 -    # such same-level-directories inside the bundle.
 -    #
 -
 -    # By default, embed things right next to the main bundle executable:
 -    #
 -    set(path "@executable_path/../../Contents/MacOS")
 -
 -    # Embed .dylibs right next to the main bundle executable:
 -    #
 -    if(item MATCHES "\\.dylib$")
 -      set(path "@executable_path/../MacOS")
 -      set(overridden 1)
 -    endif(item MATCHES "\\.dylib$")
 -
 -    # Embed frameworks in the embedded "Frameworks" directory (sibling of MacOS):
 -    #
 -    if(NOT overridden)
 -      if(item MATCHES "[^/]+\\.framework/")
 -        set(path "@executable_path/../Frameworks")
 -        set(overridden 1)
 -      endif(item MATCHES "[^/]+\\.framework/")
 -    endif(NOT overridden)
 -  endif()
 -
 -  # Provide a hook so that projects can override the default embedded location
 -  # of any given library by whatever logic they choose:
 -  #
 -  if(COMMAND gp_item_default_embedded_path_override)
 -    gp_item_default_embedded_path_override("${item}" path)
 -  endif(COMMAND gp_item_default_embedded_path_override)
 -
 -  set(${default_embedded_path_var} "${path}" PARENT_SCOPE)
 -endfunction(gp_item_default_embedded_path)
 -
 -
 -# gp_resolve_item context item exepath dirs resolved_item_var
 -#
 -# Resolve an item into an existing full path file.
 -#
 -# Override on a per-project basis by providing a project-specific
 -# gp_resolve_item_override function.
 -#
 -function(gp_resolve_item context item exepath dirs resolved_item_var)
 -  set(resolved 0)
 -  set(resolved_item "${item}")
 -
 -  # Is it already resolved?
 -  #
 -  if(EXISTS "${resolved_item}")
 -    set(resolved 1)
 -  endif(EXISTS "${resolved_item}")
 -
 -  if(NOT resolved)
 -    if(item MATCHES "@executable_path")
 -      #
 -      # @executable_path references are assumed relative to exepath
 -      #
 -      string(REPLACE "@executable_path" "${exepath}" ri "${item}")
 -      get_filename_component(ri "${ri}" ABSOLUTE)
 -
 -      if(EXISTS "${ri}")
 -        #message(STATUS "info: embedded item exists (${ri})")
 -        set(resolved 1)
 -        set(resolved_item "${ri}")
 -      else(EXISTS "${ri}")
 -        message(STATUS "warning: embedded item does not exist '${ri}'")
 -      endif(EXISTS "${ri}")
 -    endif(item MATCHES "@executable_path")
 -  endif(NOT resolved)
 -
 -  if(NOT resolved)
 -    if(item MATCHES "@loader_path")
 -      #
 -      # @loader_path references are assumed relative to the
 -      # PATH of the given "context" (presumably another library)
 -      #
 -      get_filename_component(contextpath "${context}" PATH)
 -      string(REPLACE "@loader_path" "${contextpath}" ri "${item}")
 -      get_filename_component(ri "${ri}" ABSOLUTE)
 -
 -      if(EXISTS "${ri}")
 -        #message(STATUS "info: embedded item exists (${ri})")
 -        set(resolved 1)
 -        set(resolved_item "${ri}")
 -      else(EXISTS "${ri}")
 -        message(STATUS "warning: embedded item does not exist '${ri}'")
 -      endif(EXISTS "${ri}")
 -    endif(item MATCHES "@loader_path")
 -  endif(NOT resolved)
 -
 -  if(NOT resolved)
 -    set(ri "ri-NOTFOUND")
 -    find_file(ri "${item}" ${exepath} ${dirs} NO_DEFAULT_PATH)
 -    find_file(ri "${item}" ${exepath} ${dirs} /usr/lib)
 -    if(ri)
 -      #message(STATUS "info: 'find_file' in exepath/dirs (${ri})")
 -      set(resolved 1)
 -      set(resolved_item "${ri}")
 -      set(ri "ri-NOTFOUND")
 -    endif(ri)
 -  endif(NOT resolved)
 -
 -  if(NOT resolved)
 -    if(item MATCHES "[^/]+\\.framework/")
 -      set(fw "fw-NOTFOUND")
 -      find_file(fw "${item}"
 -        "~/Library/Frameworks"
 -        "/Library/Frameworks"
 -        "/System/Library/Frameworks"
 -      )
 -      if(fw)
 -        #message(STATUS "info: 'find_file' found framework (${fw})")
 -        set(resolved 1)
 -        set(resolved_item "${fw}")
 -        set(fw "fw-NOTFOUND")
 -      endif(fw)
 -    endif(item MATCHES "[^/]+\\.framework/")
 -  endif(NOT resolved)
 -
 -  # Using find_program on Windows will find dll files that are in the PATH.
 -  # (Converting simple file names into full path names if found.)
 -  #
 -  if(WIN32)
 -  if(NOT resolved)
 -    set(ri "ri-NOTFOUND")
 -    find_program(ri "${item}" PATHS "${exepath};${dirs}" NO_DEFAULT_PATH)
 -    find_program(ri "${item}" PATHS "${exepath};${dirs}")
 -    if(ri)
 -      #message(STATUS "info: 'find_program' in exepath/dirs (${ri})")
 -      set(resolved 1)
 -      set(resolved_item "${ri}")
 -      set(ri "ri-NOTFOUND")
 -    endif(ri)
 -  endif(NOT resolved)
 -  endif(WIN32)
 -
 -  # Provide a hook so that projects can override item resolution
 -  # by whatever logic they choose:
 -  #
 -  if(COMMAND gp_resolve_item_override)
 -    gp_resolve_item_override("${context}" "${item}" "${exepath}" "${dirs}" resolved_item resolved)
 -  endif(COMMAND gp_resolve_item_override)
 -
 -  if(NOT resolved)
 -    message(STATUS "
 -warning: cannot resolve item '${item}'
 -
 -  possible problems:
 -    need more directories?
 -    need to use InstallRequiredSystemLibraries?
 -    run in install tree instead of build tree?
 -")
 -#    message(STATUS "
 -#******************************************************************************
 -#warning: cannot resolve item '${item}'
 -#
 -#  possible problems:
 -#    need more directories?
 -#    need to use InstallRequiredSystemLibraries?
 -#    run in install tree instead of build tree?
 -#
 -#    context='${context}'
 -#    item='${item}'
 -#    exepath='${exepath}'
 -#    dirs='${dirs}'
 -#    resolved_item_var='${resolved_item_var}'
 -#******************************************************************************
 -#")
 -  endif(NOT resolved)
 -
 -  set(${resolved_item_var} "${resolved_item}" PARENT_SCOPE)
 -endfunction(gp_resolve_item)
 -
 -
 -# gp_resolved_file_type original_file file exepath dirs type_var
 -#
 -# Return the type of ${file} with respect to ${original_file}. String
 -# describing type of prerequisite is returned in variable named ${type_var}.
 -#
 -# Use ${exepath} and ${dirs} if necessary to resolve non-absolute ${file}
 -# values -- but only for non-embedded items.
 -#
 -# Possible types are:
 -#   system
 -#   local
 -#   embedded
 -#   other
 -#
 -function(gp_resolved_file_type original_file file exepath dirs type_var)
 -  #message(STATUS "**")
 -
 -  if(NOT IS_ABSOLUTE "${original_file}")
 -    message(STATUS "warning: gp_resolved_file_type expects absolute full path for first arg original_file")
 -  endif()
 -
 -  set(is_embedded 0)
 -  set(is_local 0)
 -  set(is_system 0)
 -
 -  set(resolved_file "${file}")
 -
 -  if("${file}" MATCHES "^@(executable|loader)_path")
 -    set(is_embedded 1)
 -  endif()
 -
 -  if(NOT is_embedded)
 -    if(NOT IS_ABSOLUTE "${file}")
 -      gp_resolve_item("${original_file}" "${file}" "${exepath}" "${dirs}" resolved_file)
 -    endif()
 -
 -    string(TOLOWER "${original_file}" original_lower)
 -    string(TOLOWER "${resolved_file}" lower)
 -
 -    if(UNIX)
 -      if(resolved_file MATCHES "^(/lib/|/lib32/|/lib64/|/usr/lib/|/usr/lib32/|/usr/lib64/|/usr/X11R6/)")
 -        set(is_system 1)
 -      endif()
 -    endif()
 -
 -    if(APPLE)
 -      if(resolved_file MATCHES "^(/System/Library/|/usr/lib/)")
 -        set(is_system 1)
 -      endif()
 -    endif()
 -
 -    if(WIN32)
 -      string(TOLOWER "$ENV{SystemRoot}" sysroot)
 -      string(REGEX REPLACE "\\\\" "/" sysroot "${sysroot}")
 -
 -      string(TOLOWER "$ENV{windir}" windir)
 -      string(REGEX REPLACE "\\\\" "/" windir "${windir}")
 -
 -      if(lower MATCHES "^(${sysroot}/system|${windir}/system|${sysroot}/syswow|${windir}/syswow|(.*/)*msvc[^/]+dll)")
 -        set(is_system 1)
 -      endif()
 -    endif()
 -
 -    if(NOT is_system)
 -      get_filename_component(original_path "${original_lower}" PATH)
 -      get_filename_component(path "${lower}" PATH)
 -      if("${original_path}" STREQUAL "${path}")
 -        set(is_local 1)
 -      endif()
 -    endif()
 -  endif()
 -
 -  # Return type string based on computed booleans:
 -  #
 -  set(type "other")
 -
 -  if(is_system)
 -    set(type "system")
 -  elseif(is_embedded)
 -    set(type "embedded")
 -  elseif(is_local)
 -    set(type "local")
 -  endif()
 -
 -  #message(STATUS "gp_resolved_file_type: '${file}' '${resolved_file}'")
 -  #message(STATUS "                type: '${type}'")
 -
 -  if(NOT is_embedded)
 -    if(NOT IS_ABSOLUTE "${resolved_file}")
 -      if(lower MATCHES "^msvc[^/]+dll" AND is_system)
 -        message(STATUS "info: non-absolute msvc file '${file}' returning type '${type}'")
 -      else()
 -        message(STATUS "warning: gp_resolved_file_type non-absolute file '${file}' returning type '${type}' -- possibly incorrect")
 -      endif()
 -    endif()
 -  endif()
 -
 -  set(${type_var} "${type}" PARENT_SCOPE)
 -
 -  #message(STATUS "**")
 -endfunction()
 -
 -
 -# gp_file_type original_file file type_var
 -#
 -# Return the type of ${file} with respect to ${original_file}. String
 -# describing type of prerequisite is returned in variable named ${type_var}.
 -#
 -# Possible types are:
 -#   system
 -#   local
 -#   embedded
 -#   other
 -#
 -function(gp_file_type original_file file type_var)
 -  if(NOT IS_ABSOLUTE "${original_file}")
 -    message(STATUS "warning: gp_file_type expects absolute full path for first arg original_file")
 -  endif()
 -
 -  get_filename_component(exepath "${original_file}" PATH)
 -
 -  set(type "")
 -  gp_resolved_file_type("${original_file}" "${file}" "${exepath}" "" type)
 -
 -  set(${type_var} "${type}" PARENT_SCOPE)
 -endfunction(gp_file_type)
 -
 -
 -# get_prerequisites target prerequisites_var exclude_system recurse dirs
 -#
 -# Get the list of shared library files required by ${target}. The list in
 -# the variable named ${prerequisites_var} should be empty on first entry to
 -# this function. On exit, ${prerequisites_var} will contain the list of
 -# required shared library files.
 -#
 -#  target is the full path to an executable file
 -#
 -#  prerequisites_var is the name of a CMake variable to contain the results
 -#
 -#  exclude_system is 0 or 1: 0 to include "system" prerequisites , 1 to
 -#   exclude them
 -#
 -#  recurse is 0 or 1: 0 for direct prerequisites only, 1 for all prerequisites
 -#   recursively
 -#
 -#  exepath is the path to the top level executable used for @executable_path
 -#   replacment on the Mac
 -#
 -#  dirs is a list of paths where libraries might be found: these paths are
 -#   searched first when a target without any path info is given. Then standard
 -#   system locations are also searched: PATH, Framework locations, /usr/lib...
 -#
 -function(get_prerequisites target prerequisites_var exclude_system recurse exepath dirs)
 -  set(verbose 0)
 -  set(eol_char "E")
 -
 -  if(NOT IS_ABSOLUTE "${target}")
 -    message("warning: target '${target}' is not absolute...")
 -  endif(NOT IS_ABSOLUTE "${target}")
 -
 -  if(NOT EXISTS "${target}")
 -    message("warning: target '${target}' does not exist...")
 -  endif(NOT EXISTS "${target}")
 -
 -  # <setup-gp_tool-vars>
 -  #
 -  # Try to choose the right tool by default. Caller can set gp_tool prior to
 -  # calling this function to force using a different tool.
 -  #
 -  if("${gp_tool}" STREQUAL "")
 -    set(gp_tool "ldd")
 -    if(APPLE)
 -      set(gp_tool "otool")
 -    endif(APPLE)
 -    if(WIN32)
 -      set(gp_tool "dumpbin")
 -    endif(WIN32)
 -  endif("${gp_tool}" STREQUAL "")
 -
 -  set(gp_tool_known 0)
 -
 -  if("${gp_tool}" STREQUAL "ldd")
 -    set(gp_cmd_args "")
 -    set(gp_regex "^[\t ]*[^\t ]+ => ([^\t ]+).*${eol_char}$")
 -    set(gp_regex_cmp_count 1)
 -    set(gp_tool_known 1)
 -  endif("${gp_tool}" STREQUAL "ldd")
 -
 -  if("${gp_tool}" STREQUAL "otool")
 -    set(gp_cmd_args "-L")
 -    set(gp_regex "^\t([^\t]+) \\(compatibility version ([0-9]+.[0-9]+.[0-9]+), current version ([0-9]+.[0-9]+.[0-9]+)\\)${eol_char}$")
 -    set(gp_regex_cmp_count 3)
 -    set(gp_tool_known 1)
 -  endif("${gp_tool}" STREQUAL "otool")
 -
 -  if("${gp_tool}" STREQUAL "dumpbin")
 -    set(gp_cmd_args "/dependents")
 -    set(gp_regex "^    ([^ ].*[Dd][Ll][Ll])${eol_char}$")
 -    set(gp_regex_cmp_count 1)
 -    set(gp_tool_known 1)
 -    set(ENV{VS_UNICODE_OUTPUT} "") # Block extra output from inside VS IDE.
 -  endif("${gp_tool}" STREQUAL "dumpbin")
 -
 -  if(NOT gp_tool_known)
 -    message(STATUS "warning: gp_tool='${gp_tool}' is an unknown tool...")
 -    message(STATUS "CMake function get_prerequisites needs more code to handle '${gp_tool}'")
 -    message(STATUS "Valid gp_tool values are dumpbin, ldd and otool.")
 -    return()
 -  endif(NOT gp_tool_known)
 -
 -  set(gp_cmd_paths ${gp_cmd_paths}
 -    "C:/Program Files/Microsoft Visual Studio 9.0/VC/bin"
 -    "C:/Program Files (x86)/Microsoft Visual Studio 9.0/VC/bin"
 -    "C:/Program Files/Microsoft Visual Studio 8/VC/BIN"
 -    "C:/Program Files (x86)/Microsoft Visual Studio 8/VC/BIN"
 -    "C:/Program Files/Microsoft Visual Studio .NET 2003/VC7/BIN"
 -    "C:/Program Files (x86)/Microsoft Visual Studio .NET 2003/VC7/BIN"
 -    "/usr/local/bin"
 -    "/usr/bin"
 -    )
 -
 -  find_program(gp_cmd ${gp_tool} PATHS ${gp_cmd_paths})
 -
 -  if(NOT gp_cmd)
 -    message(STATUS "warning: could not find '${gp_tool}' - cannot analyze prerequisites...")
 -    return()
 -  endif(NOT gp_cmd)
 -
 -  if("${gp_tool}" STREQUAL "dumpbin")
 -    # When running dumpbin, it also needs the "Common7/IDE" directory in the
 -    # PATH. It will already be in the PATH if being run from a Visual Studio
 -    # command prompt. Add it to the PATH here in case we are running from a
 -    # different command prompt.
 -    #
 -    get_filename_component(gp_cmd_dir "${gp_cmd}" PATH)
 -    get_filename_component(gp_cmd_dlls_dir "${gp_cmd_dir}/../../Common7/IDE" ABSOLUTE)
 -    if(EXISTS "${gp_cmd_dlls_dir}")
 -      # only add to the path if it is not already in the path
 -      if(NOT "$ENV{PATH}" MATCHES "${gp_cmd_dlls_dir}")
 -        set(ENV{PATH} "$ENV{PATH};${gp_cmd_dlls_dir}")
 -      endif(NOT "$ENV{PATH}" MATCHES "${gp_cmd_dlls_dir}")
 -    endif(EXISTS "${gp_cmd_dlls_dir}")
 -  endif("${gp_tool}" STREQUAL "dumpbin")
 -  #
 -  # </setup-gp_tool-vars>
 -
 -  if("${gp_tool}" STREQUAL "ldd")
 -    set(old_ld_env "$ENV{LD_LIBRARY_PATH}")
 -    foreach(dir ${exepath} ${dirs})
 -      set(ENV{LD_LIBRARY_PATH} "${dir}:$ENV{LD_LIBRARY_PATH}")
 -    endforeach(dir)
 -  endif("${gp_tool}" STREQUAL "ldd")
 -
 -
 -  # Track new prerequisites at each new level of recursion. Start with an
 -  # empty list at each level:
 -  #
 -  set(unseen_prereqs)
 -
 -  # Run gp_cmd on the target:
 -  #
 -  execute_process(
 -    COMMAND ${gp_cmd} ${gp_cmd_args} ${target}
 -    OUTPUT_VARIABLE gp_cmd_ov
 -    )
 -
 -  if("${gp_tool}" STREQUAL "ldd")
 -    set(ENV{LD_LIBRARY_PATH} "${old_ld_env}")
 -  endif("${gp_tool}" STREQUAL "ldd")
 -
 -  if(verbose)
 -    message(STATUS "<RawOutput cmd='${gp_cmd} ${gp_cmd_args} ${target}'>")
 -    message(STATUS "gp_cmd_ov='${gp_cmd_ov}'")
 -    message(STATUS "</RawOutput>")
 -  endif(verbose)
 -
 -  get_filename_component(target_dir "${target}" PATH)
 -
 -  # Convert to a list of lines:
 -  #
 -  string(REGEX REPLACE ";" "\\\\;" candidates "${gp_cmd_ov}")
 -  string(REGEX REPLACE "\n" "${eol_char};" candidates "${candidates}")
 -
 -  # Analyze each line for file names that match the regular expression:
 -  #
 -  foreach(candidate ${candidates})
 -  if("${candidate}" MATCHES "${gp_regex}")
 -    # Extract information from each candidate:
 -    string(REGEX REPLACE "${gp_regex}" "\\1" raw_item "${candidate}")
 -
 -    if(gp_regex_cmp_count GREATER 1)
 -      string(REGEX REPLACE "${gp_regex}" "\\2" raw_compat_version "${candidate}")
 -      string(REGEX REPLACE "^([0-9]+)\\.([0-9]+)\\.([0-9]+)$" "\\1" compat_major_version "${raw_compat_version}")
 -      string(REGEX REPLACE "^([0-9]+)\\.([0-9]+)\\.([0-9]+)$" "\\2" compat_minor_version "${raw_compat_version}")
 -      string(REGEX REPLACE "^([0-9]+)\\.([0-9]+)\\.([0-9]+)$" "\\3" compat_patch_version "${raw_compat_version}")
 -    endif(gp_regex_cmp_count GREATER 1)
 -
 -    if(gp_regex_cmp_count GREATER 2)
 -      string(REGEX REPLACE "${gp_regex}" "\\3" raw_current_version "${candidate}")
 -      string(REGEX REPLACE "^([0-9]+)\\.([0-9]+)\\.([0-9]+)$" "\\1" current_major_version "${raw_current_version}")
 -      string(REGEX REPLACE "^([0-9]+)\\.([0-9]+)\\.([0-9]+)$" "\\2" current_minor_version "${raw_current_version}")
 -      string(REGEX REPLACE "^([0-9]+)\\.([0-9]+)\\.([0-9]+)$" "\\3" current_patch_version "${raw_current_version}")
 -    endif(gp_regex_cmp_count GREATER 2)
 -
 -    # Use the raw_item as the list entries returned by this function. Use the
 -    # gp_resolve_item function to resolve it to an actual full path file if
 -    # necessary.
 -    #
 -    set(item "${raw_item}")
 -
 -    # Add each item unless it is excluded:
 -    #
 -    set(add_item 1)
 -
 -    if(${exclude_system})
 -      set(type "")
 -      gp_resolved_file_type("${target}" "${item}" "${exepath}" "${dirs}" type)
 -      if("${type}" STREQUAL "system")
 -        set(add_item 0)
 -      endif("${type}" STREQUAL "system")
 -    endif(${exclude_system})
 -
 -    if(add_item)
 -      list(LENGTH ${prerequisites_var} list_length_before_append)
 -      gp_append_unique(${prerequisites_var} "${item}")
 -      list(LENGTH ${prerequisites_var} list_length_after_append)
 -
 -      if(${recurse})
 -        # If item was really added, this is the first time we have seen it.
 -        # Add it to unseen_prereqs so that we can recursively add *its*
 -        # prerequisites...
 -        #
 -        # But first: resolve its name to an absolute full path name such
 -        # that the analysis tools can simply accept it as input.
 -        #
 -        if(NOT list_length_before_append EQUAL list_length_after_append)
 -          gp_resolve_item("${target}" "${item}" "${exepath}" "${dirs}" resolved_item)
 -          set(unseen_prereqs ${unseen_prereqs} "${resolved_item}")
 -        endif(NOT list_length_before_append EQUAL list_length_after_append)
 -      endif(${recurse})
 -    endif(add_item)
 -  else("${candidate}" MATCHES "${gp_regex}")
 -    if(verbose)
 -      message(STATUS "ignoring non-matching line: '${candidate}'")
 -    endif(verbose)
 -  endif("${candidate}" MATCHES "${gp_regex}")
 -  endforeach(candidate)
 -
 -  list(LENGTH ${prerequisites_var} prerequisites_var_length)
 -  if(prerequisites_var_length GREATER 0)
 -    list(SORT ${prerequisites_var})
 -  endif(prerequisites_var_length GREATER 0)
 -  if(${recurse})
 -    set(more_inputs ${unseen_prereqs})
 -    foreach(input ${more_inputs})
 -      get_prerequisites("${input}" ${prerequisites_var} ${exclude_system} ${recurse} "${exepath}" "${dirs}")
 -    endforeach(input)
 -  endif(${recurse})
 -
 -  set(${prerequisites_var} ${${prerequisites_var}} PARENT_SCOPE)
 -endfunction(get_prerequisites)
 -
 -
 -# list_prerequisites target all exclude_system verbose
 -#
 -#  ARGV0 (target) is the full path to an executable file
 -#
 -#  optional ARGV1 (all) is 0 or 1: 0 for direct prerequisites only,
 -#   1 for all prerequisites recursively
 -#
 -#  optional ARGV2 (exclude_system) is 0 or 1: 0 to include "system"
 -#   prerequisites , 1 to exclude them
 -#
 -#  optional ARGV3 (verbose) is 0 or 1: 0 to print only full path
 -#   names of prerequisites, 1 to print extra information
 -#
 -function(list_prerequisites target)
 -  if("${ARGV1}" STREQUAL "")
 -    set(all 1)
 -  else("${ARGV1}" STREQUAL "")
 -    set(all "${ARGV1}")
 -  endif("${ARGV1}" STREQUAL "")
 -
 -  if("${ARGV2}" STREQUAL "")
 -    set(exclude_system 0)
 -  else("${ARGV2}" STREQUAL "")
 -    set(exclude_system "${ARGV2}")
 -  endif("${ARGV2}" STREQUAL "")
 -
 -  if("${ARGV3}" STREQUAL "")
 -    set(verbose 0)
 -  else("${ARGV3}" STREQUAL "")
 -    set(verbose "${ARGV3}")
 -  endif("${ARGV3}" STREQUAL "")
 -
 -  set(count 0)
 -  set(count_str "")
 -  set(print_count "${verbose}")
 -  set(print_prerequisite_type "${verbose}")
 -  set(print_target "${verbose}")
 -  set(type_str "")
 -
 -  get_filename_component(exepath "${target}" PATH)
 -
 -  set(prereqs "")
 -  get_prerequisites("${target}" prereqs ${exclude_system} ${all} "${exepath}" "")
 -
 -  if(print_target)
 -    message(STATUS "File '${target}' depends on:")
 -  endif(print_target)
 -
 -  foreach(d ${prereqs})
 -    math(EXPR count "${count} + 1")
 -
 -    if(print_count)
 -      set(count_str "${count}. ")
 -    endif(print_count)
 -
 -    if(print_prerequisite_type)
 -      gp_file_type("${target}" "${d}" type)
 -      set(type_str " (${type})")
 -    endif(print_prerequisite_type)
 -
 -    message(STATUS "${count_str}${d}${type_str}")
 -  endforeach(d)
 -endfunction(list_prerequisites)
 -
 -
 -# list_prerequisites_by_glob glob_arg glob_exp
 -#
 -#  glob_arg is GLOB or GLOB_RECURSE
 -#
 -#  glob_exp is a globbing expression used with "file(GLOB" to retrieve a list
 -#   of matching files. If a matching file is executable, its prerequisites are
 -#   listed.
 -#
 -# Any additional (optional) arguments provided are passed along as the
 -# optional arguments to the list_prerequisites calls.
 -#
 -function(list_prerequisites_by_glob glob_arg glob_exp)
 -  message(STATUS "=============================================================================")
 -  message(STATUS "List prerequisites of executables matching ${glob_arg} '${glob_exp}'")
 -  message(STATUS "")
 -  file(${glob_arg} file_list ${glob_exp})
 -  foreach(f ${file_list})
 -    is_file_executable("${f}" is_f_executable)
 -    if(is_f_executable)
 -      message(STATUS "=============================================================================")
 -      list_prerequisites("${f}" ${ARGN})
 -      message(STATUS "")
 -    endif(is_f_executable)
 -  endforeach(f)
 -endfunction(list_prerequisites_by_glob)
 +# GetPrerequisites.cmake +# +# This script provides functions to list the .dll, .dylib or .so files that an +# executable or shared library file depends on. (Its prerequisites.) +# +# It uses various tools to obtain the list of required shared library files: +#   dumpbin (Windows) +#   ldd (Linux/Unix) +#   otool (Mac OSX) +# +# The following functions are provided by this script: +#   gp_append_unique +#   is_file_executable +#   gp_item_default_embedded_path +#     (projects can override with gp_item_default_embedded_path_override) +#   gp_resolve_item +#     (projects can override with gp_resolve_item_override) +#   gp_resolved_file_type +#   gp_file_type +#   get_prerequisites +#   list_prerequisites +#   list_prerequisites_by_glob +# +# Requires CMake 2.6 or greater because it uses function, break, return and +# PARENT_SCOPE. + +#============================================================================= +# Copyright 2008-2009 Kitware, Inc. +# +# Distributed under the OSI-approved BSD License (the "License"); +# see accompanying file Copyright.txt for details. +# +# This software is distributed WITHOUT ANY WARRANTY; without even the +# implied warranty of MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. +# See the License for more information. +#============================================================================= +# (To distributed this file outside of CMake, substitute the full +#  License text for the above reference.) + +# gp_append_unique list_var value +# +# Append value to the list variable ${list_var} only if the value is not +# already in the list. +# +function(gp_append_unique list_var value) +  set(contains 0) + +  foreach(item ${${list_var}}) +    if("${item}" STREQUAL "${value}") +      set(contains 1) +      break() +    endif("${item}" STREQUAL "${value}") +  endforeach(item) + +  if(NOT contains) +    set(${list_var} ${${list_var}} "${value}" PARENT_SCOPE) +  endif(NOT contains) +endfunction(gp_append_unique) + + +# is_file_executable file result_var +# +# Return 1 in ${result_var} if ${file} is a binary executable. +# +# Return 0 in ${result_var} otherwise. +# +function(is_file_executable file result_var) +  # +  # A file is not executable until proven otherwise: +  # +  set(${result_var} 0 PARENT_SCOPE) + +  get_filename_component(file_full "${file}" ABSOLUTE) +  string(TOLOWER "${file_full}" file_full_lower) + +  # If file name ends in .exe on Windows, *assume* executable: +  # +  if(WIN32) +    if("${file_full_lower}" MATCHES "\\.exe$") +      set(${result_var} 1 PARENT_SCOPE) +      return() +    endif("${file_full_lower}" MATCHES "\\.exe$") + +    # A clause could be added here that uses output or return value of dumpbin +    # to determine ${result_var}. In 99%+? practical cases, the exe name +    # match will be sufficient... +    # +  endif(WIN32) + +  # Use the information returned from the Unix shell command "file" to +  # determine if ${file_full} should be considered an executable file... +  # +  # If the file command's output contains "executable" and does *not* contain +  # "text" then it is likely an executable suitable for prerequisite analysis +  # via the get_prerequisites macro. +  # +  if(UNIX) +    if(NOT file_cmd) +      find_program(file_cmd "file") +    endif(NOT file_cmd) + +    if(file_cmd) +      execute_process(COMMAND "${file_cmd}" "${file_full}" +        OUTPUT_VARIABLE file_ov +        OUTPUT_STRIP_TRAILING_WHITESPACE +        ) + +      # Replace the name of the file in the output with a placeholder token +      # (the string " _file_full_ ") so that just in case the path name of +      # the file contains the word "text" or "executable" we are not fooled +      # into thinking "the wrong thing" because the file name matches the +      # other 'file' command output we are looking for... +      # +      string(REPLACE "${file_full}" " _file_full_ " file_ov "${file_ov}") +      string(TOLOWER "${file_ov}" file_ov) + +      #message(STATUS "file_ov='${file_ov}'") +      if("${file_ov}" MATCHES "executable") +        #message(STATUS "executable!") +        if("${file_ov}" MATCHES "text") +          #message(STATUS "but text, so *not* a binary executable!") +        else("${file_ov}" MATCHES "text") +          set(${result_var} 1 PARENT_SCOPE) +          return() +        endif("${file_ov}" MATCHES "text") +      endif("${file_ov}" MATCHES "executable") +    else(file_cmd) +      message(STATUS "warning: No 'file' command, skipping execute_process...") +    endif(file_cmd) +  endif(UNIX) +endfunction(is_file_executable) + + +# gp_item_default_embedded_path item default_embedded_path_var +# +# Return the path that others should refer to the item by when the item +# is embedded inside a bundle. +# +# Override on a per-project basis by providing a project-specific +# gp_item_default_embedded_path_override function. +# +function(gp_item_default_embedded_path item default_embedded_path_var) + +  # On Windows and Linux, "embed" prerequisites in the same directory +  # as the executable by default: +  # +  set(path "@executable_path") +  set(overridden 0) + +  # On the Mac, relative to the executable depending on the type +  # of the thing we are embedding: +  # +  if(APPLE) +    # +    # The assumption here is that all executables in the bundle will be +    # in same-level-directories inside the bundle. The parent directory +    # of an executable inside the bundle should be MacOS or a sibling of +    # MacOS and all embedded paths returned from here will begin with +    # "@executable_path/../" and will work from all executables in all +    # such same-level-directories inside the bundle. +    # + +    # By default, embed things right next to the main bundle executable: +    # +    set(path "@executable_path/../../Contents/MacOS") + +    # Embed .dylibs right next to the main bundle executable: +    # +    if(item MATCHES "\\.dylib$") +      set(path "@executable_path/../MacOS") +      set(overridden 1) +    endif(item MATCHES "\\.dylib$") + +    # Embed frameworks in the embedded "Frameworks" directory (sibling of MacOS): +    # +    if(NOT overridden) +      if(item MATCHES "[^/]+\\.framework/") +        set(path "@executable_path/../Frameworks") +        set(overridden 1) +      endif(item MATCHES "[^/]+\\.framework/") +    endif(NOT overridden) +  endif() + +  # Provide a hook so that projects can override the default embedded location +  # of any given library by whatever logic they choose: +  # +  if(COMMAND gp_item_default_embedded_path_override) +    gp_item_default_embedded_path_override("${item}" path) +  endif(COMMAND gp_item_default_embedded_path_override) + +  set(${default_embedded_path_var} "${path}" PARENT_SCOPE) +endfunction(gp_item_default_embedded_path) + + +# gp_resolve_item context item exepath dirs resolved_item_var +# +# Resolve an item into an existing full path file. +# +# Override on a per-project basis by providing a project-specific +# gp_resolve_item_override function. +# +function(gp_resolve_item context item exepath dirs resolved_item_var) +  set(resolved 0) +  set(resolved_item "${item}") + +  # Is it already resolved? +  # +  if(EXISTS "${resolved_item}") +    set(resolved 1) +  endif(EXISTS "${resolved_item}") + +  if(NOT resolved) +    if(item MATCHES "@executable_path") +      # +      # @executable_path references are assumed relative to exepath +      # +      string(REPLACE "@executable_path" "${exepath}" ri "${item}") +      get_filename_component(ri "${ri}" ABSOLUTE) + +      if(EXISTS "${ri}") +        #message(STATUS "info: embedded item exists (${ri})") +        set(resolved 1) +        set(resolved_item "${ri}") +      else(EXISTS "${ri}") +        message(STATUS "warning: embedded item does not exist '${ri}'") +      endif(EXISTS "${ri}") +    endif(item MATCHES "@executable_path") +  endif(NOT resolved) + +  if(NOT resolved) +    if(item MATCHES "@loader_path") +      # +      # @loader_path references are assumed relative to the +      # PATH of the given "context" (presumably another library) +      # +      get_filename_component(contextpath "${context}" PATH) +      string(REPLACE "@loader_path" "${contextpath}" ri "${item}") +      get_filename_component(ri "${ri}" ABSOLUTE) + +      if(EXISTS "${ri}") +        #message(STATUS "info: embedded item exists (${ri})") +        set(resolved 1) +        set(resolved_item "${ri}") +      else(EXISTS "${ri}") +        message(STATUS "warning: embedded item does not exist '${ri}'") +      endif(EXISTS "${ri}") +    endif(item MATCHES "@loader_path") +  endif(NOT resolved) + +  if(NOT resolved) +    set(ri "ri-NOTFOUND") +    find_file(ri "${item}" ${exepath} ${dirs} NO_DEFAULT_PATH) +    find_file(ri "${item}" ${exepath} ${dirs} /usr/lib) +    if(ri) +      #message(STATUS "info: 'find_file' in exepath/dirs (${ri})") +      set(resolved 1) +      set(resolved_item "${ri}") +      set(ri "ri-NOTFOUND") +    endif(ri) +  endif(NOT resolved) + +  if(NOT resolved) +    if(item MATCHES "[^/]+\\.framework/") +      set(fw "fw-NOTFOUND") +      find_file(fw "${item}" +        "~/Library/Frameworks" +        "/Library/Frameworks" +        "/System/Library/Frameworks" +      ) +      if(fw) +        #message(STATUS "info: 'find_file' found framework (${fw})") +        set(resolved 1) +        set(resolved_item "${fw}") +        set(fw "fw-NOTFOUND") +      endif(fw) +    endif(item MATCHES "[^/]+\\.framework/") +  endif(NOT resolved) + +  # Using find_program on Windows will find dll files that are in the PATH. +  # (Converting simple file names into full path names if found.) +  # +  if(WIN32) +  if(NOT resolved) +    set(ri "ri-NOTFOUND") +    find_program(ri "${item}" PATHS "${exepath};${dirs}" NO_DEFAULT_PATH) +    find_program(ri "${item}" PATHS "${exepath};${dirs}") +    if(ri) +      #message(STATUS "info: 'find_program' in exepath/dirs (${ri})") +      set(resolved 1) +      set(resolved_item "${ri}") +      set(ri "ri-NOTFOUND") +    endif(ri) +  endif(NOT resolved) +  endif(WIN32) + +  # Provide a hook so that projects can override item resolution +  # by whatever logic they choose: +  # +  if(COMMAND gp_resolve_item_override) +    gp_resolve_item_override("${context}" "${item}" "${exepath}" "${dirs}" resolved_item resolved) +  endif(COMMAND gp_resolve_item_override) + +  if(NOT resolved) +    message(STATUS " +warning: cannot resolve item '${item}' + +  possible problems: +    need more directories? +    need to use InstallRequiredSystemLibraries? +    run in install tree instead of build tree? +") +#    message(STATUS " +#****************************************************************************** +#warning: cannot resolve item '${item}' +# +#  possible problems: +#    need more directories? +#    need to use InstallRequiredSystemLibraries? +#    run in install tree instead of build tree? +# +#    context='${context}' +#    item='${item}' +#    exepath='${exepath}' +#    dirs='${dirs}' +#    resolved_item_var='${resolved_item_var}' +#****************************************************************************** +#") +  endif(NOT resolved) + +  set(${resolved_item_var} "${resolved_item}" PARENT_SCOPE) +endfunction(gp_resolve_item) + + +# gp_resolved_file_type original_file file exepath dirs type_var +# +# Return the type of ${file} with respect to ${original_file}. String +# describing type of prerequisite is returned in variable named ${type_var}. +# +# Use ${exepath} and ${dirs} if necessary to resolve non-absolute ${file} +# values -- but only for non-embedded items. +# +# Possible types are: +#   system +#   local +#   embedded +#   other +# +function(gp_resolved_file_type original_file file exepath dirs type_var) +  #message(STATUS "**") + +  if(NOT IS_ABSOLUTE "${original_file}") +    message(STATUS "warning: gp_resolved_file_type expects absolute full path for first arg original_file") +  endif() + +  set(is_embedded 0) +  set(is_local 0) +  set(is_system 0) + +  set(resolved_file "${file}") + +  if("${file}" MATCHES "^@(executable|loader)_path") +    set(is_embedded 1) +  endif() + +  if(NOT is_embedded) +    if(NOT IS_ABSOLUTE "${file}") +      gp_resolve_item("${original_file}" "${file}" "${exepath}" "${dirs}" resolved_file) +    endif() + +    string(TOLOWER "${original_file}" original_lower) +    string(TOLOWER "${resolved_file}" lower) + +    if(UNIX) +      if(resolved_file MATCHES "^(/lib/|/lib32/|/lib64/|/usr/lib/|/usr/lib32/|/usr/lib64/|/usr/X11R6/)") +        set(is_system 1) +      endif() +    endif() + +    if(APPLE) +      if(resolved_file MATCHES "^(/System/Library/|/usr/lib/)") +        set(is_system 1) +      endif() +    endif() + +    if(WIN32) +      string(TOLOWER "$ENV{SystemRoot}" sysroot) +      string(REGEX REPLACE "\\\\" "/" sysroot "${sysroot}") + +      string(TOLOWER "$ENV{windir}" windir) +      string(REGEX REPLACE "\\\\" "/" windir "${windir}") + +      if(lower MATCHES "^(${sysroot}/system|${windir}/system|${sysroot}/syswow|${windir}/syswow|(.*/)*msvc[^/]+dll)") +        set(is_system 1) +      endif() +    endif() + +    if(NOT is_system) +      get_filename_component(original_path "${original_lower}" PATH) +      get_filename_component(path "${lower}" PATH) +      if("${original_path}" STREQUAL "${path}") +        set(is_local 1) +      endif() +    endif() +  endif() + +  # Return type string based on computed booleans: +  # +  set(type "other") + +  if(is_system) +    set(type "system") +  elseif(is_embedded) +    set(type "embedded") +  elseif(is_local) +    set(type "local") +  endif() + +  #message(STATUS "gp_resolved_file_type: '${file}' '${resolved_file}'") +  #message(STATUS "                type: '${type}'") + +  if(NOT is_embedded) +    if(NOT IS_ABSOLUTE "${resolved_file}") +      if(lower MATCHES "^msvc[^/]+dll" AND is_system) +        message(STATUS "info: non-absolute msvc file '${file}' returning type '${type}'") +      else() +        message(STATUS "warning: gp_resolved_file_type non-absolute file '${file}' returning type '${type}' -- possibly incorrect") +      endif() +    endif() +  endif() + +  set(${type_var} "${type}" PARENT_SCOPE) + +  #message(STATUS "**") +endfunction() + + +# gp_file_type original_file file type_var +# +# Return the type of ${file} with respect to ${original_file}. String +# describing type of prerequisite is returned in variable named ${type_var}. +# +# Possible types are: +#   system +#   local +#   embedded +#   other +# +function(gp_file_type original_file file type_var) +  if(NOT IS_ABSOLUTE "${original_file}") +    message(STATUS "warning: gp_file_type expects absolute full path for first arg original_file") +  endif() + +  get_filename_component(exepath "${original_file}" PATH) + +  set(type "") +  gp_resolved_file_type("${original_file}" "${file}" "${exepath}" "" type) + +  set(${type_var} "${type}" PARENT_SCOPE) +endfunction(gp_file_type) + + +# get_prerequisites target prerequisites_var exclude_system recurse dirs +# +# Get the list of shared library files required by ${target}. The list in +# the variable named ${prerequisites_var} should be empty on first entry to +# this function. On exit, ${prerequisites_var} will contain the list of +# required shared library files. +# +#  target is the full path to an executable file +# +#  prerequisites_var is the name of a CMake variable to contain the results +# +#  exclude_system is 0 or 1: 0 to include "system" prerequisites , 1 to +#   exclude them +# +#  recurse is 0 or 1: 0 for direct prerequisites only, 1 for all prerequisites +#   recursively +# +#  exepath is the path to the top level executable used for @executable_path +#   replacment on the Mac +# +#  dirs is a list of paths where libraries might be found: these paths are +#   searched first when a target without any path info is given. Then standard +#   system locations are also searched: PATH, Framework locations, /usr/lib... +# +function(get_prerequisites target prerequisites_var exclude_system recurse exepath dirs) +  set(verbose 0) +  set(eol_char "E") + +  if(NOT IS_ABSOLUTE "${target}") +    message("warning: target '${target}' is not absolute...") +  endif(NOT IS_ABSOLUTE "${target}") + +  if(NOT EXISTS "${target}") +    message("warning: target '${target}' does not exist...") +  endif(NOT EXISTS "${target}") + +  # <setup-gp_tool-vars> +  # +  # Try to choose the right tool by default. Caller can set gp_tool prior to +  # calling this function to force using a different tool. +  # +  if("${gp_tool}" STREQUAL "") +    set(gp_tool "ldd") +    if(APPLE) +      set(gp_tool "otool") +    endif(APPLE) +    if(WIN32) +      set(gp_tool "dumpbin") +    endif(WIN32) +  endif("${gp_tool}" STREQUAL "") + +  set(gp_tool_known 0) + +  if("${gp_tool}" STREQUAL "ldd") +    set(gp_cmd_args "") +    set(gp_regex "^[\t ]*[^\t ]+ => ([^\t ]+).*${eol_char}$") +    set(gp_regex_cmp_count 1) +    set(gp_tool_known 1) +  endif("${gp_tool}" STREQUAL "ldd") + +  if("${gp_tool}" STREQUAL "otool") +    set(gp_cmd_args "-L") +    set(gp_regex "^\t([^\t]+) \\(compatibility version ([0-9]+.[0-9]+.[0-9]+), current version ([0-9]+.[0-9]+.[0-9]+)\\)${eol_char}$") +    set(gp_regex_cmp_count 3) +    set(gp_tool_known 1) +  endif("${gp_tool}" STREQUAL "otool") + +  if("${gp_tool}" STREQUAL "dumpbin") +    set(gp_cmd_args "/dependents") +    set(gp_regex "^    ([^ ].*[Dd][Ll][Ll])${eol_char}$") +    set(gp_regex_cmp_count 1) +    set(gp_tool_known 1) +    set(ENV{VS_UNICODE_OUTPUT} "") # Block extra output from inside VS IDE. +  endif("${gp_tool}" STREQUAL "dumpbin") + +  if(NOT gp_tool_known) +    message(STATUS "warning: gp_tool='${gp_tool}' is an unknown tool...") +    message(STATUS "CMake function get_prerequisites needs more code to handle '${gp_tool}'") +    message(STATUS "Valid gp_tool values are dumpbin, ldd and otool.") +    return() +  endif(NOT gp_tool_known) + +  set(gp_cmd_paths ${gp_cmd_paths} +    "C:/Program Files/Microsoft Visual Studio 9.0/VC/bin" +    "C:/Program Files (x86)/Microsoft Visual Studio 9.0/VC/bin" +    "C:/Program Files/Microsoft Visual Studio 8/VC/BIN" +    "C:/Program Files (x86)/Microsoft Visual Studio 8/VC/BIN" +    "C:/Program Files/Microsoft Visual Studio .NET 2003/VC7/BIN" +    "C:/Program Files (x86)/Microsoft Visual Studio .NET 2003/VC7/BIN" +    "/usr/local/bin" +    "/usr/bin" +    ) + +  find_program(gp_cmd ${gp_tool} PATHS ${gp_cmd_paths}) + +  if(NOT gp_cmd) +    message(STATUS "warning: could not find '${gp_tool}' - cannot analyze prerequisites...") +    return() +  endif(NOT gp_cmd) + +  if("${gp_tool}" STREQUAL "dumpbin") +    # When running dumpbin, it also needs the "Common7/IDE" directory in the +    # PATH. It will already be in the PATH if being run from a Visual Studio +    # command prompt. Add it to the PATH here in case we are running from a +    # different command prompt. +    # +    get_filename_component(gp_cmd_dir "${gp_cmd}" PATH) +    get_filename_component(gp_cmd_dlls_dir "${gp_cmd_dir}/../../Common7/IDE" ABSOLUTE) +    if(EXISTS "${gp_cmd_dlls_dir}") +      # only add to the path if it is not already in the path +      if(NOT "$ENV{PATH}" MATCHES "${gp_cmd_dlls_dir}") +        set(ENV{PATH} "$ENV{PATH};${gp_cmd_dlls_dir}") +      endif(NOT "$ENV{PATH}" MATCHES "${gp_cmd_dlls_dir}") +    endif(EXISTS "${gp_cmd_dlls_dir}") +  endif("${gp_tool}" STREQUAL "dumpbin") +  # +  # </setup-gp_tool-vars> + +  if("${gp_tool}" STREQUAL "ldd") +    set(old_ld_env "$ENV{LD_LIBRARY_PATH}") +    foreach(dir ${exepath} ${dirs}) +      set(ENV{LD_LIBRARY_PATH} "${dir}:$ENV{LD_LIBRARY_PATH}") +    endforeach(dir) +  endif("${gp_tool}" STREQUAL "ldd") + + +  # Track new prerequisites at each new level of recursion. Start with an +  # empty list at each level: +  # +  set(unseen_prereqs) + +  # Run gp_cmd on the target: +  # +  execute_process( +    COMMAND ${gp_cmd} ${gp_cmd_args} ${target} +    OUTPUT_VARIABLE gp_cmd_ov +    ) + +  if("${gp_tool}" STREQUAL "ldd") +    set(ENV{LD_LIBRARY_PATH} "${old_ld_env}") +  endif("${gp_tool}" STREQUAL "ldd") + +  if(verbose) +    message(STATUS "<RawOutput cmd='${gp_cmd} ${gp_cmd_args} ${target}'>") +    message(STATUS "gp_cmd_ov='${gp_cmd_ov}'") +    message(STATUS "</RawOutput>") +  endif(verbose) + +  get_filename_component(target_dir "${target}" PATH) + +  # Convert to a list of lines: +  # +  string(REGEX REPLACE ";" "\\\\;" candidates "${gp_cmd_ov}") +  string(REGEX REPLACE "\n" "${eol_char};" candidates "${candidates}") + +  # Analyze each line for file names that match the regular expression: +  # +  foreach(candidate ${candidates}) +  if("${candidate}" MATCHES "${gp_regex}") +    # Extract information from each candidate: +    string(REGEX REPLACE "${gp_regex}" "\\1" raw_item "${candidate}") + +    if(gp_regex_cmp_count GREATER 1) +      string(REGEX REPLACE "${gp_regex}" "\\2" raw_compat_version "${candidate}") +      string(REGEX REPLACE "^([0-9]+)\\.([0-9]+)\\.([0-9]+)$" "\\1" compat_major_version "${raw_compat_version}") +      string(REGEX REPLACE "^([0-9]+)\\.([0-9]+)\\.([0-9]+)$" "\\2" compat_minor_version "${raw_compat_version}") +      string(REGEX REPLACE "^([0-9]+)\\.([0-9]+)\\.([0-9]+)$" "\\3" compat_patch_version "${raw_compat_version}") +    endif(gp_regex_cmp_count GREATER 1) + +    if(gp_regex_cmp_count GREATER 2) +      string(REGEX REPLACE "${gp_regex}" "\\3" raw_current_version "${candidate}") +      string(REGEX REPLACE "^([0-9]+)\\.([0-9]+)\\.([0-9]+)$" "\\1" current_major_version "${raw_current_version}") +      string(REGEX REPLACE "^([0-9]+)\\.([0-9]+)\\.([0-9]+)$" "\\2" current_minor_version "${raw_current_version}") +      string(REGEX REPLACE "^([0-9]+)\\.([0-9]+)\\.([0-9]+)$" "\\3" current_patch_version "${raw_current_version}") +    endif(gp_regex_cmp_count GREATER 2) + +    # Use the raw_item as the list entries returned by this function. Use the +    # gp_resolve_item function to resolve it to an actual full path file if +    # necessary. +    # +    set(item "${raw_item}") + +    # Add each item unless it is excluded: +    # +    set(add_item 1) + +    if(${exclude_system}) +      set(type "") +      gp_resolved_file_type("${target}" "${item}" "${exepath}" "${dirs}" type) +      if("${type}" STREQUAL "system") +        set(add_item 0) +      endif("${type}" STREQUAL "system") +    endif(${exclude_system}) + +    if(add_item) +      list(LENGTH ${prerequisites_var} list_length_before_append) +      gp_append_unique(${prerequisites_var} "${item}") +      list(LENGTH ${prerequisites_var} list_length_after_append) + +      if(${recurse}) +        # If item was really added, this is the first time we have seen it. +        # Add it to unseen_prereqs so that we can recursively add *its* +        # prerequisites... +        # +        # But first: resolve its name to an absolute full path name such +        # that the analysis tools can simply accept it as input. +        # +        if(NOT list_length_before_append EQUAL list_length_after_append) +          gp_resolve_item("${target}" "${item}" "${exepath}" "${dirs}" resolved_item) +          set(unseen_prereqs ${unseen_prereqs} "${resolved_item}") +        endif(NOT list_length_before_append EQUAL list_length_after_append) +      endif(${recurse}) +    endif(add_item) +  else("${candidate}" MATCHES "${gp_regex}") +    if(verbose) +      message(STATUS "ignoring non-matching line: '${candidate}'") +    endif(verbose) +  endif("${candidate}" MATCHES "${gp_regex}") +  endforeach(candidate) + +  list(LENGTH ${prerequisites_var} prerequisites_var_length) +  if(prerequisites_var_length GREATER 0) +    list(SORT ${prerequisites_var}) +  endif(prerequisites_var_length GREATER 0) +  if(${recurse}) +    set(more_inputs ${unseen_prereqs}) +    foreach(input ${more_inputs}) +      get_prerequisites("${input}" ${prerequisites_var} ${exclude_system} ${recurse} "${exepath}" "${dirs}") +    endforeach(input) +  endif(${recurse}) + +  set(${prerequisites_var} ${${prerequisites_var}} PARENT_SCOPE) +endfunction(get_prerequisites) + + +# list_prerequisites target all exclude_system verbose +# +#  ARGV0 (target) is the full path to an executable file +# +#  optional ARGV1 (all) is 0 or 1: 0 for direct prerequisites only, +#   1 for all prerequisites recursively +# +#  optional ARGV2 (exclude_system) is 0 or 1: 0 to include "system" +#   prerequisites , 1 to exclude them +# +#  optional ARGV3 (verbose) is 0 or 1: 0 to print only full path +#   names of prerequisites, 1 to print extra information +# +function(list_prerequisites target) +  if("${ARGV1}" STREQUAL "") +    set(all 1) +  else("${ARGV1}" STREQUAL "") +    set(all "${ARGV1}") +  endif("${ARGV1}" STREQUAL "") + +  if("${ARGV2}" STREQUAL "") +    set(exclude_system 0) +  else("${ARGV2}" STREQUAL "") +    set(exclude_system "${ARGV2}") +  endif("${ARGV2}" STREQUAL "") + +  if("${ARGV3}" STREQUAL "") +    set(verbose 0) +  else("${ARGV3}" STREQUAL "") +    set(verbose "${ARGV3}") +  endif("${ARGV3}" STREQUAL "") + +  set(count 0) +  set(count_str "") +  set(print_count "${verbose}") +  set(print_prerequisite_type "${verbose}") +  set(print_target "${verbose}") +  set(type_str "") + +  get_filename_component(exepath "${target}" PATH) + +  set(prereqs "") +  get_prerequisites("${target}" prereqs ${exclude_system} ${all} "${exepath}" "") + +  if(print_target) +    message(STATUS "File '${target}' depends on:") +  endif(print_target) + +  foreach(d ${prereqs}) +    math(EXPR count "${count} + 1") + +    if(print_count) +      set(count_str "${count}. ") +    endif(print_count) + +    if(print_prerequisite_type) +      gp_file_type("${target}" "${d}" type) +      set(type_str " (${type})") +    endif(print_prerequisite_type) + +    message(STATUS "${count_str}${d}${type_str}") +  endforeach(d) +endfunction(list_prerequisites) + + +# list_prerequisites_by_glob glob_arg glob_exp +# +#  glob_arg is GLOB or GLOB_RECURSE +# +#  glob_exp is a globbing expression used with "file(GLOB" to retrieve a list +#   of matching files. If a matching file is executable, its prerequisites are +#   listed. +# +# Any additional (optional) arguments provided are passed along as the +# optional arguments to the list_prerequisites calls. +# +function(list_prerequisites_by_glob glob_arg glob_exp) +  message(STATUS "=============================================================================") +  message(STATUS "List prerequisites of executables matching ${glob_arg} '${glob_exp}'") +  message(STATUS "") +  file(${glob_arg} file_list ${glob_exp}) +  foreach(f ${file_list}) +    is_file_executable("${f}" is_f_executable) +    if(is_f_executable) +      message(STATUS "=============================================================================") +      list_prerequisites("${f}" ${ARGN}) +      message(STATUS "") +    endif(is_f_executable) +  endforeach(f) +endfunction(list_prerequisites_by_glob) diff --git a/indra/cmake/LLAddBuildTest.cmake b/indra/cmake/LLAddBuildTest.cmake index 62b764bb30..05f0492234 100644 --- a/indra/cmake/LLAddBuildTest.cmake +++ b/indra/cmake/LLAddBuildTest.cmake @@ -1,276 +1,276 @@ -# -*- cmake -*-
 -include(LLTestCommand)
 -include(GoogleMock)
 -
 -MACRO(LL_ADD_PROJECT_UNIT_TESTS project sources)
 -  # Given a project name and a list of sourcefiles (with optional properties on each),
 -  # add targets to build and run the tests specified.
 -  # ASSUMPTIONS:
 -  # * this macro is being executed in the project file that is passed in
 -  # * current working SOURCE dir is that project dir
 -  # * there is a subfolder tests/ with test code corresponding to the filenames passed in
 -  # * properties for each sourcefile passed in indicate what libs to link that file with (MAKE NO ASSUMPTIONS ASIDE FROM TUT)
 -  #
 -  # More info and examples at: https://wiki.secondlife.com/wiki/How_to_add_unit_tests_to_indra_code
 -  #
 -  # WARNING: do NOT modify this code without working with poppy -
 -  # there is another branch that will conflict heavily with any changes here.
 -INCLUDE(GoogleMock)
 -
 -
 -  IF(LL_TEST_VERBOSE)
 -    MESSAGE("LL_ADD_PROJECT_UNIT_TESTS UNITTEST_PROJECT_${project} sources: ${sources}")
 -  ENDIF(LL_TEST_VERBOSE)
 -
 -  # Start with the header and project-wide setup before making targets
 -  #project(UNITTEST_PROJECT_${project})
 -  # Setup includes, paths, etc
 -  SET(alltest_SOURCE_FILES
 -    ${CMAKE_SOURCE_DIR}/test/test.cpp
 -    ${CMAKE_SOURCE_DIR}/test/lltut.cpp
 -    )
 -  SET(alltest_DEP_TARGETS
 -    # needed by the test harness itself
 -    ${APRUTIL_LIBRARIES}
 -    ${APR_LIBRARIES}
 -    llcommon
 -    )
 -  IF(NOT "${project}" STREQUAL "llmath")
 -    # add llmath as a dep unless the tested module *is* llmath!
 -    LIST(APPEND alltest_DEP_TARGETS
 -      llmath
 -      )
 -  ENDIF(NOT "${project}" STREQUAL "llmath")
 -  SET(alltest_INCLUDE_DIRS
 -    ${LLMATH_INCLUDE_DIRS}
 -    ${LLCOMMON_INCLUDE_DIRS}
 -    ${LIBS_OPEN_DIR}/test
 -    ${GOOGLEMOCK_INCLUDE_DIRS}
 -    )
 -  SET(alltest_LIBRARIES
 -    ${GOOGLEMOCK_LIBRARIES}
 -    ${PTHREAD_LIBRARY}
 -    ${WINDOWS_LIBRARIES}
 -    )
 -  # Headers, for convenience in targets.
 -  SET(alltest_HEADER_FILES
 -    ${CMAKE_SOURCE_DIR}/test/test.h
 -    )
 -
 -  # Use the default flags
 -  if (LINUX)
 -    SET(CMAKE_EXE_LINKER_FLAGS "")
 -  endif (LINUX)
 -
 -  # start the source test executable definitions
 -  SET(${project}_TEST_OUTPUT "")
 -  FOREACH (source ${sources})
 -    STRING( REGEX REPLACE "(.*)\\.[^.]+$" "\\1" name ${source} )
 -    STRING( REGEX REPLACE ".*\\.([^.]+)$" "\\1" extension ${source} )
 -    IF(LL_TEST_VERBOSE)
 -      MESSAGE("LL_ADD_PROJECT_UNIT_TESTS UNITTEST_PROJECT_${project} individual source: ${source} (${name}.${extension})")
 -    ENDIF(LL_TEST_VERBOSE)
 -
 -    #
 -    # Per-codefile additional / external source, header, and include dir property extraction
 -    #
 -    # Source
 -    GET_SOURCE_FILE_PROPERTY(${name}_test_additional_SOURCE_FILES ${source} LL_TEST_ADDITIONAL_SOURCE_FILES)
 -    IF(${name}_test_additional_SOURCE_FILES MATCHES NOTFOUND)
 -      SET(${name}_test_additional_SOURCE_FILES "")
 -    ENDIF(${name}_test_additional_SOURCE_FILES MATCHES NOTFOUND)
 -    SET(${name}_test_SOURCE_FILES ${source} tests/${name}_test.${extension} ${alltest_SOURCE_FILES} ${${name}_test_additional_SOURCE_FILES} )
 -    IF(LL_TEST_VERBOSE)
 -      MESSAGE("LL_ADD_PROJECT_UNIT_TESTS ${name}_test_SOURCE_FILES ${${name}_test_SOURCE_FILES}")
 -    ENDIF(LL_TEST_VERBOSE)
 -    # Headers
 -    GET_SOURCE_FILE_PROPERTY(${name}_test_additional_HEADER_FILES ${source} LL_TEST_ADDITIONAL_HEADER_FILES)
 -    IF(${name}_test_additional_HEADER_FILES MATCHES NOTFOUND)
 -      SET(${name}_test_additional_HEADER_FILES "")
 -    ENDIF(${name}_test_additional_HEADER_FILES MATCHES NOTFOUND)
 -    SET(${name}_test_HEADER_FILES ${name}.h ${${name}_test_additional_HEADER_FILES})
 -    set_source_files_properties(${${name}_test_HEADER_FILES} PROPERTIES HEADER_FILE_ONLY TRUE)
 -    LIST(APPEND ${name}_test_SOURCE_FILES ${${name}_test_HEADER_FILES})
 -    IF(LL_TEST_VERBOSE)
 -      MESSAGE("LL_ADD_PROJECT_UNIT_TESTS ${name}_test_HEADER_FILES ${${name}_test_HEADER_FILES}")
 -    ENDIF(LL_TEST_VERBOSE)
 -    # Include dirs
 -    GET_SOURCE_FILE_PROPERTY(${name}_test_additional_INCLUDE_DIRS ${source} LL_TEST_ADDITIONAL_INCLUDE_DIRS)
 -    IF(${name}_test_additional_INCLUDE_DIRS MATCHES NOTFOUND)
 -      SET(${name}_test_additional_INCLUDE_DIRS "")
 -    ENDIF(${name}_test_additional_INCLUDE_DIRS MATCHES NOTFOUND)
 -    INCLUDE_DIRECTORIES(${alltest_INCLUDE_DIRS} ${name}_test_additional_INCLUDE_DIRS )
 -    IF(LL_TEST_VERBOSE)
 -      MESSAGE("LL_ADD_PROJECT_UNIT_TESTS ${name}_test_additional_INCLUDE_DIRS ${${name}_test_additional_INCLUDE_DIRS}")
 -    ENDIF(LL_TEST_VERBOSE)
 -
 -
 -    # Setup target
 -    ADD_EXECUTABLE(PROJECT_${project}_TEST_${name} ${${name}_test_SOURCE_FILES})
 -    SET_TARGET_PROPERTIES(PROJECT_${project}_TEST_${name} PROPERTIES RUNTIME_OUTPUT_DIRECTORY "${EXE_STAGING_DIR}")
 -
 -    #
 -    # Per-codefile additional / external project dep and lib dep property extraction
 -    #
 -    # WARNING: it's REALLY IMPORTANT to not mix these. I guarantee it will not work in the future. + poppy 2009-04-19
 -    # Projects
 -    GET_SOURCE_FILE_PROPERTY(${name}_test_additional_PROJECTS ${source} LL_TEST_ADDITIONAL_PROJECTS)
 -    IF(${name}_test_additional_PROJECTS MATCHES NOTFOUND)
 -      SET(${name}_test_additional_PROJECTS "")
 -    ENDIF(${name}_test_additional_PROJECTS MATCHES NOTFOUND)
 -    # Libraries
 -    GET_SOURCE_FILE_PROPERTY(${name}_test_additional_LIBRARIES ${source} LL_TEST_ADDITIONAL_LIBRARIES)
 -    IF(${name}_test_additional_LIBRARIES MATCHES NOTFOUND)
 -      SET(${name}_test_additional_LIBRARIES "")
 -    ENDIF(${name}_test_additional_LIBRARIES MATCHES NOTFOUND)
 -    IF(LL_TEST_VERBOSE)
 -      MESSAGE("LL_ADD_PROJECT_UNIT_TESTS ${name}_test_additional_PROJECTS ${${name}_test_additional_PROJECTS}")
 -      MESSAGE("LL_ADD_PROJECT_UNIT_TESTS ${name}_test_additional_LIBRARIES ${${name}_test_additional_LIBRARIES}")
 -    ENDIF(LL_TEST_VERBOSE)
 -    # Add to project
 -    TARGET_LINK_LIBRARIES(PROJECT_${project}_TEST_${name} ${alltest_LIBRARIES} ${alltest_DEP_TARGETS} ${${name}_test_additional_PROJECTS} ${${name}_test_additional_LIBRARIES} )
 -    # Compile-time Definitions
 -    GET_SOURCE_FILE_PROPERTY(${name}_test_additional_CFLAGS ${source} LL_TEST_ADDITIONAL_CFLAGS)
 -     IF(NOT ${name}_test_additional_CFLAGS MATCHES NOTFOUND)
 -       SET_TARGET_PROPERTIES(PROJECT_${project}_TEST_${name} PROPERTIES COMPILE_FLAGS ${${name}_test_additional_CFLAGS} )
 -       IF(LL_TEST_VERBOSE)
 -         MESSAGE("LL_ADD_PROJECT_UNIT_TESTS ${name}_test_additional_CFLAGS ${${name}_test_additional_CFLAGS}")
 -       ENDIF(LL_TEST_VERBOSE)
 -     ENDIF(NOT ${name}_test_additional_CFLAGS MATCHES NOTFOUND)
 -     
 -    #
 -    # Setup test targets
 -    #
 -    GET_TARGET_PROPERTY(TEST_EXE PROJECT_${project}_TEST_${name} LOCATION)
 -    SET(TEST_OUTPUT ${CMAKE_CURRENT_BINARY_DIR}/PROJECT_${project}_TEST_${name}_ok.txt)
 -    SET(TEST_CMD ${TEST_EXE} --touch=${TEST_OUTPUT} --sourcedir=${CMAKE_CURRENT_SOURCE_DIR})
 -
 -    # daveh - what configuration does this use? Debug? it's cmake-time, not build time. + poppy 2009-04-19
 -    IF(LL_TEST_VERBOSE)
 -      MESSAGE(STATUS "LL_ADD_PROJECT_UNIT_TESTS ${name} test_cmd  = ${TEST_CMD}")
 -    ENDIF(LL_TEST_VERBOSE)
 -
 -    SET_TEST_PATH(LD_LIBRARY_PATH)
 -    LL_TEST_COMMAND(TEST_SCRIPT_CMD "${LD_LIBRARY_PATH}" ${TEST_CMD})
 -    IF(LL_TEST_VERBOSE)
 -      MESSAGE(STATUS "LL_ADD_PROJECT_UNIT_TESTS ${name} test_script  = ${TEST_SCRIPT_CMD}")
 -    ENDIF(LL_TEST_VERBOSE)
 -    # Add test 
 -    ADD_CUSTOM_COMMAND(
 -        OUTPUT ${TEST_OUTPUT}
 -        COMMAND ${TEST_SCRIPT_CMD}
 -        DEPENDS PROJECT_${project}_TEST_${name}
 -        WORKING_DIRECTORY ${CMAKE_CURRENT_BINARY_DIR}
 -        )
 -    # Why not add custom target and add POST_BUILD command?
 -    # Slightly less uncertain behavior
 -    # (OUTPUT commands run non-deterministically AFAIK) + poppy 2009-04-19
 -    # > I did not use a post build step as I could not make it notify of a 
 -    # > failure after the first time you build and fail a test. - daveh 2009-04-20
 -    LIST(APPEND ${project}_TEST_OUTPUT ${TEST_OUTPUT})
 -  ENDFOREACH (source)
 -
 -  # Add the test runner target per-project
 -  # (replaces old _test_ok targets all over the place)
 -  ADD_CUSTOM_TARGET(${project}_tests ALL DEPENDS ${${project}_TEST_OUTPUT})
 -  ADD_DEPENDENCIES(${project} ${project}_tests)
 -ENDMACRO(LL_ADD_PROJECT_UNIT_TESTS)
 -
 -FUNCTION(LL_ADD_INTEGRATION_TEST 
 -    testname
 -    additional_source_files
 -    library_dependencies
 -# variable args
 -    )
 -  if(TEST_DEBUG)
 -    message(STATUS "Adding INTEGRATION_TEST_${testname} - debug output is on")
 -  endif(TEST_DEBUG)
 -  
 -  SET(source_files
 -    tests/${testname}_test.cpp
 -    ${CMAKE_SOURCE_DIR}/test/test.cpp
 -    ${CMAKE_SOURCE_DIR}/test/lltut.cpp
 -    ${additional_source_files}
 -    )
 -
 -  SET(libraries
 -    ${library_dependencies}
 -    ${GOOGLEMOCK_LIBRARIES}
 -    ${PTHREAD_LIBRARY}
 -    )
 -
 -  # Add test executable build target
 -  if(TEST_DEBUG)
 -    message(STATUS "ADD_EXECUTABLE(INTEGRATION_TEST_${testname} ${source_files})")
 -  endif(TEST_DEBUG)
 -  ADD_EXECUTABLE(INTEGRATION_TEST_${testname} ${source_files})
 -  SET_TARGET_PROPERTIES(INTEGRATION_TEST_${testname} PROPERTIES RUNTIME_OUTPUT_DIRECTORY "${EXE_STAGING_DIR}")
 -  if(STANDALONE)
 -    SET_TARGET_PROPERTIES(INTEGRATION_TEST_${testname} PROPERTIES COMPILE_FLAGS -I"${TUT_INCLUDE_DIR}")
 -  endif(STANDALONE)
 -
 -  # Add link deps to the executable
 -  if(TEST_DEBUG)
 -    message(STATUS "TARGET_LINK_LIBRARIES(INTEGRATION_TEST_${testname} ${libraries})")
 -  endif(TEST_DEBUG)
 -  TARGET_LINK_LIBRARIES(INTEGRATION_TEST_${testname} ${libraries})
 -
 -  # Create the test running command
 -  SET(test_command ${ARGN})
 -  GET_TARGET_PROPERTY(TEST_EXE INTEGRATION_TEST_${testname} LOCATION)
 -  LIST(FIND test_command "{}" test_exe_pos)
 -  IF(test_exe_pos LESS 0)
 -    # The {} marker means "the full pathname of the test executable."
 -    # test_exe_pos -1 means we didn't find it -- so append the test executable
 -    # name to $ARGN, the variable part of the arg list. This is convenient
 -    # shorthand for both straightforward execution of the test program (empty
 -    # $ARGN) and for running a "wrapper" program of some kind accepting the
 -    # pathname of the test program as the last of its args. You need specify
 -    # {} only if the test program's pathname isn't the last argument in the
 -    # desired command line.
 -    LIST(APPEND test_command "${TEST_EXE}")
 -  ELSE (test_exe_pos LESS 0)
 -    # Found {} marker at test_exe_pos. Remove the {}...
 -    LIST(REMOVE_AT test_command test_exe_pos)
 -    # ...and replace it with the actual name of the test executable.
 -    LIST(INSERT test_command test_exe_pos "${TEST_EXE}")
 -  ENDIF (test_exe_pos LESS 0)
 -
 -  SET_TEST_PATH(LD_LIBRARY_PATH)
 -  LL_TEST_COMMAND(TEST_SCRIPT_CMD "${LD_LIBRARY_PATH}" ${test_command})
 -
 -  if(TEST_DEBUG)
 -    message(STATUS "TEST_SCRIPT_CMD: ${TEST_SCRIPT_CMD}")
 -  endif(TEST_DEBUG)
 -
 -  ADD_CUSTOM_COMMAND(
 -    TARGET INTEGRATION_TEST_${testname}
 -    POST_BUILD
 -    COMMAND ${TEST_SCRIPT_CMD}
 -    )
 -
 -  # Use CTEST? Not sure how to yet...
 -  # ADD_TEST(INTEGRATION_TEST_RUNNER_${testname} ${TEST_SCRIPT_CMD})
 -
 -ENDFUNCTION(LL_ADD_INTEGRATION_TEST)
 -
 -MACRO(SET_TEST_PATH LISTVAR)
 -  IF(WINDOWS)
 -    # We typically build/package only Release variants of third-party
 -    # libraries, so append the Release staging dir in case the library being
 -    # sought doesn't have a debug variant.
 -    set(${LISTVAR} ${SHARED_LIB_STAGING_DIR}/${CMAKE_CFG_INTDIR} ${SHARED_LIB_STAGING_DIR}/Release)
 -  ELSEIF(DARWIN)
 -    # We typically build/package only Release variants of third-party
 -    # libraries, so append the Release staging dir in case the library being
 -    # sought doesn't have a debug variant.
 -    set(${LISTVAR} ${SHARED_LIB_STAGING_DIR}/${CMAKE_CFG_INTDIR}/Resources ${SHARED_LIB_STAGING_DIR}/Release/Resources /usr/lib)
 -  ELSE(WINDOWS)
 -    # Linux uses a single staging directory anyway.
 -    IF (STANDALONE)
 -      set(${LISTVAR} ${CMAKE_BINARY_DIR}/llcommon /usr/lib /usr/local/lib)
 -    ELSE (STANDALONE)
 -      set(${LISTVAR} ${SHARED_LIB_STAGING_DIR} /usr/lib)
 -    ENDIF (STANDALONE)
 -  ENDIF(WINDOWS)
 -ENDMACRO(SET_TEST_PATH)
 +# -*- cmake -*- +include(LLTestCommand) +include(GoogleMock) + +MACRO(LL_ADD_PROJECT_UNIT_TESTS project sources) +  # Given a project name and a list of sourcefiles (with optional properties on each), +  # add targets to build and run the tests specified. +  # ASSUMPTIONS: +  # * this macro is being executed in the project file that is passed in +  # * current working SOURCE dir is that project dir +  # * there is a subfolder tests/ with test code corresponding to the filenames passed in +  # * properties for each sourcefile passed in indicate what libs to link that file with (MAKE NO ASSUMPTIONS ASIDE FROM TUT) +  # +  # More info and examples at: https://wiki.secondlife.com/wiki/How_to_add_unit_tests_to_indra_code +  # +  # WARNING: do NOT modify this code without working with poppy - +  # there is another branch that will conflict heavily with any changes here. +INCLUDE(GoogleMock) + + +  IF(LL_TEST_VERBOSE) +    MESSAGE("LL_ADD_PROJECT_UNIT_TESTS UNITTEST_PROJECT_${project} sources: ${sources}") +  ENDIF(LL_TEST_VERBOSE) + +  # Start with the header and project-wide setup before making targets +  #project(UNITTEST_PROJECT_${project}) +  # Setup includes, paths, etc +  SET(alltest_SOURCE_FILES +    ${CMAKE_SOURCE_DIR}/test/test.cpp +    ${CMAKE_SOURCE_DIR}/test/lltut.cpp +    ) +  SET(alltest_DEP_TARGETS +    # needed by the test harness itself +    ${APRUTIL_LIBRARIES} +    ${APR_LIBRARIES} +    llcommon +    ) +  IF(NOT "${project}" STREQUAL "llmath") +    # add llmath as a dep unless the tested module *is* llmath! +    LIST(APPEND alltest_DEP_TARGETS +      llmath +      ) +  ENDIF(NOT "${project}" STREQUAL "llmath") +  SET(alltest_INCLUDE_DIRS +    ${LLMATH_INCLUDE_DIRS} +    ${LLCOMMON_INCLUDE_DIRS} +    ${LIBS_OPEN_DIR}/test +    ${GOOGLEMOCK_INCLUDE_DIRS} +    ) +  SET(alltest_LIBRARIES +    ${GOOGLEMOCK_LIBRARIES} +    ${PTHREAD_LIBRARY} +    ${WINDOWS_LIBRARIES} +    ) +  # Headers, for convenience in targets. +  SET(alltest_HEADER_FILES +    ${CMAKE_SOURCE_DIR}/test/test.h +    ) + +  # Use the default flags +  if (LINUX) +    SET(CMAKE_EXE_LINKER_FLAGS "") +  endif (LINUX) + +  # start the source test executable definitions +  SET(${project}_TEST_OUTPUT "") +  FOREACH (source ${sources}) +    STRING( REGEX REPLACE "(.*)\\.[^.]+$" "\\1" name ${source} ) +    STRING( REGEX REPLACE ".*\\.([^.]+)$" "\\1" extension ${source} ) +    IF(LL_TEST_VERBOSE) +      MESSAGE("LL_ADD_PROJECT_UNIT_TESTS UNITTEST_PROJECT_${project} individual source: ${source} (${name}.${extension})") +    ENDIF(LL_TEST_VERBOSE) + +    # +    # Per-codefile additional / external source, header, and include dir property extraction +    # +    # Source +    GET_SOURCE_FILE_PROPERTY(${name}_test_additional_SOURCE_FILES ${source} LL_TEST_ADDITIONAL_SOURCE_FILES) +    IF(${name}_test_additional_SOURCE_FILES MATCHES NOTFOUND) +      SET(${name}_test_additional_SOURCE_FILES "") +    ENDIF(${name}_test_additional_SOURCE_FILES MATCHES NOTFOUND) +    SET(${name}_test_SOURCE_FILES ${source} tests/${name}_test.${extension} ${alltest_SOURCE_FILES} ${${name}_test_additional_SOURCE_FILES} ) +    IF(LL_TEST_VERBOSE) +      MESSAGE("LL_ADD_PROJECT_UNIT_TESTS ${name}_test_SOURCE_FILES ${${name}_test_SOURCE_FILES}") +    ENDIF(LL_TEST_VERBOSE) +    # Headers +    GET_SOURCE_FILE_PROPERTY(${name}_test_additional_HEADER_FILES ${source} LL_TEST_ADDITIONAL_HEADER_FILES) +    IF(${name}_test_additional_HEADER_FILES MATCHES NOTFOUND) +      SET(${name}_test_additional_HEADER_FILES "") +    ENDIF(${name}_test_additional_HEADER_FILES MATCHES NOTFOUND) +    SET(${name}_test_HEADER_FILES ${name}.h ${${name}_test_additional_HEADER_FILES}) +    set_source_files_properties(${${name}_test_HEADER_FILES} PROPERTIES HEADER_FILE_ONLY TRUE) +    LIST(APPEND ${name}_test_SOURCE_FILES ${${name}_test_HEADER_FILES}) +    IF(LL_TEST_VERBOSE) +      MESSAGE("LL_ADD_PROJECT_UNIT_TESTS ${name}_test_HEADER_FILES ${${name}_test_HEADER_FILES}") +    ENDIF(LL_TEST_VERBOSE) +    # Include dirs +    GET_SOURCE_FILE_PROPERTY(${name}_test_additional_INCLUDE_DIRS ${source} LL_TEST_ADDITIONAL_INCLUDE_DIRS) +    IF(${name}_test_additional_INCLUDE_DIRS MATCHES NOTFOUND) +      SET(${name}_test_additional_INCLUDE_DIRS "") +    ENDIF(${name}_test_additional_INCLUDE_DIRS MATCHES NOTFOUND) +    INCLUDE_DIRECTORIES(${alltest_INCLUDE_DIRS} ${name}_test_additional_INCLUDE_DIRS ) +    IF(LL_TEST_VERBOSE) +      MESSAGE("LL_ADD_PROJECT_UNIT_TESTS ${name}_test_additional_INCLUDE_DIRS ${${name}_test_additional_INCLUDE_DIRS}") +    ENDIF(LL_TEST_VERBOSE) + + +    # Setup target +    ADD_EXECUTABLE(PROJECT_${project}_TEST_${name} ${${name}_test_SOURCE_FILES}) +    SET_TARGET_PROPERTIES(PROJECT_${project}_TEST_${name} PROPERTIES RUNTIME_OUTPUT_DIRECTORY "${EXE_STAGING_DIR}") + +    # +    # Per-codefile additional / external project dep and lib dep property extraction +    # +    # WARNING: it's REALLY IMPORTANT to not mix these. I guarantee it will not work in the future. + poppy 2009-04-19 +    # Projects +    GET_SOURCE_FILE_PROPERTY(${name}_test_additional_PROJECTS ${source} LL_TEST_ADDITIONAL_PROJECTS) +    IF(${name}_test_additional_PROJECTS MATCHES NOTFOUND) +      SET(${name}_test_additional_PROJECTS "") +    ENDIF(${name}_test_additional_PROJECTS MATCHES NOTFOUND) +    # Libraries +    GET_SOURCE_FILE_PROPERTY(${name}_test_additional_LIBRARIES ${source} LL_TEST_ADDITIONAL_LIBRARIES) +    IF(${name}_test_additional_LIBRARIES MATCHES NOTFOUND) +      SET(${name}_test_additional_LIBRARIES "") +    ENDIF(${name}_test_additional_LIBRARIES MATCHES NOTFOUND) +    IF(LL_TEST_VERBOSE) +      MESSAGE("LL_ADD_PROJECT_UNIT_TESTS ${name}_test_additional_PROJECTS ${${name}_test_additional_PROJECTS}") +      MESSAGE("LL_ADD_PROJECT_UNIT_TESTS ${name}_test_additional_LIBRARIES ${${name}_test_additional_LIBRARIES}") +    ENDIF(LL_TEST_VERBOSE) +    # Add to project +    TARGET_LINK_LIBRARIES(PROJECT_${project}_TEST_${name} ${alltest_LIBRARIES} ${alltest_DEP_TARGETS} ${${name}_test_additional_PROJECTS} ${${name}_test_additional_LIBRARIES} ) +    # Compile-time Definitions +    GET_SOURCE_FILE_PROPERTY(${name}_test_additional_CFLAGS ${source} LL_TEST_ADDITIONAL_CFLAGS) +     IF(NOT ${name}_test_additional_CFLAGS MATCHES NOTFOUND) +       SET_TARGET_PROPERTIES(PROJECT_${project}_TEST_${name} PROPERTIES COMPILE_FLAGS ${${name}_test_additional_CFLAGS} ) +       IF(LL_TEST_VERBOSE) +         MESSAGE("LL_ADD_PROJECT_UNIT_TESTS ${name}_test_additional_CFLAGS ${${name}_test_additional_CFLAGS}") +       ENDIF(LL_TEST_VERBOSE) +     ENDIF(NOT ${name}_test_additional_CFLAGS MATCHES NOTFOUND) +      +    # +    # Setup test targets +    # +    GET_TARGET_PROPERTY(TEST_EXE PROJECT_${project}_TEST_${name} LOCATION) +    SET(TEST_OUTPUT ${CMAKE_CURRENT_BINARY_DIR}/PROJECT_${project}_TEST_${name}_ok.txt) +    SET(TEST_CMD ${TEST_EXE} --touch=${TEST_OUTPUT} --sourcedir=${CMAKE_CURRENT_SOURCE_DIR}) + +    # daveh - what configuration does this use? Debug? it's cmake-time, not build time. + poppy 2009-04-19 +    IF(LL_TEST_VERBOSE) +      MESSAGE(STATUS "LL_ADD_PROJECT_UNIT_TESTS ${name} test_cmd  = ${TEST_CMD}") +    ENDIF(LL_TEST_VERBOSE) + +    SET_TEST_PATH(LD_LIBRARY_PATH) +    LL_TEST_COMMAND(TEST_SCRIPT_CMD "${LD_LIBRARY_PATH}" ${TEST_CMD}) +    IF(LL_TEST_VERBOSE) +      MESSAGE(STATUS "LL_ADD_PROJECT_UNIT_TESTS ${name} test_script  = ${TEST_SCRIPT_CMD}") +    ENDIF(LL_TEST_VERBOSE) +    # Add test  +    ADD_CUSTOM_COMMAND( +        OUTPUT ${TEST_OUTPUT} +        COMMAND ${TEST_SCRIPT_CMD} +        DEPENDS PROJECT_${project}_TEST_${name} +        WORKING_DIRECTORY ${CMAKE_CURRENT_BINARY_DIR} +        ) +    # Why not add custom target and add POST_BUILD command? +    # Slightly less uncertain behavior +    # (OUTPUT commands run non-deterministically AFAIK) + poppy 2009-04-19 +    # > I did not use a post build step as I could not make it notify of a  +    # > failure after the first time you build and fail a test. - daveh 2009-04-20 +    LIST(APPEND ${project}_TEST_OUTPUT ${TEST_OUTPUT}) +  ENDFOREACH (source) + +  # Add the test runner target per-project +  # (replaces old _test_ok targets all over the place) +  ADD_CUSTOM_TARGET(${project}_tests ALL DEPENDS ${${project}_TEST_OUTPUT}) +  ADD_DEPENDENCIES(${project} ${project}_tests) +ENDMACRO(LL_ADD_PROJECT_UNIT_TESTS) + +FUNCTION(LL_ADD_INTEGRATION_TEST  +    testname +    additional_source_files +    library_dependencies +# variable args +    ) +  if(TEST_DEBUG) +    message(STATUS "Adding INTEGRATION_TEST_${testname} - debug output is on") +  endif(TEST_DEBUG) +   +  SET(source_files +    tests/${testname}_test.cpp +    ${CMAKE_SOURCE_DIR}/test/test.cpp +    ${CMAKE_SOURCE_DIR}/test/lltut.cpp +    ${additional_source_files} +    ) + +  SET(libraries +    ${library_dependencies} +    ${GOOGLEMOCK_LIBRARIES} +    ${PTHREAD_LIBRARY} +    ) + +  # Add test executable build target +  if(TEST_DEBUG) +    message(STATUS "ADD_EXECUTABLE(INTEGRATION_TEST_${testname} ${source_files})") +  endif(TEST_DEBUG) +  ADD_EXECUTABLE(INTEGRATION_TEST_${testname} ${source_files}) +  SET_TARGET_PROPERTIES(INTEGRATION_TEST_${testname} PROPERTIES RUNTIME_OUTPUT_DIRECTORY "${EXE_STAGING_DIR}") +  if(STANDALONE) +    SET_TARGET_PROPERTIES(INTEGRATION_TEST_${testname} PROPERTIES COMPILE_FLAGS -I"${TUT_INCLUDE_DIR}") +  endif(STANDALONE) + +  # Add link deps to the executable +  if(TEST_DEBUG) +    message(STATUS "TARGET_LINK_LIBRARIES(INTEGRATION_TEST_${testname} ${libraries})") +  endif(TEST_DEBUG) +  TARGET_LINK_LIBRARIES(INTEGRATION_TEST_${testname} ${libraries}) + +  # Create the test running command +  SET(test_command ${ARGN}) +  GET_TARGET_PROPERTY(TEST_EXE INTEGRATION_TEST_${testname} LOCATION) +  LIST(FIND test_command "{}" test_exe_pos) +  IF(test_exe_pos LESS 0) +    # The {} marker means "the full pathname of the test executable." +    # test_exe_pos -1 means we didn't find it -- so append the test executable +    # name to $ARGN, the variable part of the arg list. This is convenient +    # shorthand for both straightforward execution of the test program (empty +    # $ARGN) and for running a "wrapper" program of some kind accepting the +    # pathname of the test program as the last of its args. You need specify +    # {} only if the test program's pathname isn't the last argument in the +    # desired command line. +    LIST(APPEND test_command "${TEST_EXE}") +  ELSE (test_exe_pos LESS 0) +    # Found {} marker at test_exe_pos. Remove the {}... +    LIST(REMOVE_AT test_command test_exe_pos) +    # ...and replace it with the actual name of the test executable. +    LIST(INSERT test_command test_exe_pos "${TEST_EXE}") +  ENDIF (test_exe_pos LESS 0) + +  SET_TEST_PATH(LD_LIBRARY_PATH) +  LL_TEST_COMMAND(TEST_SCRIPT_CMD "${LD_LIBRARY_PATH}" ${test_command}) + +  if(TEST_DEBUG) +    message(STATUS "TEST_SCRIPT_CMD: ${TEST_SCRIPT_CMD}") +  endif(TEST_DEBUG) + +  ADD_CUSTOM_COMMAND( +    TARGET INTEGRATION_TEST_${testname} +    POST_BUILD +    COMMAND ${TEST_SCRIPT_CMD} +    ) + +  # Use CTEST? Not sure how to yet... +  # ADD_TEST(INTEGRATION_TEST_RUNNER_${testname} ${TEST_SCRIPT_CMD}) + +ENDFUNCTION(LL_ADD_INTEGRATION_TEST) + +MACRO(SET_TEST_PATH LISTVAR) +  IF(WINDOWS) +    # We typically build/package only Release variants of third-party +    # libraries, so append the Release staging dir in case the library being +    # sought doesn't have a debug variant. +    set(${LISTVAR} ${SHARED_LIB_STAGING_DIR}/${CMAKE_CFG_INTDIR} ${SHARED_LIB_STAGING_DIR}/Release) +  ELSEIF(DARWIN) +    # We typically build/package only Release variants of third-party +    # libraries, so append the Release staging dir in case the library being +    # sought doesn't have a debug variant. +    set(${LISTVAR} ${SHARED_LIB_STAGING_DIR}/${CMAKE_CFG_INTDIR}/Resources ${SHARED_LIB_STAGING_DIR}/Release/Resources /usr/lib) +  ELSE(WINDOWS) +    # Linux uses a single staging directory anyway. +    IF (STANDALONE) +      set(${LISTVAR} ${CMAKE_BINARY_DIR}/llcommon /usr/lib /usr/local/lib) +    ELSE (STANDALONE) +      set(${LISTVAR} ${SHARED_LIB_STAGING_DIR} /usr/lib) +    ENDIF (STANDALONE) +  ENDIF(WINDOWS) +ENDMACRO(SET_TEST_PATH) diff --git a/indra/newview/llassetuploadresponders.cpp b/indra/newview/llassetuploadresponders.cpp index f12bc16d4b..dd5bc74b2a 100644 --- a/indra/newview/llassetuploadresponders.cpp +++ b/indra/newview/llassetuploadresponders.cpp @@ -126,6 +126,7 @@ void LLAssetUploadResponder::error(U32 statusNum, const std::string& reason)  			break;  	}  	LLUploadDialog::modalUploadFinished(); +	LLFilePicker::instance().reset();  // unlock file picker when bulk upload fails  }  //virtual  diff --git a/indra/newview/llfloaterwebcontent.cpp b/indra/newview/llfloaterwebcontent.cpp index 51726112a0..058567492b 100644 --- a/indra/newview/llfloaterwebcontent.cpp +++ b/indra/newview/llfloaterwebcontent.cpp @@ -1,402 +1,402 @@ -/**
 - * @file llfloaterwebcontent.cpp
 - * @brief floater for displaying web content - e.g. profiles and search (eventually)
 - *
 - * $LicenseInfo:firstyear=2006&license=viewerlgpl$
 - * Second Life Viewer Source Code
 - * Copyright (C) 2010, Linden Research, Inc.
 - *
 - * This library is free software; you can redistribute it and/or
 - * modify it under the terms of the GNU Lesser General Public
 - * License as published by the Free Software Foundation;
 - * version 2.1 of the License only.
 - *
 - * This library 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
 - * Lesser General Public License for more details.
 - *
 - * You should have received a copy of the GNU Lesser General Public
 - * License along with this library; if not, write to the Free Software
 - * Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA  02110-1301  USA
 - *
 - * Linden Research, Inc., 945 Battery Street, San Francisco, CA  94111  USA
 - * $/LicenseInfo$
 - */
 -
 -#include "llviewerprecompiledheaders.h"
 -
 -#include "llcombobox.h"
 -#include "lliconctrl.h"
 -#include "llfloaterreg.h"
 -#include "lllayoutstack.h"
 -#include "llpluginclassmedia.h"
 -#include "llprogressbar.h"
 -#include "lltextbox.h"
 -#include "llurlhistory.h"
 -#include "llviewercontrol.h"
 -#include "llweb.h"
 -#include "llwindow.h"
 -
 -#include "llfloaterwebcontent.h"
 -
 -LLFloaterWebContent::LLFloaterWebContent( const LLSD& key )
 -	: LLFloater( key )
 -{
 -	mCommitCallbackRegistrar.add( "WebContent.Back", boost::bind( &LLFloaterWebContent::onClickBack, this ));
 -	mCommitCallbackRegistrar.add( "WebContent.Forward", boost::bind( &LLFloaterWebContent::onClickForward, this ));
 -	mCommitCallbackRegistrar.add( "WebContent.Reload", boost::bind( &LLFloaterWebContent::onClickReload, this ));
 -	mCommitCallbackRegistrar.add( "WebContent.Stop", boost::bind( &LLFloaterWebContent::onClickStop, this ));
 -	mCommitCallbackRegistrar.add( "WebContent.EnterAddress", boost::bind( &LLFloaterWebContent::onEnterAddress, this ));
 -	mCommitCallbackRegistrar.add( "WebContent.PopExternal", boost::bind( &LLFloaterWebContent::onPopExternal, this ));
 -}
 -
 -BOOL LLFloaterWebContent::postBuild()
 -{
 -	// these are used in a bunch of places so cache them
 -	mWebBrowser = getChild< LLMediaCtrl >( "webbrowser" );
 -	mAddressCombo = getChild< LLComboBox >( "address" );
 -	mStatusBarText = getChild< LLTextBox >( "statusbartext" );
 -	mStatusBarProgress = getChild<LLProgressBar>("statusbarprogress" );
 -
 -	// observe browser events
 -	mWebBrowser->addObserver( this );
 -
 -	// these buttons are always enabled
 -	getChildView("reload")->setEnabled( true );
 -	getChildView("popexternal")->setEnabled( true );
 -
 -	// cache image for secure browsing
 -	mSecureLockIcon = getChild< LLIconCtrl >("media_secure_lock_flag");
 -
 -	// initialize the URL history using the system URL History manager
 -	initializeURLHistory();
 -
 -	return TRUE;
 -}
 -
 -void LLFloaterWebContent::initializeURLHistory()
 -{
 -	// start with an empty list
 -	LLCtrlListInterface* url_list = childGetListInterface("address");
 -	if (url_list)
 -	{
 -		url_list->operateOnAll(LLCtrlListInterface::OP_DELETE);
 -	}
 -
 -	// Get all of the entries in the "browser" collection
 -	LLSD browser_history = LLURLHistory::getURLHistory("browser");
 -	LLSD::array_iterator iter_history =
 -		browser_history.beginArray();
 -	LLSD::array_iterator end_history =
 -		browser_history.endArray();
 -	for(; iter_history != end_history; ++iter_history)
 -	{
 -		std::string url = (*iter_history).asString();
 -		if(! url.empty())
 -			url_list->addSimpleElement(url);
 -	}
 -}
 -
 -//static
 -void LLFloaterWebContent::create( const std::string &url, const std::string& target, const std::string& uuid )
 -{
 -	lldebugs << "url = " << url << ", target = " << target << ", uuid = " << uuid << llendl;
 -
 -	std::string tag = target;
 -
 -	if(target.empty() || target == "_blank")
 -	{
 -		if(!uuid.empty())
 -		{
 -			tag = uuid;
 -		}
 -		else
 -		{
 -			// create a unique tag for this instance
 -			LLUUID id;
 -			id.generate();
 -			tag = id.asString();
 -		}
 -	}
 -
 -	S32 browser_window_limit = gSavedSettings.getS32("WebContentWindowLimit");
 -
 -	if(LLFloaterReg::findInstance("web_content", tag) != NULL)
 -	{
 -		// There's already a web browser for this tag, so we won't be opening a new window.
 -	}
 -	else if(browser_window_limit != 0)
 -	{
 -		// showInstance will open a new window.  Figure out how many web browsers are already open,
 -		// and close the least recently opened one if this will put us over the limit.
 -
 -		LLFloaterReg::const_instance_list_t &instances = LLFloaterReg::getFloaterList("web_content");
 -		lldebugs << "total instance count is " << instances.size() << llendl;
 -
 -		for(LLFloaterReg::const_instance_list_t::const_iterator iter = instances.begin(); iter != instances.end(); iter++)
 -		{
 -			lldebugs << "    " << (*iter)->getKey() << llendl;
 -		}
 -
 -		if(instances.size() >= (size_t)browser_window_limit)
 -		{
 -			// Destroy the least recently opened instance
 -			(*instances.begin())->closeFloater();
 -		}
 -	}
 -
 -	LLFloaterWebContent *browser = dynamic_cast<LLFloaterWebContent*> (LLFloaterReg::showInstance("web_content", tag));
 -	llassert(browser);
 -	if(browser)
 -	{
 -		browser->mUUID = uuid;
 -
 -		// tell the browser instance to load the specified URL
 -		browser->open_media(url, target);
 -		LLViewerMedia::proxyWindowOpened(target, uuid);
 -	}
 -}
 -
 -//static
 -void LLFloaterWebContent::closeRequest(const std::string &uuid)
 -{
 -	LLFloaterReg::const_instance_list_t& inst_list = LLFloaterReg::getFloaterList("web_content");
 -	lldebugs << "instance list size is " << inst_list.size() << ", incoming uuid is " << uuid << llendl;
 -	for (LLFloaterReg::const_instance_list_t::const_iterator iter = inst_list.begin(); iter != inst_list.end(); ++iter)
 -	{
 -		LLFloaterWebContent* i = dynamic_cast<LLFloaterWebContent*>(*iter);
 -		lldebugs << "    " << i->mUUID << llendl;
 -		if (i && i->mUUID == uuid)
 -		{
 -			i->closeFloater(false);
 -			return;
 - 		}
 - 	}
 -}
 -
 -//static
 -void LLFloaterWebContent::geometryChanged(const std::string &uuid, S32 x, S32 y, S32 width, S32 height)
 -{
 -	LLFloaterReg::const_instance_list_t& inst_list = LLFloaterReg::getFloaterList("web_content");
 -	lldebugs << "instance list size is " << inst_list.size() << ", incoming uuid is " << uuid << llendl;
 -	for (LLFloaterReg::const_instance_list_t::const_iterator iter = inst_list.begin(); iter != inst_list.end(); ++iter)
 -	{
 -		LLFloaterWebContent* i = dynamic_cast<LLFloaterWebContent*>(*iter);
 -		lldebugs << "    " << i->mUUID << llendl;
 -		if (i && i->mUUID == uuid)
 -		{
 -			i->geometryChanged(x, y, width, height);
 -			return;
 -		}
 -	}
 -}
 -
 -void LLFloaterWebContent::geometryChanged(S32 x, S32 y, S32 width, S32 height)
 -{
 -	// Make sure the layout of the browser control is updated, so this calculation is correct.
 -	LLLayoutStack::updateClass();
 -
 -	// TODO: need to adjust size and constrain position to make sure floaters aren't moved outside the window view, etc.
 -	LLCoordWindow window_size;
 -	getWindow()->getSize(&window_size);
 -
 -	// Adjust width and height for the size of the chrome on the web Browser window.
 -	width += getRect().getWidth() - mWebBrowser->getRect().getWidth();
 -	height += getRect().getHeight() - mWebBrowser->getRect().getHeight();
 -
 -	LLRect geom;
 -	geom.setOriginAndSize(x, window_size.mY - (y + height), width, height);
 -
 -	lldebugs << "geometry change: " << geom << llendl;
 -
 -	handleReshape(geom,false);
 -}
 -
 -void LLFloaterWebContent::open_media(const std::string& web_url, const std::string& target)
 -{
 -	// Specifying a mime type of text/html here causes the plugin system to skip the MIME type probe and just open a browser plugin.
 -	mWebBrowser->setHomePageUrl(web_url, "text/html");
 -	mWebBrowser->setTarget(target);
 -	mWebBrowser->navigateTo(web_url, "text/html");
 -	set_current_url(web_url);
 -}
 -
 -//virtual
 -void LLFloaterWebContent::onClose(bool app_quitting)
 -{
 -	LLViewerMedia::proxyWindowClosed(mUUID);
 -	destroy();
 -}
 -
 -// virtual
 -void LLFloaterWebContent::draw()
 -{
 -	// this is asychronous so we need to keep checking
 -	getChildView( "back" )->setEnabled( mWebBrowser->canNavigateBack() );
 -	getChildView( "forward" )->setEnabled( mWebBrowser->canNavigateForward() );
 -
 -	LLFloater::draw();
 -}
 -
 -// virtual
 -void LLFloaterWebContent::handleMediaEvent(LLPluginClassMedia* self, EMediaEvent event)
 -{
 -	if(event == MEDIA_EVENT_LOCATION_CHANGED)
 -	{
 -		const std::string url = self->getLocation();
 -
 -		if ( url.length() )
 -			mStatusBarText->setText( url );
 -
 -		set_current_url( url );
 -	}
 -	else if(event == MEDIA_EVENT_NAVIGATE_BEGIN)
 -	{
 -		// flags are sent with this event
 -		getChildView("back")->setEnabled( self->getHistoryBackAvailable() );
 -		getChildView("forward")->setEnabled( self->getHistoryForwardAvailable() );
 -
 -		// toggle visibility of these buttons based on browser state
 -		getChildView("reload")->setVisible( false );
 -		getChildView("stop")->setVisible( true );
 -
 -		// turn "on" progress bar now we're about to start loading
 -		mStatusBarProgress->setVisible( true );
 -	}
 -	else if(event == MEDIA_EVENT_NAVIGATE_COMPLETE)
 -	{
 -		// flags are sent with this event
 -		getChildView("back")->setEnabled( self->getHistoryBackAvailable() );
 -		getChildView("forward")->setEnabled( self->getHistoryForwardAvailable() );
 -
 -		// toggle visibility of these buttons based on browser state
 -		getChildView("reload")->setVisible( true );
 -		getChildView("stop")->setVisible( false );
 -
 -		// turn "off" progress bar now we're loaded
 -		mStatusBarProgress->setVisible( false );
 -
 -		// we populate the status bar with URLs as they change so clear it now we're done
 -		const std::string end_str = "";
 -		mStatusBarText->setText( end_str );
 -
 -		// decide if secure browsing icon should be displayed
 -		std::string prefix =  std::string("https://");
 -		std::string test_prefix = mCurrentURL.substr(0, prefix.length());
 -		LLStringUtil::toLower(test_prefix);
 -		if(test_prefix == prefix)
 -		{
 -			mSecureLockIcon->setVisible(true);
 -		}
 -		else
 -		{
 -			mSecureLockIcon->setVisible(false);
 -		}
 -	}
 -	else if(event == MEDIA_EVENT_CLOSE_REQUEST)
 -	{
 -		// The browser instance wants its window closed.
 -		closeFloater();
 -	}
 -	else if(event == MEDIA_EVENT_GEOMETRY_CHANGE)
 -	{
 -		geometryChanged(self->getGeometryX(), self->getGeometryY(), self->getGeometryWidth(), self->getGeometryHeight());
 -	}
 -	else if(event == MEDIA_EVENT_STATUS_TEXT_CHANGED )
 -	{
 -		const std::string text = self->getStatusText();
 -		if ( text.length() )
 -			mStatusBarText->setText( text );
 -	}
 -	else if(event == MEDIA_EVENT_PROGRESS_UPDATED )
 -	{
 -		int percent = (int)self->getProgressPercent();
 -		mStatusBarProgress->setValue( percent );
 -	}
 -	else if(event == MEDIA_EVENT_NAME_CHANGED )
 -	{
 -		std::string page_title = self->getMediaName();
 -		// simulate browser behavior - title is empty, use the current URL
 -		if ( page_title.length() > 0 )
 -			setTitle( page_title );
 -		else
 -			setTitle( mCurrentURL );
 -	}
 -	else if(event == MEDIA_EVENT_LINK_HOVERED )
 -	{
 -		const std::string link = self->getHoverLink();
 -		mStatusBarText->setText( link );
 -	}
 -}
 -
 -void LLFloaterWebContent::set_current_url(const std::string& url)
 -{
 -	mCurrentURL = url;
 -
 -	// serialize url history into the system URL History manager
 -	LLURLHistory::removeURL("browser", mCurrentURL);
 -	LLURLHistory::addURL("browser", mCurrentURL);
 -
 -	mAddressCombo->remove( mCurrentURL );
 -	mAddressCombo->add( mCurrentURL );
 -	mAddressCombo->selectByValue( mCurrentURL );
 -}
 -
 -void LLFloaterWebContent::onClickForward()
 -{
 -	mWebBrowser->navigateForward();
 -}
 -
 -void LLFloaterWebContent::onClickBack()
 -{
 -	mWebBrowser->navigateBack();
 -}
 -
 -void LLFloaterWebContent::onClickReload()
 -{
 -
 -	if( mWebBrowser->getMediaPlugin() )
 -	{
 -		bool ignore_cache = true;
 -		mWebBrowser->getMediaPlugin()->browse_reload( ignore_cache );
 -	}
 -	else
 -	{
 -		mWebBrowser->navigateTo(mCurrentURL);
 -	}
 -}
 -
 -void LLFloaterWebContent::onClickStop()
 -{
 -	if( mWebBrowser->getMediaPlugin() )
 -		mWebBrowser->getMediaPlugin()->browse_stop();
 -
 -	// still should happen when we catch the navigate complete event
 -	// but sometimes (don't know why) that event isn't sent from Qt
 -	// and we getto a point where the stop button stays active.
 -	getChildView("reload")->setVisible( true );
 -	getChildView("stop")->setVisible( false );
 -}
 -
 -void LLFloaterWebContent::onEnterAddress()
 -{
 -	// make sure there is at least something there.
 -	// (perhaps this test should be for minimum length of a URL)
 -	std::string url = mAddressCombo->getValue().asString();
 -	if ( url.length() > 0 )
 -	{
 -		mWebBrowser->navigateTo( url, "text/html");
 -	};
 -}
 -
 -void LLFloaterWebContent::onPopExternal()
 -{
 -	// make sure there is at least something there.
 -	// (perhaps this test should be for minimum length of a URL)
 -	std::string url = mAddressCombo->getValue().asString();
 -	if ( url.length() > 0 )
 -	{
 -		LLWeb::loadURLExternal( url );
 -	};
 -}
 +/** + * @file llfloaterwebcontent.cpp + * @brief floater for displaying web content - e.g. profiles and search (eventually) + * + * $LicenseInfo:firstyear=2006&license=viewerlgpl$ + * Second Life Viewer Source Code + * Copyright (C) 2010, Linden Research, Inc. + * + * This library is free software; you can redistribute it and/or + * modify it under the terms of the GNU Lesser General Public + * License as published by the Free Software Foundation; + * version 2.1 of the License only. + * + * This library 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 + * Lesser General Public License for more details. + * + * You should have received a copy of the GNU Lesser General Public + * License along with this library; if not, write to the Free Software + * Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA  02110-1301  USA + * + * Linden Research, Inc., 945 Battery Street, San Francisco, CA  94111  USA + * $/LicenseInfo$ + */ + +#include "llviewerprecompiledheaders.h" + +#include "llcombobox.h" +#include "lliconctrl.h" +#include "llfloaterreg.h" +#include "lllayoutstack.h" +#include "llpluginclassmedia.h" +#include "llprogressbar.h" +#include "lltextbox.h" +#include "llurlhistory.h" +#include "llviewercontrol.h" +#include "llweb.h" +#include "llwindow.h" + +#include "llfloaterwebcontent.h" + +LLFloaterWebContent::LLFloaterWebContent( const LLSD& key ) +	: LLFloater( key ) +{ +	mCommitCallbackRegistrar.add( "WebContent.Back", boost::bind( &LLFloaterWebContent::onClickBack, this )); +	mCommitCallbackRegistrar.add( "WebContent.Forward", boost::bind( &LLFloaterWebContent::onClickForward, this )); +	mCommitCallbackRegistrar.add( "WebContent.Reload", boost::bind( &LLFloaterWebContent::onClickReload, this )); +	mCommitCallbackRegistrar.add( "WebContent.Stop", boost::bind( &LLFloaterWebContent::onClickStop, this )); +	mCommitCallbackRegistrar.add( "WebContent.EnterAddress", boost::bind( &LLFloaterWebContent::onEnterAddress, this )); +	mCommitCallbackRegistrar.add( "WebContent.PopExternal", boost::bind( &LLFloaterWebContent::onPopExternal, this )); +} + +BOOL LLFloaterWebContent::postBuild() +{ +	// these are used in a bunch of places so cache them +	mWebBrowser = getChild< LLMediaCtrl >( "webbrowser" ); +	mAddressCombo = getChild< LLComboBox >( "address" ); +	mStatusBarText = getChild< LLTextBox >( "statusbartext" ); +	mStatusBarProgress = getChild<LLProgressBar>("statusbarprogress" ); + +	// observe browser events +	mWebBrowser->addObserver( this ); + +	// these buttons are always enabled +	getChildView("reload")->setEnabled( true ); +	getChildView("popexternal")->setEnabled( true ); + +	// cache image for secure browsing +	mSecureLockIcon = getChild< LLIconCtrl >("media_secure_lock_flag"); + +	// initialize the URL history using the system URL History manager +	initializeURLHistory(); + +	return TRUE; +} + +void LLFloaterWebContent::initializeURLHistory() +{ +	// start with an empty list +	LLCtrlListInterface* url_list = childGetListInterface("address"); +	if (url_list) +	{ +		url_list->operateOnAll(LLCtrlListInterface::OP_DELETE); +	} + +	// Get all of the entries in the "browser" collection +	LLSD browser_history = LLURLHistory::getURLHistory("browser"); +	LLSD::array_iterator iter_history = +		browser_history.beginArray(); +	LLSD::array_iterator end_history = +		browser_history.endArray(); +	for(; iter_history != end_history; ++iter_history) +	{ +		std::string url = (*iter_history).asString(); +		if(! url.empty()) +			url_list->addSimpleElement(url); +	} +} + +//static +void LLFloaterWebContent::create( const std::string &url, const std::string& target, const std::string& uuid ) +{ +	lldebugs << "url = " << url << ", target = " << target << ", uuid = " << uuid << llendl; + +	std::string tag = target; + +	if(target.empty() || target == "_blank") +	{ +		if(!uuid.empty()) +		{ +			tag = uuid; +		} +		else +		{ +			// create a unique tag for this instance +			LLUUID id; +			id.generate(); +			tag = id.asString(); +		} +	} + +	S32 browser_window_limit = gSavedSettings.getS32("WebContentWindowLimit"); + +	if(LLFloaterReg::findInstance("web_content", tag) != NULL) +	{ +		// There's already a web browser for this tag, so we won't be opening a new window. +	} +	else if(browser_window_limit != 0) +	{ +		// showInstance will open a new window.  Figure out how many web browsers are already open, +		// and close the least recently opened one if this will put us over the limit. + +		LLFloaterReg::const_instance_list_t &instances = LLFloaterReg::getFloaterList("web_content"); +		lldebugs << "total instance count is " << instances.size() << llendl; + +		for(LLFloaterReg::const_instance_list_t::const_iterator iter = instances.begin(); iter != instances.end(); iter++) +		{ +			lldebugs << "    " << (*iter)->getKey() << llendl; +		} + +		if(instances.size() >= (size_t)browser_window_limit) +		{ +			// Destroy the least recently opened instance +			(*instances.begin())->closeFloater(); +		} +	} + +	LLFloaterWebContent *browser = dynamic_cast<LLFloaterWebContent*> (LLFloaterReg::showInstance("web_content", tag)); +	llassert(browser); +	if(browser) +	{ +		browser->mUUID = uuid; + +		// tell the browser instance to load the specified URL +		browser->open_media(url, target); +		LLViewerMedia::proxyWindowOpened(target, uuid); +	} +} + +//static +void LLFloaterWebContent::closeRequest(const std::string &uuid) +{ +	LLFloaterReg::const_instance_list_t& inst_list = LLFloaterReg::getFloaterList("web_content"); +	lldebugs << "instance list size is " << inst_list.size() << ", incoming uuid is " << uuid << llendl; +	for (LLFloaterReg::const_instance_list_t::const_iterator iter = inst_list.begin(); iter != inst_list.end(); ++iter) +	{ +		LLFloaterWebContent* i = dynamic_cast<LLFloaterWebContent*>(*iter); +		lldebugs << "    " << i->mUUID << llendl; +		if (i && i->mUUID == uuid) +		{ +			i->closeFloater(false); +			return; + 		} + 	} +} + +//static +void LLFloaterWebContent::geometryChanged(const std::string &uuid, S32 x, S32 y, S32 width, S32 height) +{ +	LLFloaterReg::const_instance_list_t& inst_list = LLFloaterReg::getFloaterList("web_content"); +	lldebugs << "instance list size is " << inst_list.size() << ", incoming uuid is " << uuid << llendl; +	for (LLFloaterReg::const_instance_list_t::const_iterator iter = inst_list.begin(); iter != inst_list.end(); ++iter) +	{ +		LLFloaterWebContent* i = dynamic_cast<LLFloaterWebContent*>(*iter); +		lldebugs << "    " << i->mUUID << llendl; +		if (i && i->mUUID == uuid) +		{ +			i->geometryChanged(x, y, width, height); +			return; +		} +	} +} + +void LLFloaterWebContent::geometryChanged(S32 x, S32 y, S32 width, S32 height) +{ +	// Make sure the layout of the browser control is updated, so this calculation is correct. +	LLLayoutStack::updateClass(); + +	// TODO: need to adjust size and constrain position to make sure floaters aren't moved outside the window view, etc. +	LLCoordWindow window_size; +	getWindow()->getSize(&window_size); + +	// Adjust width and height for the size of the chrome on the web Browser window. +	width += getRect().getWidth() - mWebBrowser->getRect().getWidth(); +	height += getRect().getHeight() - mWebBrowser->getRect().getHeight(); + +	LLRect geom; +	geom.setOriginAndSize(x, window_size.mY - (y + height), width, height); + +	lldebugs << "geometry change: " << geom << llendl; + +	handleReshape(geom,false); +} + +void LLFloaterWebContent::open_media(const std::string& web_url, const std::string& target) +{ +	// Specifying a mime type of text/html here causes the plugin system to skip the MIME type probe and just open a browser plugin. +	mWebBrowser->setHomePageUrl(web_url, "text/html"); +	mWebBrowser->setTarget(target); +	mWebBrowser->navigateTo(web_url, "text/html"); +	set_current_url(web_url); +} + +//virtual +void LLFloaterWebContent::onClose(bool app_quitting) +{ +	LLViewerMedia::proxyWindowClosed(mUUID); +	destroy(); +} + +// virtual +void LLFloaterWebContent::draw() +{ +	// this is asychronous so we need to keep checking +	getChildView( "back" )->setEnabled( mWebBrowser->canNavigateBack() ); +	getChildView( "forward" )->setEnabled( mWebBrowser->canNavigateForward() ); + +	LLFloater::draw(); +} + +// virtual +void LLFloaterWebContent::handleMediaEvent(LLPluginClassMedia* self, EMediaEvent event) +{ +	if(event == MEDIA_EVENT_LOCATION_CHANGED) +	{ +		const std::string url = self->getLocation(); + +		if ( url.length() ) +			mStatusBarText->setText( url ); + +		set_current_url( url ); +	} +	else if(event == MEDIA_EVENT_NAVIGATE_BEGIN) +	{ +		// flags are sent with this event +		getChildView("back")->setEnabled( self->getHistoryBackAvailable() ); +		getChildView("forward")->setEnabled( self->getHistoryForwardAvailable() ); + +		// toggle visibility of these buttons based on browser state +		getChildView("reload")->setVisible( false ); +		getChildView("stop")->setVisible( true ); + +		// turn "on" progress bar now we're about to start loading +		mStatusBarProgress->setVisible( true ); +	} +	else if(event == MEDIA_EVENT_NAVIGATE_COMPLETE) +	{ +		// flags are sent with this event +		getChildView("back")->setEnabled( self->getHistoryBackAvailable() ); +		getChildView("forward")->setEnabled( self->getHistoryForwardAvailable() ); + +		// toggle visibility of these buttons based on browser state +		getChildView("reload")->setVisible( true ); +		getChildView("stop")->setVisible( false ); + +		// turn "off" progress bar now we're loaded +		mStatusBarProgress->setVisible( false ); + +		// we populate the status bar with URLs as they change so clear it now we're done +		const std::string end_str = ""; +		mStatusBarText->setText( end_str ); + +		// decide if secure browsing icon should be displayed +		std::string prefix =  std::string("https://"); +		std::string test_prefix = mCurrentURL.substr(0, prefix.length()); +		LLStringUtil::toLower(test_prefix); +		if(test_prefix == prefix) +		{ +			mSecureLockIcon->setVisible(true); +		} +		else +		{ +			mSecureLockIcon->setVisible(false); +		} +	} +	else if(event == MEDIA_EVENT_CLOSE_REQUEST) +	{ +		// The browser instance wants its window closed. +		closeFloater(); +	} +	else if(event == MEDIA_EVENT_GEOMETRY_CHANGE) +	{ +		geometryChanged(self->getGeometryX(), self->getGeometryY(), self->getGeometryWidth(), self->getGeometryHeight()); +	} +	else if(event == MEDIA_EVENT_STATUS_TEXT_CHANGED ) +	{ +		const std::string text = self->getStatusText(); +		if ( text.length() ) +			mStatusBarText->setText( text ); +	} +	else if(event == MEDIA_EVENT_PROGRESS_UPDATED ) +	{ +		int percent = (int)self->getProgressPercent(); +		mStatusBarProgress->setValue( percent ); +	} +	else if(event == MEDIA_EVENT_NAME_CHANGED ) +	{ +		std::string page_title = self->getMediaName(); +		// simulate browser behavior - title is empty, use the current URL +		if ( page_title.length() > 0 ) +			setTitle( page_title ); +		else +			setTitle( mCurrentURL ); +	} +	else if(event == MEDIA_EVENT_LINK_HOVERED ) +	{ +		const std::string link = self->getHoverLink(); +		mStatusBarText->setText( link ); +	} +} + +void LLFloaterWebContent::set_current_url(const std::string& url) +{ +	mCurrentURL = url; + +	// serialize url history into the system URL History manager +	LLURLHistory::removeURL("browser", mCurrentURL); +	LLURLHistory::addURL("browser", mCurrentURL); + +	mAddressCombo->remove( mCurrentURL ); +	mAddressCombo->add( mCurrentURL ); +	mAddressCombo->selectByValue( mCurrentURL ); +} + +void LLFloaterWebContent::onClickForward() +{ +	mWebBrowser->navigateForward(); +} + +void LLFloaterWebContent::onClickBack() +{ +	mWebBrowser->navigateBack(); +} + +void LLFloaterWebContent::onClickReload() +{ + +	if( mWebBrowser->getMediaPlugin() ) +	{ +		bool ignore_cache = true; +		mWebBrowser->getMediaPlugin()->browse_reload( ignore_cache ); +	} +	else +	{ +		mWebBrowser->navigateTo(mCurrentURL); +	} +} + +void LLFloaterWebContent::onClickStop() +{ +	if( mWebBrowser->getMediaPlugin() ) +		mWebBrowser->getMediaPlugin()->browse_stop(); + +	// still should happen when we catch the navigate complete event +	// but sometimes (don't know why) that event isn't sent from Qt +	// and we getto a point where the stop button stays active. +	getChildView("reload")->setVisible( true ); +	getChildView("stop")->setVisible( false ); +} + +void LLFloaterWebContent::onEnterAddress() +{ +	// make sure there is at least something there. +	// (perhaps this test should be for minimum length of a URL) +	std::string url = mAddressCombo->getValue().asString(); +	if ( url.length() > 0 ) +	{ +		mWebBrowser->navigateTo( url, "text/html"); +	}; +} + +void LLFloaterWebContent::onPopExternal() +{ +	// make sure there is at least something there. +	// (perhaps this test should be for minimum length of a URL) +	std::string url = mAddressCombo->getValue().asString(); +	if ( url.length() > 0 ) +	{ +		LLWeb::loadURLExternal( url ); +	}; +} diff --git a/indra/newview/llfloaterwebcontent.h b/indra/newview/llfloaterwebcontent.h index 001d822ada..ecc7e970d8 100644 --- a/indra/newview/llfloaterwebcontent.h +++ b/indra/newview/llfloaterwebcontent.h @@ -1,82 +1,82 @@ -/**
 - * @file llfloaterwebcontent.h
 - * @brief floater for displaying web content - e.g. profiles and search (eventually)
 - *
 - * $LicenseInfo:firstyear=2006&license=viewerlgpl$
 - * Second Life Viewer Source Code
 - * Copyright (C) 2010, Linden Research, Inc.
 - *
 - * This library is free software; you can redistribute it and/or
 - * modify it under the terms of the GNU Lesser General Public
 - * License as published by the Free Software Foundation;
 - * version 2.1 of the License only.
 - *
 - * This library 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
 - * Lesser General Public License for more details.
 - *
 - * You should have received a copy of the GNU Lesser General Public
 - * License along with this library; if not, write to the Free Software
 - * Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA  02110-1301  USA
 - *
 - * Linden Research, Inc., 945 Battery Street, San Francisco, CA  94111  USA
 - * $/LicenseInfo$
 - */
 -
 -#ifndef LL_LLFLOATERWEBCONTENT_H
 -#define LL_LLFLOATERWEBCONTENT_H
 -
 -#include "llfloater.h"
 -#include "llmediactrl.h"
 -
 -class LLMediaCtrl;
 -class LLComboBox;
 -class LLTextBox;
 -class LLProgressBar;
 -class LLIconCtrl;
 -
 -class LLFloaterWebContent :
 -	public LLFloater,
 -	public LLViewerMediaObserver
 -{
 -public:
 -    LOG_CLASS(LLFloaterWebContent);
 -	LLFloaterWebContent(const LLSD& key);
 -
 +/** + * @file llfloaterwebcontent.h + * @brief floater for displaying web content - e.g. profiles and search (eventually) + * + * $LicenseInfo:firstyear=2006&license=viewerlgpl$ + * Second Life Viewer Source Code + * Copyright (C) 2010, Linden Research, Inc. + * + * This library is free software; you can redistribute it and/or + * modify it under the terms of the GNU Lesser General Public + * License as published by the Free Software Foundation; + * version 2.1 of the License only. + * + * This library 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 + * Lesser General Public License for more details. + * + * You should have received a copy of the GNU Lesser General Public + * License along with this library; if not, write to the Free Software + * Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA  02110-1301  USA + * + * Linden Research, Inc., 945 Battery Street, San Francisco, CA  94111  USA + * $/LicenseInfo$ + */ + +#ifndef LL_LLFLOATERWEBCONTENT_H +#define LL_LLFLOATERWEBCONTENT_H + +#include "llfloater.h" +#include "llmediactrl.h" + +class LLMediaCtrl; +class LLComboBox; +class LLTextBox; +class LLProgressBar; +class LLIconCtrl; + +class LLFloaterWebContent : +	public LLFloater, +	public LLViewerMediaObserver +{ +public: +    LOG_CLASS(LLFloaterWebContent); +	LLFloaterWebContent(const LLSD& key); +  	void initializeURLHistory(); -
 -	static void create(const std::string &url, const std::string& target, const std::string& uuid = LLStringUtil::null);
 -
 -	static void closeRequest(const std::string &uuid);
 -	static void geometryChanged(const std::string &uuid, S32 x, S32 y, S32 width, S32 height);
 -	void geometryChanged(S32 x, S32 y, S32 width, S32 height);
 -
 -	/* virtual */ BOOL postBuild();
 -	/* virtual */ void onClose(bool app_quitting);
 -	/* virtual */ void draw();
 -
 -	// inherited from LLViewerMediaObserver
 -	/*virtual*/ void handleMediaEvent(LLPluginClassMedia* self, EMediaEvent event);
 -
 -	void onClickBack();
 -	void onClickForward();
 -	void onClickReload();
 -	void onClickStop();
 -	void onEnterAddress();
 -	void onPopExternal();
 -
 -private:
 -	void open_media(const std::string& media_url, const std::string& target);
 -	void set_current_url(const std::string& url);
 -
 -	LLMediaCtrl* mWebBrowser;
 -	LLComboBox* mAddressCombo;
 -	LLIconCtrl *mSecureLockIcon;
 -	LLTextBox* mStatusBarText;
 -	LLProgressBar* mStatusBarProgress;
 -	std::string mCurrentURL;
 -	std::string mUUID;
 -};
 -
 -#endif  // LL_LLFLOATERWEBCONTENT_H
 + +	static void create(const std::string &url, const std::string& target, const std::string& uuid = LLStringUtil::null); + +	static void closeRequest(const std::string &uuid); +	static void geometryChanged(const std::string &uuid, S32 x, S32 y, S32 width, S32 height); +	void geometryChanged(S32 x, S32 y, S32 width, S32 height); + +	/* virtual */ BOOL postBuild(); +	/* virtual */ void onClose(bool app_quitting); +	/* virtual */ void draw(); + +	// inherited from LLViewerMediaObserver +	/*virtual*/ void handleMediaEvent(LLPluginClassMedia* self, EMediaEvent event); + +	void onClickBack(); +	void onClickForward(); +	void onClickReload(); +	void onClickStop(); +	void onEnterAddress(); +	void onPopExternal(); + +private: +	void open_media(const std::string& media_url, const std::string& target); +	void set_current_url(const std::string& url); + +	LLMediaCtrl* mWebBrowser; +	LLComboBox* mAddressCombo; +	LLIconCtrl *mSecureLockIcon; +	LLTextBox* mStatusBarText; +	LLProgressBar* mStatusBarProgress; +	std::string mCurrentURL; +	std::string mUUID; +}; + +#endif  // LL_LLFLOATERWEBCONTENT_H diff --git a/indra/newview/llimview.cpp b/indra/newview/llimview.cpp index ce305dcd89..afd565bb26 100644 --- a/indra/newview/llimview.cpp +++ b/indra/newview/llimview.cpp @@ -280,19 +280,19 @@ LLIMModel::LLIMSession::LLIMSession(const LLUUID& session_id, const std::string&  void LLIMModel::LLIMSession::onAdHocNameCache(const LLAvatarName& av_name)  {  	if (av_name.mIsDummy) -	{
 -		S32 separator_index = mName.rfind(" ");
 -		std::string name = mName.substr(0, separator_index);
 -		++separator_index;
 -		std::string conference_word = mName.substr(separator_index, mName.length());
 -
 -		// additional check that session name is what we expected
 -		if ("Conference" == conference_word)
 -		{
 -			LLStringUtil::format_map_t args;
 -			args["[AGENT_NAME]"] = name;
 -			LLTrans::findString(mName, "conference-title-incoming", args);
 -		}
 +	{ +		S32 separator_index = mName.rfind(" "); +		std::string name = mName.substr(0, separator_index); +		++separator_index; +		std::string conference_word = mName.substr(separator_index, mName.length()); + +		// additional check that session name is what we expected +		if ("Conference" == conference_word) +		{ +			LLStringUtil::format_map_t args; +			args["[AGENT_NAME]"] = name; +			LLTrans::findString(mName, "conference-title-incoming", args); +		}  	}  	else  	{ diff --git a/indra/newview/llimview.h b/indra/newview/llimview.h index e765a8da2f..a15776c207 100644 --- a/indra/newview/llimview.h +++ b/indra/newview/llimview.h @@ -100,7 +100,7 @@ public:  		void onAvatarNameCache(const LLUUID& avatar_id, const LLAvatarName& av_name); -		void onAdHocNameCache(const LLAvatarName& av_name);
 +		void onAdHocNameCache(const LLAvatarName& av_name);  		//*TODO make private  		static std::string generateHash(const std::set<LLUUID>& sorted_uuids); diff --git a/indra/newview/lllogchat.cpp b/indra/newview/lllogchat.cpp index 0121bbb1ed..9adf374c71 100644 --- a/indra/newview/lllogchat.cpp +++ b/indra/newview/lllogchat.cpp @@ -89,15 +89,15 @@ const static boost::regex TIMESTAMP_AND_STUFF("^(\\[\\d{4}/\\d{1,2}/\\d{1,2}\\s+   */  const static boost::regex NAME_AND_TEXT("([^:]+[:]{1})?(\\s*)(.*)"); -/**
 - * These are recognizers for matching the names of ad-hoc conferences when generating the log file name
 - * On invited side, an ad-hoc is named like "<first name> <last name> Conference 2010/11/19 03:43 f0f4"
 - * On initiating side, an ad-hoc is named like Ad-hoc Conference hash<hash>"
 - * If the naming system for ad-hoc conferences are change in LLIMModel::LLIMSession::buildHistoryFileName()
 - * then these definition need to be adjusted as well.
 - */
 -const static boost::regex INBOUND_CONFERENCE("^[a-zA-Z]{1,31} [a-zA-Z]{1,31} Conference [0-9]{4}/[0-9]{2}/[0-9]{2} [0-9]{2}:[0-9]{2} [0-9a-f]{4}");
 -const static boost::regex OUTBOUND_CONFERENCE("^Ad-hoc Conference hash[a-f0-9]{8}-[a-f0-9]{4}-[a-f0-9]{4}-[a-f0-9]{4}-[a-f0-9]{12}");
 +/** + * These are recognizers for matching the names of ad-hoc conferences when generating the log file name + * On invited side, an ad-hoc is named like "<first name> <last name> Conference 2010/11/19 03:43 f0f4" + * On initiating side, an ad-hoc is named like Ad-hoc Conference hash<hash>" + * If the naming system for ad-hoc conferences are change in LLIMModel::LLIMSession::buildHistoryFileName() + * then these definition need to be adjusted as well. + */ +const static boost::regex INBOUND_CONFERENCE("^[a-zA-Z]{1,31} [a-zA-Z]{1,31} Conference [0-9]{4}/[0-9]{2}/[0-9]{2} [0-9]{2}:[0-9]{2} [0-9a-f]{4}"); +const static boost::regex OUTBOUND_CONFERENCE("^Ad-hoc Conference hash[a-f0-9]{8}-[a-f0-9]{4}-[a-f0-9]{4}-[a-f0-9]{4}-[a-f0-9]{12}");  //is used to parse complex object names like "Xstreet SL Terminal v2.2.5 st"  const static std::string NAME_TEXT_DIVIDER(": "); diff --git a/indra/newview/skins/default/xui/en/floater_web_content.xml b/indra/newview/skins/default/xui/en/floater_web_content.xml index 2ad46824c2..d90d0feda9 100644 --- a/indra/newview/skins/default/xui/en/floater_web_content.xml +++ b/indra/newview/skins/default/xui/en/floater_web_content.xml @@ -1,190 +1,190 @@ -<?xml version="1.0" encoding="utf-8" standalone="yes" ?>
 -<floater
 -  legacy_header_height="18"
 -  can_resize="true"
 -  height="440"
 -  layout="topleft"
 -  min_height="140"
 -  min_width="467"
 -  name="floater_web_content"
 -  help_topic="floater_web_content"
 -  save_rect="true"
 -  auto_tile="true"
 -  title=""
 -  initial_mime_type="text/html"
 -  width="820">
 -  <layout_stack
 -    bottom="440"
 -    follows="left|right|top|bottom"
 -    layout="topleft"
 -    left="5"
 -    name="stack1"
 -    orientation="vertical"
 -    top="20"
 -    width="810">
 -    <layout_panel
 -      auto_resize="false"
 -      default_tab_group="1"
 -      height="22"
 -      layout="topleft"
 -      left="0"
 -      min_height="20"
 -      name="nav_controls"
 -      top="400"
 -      user_resize="false"
 -      width="800">
 -      <button
 -        image_overlay="Arrow_Left_Off"
 -		    image_disabled="PushButton_Disabled"
 -		    image_disabled_selected="PushButton_Disabled"
 -		    image_selected="PushButton_Selected"
 -		    image_unselected="PushButton_Off"
 -		    hover_glow_amount="0.15"
 -        tool_tip="Navigate back"
 -        follows="left|top"
 -        height="22"
 -        layout="topleft"
 -        left="1"
 -        name="back"
 -        top="0"
 -        width="22">
 -        <button.commit_callback
 -          function="WebContent.Back" />
 -      </button>
 -      <button
 -        image_overlay="Arrow_Right_Off"
 -		    image_disabled="PushButton_Disabled"
 -		    image_disabled_selected="PushButton_Disabled"
 -		    image_selected="PushButton_Selected"
 -		    image_unselected="PushButton_Off"
 -        tool_tip="Navigate forward"
 -        follows="left|top"
 -        height="22"
 -        layout="topleft"
 -        left="27"
 -        name="forward"
 -        top_delta="0"
 -        width="22">
 -        <button.commit_callback
 -          function="WebContent.Forward" />
 -      </button>
 -      <button
 -        image_overlay="Stop_Off"
 -		    image_disabled="PushButton_Disabled"
 -		    image_disabled_selected="PushButton_Disabled"
 -		    image_selected="PushButton_Selected"
 -		    image_unselected="PushButton_Off"
 -        tool_tip="Stop navigation"
 -        enabled="true"
 -        follows="left|top"
 -        height="22"
 -        layout="topleft"
 -        left="51"
 -        name="stop"
 -        top_delta="0"
 -        width="22">
 -        <button.commit_callback
 -          function="WebContent.Stop" />
 -      </button>
 -      <button
 -        image_overlay="Refresh_Off"
 -		    image_disabled="PushButton_Disabled"
 -		    image_disabled_selected="PushButton_Disabled"
 -		    image_selected="PushButton_Selected"
 -		    image_unselected="PushButton_Off"
 -        tool_tip="Reload page"
 -        follows="left|top"
 -        height="22"
 -        layout="topleft"
 -        left="51"
 -        name="reload"
 -        top_delta="0"
 -        width="22">
 -        <button.commit_callback
 -          function="WebContent.Reload" />
 -      </button>
 -      <combo_box
 -        allow_text_entry="true"
 -        follows="left|top|right"
 -        tab_group="1"
 -        height="22"
 -        layout="topleft"
 -        left_pad="4"
 -        max_chars="1024"
 -        name="address"
 -        combo_editor.select_on_focus="true"
 -        tool_tip="Enter URL here"
 -        top_delta="0"
 -        width="702">
 -        <combo_box.commit_callback
 -          function="WebContent.EnterAddress" />
 -      </combo_box>
 -      <icon
 -        name="media_secure_lock_flag"
 -        height="16"
 -        follows="top|right"
 -        image_name="Lock2"
 -        layout="topleft"
 -        left_delta="656"
 -        top_delta="2"
 -        visible="false" 
 -        tool_tip="Secured Browsing"
 -        width="16" />
 -      <button
 -        image_overlay="ExternalBrowser_Off"
 -		    image_disabled="PushButton_Disabled"
 -		    image_disabled_selected="PushButton_Disabled"
 -		    image_selected="PushButton_Selected"
 -		    image_unselected="PushButton_Off"
 -        tool_tip="Open current URL in your desktop browser"
 -        follows="right|top"
 -        enabled="true" 
 -        height="22"
 -        layout="topleft"
 -        name="popexternal"
 -        right="800"
 -        top_delta="-2"
 -        width="22">
 -        <button.commit_callback
 -          function="WebContent.PopExternal" />
 -      </button>
 -    </layout_panel>
 -    <layout_panel
 -      height="40"
 -      layout="topleft"
 -      left_delta="0"
 -      name="external_controls"
 -      top_delta="0"
 -      user_resize="false"
 -      width="540">
 -      <web_browser
 -        bottom="-22"
 -        follows="all"
 -        layout="topleft"
 -        left="0"
 -        name="webbrowser"
 -        top="0"/>
 -      <text
 -        type="string"
 -        length="100"
 -        follows="bottom|left"
 -        height="20"
 -        layout="topleft"
 -        left_delta="0"
 -        name="statusbartext"
 -        parse_urls="false"
 -        text_color="0.4 0.4 0.4 1" 
 -        top_pad="5"
 -        width="452"/>
 -      <progress_bar
 -        color_bar="0.3 1.0 0.3 1"
 -        follows="bottom|right"
 -        height="16"
 -        top_delta="-1"
 -        left_pad="24"
 -        layout="topleft"
 -        name="statusbarprogress"
 -        width="64"/>
 -    </layout_panel>
 -  </layout_stack>
 -</floater>
 +<?xml version="1.0" encoding="utf-8" standalone="yes" ?> +<floater +  legacy_header_height="18" +  can_resize="true" +  height="440" +  layout="topleft" +  min_height="140" +  min_width="467" +  name="floater_web_content" +  help_topic="floater_web_content" +  save_rect="true" +  auto_tile="true" +  title="" +  initial_mime_type="text/html" +  width="820"> +  <layout_stack +    bottom="440" +    follows="left|right|top|bottom" +    layout="topleft" +    left="5" +    name="stack1" +    orientation="vertical" +    top="20" +    width="810"> +    <layout_panel +      auto_resize="false" +      default_tab_group="1" +      height="22" +      layout="topleft" +      left="0" +      min_height="20" +      name="nav_controls" +      top="400" +      user_resize="false" +      width="800"> +      <button +        image_overlay="Arrow_Left_Off" +        image_disabled="PushButton_Disabled" +        image_disabled_selected="PushButton_Disabled" +        image_selected="PushButton_Selected" +        image_unselected="PushButton_Off" +        hover_glow_amount="0.15" +        tool_tip="Navigate back" +        follows="left|top" +        height="22" +        layout="topleft" +        left="1" +        name="back" +        top="0" +        width="22"> +        <button.commit_callback +          function="WebContent.Back" /> +      </button> +      <button +        image_overlay="Arrow_Right_Off" +        image_disabled="PushButton_Disabled" +        image_disabled_selected="PushButton_Disabled" +        image_selected="PushButton_Selected" +        image_unselected="PushButton_Off" +        tool_tip="Navigate forward" +        follows="left|top" +        height="22" +        layout="topleft" +        left="27" +        name="forward" +        top_delta="0" +        width="22"> +        <button.commit_callback +          function="WebContent.Forward" /> +      </button> +      <button +        image_overlay="Stop_Off" +        image_disabled="PushButton_Disabled" +        image_disabled_selected="PushButton_Disabled" +        image_selected="PushButton_Selected" +        image_unselected="PushButton_Off" +        tool_tip="Stop navigation" +        enabled="true" +        follows="left|top" +        height="22" +        layout="topleft" +        left="51" +        name="stop" +        top_delta="0" +        width="22"> +        <button.commit_callback +          function="WebContent.Stop" /> +      </button> +      <button +        image_overlay="Refresh_Off" +        image_disabled="PushButton_Disabled" +        image_disabled_selected="PushButton_Disabled" +        image_selected="PushButton_Selected" +        image_unselected="PushButton_Off" +        tool_tip="Reload page" +        follows="left|top" +        height="22" +        layout="topleft" +        left="51" +        name="reload" +        top_delta="0" +        width="22"> +        <button.commit_callback +          function="WebContent.Reload" /> +      </button> +      <combo_box +        allow_text_entry="true" +        follows="left|top|right" +        tab_group="1" +        height="22" +        layout="topleft" +        left_pad="4" +        max_chars="1024" +        name="address" +        combo_editor.select_on_focus="true" +        tool_tip="Enter URL here" +        top_delta="0" +        width="702"> +        <combo_box.commit_callback +          function="WebContent.EnterAddress" /> +      </combo_box> +      <icon +        name="media_secure_lock_flag" +        height="16" +        follows="top|right" +        image_name="Lock2" +        layout="topleft" +        left_delta="656" +        top_delta="2" +        visible="false"  +        tool_tip="Secured Browsing" +        width="16" /> +      <button +        image_overlay="ExternalBrowser_Off" +        image_disabled="PushButton_Disabled" +        image_disabled_selected="PushButton_Disabled" +        image_selected="PushButton_Selected" +        image_unselected="PushButton_Off" +        tool_tip="Open current URL in your desktop browser" +        follows="right|top" +        enabled="true"  +        height="22" +        layout="topleft" +        name="popexternal" +        right="800" +        top_delta="-2" +        width="22"> +        <button.commit_callback +          function="WebContent.PopExternal" /> +      </button> +    </layout_panel> +    <layout_panel +      height="40" +      layout="topleft" +      left_delta="0" +      name="external_controls" +      top_delta="0" +      user_resize="false" +      width="540"> +      <web_browser +        bottom="-22" +        follows="all" +        layout="topleft" +        left="0" +        name="webbrowser" +        top="0"/> +      <text +        type="string" +        length="100" +        follows="bottom|left" +        height="20" +        layout="topleft" +        left_delta="0" +        name="statusbartext" +        parse_urls="false" +        text_color="0.4 0.4 0.4 1"  +        top_pad="5" +        width="452"/> +      <progress_bar +        color_bar="0.3 1.0 0.3 1" +        follows="bottom|right" +        height="16" +        top_delta="-1" +        left_pad="24" +        layout="topleft" +        name="statusbarprogress" +        width="64"/> +    </layout_panel> +  </layout_stack> +</floater> diff --git a/indra/newview/tests/llremoteparcelrequest_test.cpp b/indra/newview/tests/llremoteparcelrequest_test.cpp index a6c1f69c82..dae22521bb 100644 --- a/indra/newview/tests/llremoteparcelrequest_test.cpp +++ b/indra/newview/tests/llremoteparcelrequest_test.cpp @@ -1,134 +1,134 @@ -/** 
 - * @file llremoteparcelrequest_test.cpp
 - * @author Brad Kittenbrink <brad@lindenlab.com>
 - *
 - * $LicenseInfo:firstyear=2010&license=viewerlgpl$
 - * Second Life Viewer Source Code
 - * Copyright (C) 2010, Linden Research, Inc.
 - * 
 - * This library is free software; you can redistribute it and/or
 - * modify it under the terms of the GNU Lesser General Public
 - * License as published by the Free Software Foundation;
 - * version 2.1 of the License only.
 - * 
 - * This library 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
 - * Lesser General Public License for more details.
 - * 
 - * You should have received a copy of the GNU Lesser General Public
 - * License along with this library; if not, write to the Free Software
 - * Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA  02110-1301  USA
 - * 
 - * Linden Research, Inc., 945 Battery Street, San Francisco, CA  94111  USA
 - * $/LicenseInfo$
 - */
 -
 -#include "linden_common.h"
 -
 -#include "../test/lltut.h"
 -
 -#include "../llremoteparcelrequest.h"
 -
 -#include "../llagent.h"
 -#include "message.h"
 -
 -namespace {
 -	LLControlGroup s_saved_settings("dummy_settings");
 -	const LLUUID TEST_PARCEL_ID("11111111-1111-1111-1111-111111111111");
 -}
 -
 -LLCurl::Responder::Responder() { }
 -LLCurl::Responder::~Responder() { }
 -void LLCurl::Responder::error(U32,std::string const &) { }
 -void LLCurl::Responder::result(LLSD const &) { }
 -void LLCurl::Responder::errorWithContent(U32 status,std::string const &,LLSD const &) { }
 -void LLCurl::Responder::completedRaw(U32 status, std::string const &, LLChannelDescriptors const &,boost::shared_ptr<LLBufferArray> const &) { }
 -void LLCurl::Responder::completed(U32 status, std::string const &, LLSD const &) { }
 -void LLCurl::Responder::completedHeader(U32 status, std::string const &, LLSD const &) { }
 -void LLMessageSystem::getF32(char const *,char const *,F32 &,S32) { }
 -void LLMessageSystem::getU8(char const *,char const *,U8 &,S32) { }
 -void LLMessageSystem::getS32(char const *,char const *,S32 &,S32) { }
 -void LLMessageSystem::getString(char const *,char const *, std::string &,S32) { }
 -void LLMessageSystem::getUUID(char const *,char const *, LLUUID & out_id,S32)
 -{
 -	out_id = TEST_PARCEL_ID;
 -}
 -void LLMessageSystem::nextBlock(char const *) { }
 -void LLMessageSystem::addUUID(char const *,LLUUID const &) { }
 -void LLMessageSystem::addUUIDFast(char const *,LLUUID const &) { }
 -void LLMessageSystem::nextBlockFast(char const *) { }
 -void LLMessageSystem::newMessage(char const *) { }
 -LLMessageSystem * gMessageSystem;
 -char * _PREHASH_AgentID;
 -char * _PREHASH_AgentData;
 -LLAgent gAgent;
 -LLAgent::LLAgent() : mAgentAccess(s_saved_settings) { }
 -LLAgent::~LLAgent() { }
 -void LLAgent::sendReliableMessage(void) { }
 -LLUUID gAgentSessionID;
 -LLUUID gAgentID;
 -LLUIColor::LLUIColor(void) { }
 -LLAgentAccess::LLAgentAccess(LLControlGroup & settings) : mSavedSettings(settings) { }
 -LLControlGroup::LLControlGroup(std::string const & name) : LLInstanceTracker<LLControlGroup, std::string>(name) { }
 -LLControlGroup::~LLControlGroup(void) { }
 -
 -namespace tut
 -{
 -	struct TestObserver : public LLRemoteParcelInfoObserver {
 -		TestObserver() : mProcessed(false) { }
 -
 -		virtual void processParcelInfo(const LLParcelData& parcel_data)
 -		{
 -			mProcessed = true;
 -		}
 -
 -		virtual void setParcelID(const LLUUID& parcel_id) { }
 -
 -		virtual void setErrorStatus(U32 status, const std::string& reason) { }
 -
 -		bool mProcessed;
 -	};
 -
 -    struct RemoteParcelRequestData
 -    {
 -		RemoteParcelRequestData()
 -		{
 -		}
 -    };
 -    
 -	typedef test_group<RemoteParcelRequestData> remoteparcelrequest_t;
 -	typedef remoteparcelrequest_t::object remoteparcelrequest_object_t;
 -	tut::remoteparcelrequest_t tut_remoteparcelrequest("LLRemoteParcelRequest");
 -
 -	template<> template<>
 -	void remoteparcelrequest_object_t::test<1>()
 -	{
 -		set_test_name("observer pointer");
 -
 -		boost::scoped_ptr<TestObserver> observer(new TestObserver());
 -
 -		LLRemoteParcelInfoProcessor & processor = LLRemoteParcelInfoProcessor::instance();
 -		processor.addObserver(LLUUID(TEST_PARCEL_ID), observer.get());
 -
 -		processor.processParcelInfoReply(gMessageSystem, NULL);
 -
 -		ensure(observer->mProcessed);
 -	}
 -
 -	template<> template<>
 -	void remoteparcelrequest_object_t::test<2>()
 -	{
 -		set_test_name("CHOP-220: dangling observer pointer");
 -
 -		LLRemoteParcelInfoObserver * observer = new TestObserver();
 -
 -		LLRemoteParcelInfoProcessor & processor = LLRemoteParcelInfoProcessor::instance();
 -		processor.addObserver(LLUUID(TEST_PARCEL_ID), observer);
 -
 -		delete observer;
 -		observer = NULL;
 -
 -		processor.processParcelInfoReply(gMessageSystem, NULL);
 -	}
 -}
 +/**  + * @file llremoteparcelrequest_test.cpp + * @author Brad Kittenbrink <brad@lindenlab.com> + * + * $LicenseInfo:firstyear=2010&license=viewerlgpl$ + * Second Life Viewer Source Code + * Copyright (C) 2010, Linden Research, Inc. + *  + * This library is free software; you can redistribute it and/or + * modify it under the terms of the GNU Lesser General Public + * License as published by the Free Software Foundation; + * version 2.1 of the License only. + *  + * This library 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 + * Lesser General Public License for more details. + *  + * You should have received a copy of the GNU Lesser General Public + * License along with this library; if not, write to the Free Software + * Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA  02110-1301  USA + *  + * Linden Research, Inc., 945 Battery Street, San Francisco, CA  94111  USA + * $/LicenseInfo$ + */ + +#include "linden_common.h" + +#include "../test/lltut.h" + +#include "../llremoteparcelrequest.h" + +#include "../llagent.h" +#include "message.h" + +namespace { +	LLControlGroup s_saved_settings("dummy_settings"); +	const LLUUID TEST_PARCEL_ID("11111111-1111-1111-1111-111111111111"); +} + +LLCurl::Responder::Responder() { } +LLCurl::Responder::~Responder() { } +void LLCurl::Responder::error(U32,std::string const &) { } +void LLCurl::Responder::result(LLSD const &) { } +void LLCurl::Responder::errorWithContent(U32 status,std::string const &,LLSD const &) { } +void LLCurl::Responder::completedRaw(U32 status, std::string const &, LLChannelDescriptors const &,boost::shared_ptr<LLBufferArray> const &) { } +void LLCurl::Responder::completed(U32 status, std::string const &, LLSD const &) { } +void LLCurl::Responder::completedHeader(U32 status, std::string const &, LLSD const &) { } +void LLMessageSystem::getF32(char const *,char const *,F32 &,S32) { } +void LLMessageSystem::getU8(char const *,char const *,U8 &,S32) { } +void LLMessageSystem::getS32(char const *,char const *,S32 &,S32) { } +void LLMessageSystem::getString(char const *,char const *, std::string &,S32) { } +void LLMessageSystem::getUUID(char const *,char const *, LLUUID & out_id,S32) +{ +	out_id = TEST_PARCEL_ID; +} +void LLMessageSystem::nextBlock(char const *) { } +void LLMessageSystem::addUUID(char const *,LLUUID const &) { } +void LLMessageSystem::addUUIDFast(char const *,LLUUID const &) { } +void LLMessageSystem::nextBlockFast(char const *) { } +void LLMessageSystem::newMessage(char const *) { } +LLMessageSystem * gMessageSystem; +char * _PREHASH_AgentID; +char * _PREHASH_AgentData; +LLAgent gAgent; +LLAgent::LLAgent() : mAgentAccess(s_saved_settings) { } +LLAgent::~LLAgent() { } +void LLAgent::sendReliableMessage(void) { } +LLUUID gAgentSessionID; +LLUUID gAgentID; +LLUIColor::LLUIColor(void) { } +LLAgentAccess::LLAgentAccess(LLControlGroup & settings) : mSavedSettings(settings) { } +LLControlGroup::LLControlGroup(std::string const & name) : LLInstanceTracker<LLControlGroup, std::string>(name) { } +LLControlGroup::~LLControlGroup(void) { } + +namespace tut +{ +	struct TestObserver : public LLRemoteParcelInfoObserver { +		TestObserver() : mProcessed(false) { } + +		virtual void processParcelInfo(const LLParcelData& parcel_data) +		{ +			mProcessed = true; +		} + +		virtual void setParcelID(const LLUUID& parcel_id) { } + +		virtual void setErrorStatus(U32 status, const std::string& reason) { } + +		bool mProcessed; +	}; + +    struct RemoteParcelRequestData +    { +		RemoteParcelRequestData() +		{ +		} +    }; +     +	typedef test_group<RemoteParcelRequestData> remoteparcelrequest_t; +	typedef remoteparcelrequest_t::object remoteparcelrequest_object_t; +	tut::remoteparcelrequest_t tut_remoteparcelrequest("LLRemoteParcelRequest"); + +	template<> template<> +	void remoteparcelrequest_object_t::test<1>() +	{ +		set_test_name("observer pointer"); + +		boost::scoped_ptr<TestObserver> observer(new TestObserver()); + +		LLRemoteParcelInfoProcessor & processor = LLRemoteParcelInfoProcessor::instance(); +		processor.addObserver(LLUUID(TEST_PARCEL_ID), observer.get()); + +		processor.processParcelInfoReply(gMessageSystem, NULL); + +		ensure(observer->mProcessed); +	} + +	template<> template<> +	void remoteparcelrequest_object_t::test<2>() +	{ +		set_test_name("CHOP-220: dangling observer pointer"); + +		LLRemoteParcelInfoObserver * observer = new TestObserver(); + +		LLRemoteParcelInfoProcessor & processor = LLRemoteParcelInfoProcessor::instance(); +		processor.addObserver(LLUUID(TEST_PARCEL_ID), observer); + +		delete observer; +		observer = NULL; + +		processor.processParcelInfoReply(gMessageSystem, NULL); +	} +} diff --git a/indra/viewer_components/updater/tests/llupdaterservice_test.cpp b/indra/viewer_components/updater/tests/llupdaterservice_test.cpp index 5f8cd28f29..88ab5a2284 100644 --- a/indra/viewer_components/updater/tests/llupdaterservice_test.cpp +++ b/indra/viewer_components/updater/tests/llupdaterservice_test.cpp @@ -1,200 +1,200 @@ -/**
 - * @file   llupdaterservice_test.cpp
 - * @brief  Tests of llupdaterservice.cpp.
 - * 
 - * $LicenseInfo:firstyear=2010&license=viewerlgpl$
 - * Second Life Viewer Source Code
 - * Copyright (C) 2010, Linden Research, Inc.
 - * 
 - * This library is free software; you can redistribute it and/or
 - * modify it under the terms of the GNU Lesser General Public
 - * License as published by the Free Software Foundation;
 - * version 2.1 of the License only.
 - * 
 - * This library 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
 - * Lesser General Public License for more details.
 - * 
 - * You should have received a copy of the GNU Lesser General Public
 - * License along with this library; if not, write to the Free Software
 - * Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA  02110-1301  USA
 - * 
 - * Linden Research, Inc., 945 Battery Street, San Francisco, CA  94111  USA
 - * $/LicenseInfo$
 - */
 -
 -// Precompiled header
 -#include "linden_common.h"
 -// associated header
 -#include "../llupdaterservice.h"
 -#include "../llupdatechecker.h"
 -#include "../llupdatedownloader.h"
 -#include "../llupdateinstaller.h"
 -
 -#include "../../../test/lltut.h"
 -//#define DEBUG_ON
 -#include "../../../test/debug.h"
 -
 -#include "llevents.h"
 -#include "lldir.h"
 -
 -/*****************************************************************************
 -*   MOCK'd
 -*****************************************************************************/
 -LLUpdateChecker::LLUpdateChecker(LLUpdateChecker::Client & client)
 -{}
 -void LLUpdateChecker::check(std::string const & protocolVersion, std::string const & hostUrl, 
 -								  std::string const & servicePath, std::string channel, std::string version)
 -{}
 -LLUpdateDownloader::LLUpdateDownloader(Client & ) {}
 -void LLUpdateDownloader::download(LLURI const & , std::string const &, std::string const &, bool){}
 -
 -class LLDir_Mock : public LLDir
 -{
 -	void initAppDirs(const std::string &app_name, 
 -		   			 const std::string& app_read_only_data_dir = "") {}
 -	U32 countFilesInDir(const std::string &dirname, const std::string &mask) 
 -	{
 -		return 0;
 -	}
 -
 -	BOOL getNextFileInDir(const std::string &dirname, 
 -						  const std::string &mask, 
 -						  std::string &fname) 
 -	{
 -		return false;
 -	}
 -	void getRandomFileInDir(const std::string &dirname, 
 -							const std::string &mask, 
 -							std::string &fname) {}
 -	std::string getCurPath() { return ""; }
 -	BOOL fileExists(const std::string &filename) const { return false; }
 -	std::string getLLPluginLauncher() { return ""; }
 -	std::string getLLPluginFilename(std::string base_name) { return ""; }
 -
 -} gDirUtil;
 -LLDir* gDirUtilp = &gDirUtil;
 -LLDir::LLDir() {}
 -LLDir::~LLDir() {}
 -S32 LLDir::deleteFilesInDir(const std::string &dirname, 
 -							const std::string &mask)
 -{ return 0; }
 -
 -void LLDir::setChatLogsDir(const std::string &path){}		
 -void LLDir::setPerAccountChatLogsDir(const std::string &username){}
 -void LLDir::setLindenUserDir(const std::string &username){}		
 -void LLDir::setSkinFolder(const std::string &skin_folder){}
 -bool LLDir::setCacheDir(const std::string &path){ return true; }
 -void LLDir::dumpCurrentDirectories() {}
 -
 -std::string LLDir::getExpandedFilename(ELLPath location, 
 -									   const std::string &filename) const 
 -{
 -	return "";
 -}
 -
 -std::string LLUpdateDownloader::downloadMarkerPath(void)
 -{
 -	return "";
 -}
 -
 -void LLUpdateDownloader::resume(void) {}
 -void LLUpdateDownloader::cancel(void) {}
 -void LLUpdateDownloader::setBandwidthLimit(U64 bytesPerSecond) {}
 -
 -int ll_install_update(std::string const &, std::string const &, bool, LLInstallScriptMode)
 -{
 -	return 0;
 -}
 -
 -std::string const & ll_install_failed_marker_path()
 -{
 -	static std::string wubba;
 -	return wubba;
 -}
 -
 -/*
 -#pragma warning(disable: 4273)
 -llus_mock_llifstream::llus_mock_llifstream(const std::string& _Filename,
 -										   ios_base::openmode _Mode,
 -										   int _Prot) :
 -	std::basic_istream<char,std::char_traits< char > >(NULL,true)
 -{}
 -
 -llus_mock_llifstream::~llus_mock_llifstream() {}
 -bool llus_mock_llifstream::is_open() const {return true;}
 -void llus_mock_llifstream::close() {}
 -*/
 -
 -/*****************************************************************************
 -*   TUT
 -*****************************************************************************/
 -namespace tut
 -{
 -    struct llupdaterservice_data
 -    {
 -		llupdaterservice_data() :
 -            pumps(LLEventPumps::instance()),
 -			test_url("dummy_url"),
 -			test_channel("dummy_channel"),
 -			test_version("dummy_version")
 -		{}
 -		LLEventPumps& pumps;
 -		std::string test_url;
 -		std::string test_channel;
 -		std::string test_version;
 -	};
 -
 -    typedef test_group<llupdaterservice_data> llupdaterservice_group;
 -    typedef llupdaterservice_group::object llupdaterservice_object;
 -    llupdaterservice_group llupdaterservicegrp("LLUpdaterService");
 -
 -    template<> template<>
 -    void llupdaterservice_object::test<1>()
 -    {
 -        DEBUG;
 -		LLUpdaterService updater;
 -		bool got_usage_error = false;
 -		try
 -		{
 -			updater.startChecking();
 -		}
 -		catch(LLUpdaterService::UsageError)
 -		{
 -			got_usage_error = true;
 -		}
 -		ensure("Caught start before params", got_usage_error);
 -	}
 -
 -    template<> template<>
 -    void llupdaterservice_object::test<2>()
 -    {
 -        DEBUG;
 -		LLUpdaterService updater;
 -		bool got_usage_error = false;
 -		try
 -		{
 -			updater.initialize("1.0",test_url, "update" ,test_channel, test_version);
 -			updater.startChecking();
 -			updater.initialize("1.0", "other_url", "update", test_channel, test_version);
 -		}
 -		catch(LLUpdaterService::UsageError)
 -		{
 -			got_usage_error = true;
 -		}
 -		ensure("Caught params while running", got_usage_error);
 -	}
 -
 -    template<> template<>
 -    void llupdaterservice_object::test<3>()
 -    {
 -        DEBUG;
 -		LLUpdaterService updater;
 -		updater.initialize("1.0", test_url, "update", test_channel, test_version);
 -		updater.startChecking();
 -		ensure(updater.isChecking());
 -		updater.stopChecking();
 -		ensure(!updater.isChecking());
 -	}
 -}
 +/** + * @file   llupdaterservice_test.cpp + * @brief  Tests of llupdaterservice.cpp. + *  + * $LicenseInfo:firstyear=2010&license=viewerlgpl$ + * Second Life Viewer Source Code + * Copyright (C) 2010, Linden Research, Inc. + *  + * This library is free software; you can redistribute it and/or + * modify it under the terms of the GNU Lesser General Public + * License as published by the Free Software Foundation; + * version 2.1 of the License only. + *  + * This library 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 + * Lesser General Public License for more details. + *  + * You should have received a copy of the GNU Lesser General Public + * License along with this library; if not, write to the Free Software + * Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA  02110-1301  USA + *  + * Linden Research, Inc., 945 Battery Street, San Francisco, CA  94111  USA + * $/LicenseInfo$ + */ + +// Precompiled header +#include "linden_common.h" +// associated header +#include "../llupdaterservice.h" +#include "../llupdatechecker.h" +#include "../llupdatedownloader.h" +#include "../llupdateinstaller.h" + +#include "../../../test/lltut.h" +//#define DEBUG_ON +#include "../../../test/debug.h" + +#include "llevents.h" +#include "lldir.h" + +/***************************************************************************** +*   MOCK'd +*****************************************************************************/ +LLUpdateChecker::LLUpdateChecker(LLUpdateChecker::Client & client) +{} +void LLUpdateChecker::check(std::string const & protocolVersion, std::string const & hostUrl,  +								  std::string const & servicePath, std::string channel, std::string version) +{} +LLUpdateDownloader::LLUpdateDownloader(Client & ) {} +void LLUpdateDownloader::download(LLURI const & , std::string const &, std::string const &, bool){} + +class LLDir_Mock : public LLDir +{ +	void initAppDirs(const std::string &app_name,  +		   			 const std::string& app_read_only_data_dir = "") {} +	U32 countFilesInDir(const std::string &dirname, const std::string &mask)  +	{ +		return 0; +	} + +	BOOL getNextFileInDir(const std::string &dirname,  +						  const std::string &mask,  +						  std::string &fname)  +	{ +		return false; +	} +	void getRandomFileInDir(const std::string &dirname,  +							const std::string &mask,  +							std::string &fname) {} +	std::string getCurPath() { return ""; } +	BOOL fileExists(const std::string &filename) const { return false; } +	std::string getLLPluginLauncher() { return ""; } +	std::string getLLPluginFilename(std::string base_name) { return ""; } + +} gDirUtil; +LLDir* gDirUtilp = &gDirUtil; +LLDir::LLDir() {} +LLDir::~LLDir() {} +S32 LLDir::deleteFilesInDir(const std::string &dirname,  +							const std::string &mask) +{ return 0; } + +void LLDir::setChatLogsDir(const std::string &path){}		 +void LLDir::setPerAccountChatLogsDir(const std::string &username){} +void LLDir::setLindenUserDir(const std::string &username){}		 +void LLDir::setSkinFolder(const std::string &skin_folder){} +bool LLDir::setCacheDir(const std::string &path){ return true; } +void LLDir::dumpCurrentDirectories() {} + +std::string LLDir::getExpandedFilename(ELLPath location,  +									   const std::string &filename) const  +{ +	return ""; +} + +std::string LLUpdateDownloader::downloadMarkerPath(void) +{ +	return ""; +} + +void LLUpdateDownloader::resume(void) {} +void LLUpdateDownloader::cancel(void) {} +void LLUpdateDownloader::setBandwidthLimit(U64 bytesPerSecond) {} + +int ll_install_update(std::string const &, std::string const &, bool, LLInstallScriptMode) +{ +	return 0; +} + +std::string const & ll_install_failed_marker_path() +{ +	static std::string wubba; +	return wubba; +} + +/* +#pragma warning(disable: 4273) +llus_mock_llifstream::llus_mock_llifstream(const std::string& _Filename, +										   ios_base::openmode _Mode, +										   int _Prot) : +	std::basic_istream<char,std::char_traits< char > >(NULL,true) +{} + +llus_mock_llifstream::~llus_mock_llifstream() {} +bool llus_mock_llifstream::is_open() const {return true;} +void llus_mock_llifstream::close() {} +*/ + +/***************************************************************************** +*   TUT +*****************************************************************************/ +namespace tut +{ +    struct llupdaterservice_data +    { +		llupdaterservice_data() : +            pumps(LLEventPumps::instance()), +			test_url("dummy_url"), +			test_channel("dummy_channel"), +			test_version("dummy_version") +		{} +		LLEventPumps& pumps; +		std::string test_url; +		std::string test_channel; +		std::string test_version; +	}; + +    typedef test_group<llupdaterservice_data> llupdaterservice_group; +    typedef llupdaterservice_group::object llupdaterservice_object; +    llupdaterservice_group llupdaterservicegrp("LLUpdaterService"); + +    template<> template<> +    void llupdaterservice_object::test<1>() +    { +        DEBUG; +		LLUpdaterService updater; +		bool got_usage_error = false; +		try +		{ +			updater.startChecking(); +		} +		catch(LLUpdaterService::UsageError) +		{ +			got_usage_error = true; +		} +		ensure("Caught start before params", got_usage_error); +	} + +    template<> template<> +    void llupdaterservice_object::test<2>() +    { +        DEBUG; +		LLUpdaterService updater; +		bool got_usage_error = false; +		try +		{ +			updater.initialize("1.0",test_url, "update" ,test_channel, test_version); +			updater.startChecking(); +			updater.initialize("1.0", "other_url", "update", test_channel, test_version); +		} +		catch(LLUpdaterService::UsageError) +		{ +			got_usage_error = true; +		} +		ensure("Caught params while running", got_usage_error); +	} + +    template<> template<> +    void llupdaterservice_object::test<3>() +    { +        DEBUG; +		LLUpdaterService updater; +		updater.initialize("1.0", test_url, "update", test_channel, test_version); +		updater.startChecking(); +		ensure(updater.isChecking()); +		updater.stopChecking(); +		ensure(!updater.isChecking()); +	} +} | 
