/*
 * Copyright (c) Facebook, Inc. and its affiliates.
 *
 * Licensed under the Apache License, Version 2.0 (the "License");
 * you may not use this file except in compliance with the License.
 * You may obtain a copy of the License at
 *
 *     http://www.apache.org/licenses/LICENSE-2.0
 *
 * Unless required by applicable law or agreed to in writing, software
 * distributed under the License is distributed on an "AS IS" BASIS,
 * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
 * See the License for the specific language governing permissions and
 * limitations under the License.
 */

#pragma once

#include <folly/Portability.h>

namespace folly {

enum class annotate_rwlock_level : long {
  rdlock = 0,
  wrlock = 1,
};

namespace detail {

void annotate_rwlock_create_impl(
    void const volatile* const addr, char const* const f, int const l);

void annotate_rwlock_create_static_impl(
    void const volatile* const addr, char const* const f, int const l);

void annotate_rwlock_destroy_impl(
    void const volatile* const addr, char const* const f, int const l);

void annotate_rwlock_acquired_impl(
    void const volatile* const addr,
    annotate_rwlock_level const w,
    char const* const f,
    int const l);

void annotate_rwlock_released_impl(
    void const volatile* const addr,
    annotate_rwlock_level const w,
    char const* const f,
    int const l);

void annotate_benign_race_sized_impl(
    const volatile void* addr,
    long size,
    const char* desc,
    const char* f,
    int l);

void annotate_ignore_reads_begin_impl(const char* f, int l);

void annotate_ignore_reads_end_impl(const char* f, int l);

void annotate_ignore_writes_begin_impl(const char* f, int l);

void annotate_ignore_writes_end_impl(const char* f, int l);

void annotate_ignore_sync_begin_impl(const char* f, int l);

void annotate_ignore_sync_end_impl(const char* f, int l);

} // namespace detail

FOLLY_ALWAYS_INLINE static void annotate_rwlock_create(
    void const volatile* const addr, char const* const f, int const l) {
  if (kIsSanitizeThread) {
    detail::annotate_rwlock_create_impl(addr, f, l);
  }
}

FOLLY_ALWAYS_INLINE static void annotate_rwlock_create_static(
    void const volatile* const addr, char const* const f, int const l) {
  if (kIsSanitizeThread) {
    detail::annotate_rwlock_create_static_impl(addr, f, l);
  }
}

FOLLY_ALWAYS_INLINE static void annotate_rwlock_destroy(
    void const volatile* const addr, char const* const f, int const l) {
  if (kIsSanitizeThread) {
    detail::annotate_rwlock_destroy_impl(addr, f, l);
  }
}

FOLLY_ALWAYS_INLINE static void annotate_rwlock_acquired(
    void const volatile* const addr,
    annotate_rwlock_level const w,
    char const* const f,
    int const l) {
  if (kIsSanitizeThread) {
    detail::annotate_rwlock_acquired_impl(addr, w, f, l);
  }
}

FOLLY_ALWAYS_INLINE static void annotate_rwlock_try_acquired(
    void const volatile* const addr,
    annotate_rwlock_level const w,
    bool const result,
    char const* const f,
    int const l) {
  if (result) {
    annotate_rwlock_acquired(addr, w, f, l);
  }
}

FOLLY_ALWAYS_INLINE static void annotate_rwlock_released(
    void const volatile* const addr,
    annotate_rwlock_level const w,
    char const* const f,
    int const l) {
  if (kIsSanitizeThread) {
    detail::annotate_rwlock_released_impl(addr, w, f, l);
  }
}

FOLLY_ALWAYS_INLINE static void annotate_benign_race_sized(
    void const volatile* const addr,
    long const size,
    char const* const desc,
    char const* const f,
    int const l) {
  if (kIsSanitizeThread) {
    detail::annotate_benign_race_sized_impl(addr, size, desc, f, l);
  }
}

FOLLY_ALWAYS_INLINE static void annotate_ignore_reads_begin(
    const char* f, int l) {
  if (kIsSanitizeThread) {
    detail::annotate_ignore_reads_begin_impl(f, l);
  }
}

FOLLY_ALWAYS_INLINE static void annotate_ignore_reads_end(
    const char* f, int l) {
  if (kIsSanitizeThread) {
    detail::annotate_ignore_reads_end_impl(f, l);
  }
}

FOLLY_ALWAYS_INLINE static void annotate_ignore_writes_begin(
    const char* f, int l) {
  if (kIsSanitizeThread) {
    detail::annotate_ignore_writes_begin_impl(f, l);
  }
}

FOLLY_ALWAYS_INLINE static void annotate_ignore_writes_end(
    const char* f, int l) {
  if (kIsSanitizeThread) {
    detail::annotate_ignore_writes_end_impl(f, l);
  }
}

FOLLY_ALWAYS_INLINE static void annotate_ignore_sync_begin(
    const char* f, int l) {
  if (kIsSanitizeThread) {
    detail::annotate_ignore_sync_begin_impl(f, l);
  }
}

FOLLY_ALWAYS_INLINE static void annotate_ignore_sync_end(const char* f, int l) {
  if (kIsSanitizeThread) {
    detail::annotate_ignore_sync_end_impl(f, l);
  }
}

class annotate_ignore_thread_sanitizer_guard {
 public:
  annotate_ignore_thread_sanitizer_guard(const char* file, int line)
      : file_(file), line_(line) {
    annotate_ignore_reads_begin(file_, line_);
    annotate_ignore_writes_begin(file_, line_);
  }

  annotate_ignore_thread_sanitizer_guard(
      const annotate_ignore_thread_sanitizer_guard&) = delete;
  annotate_ignore_thread_sanitizer_guard& operator=(
      const annotate_ignore_thread_sanitizer_guard&) = delete;

  ~annotate_ignore_thread_sanitizer_guard() {
    annotate_ignore_reads_end(file_, line_);
    annotate_ignore_writes_end(file_, line_);
  }

 private:
  const char* file_;
  int line_;
};

} // namespace folly