blob: c8c2cbbaf34b43c619cea290289fa22d9a23531a [file] [log] [blame]
/*
* Licensed to the Apache Software Foundation (ASF) under one
* or more contributor license agreements. See the NOTICE file
* distributed with this work for additional information
* regarding copyright ownership. The ASF licenses this file
* to you 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.
*/
#include "write_file.h"
#include <errno.h>
#include <fcntl.h>
#include <stdio.h>
#include <string.h>
#include <sys/stat.h>
#include <sys/time.h>
#include <unistd.h>
#include "common/config/config.h"
#include "common/logger/elog.h"
#include "utils/errno_define.h"
#ifdef _WIN32
int fsync(int);
#endif
using namespace common;
namespace storage {
#ifndef LIBTSFILE_SDK
int WriteFile::create(const FileID &file_id, int flags, mode_t mode) {
if (fd_ > 0) {
// log_err("file already opened, fd=%d", fd_);
ASSERT(false);
return E_ALREADY_EXIST;
}
file_id_ = file_id;
path_ = get_file_path_from_file_id(file_id_);
return do_create(flags, mode);
}
#endif
int WriteFile::create(const std::string &file_path, int flags, mode_t mode) {
if (fd_ > 0) {
// log_err("file already opened, fd=%d", fd_);
ASSERT(false);
return E_ALREADY_EXIST;
}
path_ = file_path;
return do_create(flags, mode);
}
int WriteFile::do_create(int flags, mode_t mode) {
int ret = E_OK;
// TODO make sure no same file exists
fd_ = ::open(path_.c_str(), flags, mode);
if (fd_ < 0) {
// log_err("open file error, path=%s, errno=%d", path_.c_str(), errno);
ret = E_FILE_OPEN_ERR;
} else {
}
return ret;
}
int WriteFile::write(const char *buf, uint32_t len) {
ASSERT(fd_ > 0);
#if 0 // DEBUG_SE
struct stat statbuf;
if (fstat(fd_, &statbuf) < 0) {
perror("fstat");
ASSERT(false);
}
printf("writer buffer(%u bytes) to file at %zu\n", len, statbuf.st_size);
for (uint32_t i = 0; i < len;) {
printf("0x%02x ", (uint8_t)buf[i]);
if ((++i) % 16 == 0) {
printf("\n");
}
}
printf("\n");
#endif
int ret = E_OK;
uint32_t write_done = 0;
while (write_done < len && IS_SUCC(ret)) {
int32_t cur_write = ::write(fd_, buf + write_done, len - write_done);
if (cur_write < 0) {
ret = E_FILE_WRITE_ERR;
// log_err("file writer error, path=%s, error=%d", path_.c_str(),
// errno);
} else {
write_done += cur_write;
}
}
return ret;
}
int WriteFile::sync() {
ASSERT(fd_ > 0);
if (::fsync(fd_) < 0) {
// log_err("file fsync error, path=%s, errno=%d", path_.c_str(), errno);
return E_FILE_SYNC_ERR;
}
return E_OK;
}
int WriteFile::close() {
ASSERT(fd_ > 0);
if (::close(fd_) < 0) {
#ifdef DEBUG_SE
std::cout << "failed to close " << path_ << " errorno " << errno
<< std::endl;
#endif
// log_err("file close error, path=%s, errno=%d", path_.c_str(), errno);
return E_FILE_CLOSE_ERR;
}
fd_ = -1;
#ifdef DEBUG_SE
std::cout << "close finish" << std::endl;
#endif
return E_OK;
}
} // end namespace storage
#ifdef _WIN32
int fsync(int fd) { return _commit(fd); }
#endif