Files
Linux-C-Notes/C16-并发/parallel/signal/slowcat2.c
2024-05-27 02:33:10 +08:00

103 lines
2.1 KiB
C
Raw Blame History

This file contains ambiguous Unicode characters
This file contains Unicode characters that might be confused with other characters. If you think that this is intentional, you can safely ignore this warning. Use the Escape button to reveal them.
#include <errno.h>
#include <fcntl.h>
#include <signal.h>
#include <stdio.h>
#include <stdlib.h>
#include <unistd.h>
#define CPS 10
#define BUFSIZE CPS
#define BURST 100 //!! token上限
// static volatile int token = 0;
static volatile sig_atomic_t token = 0; //!!! 保证原子
static void alrm_handler(int s)
{
alarm(1); //!! 产生新的alarm
token++;
if (token > BURST)
token = BURST;
}
/***********************************************************************
* @brief slowcat
* @details 流量控制,令牌桶实现
* @param argc
* @param argv
* @return int
***********************************************************************/
int main(int argc, char **argv)
{
int sfd, dfd = 1;
char buf[BUFSIZE];
int len, ret, pos;
if (argc < 2)
{
fprintf(stderr, "Usage: %s <src>\n", argv[0]);
exit(1);
}
signal(SIGALRM, alrm_handler);
alarm(1);
do //!!!
{
sfd = open(argv[1], O_RDONLY);
if (sfd < 0)
{
//!!!
if (errno != EINTR)
{
perror("open");
exit(1);
}
}
} while (sfd < 0);
while (1)
{
while (token <= 0)
pause( ); //!
token--; //!!! 这条指令不一定原子
//* 阻塞在这里的时候token会自增
while ((len = read(sfd, buf, BUFSIZE)) < 0)
{
//!!!
if (EINTR == errno)
continue;
perror("read");
break;
}
if (len == 0)
break;
pos = 0;
while (len > 0)
{
/* 可能没写完被别的中断打断 */
ret = write(dfd, buf + pos, len);
if (ret < 0)
{
//!!!
if (EINTR == errno)
continue;
perror("write");
exit(1);
}
pos += ret;
len -= ret;
}
// sleep(1); //! 能实现,但是移植性不好
}
close(sfd);
exit(0);
}