# CMake file to replace the string contents in ONNX, Caffe, and Caffe2 proto.
# Usage example:
#   cmake -DFILENAME=caffe2.pb.h -DLOCAL_PROTOBUF=ON -P ProtoBufPatch.cmake

file(READ ${FILENAME} content)

if(NOT SYSTEM_PROTOBUF)
  # protobuf-3.6.0 pattern
  string(
    REPLACE
    "::google::protobuf::internal::GetEmptyStringAlreadyInited"
    "GetEmptyStringAlreadyInited"
    content
    "${content}")

  # protobuf-3.8.0+ pattern
  string(
    REPLACE
    "::PROTOBUF_NAMESPACE_ID::internal::GetEmptyStringAlreadyInited"
    "GetEmptyStringAlreadyInited"
    content
    "${content}")

  string(
    REPLACE
    "PROTOBUF_CONSTEXPR"
    ""
    content
    "${content}")

  # https://github.com/protocolbuffers/protobuf/commit/0400cca3236de1ca303af38bf81eab332d042b7c
  # changes PROTOBUF_CONSTEXPR to constexpr, which breaks windows
  # build.
  if(MSVC)
    string(
      REGEX REPLACE
      "static constexpr ([^ ]+) ([^ ]+) ="
      "static \\1 const \\2 ="
      content
      "${content}")
  endif()

  foreach(ns ${NAMESPACES})
    # Insert "const ::std::string& GetEmptyStringAlreadyInited();" within
    # the namespace and make sure we only do it once in the file. Unfortunately
    # using string(REPLACE ...) doesn't work because it will replace at all
    # locations and there might be multiple declarations of the namespace
    # depending on how the proto is structured.
    set(search "namespace ${ns} {")
    string(LENGTH "${search}" search_len)
    string(FIND "${content}" "${search}" pos)
    if(${pos} GREATER -1)
      math(EXPR pos "${pos}+${search_len}")
      string(SUBSTRING "${content}" 0 ${pos} content_pre)
      string(SUBSTRING "${content}" ${pos} -1 content_post)
      string(
        CONCAT
        content
        "${content_pre}"
        " const ::std::string& GetEmptyStringAlreadyInited(); "
        "${content_post}")
    endif()
  endforeach()

  # The moving constructor is defined in the header file, which will cause
  # a link error that claims that the vftable is not found. Luckily, we
  # could move the definition into the source file to solve the problem.
  list(LENGTH NAMESPACES ns_count)
  if("${FILENAME}" MATCHES ".pb.h" AND ns_count EQUAL 1)
    string(REPLACE ".pb.h" ".pb.cc" SOURCE_FILENAME ${FILENAME})
    file(READ ${SOURCE_FILENAME} content_cc_origin)

    string(REGEX MATCHALL "([a-zA-Z_]+)\\([a-zA-Z_]+&& from\\) noexcept[^}]*}" content_cc "${content}")
    string(REGEX REPLACE "};" "}\n" content_cc "${content_cc}")
    string(REGEX REPLACE "([a-zA-Z_]+)\\([a-zA-Z_]+&& from\\) noexcept" "  \\1::\\1(\\1&& from) noexcept" content_cc "${content_cc}")
    set(content_cc "${content_cc_origin}\nnamespace ${NAMESPACES} {\n#if LANG_CXX11\n${content_cc}\n#endif\n}")

    string(REGEX REPLACE "([a-zA-Z_]+)\\([a-zA-Z_]+&& from\\) noexcept([^}]*)}" "\\1(\\1&& from) noexcept;" content "${content}")

    file(WRITE ${SOURCE_FILENAME} "${content_cc}")
  endif()
endif(NOT SYSTEM_PROTOBUF)

# constexpr int TensorBoundShape_DimType_DimType_ARRAYSIZE = TensorBoundShape_DimType_DimType_MAX + 1;
# throws
# error: more than one operator "+" matches these operands:
#     built-in operator "arithmetic + arithmetic"
#     function "c10::operator+(int, c10::BFloat16)"
#     function "c10::operator+(c10::BFloat16, int)"
#     function "c10::operator+(int, c10::Half)"
#     function "c10::operator+(c10::Half, int)"
#   operand types are: const caffe2::ExternalDataProto_SourceType + int
string(
  REGEX REPLACE
  "constexpr ([^ ]+) ([^ ]+_ARRAYSIZE) = ([^ ]+_MAX) \\+ 1;"
  "constexpr \\1 \\2 = static_cast<\\1>(\\3) + 1;"
  content
  "${content}")

file(WRITE ${FILENAME} "${content}")
