123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461 |
- #include "async_file.h"
- #include <sys/types.h>
- #include <sys/stat.h>
- #include <fcntl.h>
- #include <errno.h>
- ////////////////CMapBase
- CMapBase::CMapBase():
- _fd(-1), _rw(0), _size(0), _map(0)
- {
- bzero(_path, sizeof(_path));
- bzero(_errmsg, sizeof(_errmsg));
- }
- CMapBase::~CMapBase()
- {
- unmount();
- }
- void CMapBase::unmount()
- {
- close_fd();
- unmap();
- }
- void CMapBase::Unlink()
- {
- /*
- * 读完后,truncate文件为0并删除,防止启动磁盘flush
- */
- if (_rw == O_RDONLY) {
- //int unused;
- //unused = ftruncate(_fd, 0);
- //unused = unlink(_path);
- }
- }
- int CMapBase::Mount(const char *path, int rw, int size)
- {
- mkdir(ASYNC_FILE_PATH, 0777);
- if (access(ASYNC_FILE_PATH, W_OK | X_OK) < 0) {
- snprintf(_errmsg, sizeof(_errmsg), "dir(%s) Not writable",
- ASYNC_FILE_PATH);
- return -1;
- }
- int flag = rw;
- int prot = 0;
- switch (flag) {
- case O_WRONLY:
- case O_RDWR:
- default:
- flag = O_CREAT | O_RDWR;
- prot = PROT_WRITE | PROT_READ;
- break;
- case O_RDONLY:
- prot = PROT_READ;
- break;
- }
- if ((_fd = open(path, flag, 0644)) < 0) {
- snprintf(_errmsg, sizeof(_errmsg),
- "open failed[path:%s], errno:%d %m", path, errno);
- return -1;
- }
- struct stat st;
- fstat(_fd, &st);
- if (O_RDONLY == rw) {
- size = st.st_size;
- } else if (st.st_size != size) {
- if (ftruncate(_fd, size) != 0) {
- return -1;
- }
- }
- _map = (char *)mmap(0, size, prot, MAP_SHARED, _fd, 0);
- if (MAP_FAILED == _map) {
- snprintf(_errmsg, sizeof(_errmsg),
- "map failed[path:%s, size:%d, _fd:%d], errno:%d %m\n",
- path, size, _fd, errno);
- return -1;
- }
- _rw = rw;
- _size = size;
- snprintf(_path, sizeof(_path), "%s", path);
- return 0;
- }
- //////////////////FileImpl
- int CAsyncFileImpl::OpenForWriter(CAsyncFilePos & pos)
- {
- _pos = pos;
- char path[256] = { 0 };
- FileName(path, sizeof(path));
- return Mount(path, O_WRONLY, MAX_ASYNC_FILE_SIZE);
- }
- int CAsyncFileImpl::OpenForReader(CAsyncFilePos & pos)
- {
- _pos = pos;
- char path[256] = { 0 };
- FileName(path, sizeof(path));
- return Mount(path, O_RDONLY);
- }
- int CAsyncFileImpl::Input(buffer & buff)
- {
- if (IsWriterEnd(buff.size())) {
- WriteEndFlag();
- return CHBGlobal::ASYNC_NEED_SWTICH_FILE;
- }
- //len
- uint32_t *len = (uint32_t *) ((char *)_map + _pos.offset);
- *len = buff.size();
- _pos.Front(4);
- //buff
- memcpy((char *)_map + _pos.offset, buff.c_str(), *len);
- _pos.Front(*len);
- return CHBGlobal::ASYNC_PROCESS_OK;
- }
- int CAsyncFileImpl::Output(buffer & buff)
- {
- if (IsReaderEnd())
- return CHBGlobal::ASYNC_NEED_SWTICH_FILE;
- //len
- uint32_t *len = (uint32_t *) ((char *)_map + _pos.offset);
- _pos.Front(4);
- //buff
- buff.append((char *)_map + _pos.offset, *len);
- _pos.Front(*len);
- return CHBGlobal::ASYNC_PROCESS_OK;
- }
- //////////////////Writer /////////////////////////
- int CAsyncFileWriter::Open()
- {
- if (_controller.Init()) {
- snprintf(_errmsg, sizeof(_errmsg), "controller init failed, %s",
- _controller.ErrorMessage());
- return CHBGlobal::ASYNC_PROCESS_ERR;
- }
- CAsyncFileImpl *p = new CAsyncFileImpl();
- if (p->OpenForWriter(_controller.WriterPos())) {
- snprintf(_errmsg, sizeof(_errmsg), "open for writer failed, %s",
- p->ErrorMessage());
- return CHBGlobal::ASYNC_PROCESS_ERR;
- }
- /*
- * 加入writer的map文件列表
- */
- AddToList(p);
- return CHBGlobal::ASYNC_PROCESS_OK;
- }
- int CAsyncFileWriter::Write(buffer & buf)
- {
- if (_asyncfiles.size() <= 0) {
- snprintf(_errmsg, sizeof(_errmsg),
- "__BUG__, writer maps is zero");
- return CHBGlobal::ERR_ASYNC_WRITER_LOGIC;
- }
- //从缓冲中取一个文件来写
- int ret = _asyncfiles.front()->Input(buf);
- switch (ret) {
- case CHBGlobal::ASYNC_PROCESS_OK:
- {
- //成功写入 更新控制文件写指针
- _controller.WriterPos() =
- _asyncfiles.front()->CurrentPos();
- break;
- }
- case CHBGlobal::ASYNC_NEED_SWTICH_FILE:
- {
- //已经写满,需要切换, 先准备下一个文件,再切换。
- CAsyncFilePos pos = _controller.WriterPos();
- pos.Shift();
- CAsyncFileImpl *p = new CAsyncFileImpl();
- if (p->OpenForWriter(pos)) {
- snprintf(_errmsg, sizeof(_errmsg),
- "create CAsyncFileImpl failed, %s",
- p->ErrorMessage());
- DELETE(p);
- return CHBGlobal::ERR_ASYNC_SWTICH_FILE_ERR;
- }
- //加入缓冲
- AddToList(p);
- //更新控制文件
- _controller.SwitchWriterPos();
- //继续写
- return Write(buf);
- break;
- }
- //no default.
- }
- return CHBGlobal::ASYNC_PROCESS_OK;
- }
- /////////////////////////Reader ////////////////////
- int CAsyncFileReader::Open()
- {
- if (_controller.Init()) {
- snprintf(_errmsg, sizeof(_errmsg), "controller init failed, %s",
- _controller.ErrorMessage());
- return CHBGlobal::ASYNC_PROCESS_ERR;
- }
- _asyncfile = new CAsyncFileImpl();
- if (_asyncfile->OpenForReader(_controller.ReaderPos())) {
- snprintf(_errmsg, sizeof(_errmsg), "open for reader failed, %s",
- _asyncfile->ErrorMessage());
- return CHBGlobal::ASYNC_PROCESS_ERR;
- }
- return CHBGlobal::ASYNC_PROCESS_OK;
- }
- inline void CAsyncFileReader::Commit(void) {
- if(_processing) {
- _controller.ReaderPos() = _asyncfile->CurrentPos();
- _processing = 0;
- }
- }
- int CAsyncFileReader::Read(buffer & buff)
- {
- if(!_asyncfile){
- snprintf(_errmsg, sizeof(_errmsg), "reader encounter logic error");
- return CHBGlobal::ERR_ASYNC_READER_LOGIC;
- }
- Commit();
- /* 判断是否出现切文件暂态,如果出现则sespend读者 */
- if(_controller.ReaderPos().IsTransient(_controller.WriterPos())){
- return CHBGlobal::ASYNC_READER_WAIT_DATA;
- }
- /* 暂时没有更多数据 */
- if (_controller.ReaderPos().EQ(_controller.WriterPos()))
- return CHBGlobal::ASYNC_READER_WAIT_DATA;
- /* ERROR */
- if( _controller.ReaderPos().GT(_controller.WriterPos())) {
- /* 害怕再次遇到切换暂态 */
- usleep(1000);
- if( _controller.ReaderPos().GT(_controller.WriterPos())) {
- snprintf(_errmsg, sizeof(_errmsg), "reader pos is overflow");
- return CHBGlobal::ERR_ASYNC_READER_OVERFLOW;
- }
- }
- int ret = _asyncfile->Output(buff);
- switch (ret) {
- case CHBGlobal::ASYNC_PROCESS_OK:
- {
- // mark as processing, delay commit
- _processing = 1;
-
- //更新控制文件读指针
- //_controller.ReaderPos() = _asyncfile->CurrentPos();
- break;
- }
- case CHBGlobal::ASYNC_NEED_SWTICH_FILE:
- {
- CAsyncFilePos pos = _controller.ReaderPos();
- pos.Shift();
- //delete file
- _asyncfile->Unlink();
- DELETE(_asyncfile);
- _asyncfile = new CAsyncFileImpl();
- if (_asyncfile->OpenForReader(pos)) {
- snprintf(_errmsg, sizeof(_errmsg),
- "create CAsyncFileImpl failed, %s",
- _asyncfile->ErrorMessage());
- DELETE(_asyncfile);
- return CHBGlobal::ERR_ASYNC_SWTICH_FILE_ERR;
- }
- _controller.SwitchReaderPos();
- return Read(buff);
- break;
- }
- }
- return CHBGlobal::ASYNC_PROCESS_OK;
- }
- int CAsyncFileChecker::Check()
- {
- if (_controller.Init()) {
- snprintf(_errmsg, sizeof(_errmsg), "controller init failed");
- return CHBGlobal::ERR_ASYNC_CONTROLLER_ERR;
- }
- if (_controller.IsDirty()) {
- snprintf(_errmsg, sizeof(_errmsg),
- "full sync is not complete, status is dirty");
- return CHBGlobal::ERR_FULL_SYNC_NOT_COMPLETE;
- }
- /* 检查reader是否比write快 */
- if (_controller.ReaderPos().GT(_controller.WriterPos())) {
- snprintf(_errmsg, sizeof(_errmsg), "reader pos is overflow");
- return CHBGlobal::ERR_ASYNC_READER_OVERFLOW;
- }
- /*检查reader的有效性 */
- if (!_controller.ReaderPos().Zero()) {
- _asyncfile = new CAsyncFileImpl;
- if (_asyncfile->OpenForReader(_controller.ReaderPos())) {
- snprintf(_errmsg, sizeof(_errmsg),
- "reader pos is error");
- return CHBGlobal::ERR_ASYNC_READER_POS_ERR;
- }
- if (_controller.ReaderPos().offset >
- (unsigned)_asyncfile->Size() + 4) {
- snprintf(_errmsg, sizeof(_errmsg),
- "reader pos is error");
- return CHBGlobal::ERR_ASYNC_READER_POS_ERR;
- }
- }
- DELETE(_asyncfile);
- /*检查writer有效性 */
- if (!_controller.WriterPos().Zero()) {
- _asyncfile = new CAsyncFileImpl;
- if (_asyncfile->OpenForWriter(_controller.WriterPos())) {
- snprintf(_errmsg, sizeof(_errmsg),
- "writer pos is error");
- return CHBGlobal::ERR_ASYNC_WRITER_POS_ERR;
- }
- if (_controller.WriterPos().offset >
- (unsigned)_asyncfile->Size() + 4) {
- snprintf(_errmsg, sizeof(_errmsg),
- "writer pos is error");
- return CHBGlobal::ERR_ASYNC_WRITER_POS_ERR;
- }
- }
- DELETE(_asyncfile);
- return 0;
- }
- /*
- * 测试代码
- */
- #ifdef __UNIT_TEST__
- #include <stdio.h>
- #include <sys/types.h>
- #include <sys/wait.h>
- int W()
- {
- buffer buff;
- buff.append("just for test");
- CAsyncFileWriter writer(3);
- if (writer.Open()) {
- printf("writer Open failed\n");
- return -1;
- }
- while (1) {
- writer.Write(buff);
- //usleep(1000);
- }
- return 0;
- }
- int R()
- {
- CAsyncFileReader reader;
- if (reader.Open()) {
- printf("reader Open failed\n");
- return -1;
- }
- while (1) {
- buffer buff;
- int ret = reader.Read(buff);
- if (ret == CHBGlobal::ASYNC_READER_WAIT_DATA) {
- printf("no data. sleep 1 \n");
- sleep(1);
- } else {
- printf("error, \n");
- exit(0);
- }
- // printf("buff.len=%d, buff.content=%9s\n", buff.size(), buff.c_str());
- }
- return 0;
- }
- int main()
- {
- int ret = 0;
- CAsyncFileChecker check;
- if (ret = check.Check()) {
- printf("check not pass, ret=%d\n", ret);
- return -1;
- }
- int pid = fork();
- if (pid < 0) {
- printf("fork child failed: %m\n");
- return -1;
- } else if (pid > 0) {
- nice(5);
- W();
- } else {
- //先让writer跑起来, 否则没有文件,reader会出错
- sleep(3);
- R();
- }
- return 0;
- }
- #endif
|