2023-01-15 13:26:14 +01:00
|
|
|
#include "mainwindow.h"
|
2023-01-07 12:44:45 +01:00
|
|
|
|
|
|
|
using namespace std;
|
|
|
|
|
|
|
|
#define READ 0
|
|
|
|
#define WRITE 1
|
|
|
|
|
2023-01-23 23:42:20 +01:00
|
|
|
//Take a string and explode it in array
|
|
|
|
// s => string to explode
|
|
|
|
// c => character separator
|
|
|
|
// n => number of results in array, the last is the rest of string to end
|
2023-01-07 12:44:45 +01:00
|
|
|
const vector<string> explode(const string& s, const char& c, int n = 0)
|
|
|
|
{
|
|
|
|
string buff;
|
|
|
|
vector<string> v;
|
|
|
|
size_t pos = 0;
|
|
|
|
size_t ppos = 0;
|
|
|
|
int i = 0;
|
|
|
|
|
|
|
|
while (i < n - 1)
|
|
|
|
{
|
|
|
|
pos = s.find(c, ppos);
|
|
|
|
if (pos != string::npos)
|
|
|
|
{
|
|
|
|
buff = s.substr(ppos, pos - ppos);
|
|
|
|
if (buff != "")
|
|
|
|
{
|
|
|
|
i++;
|
|
|
|
v.push_back(s.substr(ppos, pos - ppos));
|
|
|
|
}
|
|
|
|
ppos = pos + 1;
|
|
|
|
}else
|
|
|
|
{
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
if (ppos < s.size())
|
|
|
|
{
|
|
|
|
v.push_back(s.substr(ppos));
|
|
|
|
}
|
|
|
|
return v;
|
|
|
|
}
|
|
|
|
|
2023-01-23 23:42:20 +01:00
|
|
|
// open a pipe, fork and return pid of child
|
|
|
|
// argv => array of string with command in first and parameters following
|
|
|
|
FILE * popen2(array<string,8> argv, string type, int & pid)
|
2023-01-07 12:44:45 +01:00
|
|
|
{
|
|
|
|
pid_t child_pid;
|
|
|
|
int fd[2];
|
2023-01-18 23:12:21 +01:00
|
|
|
QString message;
|
2023-01-23 10:04:54 +01:00
|
|
|
string command;
|
2023-01-07 12:44:45 +01:00
|
|
|
|
2023-01-18 23:12:21 +01:00
|
|
|
if (pipe(fd) == -1)
|
2023-01-07 12:44:45 +01:00
|
|
|
{
|
2023-01-18 23:12:21 +01:00
|
|
|
message = "Open pipe failed" + QString::fromStdString(strerror(errno));
|
|
|
|
QMessageBox::warning(
|
|
|
|
NULL,
|
|
|
|
"RsyncUI",
|
|
|
|
message);
|
2023-01-23 23:42:20 +01:00
|
|
|
exit(-1);
|
2023-01-18 23:12:21 +01:00
|
|
|
}else
|
2023-01-07 12:44:45 +01:00
|
|
|
{
|
2023-01-18 23:12:21 +01:00
|
|
|
if((child_pid = fork()) == -1)
|
2023-01-12 09:23:45 +01:00
|
|
|
{
|
2023-01-18 23:12:21 +01:00
|
|
|
perror("fork");
|
|
|
|
exit(1);
|
2023-01-12 09:23:45 +01:00
|
|
|
}
|
2023-01-18 23:12:21 +01:00
|
|
|
|
|
|
|
/* child process */
|
|
|
|
if (child_pid == 0)
|
2023-01-07 12:44:45 +01:00
|
|
|
{
|
2023-01-18 23:12:21 +01:00
|
|
|
if (type == "r")
|
|
|
|
{
|
|
|
|
close(fd[READ]); //Close the READ end of the pipe since the child's fd is write-only
|
|
|
|
dup2(fd[WRITE], 1); //Redirect stdout to pipe
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
close(fd[WRITE]); //Close the WRITE end of the pipe since the child's fd is read-only
|
|
|
|
dup2(fd[READ], 0); //Redirect stdin to pipe
|
|
|
|
}
|
|
|
|
|
|
|
|
setpgid(child_pid, child_pid); //Needed so negative PIDs can kill children of /bin/sh
|
2023-01-28 15:41:48 +01:00
|
|
|
//TODO : change for execvp
|
|
|
|
|
2023-01-23 23:42:20 +01:00
|
|
|
if (execlp(argv[0].c_str(), argv[0].c_str(), argv[1].c_str(), argv[2].c_str(), argv[3].c_str(), argv[4].c_str(), argv[5].c_str(), NULL ) == -1)
|
2023-01-18 23:12:21 +01:00
|
|
|
exit (0);
|
2023-01-07 12:44:45 +01:00
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
2023-01-18 23:12:21 +01:00
|
|
|
if (type == "r")
|
|
|
|
{
|
|
|
|
close(fd[WRITE]); //Close the WRITE end of the pipe since parent's fd is read-only
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
close(fd[READ]); //Close the READ end of the pipe since parent's fd is write-only
|
|
|
|
}
|
2023-01-07 12:44:45 +01:00
|
|
|
}
|
|
|
|
|
2023-01-18 23:12:21 +01:00
|
|
|
pid = child_pid;
|
|
|
|
if (type == "r")
|
|
|
|
{
|
|
|
|
return fdopen(fd[READ], "r");
|
|
|
|
}
|
|
|
|
return fdopen(fd[WRITE], "w");
|
|
|
|
}
|
|
|
|
return 0;
|
2023-01-07 12:44:45 +01:00
|
|
|
}
|
|
|
|
|
2023-01-23 23:42:20 +01:00
|
|
|
// close pipe open by popen2 while pid is finished
|
|
|
|
// fp => file pointer
|
|
|
|
// pid => pid of the processus open bu popen2
|
|
|
|
|
2023-01-07 12:44:45 +01:00
|
|
|
int pclose2(FILE * fp, pid_t pid)
|
|
|
|
{
|
|
|
|
int stat;
|
|
|
|
|
|
|
|
fclose(fp);
|
|
|
|
while (waitpid(pid, &stat, 0) == -1)
|
|
|
|
{
|
|
|
|
if (errno != EINTR)
|
|
|
|
{
|
|
|
|
stat = -1;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return stat;
|
|
|
|
}
|
|
|
|
|