108 lines
		
	
	
		
			2.6 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
			
		
		
	
	
			108 lines
		
	
	
		
			2.6 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
| /**
 | |
|  * Copyright (c) 2021 OceanBase
 | |
|  * OceanBase CE is licensed under Mulan PubL v2.
 | |
|  * You can use this software according to the terms and conditions of the Mulan PubL v2.
 | |
|  * You may obtain a copy of Mulan PubL v2 at:
 | |
|  *          http://license.coscl.org.cn/MulanPubL-2.0
 | |
|  * THIS SOFTWARE IS PROVIDED ON AN "AS IS" BASIS, WITHOUT WARRANTIES OF ANY KIND,
 | |
|  * EITHER EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO NON-INFRINGEMENT,
 | |
|  * MERCHANTABILITY OR FIT FOR A PARTICULAR PURPOSE.
 | |
|  * See the Mulan PubL v2 for more details.
 | |
|  */
 | |
| 
 | |
| #define _GNU_SOURCE 1
 | |
| #include <unistd.h>
 | |
| #include <stdio.h>
 | |
| #include <errno.h>
 | |
| #include <dlfcn.h>
 | |
| #include <libaio.h>
 | |
| #include <libconfig.h>
 | |
| #include <stdlib.h>
 | |
| #include <signal.h>
 | |
| #include <pthread.h>
 | |
| #include <sys/stat.h>
 | |
| #include <sys/time.h>
 | |
| #include <sys/types.h>
 | |
| #include <sys/syscall.h>
 | |
| #include <fcntl.h>
 | |
| 
 | |
| static ssize_t (*real_pwrite)(int fd, const void* buf, size_t count, off_t offset);
 | |
| 
 | |
| struct pwrite_conf_t {
 | |
|   int start_fail_index_;
 | |
|   int fail_cnt_;
 | |
|   int reset_call_cnt_;
 | |
|   int error_;
 | |
| } pwrite_conf_t;
 | |
| 
 | |
| static struct pwrite_conf_t pwrite_conf;
 | |
| static int is_inited = 0;
 | |
| static pthread_mutex_t init_mutex;
 | |
| static time_t load_conf_time = 0;
 | |
| static pthread_mutex_t load_conf_mutex;
 | |
| static int call_cnt;
 | |
| 
 | |
| void io_init(void)
 | |
| {
 | |
|   *(void**)&real_pwrite = dlsym(RTLD_NEXT, "pwrite");
 | |
|   printf("init real pwrite succeed\n");
 | |
| }
 | |
| 
 | |
| void check_init()
 | |
| {
 | |
|   pthread_mutex_lock(&init_mutex);
 | |
|   if (0 == is_inited) {
 | |
|     io_init();
 | |
|     is_inited = 1;
 | |
|   }
 | |
|   pthread_mutex_unlock(&init_mutex);
 | |
| }
 | |
| 
 | |
| void load_conf()
 | |
| {
 | |
|   int fd = open("pwrite_conf", O_RDONLY);
 | |
|   char buf[64];
 | |
|   memset(buf, 0, sizeof(buf));
 | |
|   read(fd, buf, sizeof(buf));
 | |
|   close(fd);
 | |
|   sscanf(buf,
 | |
|       "%d,%d,%d,%d",
 | |
|       &pwrite_conf.start_fail_index_,
 | |
|       &pwrite_conf.fail_cnt_,
 | |
|       &pwrite_conf.reset_call_cnt_,
 | |
|       &pwrite_conf.error_);
 | |
|   if (pwrite_conf.reset_call_cnt_) {
 | |
|     call_cnt = 0;
 | |
|     printf("reset call cnt succeed\n");
 | |
|   }
 | |
|   printf("load pwrite conf succeed, %s\n", buf);
 | |
| }
 | |
| 
 | |
| void check_load_conf()
 | |
| {
 | |
|   pthread_mutex_lock(&load_conf_mutex);
 | |
|   time_t now = time(NULL);
 | |
|   if (now - load_conf_time > 2) {
 | |
|     load_conf();
 | |
|     load_conf_time = now;
 | |
|   }
 | |
|   pthread_mutex_unlock(&load_conf_mutex);
 | |
| }
 | |
| 
 | |
| ssize_t pwrite(int fd, const void* buf, size_t count, off_t offset)
 | |
| {
 | |
|   ssize_t rt = 0;
 | |
|   check_init();
 | |
|   check_load_conf();
 | |
|   if (call_cnt < pwrite_conf.start_fail_index_ || call_cnt >= pwrite_conf.start_fail_index_ + pwrite_conf.fail_cnt_) {
 | |
|     errno = 0;
 | |
|     rt = real_pwrite(fd, buf, count, offset);
 | |
|   } else {
 | |
|     errno = pwrite_conf.error_;
 | |
|     rt = 0;
 | |
|   }
 | |
|   ++call_cnt;
 | |
|   printf("pwrite called");
 | |
|   return rt;
 | |
| }
 | 
