106f6ba60Sopenharmony_ci/*
206f6ba60Sopenharmony_ci * Copyright (c) Huawei Technologies Co., Ltd. 2021. All rights reserved.
306f6ba60Sopenharmony_ci * Licensed under the Apache License, Version 2.0 (the "License");
406f6ba60Sopenharmony_ci * you may not use this file except in compliance with the License.
506f6ba60Sopenharmony_ci * You may obtain a copy of the License at
606f6ba60Sopenharmony_ci *
706f6ba60Sopenharmony_ci *     http://www.apache.org/licenses/LICENSE-2.0
806f6ba60Sopenharmony_ci *
906f6ba60Sopenharmony_ci * Unless required by applicable law or agreed to in writing, software
1006f6ba60Sopenharmony_ci * distributed under the License is distributed on an "AS IS" BASIS,
1106f6ba60Sopenharmony_ci * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
1206f6ba60Sopenharmony_ci * See the License for the specific language governing permissions and
1306f6ba60Sopenharmony_ci * limitations under the License.
1406f6ba60Sopenharmony_ci */
1506f6ba60Sopenharmony_ci#include "event_notifier.h"
1606f6ba60Sopenharmony_ci
1706f6ba60Sopenharmony_ci#include <cerrno>
1806f6ba60Sopenharmony_ci#include <climits>
1906f6ba60Sopenharmony_ci#include <cstring>
2006f6ba60Sopenharmony_ci#include <fcntl.h>
2106f6ba60Sopenharmony_ci#include <sys/eventfd.h>
2206f6ba60Sopenharmony_ci#include <unistd.h>
2306f6ba60Sopenharmony_ci#include "logging.h"
2406f6ba60Sopenharmony_ci
2506f6ba60Sopenharmony_ci#ifndef EFD_SEMAPHORE
2606f6ba60Sopenharmony_ci#define EFD_SEMAPHORE 1
2706f6ba60Sopenharmony_ci#endif
2806f6ba60Sopenharmony_ci
2906f6ba60Sopenharmony_ciEventNotifierPtr EventNotifier::Create(unsigned int initValue, unsigned int mask)
3006f6ba60Sopenharmony_ci{
3106f6ba60Sopenharmony_ci    return std::make_shared<EventNotifier>(initValue, mask);
3206f6ba60Sopenharmony_ci}
3306f6ba60Sopenharmony_ci
3406f6ba60Sopenharmony_ciEventNotifierPtr EventNotifier::CreateWithFd(int fd)
3506f6ba60Sopenharmony_ci{
3606f6ba60Sopenharmony_ci    return std::make_shared<EventNotifier>(fd);
3706f6ba60Sopenharmony_ci}
3806f6ba60Sopenharmony_ci
3906f6ba60Sopenharmony_ciEventNotifier::EventNotifier(unsigned int initValue, unsigned int mask) : fd_(-1), flags_(O_CLOEXEC)
4006f6ba60Sopenharmony_ci{
4106f6ba60Sopenharmony_ci    if (mask & NONBLOCK) {
4206f6ba60Sopenharmony_ci        flags_ |= O_NONBLOCK;
4306f6ba60Sopenharmony_ci    }
4406f6ba60Sopenharmony_ci    if (mask & SEMAPHORE) {
4506f6ba60Sopenharmony_ci        flags_ |= EFD_SEMAPHORE;
4606f6ba60Sopenharmony_ci    }
4706f6ba60Sopenharmony_ci    fd_ = eventfd(initValue, flags_);
4806f6ba60Sopenharmony_ci    CHECK_TRUE(fd_ >= 0, NO_RETVAL, "create eventfd FAILED, %d", errno);
4906f6ba60Sopenharmony_ci    PROFILER_LOG_DEBUG(LOG_CORE, "EventNotifier create eventfd %d done!", fd_);
5006f6ba60Sopenharmony_ci}
5106f6ba60Sopenharmony_ci
5206f6ba60Sopenharmony_ciEventNotifier::EventNotifier(int fd) : fd_(fd), flags_(0)
5306f6ba60Sopenharmony_ci{
5406f6ba60Sopenharmony_ci    int flags = fcntl(fd_, F_GETFL);
5506f6ba60Sopenharmony_ci    CHECK_TRUE(flags >= 0, NO_RETVAL, "get flags of fd %d FAILED, %d", fd, errno);
5606f6ba60Sopenharmony_ci    PROFILER_LOG_DEBUG(LOG_CORE, "EventNotifier bind eventfd %d done!", fd_);
5706f6ba60Sopenharmony_ci}
5806f6ba60Sopenharmony_ci
5906f6ba60Sopenharmony_ciEventNotifier::~EventNotifier()
6006f6ba60Sopenharmony_ci{
6106f6ba60Sopenharmony_ci    PROFILER_LOG_DEBUG(LOG_CORE, "EventNotifier close eventfd %d", fd_);
6206f6ba60Sopenharmony_ci    close(fd_);
6306f6ba60Sopenharmony_ci}
6406f6ba60Sopenharmony_ci
6506f6ba60Sopenharmony_ciint EventNotifier::GetFd() const
6606f6ba60Sopenharmony_ci{
6706f6ba60Sopenharmony_ci    return fd_;
6806f6ba60Sopenharmony_ci}
6906f6ba60Sopenharmony_ci
7006f6ba60Sopenharmony_ciuint64_t EventNotifier::Take() const
7106f6ba60Sopenharmony_ci{
7206f6ba60Sopenharmony_ci    uint64_t value = UINT64_MAX;
7306f6ba60Sopenharmony_ci    int retval = TEMP_FAILURE_RETRY(read(fd_, &value, sizeof(value)));
7406f6ba60Sopenharmony_ci    CHECK_TRUE(retval == sizeof(value), 0, "read value from eventfd %d failed, %d!", fd_, errno);
7506f6ba60Sopenharmony_ci    return value;
7606f6ba60Sopenharmony_ci}
7706f6ba60Sopenharmony_ci
7806f6ba60Sopenharmony_cibool EventNotifier::Post(uint64_t value) const
7906f6ba60Sopenharmony_ci{
8006f6ba60Sopenharmony_ci    int retval = TEMP_FAILURE_RETRY(write(fd_, &value, sizeof(value)));
8106f6ba60Sopenharmony_ci    CHECK_TRUE(retval == sizeof(value), false, "write value to eventfd %d failed, %d!", fd_, errno);
8206f6ba60Sopenharmony_ci    return true;
8306f6ba60Sopenharmony_ci}
84