32#include "XrdVersion.hh"
183 XrdXrootdProtocol::ProtStack(
"ProtStack",
184 "xroot protocol anchor");
203 const char *txt =
"completed.";
207 pi->
eDest->
Say(
"Copr. 2012 Stanford University, xroot protocol "
209 pi->
eDest->
Say(
"++++++ xroot protocol initialization started.");
215 else txt =
"failed.";
216 pi->
eDest->
Say(
"------ xroot protocol initialization ", txt);
240 if (pi->
Port < 0)
return 1094;
274 static unsigned int Sid = 1;
297static struct hs_response
312 {
if (dlen <= 0) lp->
setEtext(
"handshake not received");
330 rc = lp->
Send((
char *)&hsresp,
sizeof(hsresp));
341 if (lp->
Recv(hsbuff, hsSZ) != hsSZ)
366#define TRACELINK Link
377 else if ((rc = (*this.*
Resume)()) != 0)
return rc;
378 else {
Resume = 0;
return 0;}
387 reqID = ntohs(
Request.header.requestid);
395 Request.header.requestid = reqID;
399 <<
" dlen=" <<
Request.header.dlen);
406 return Link->setEtext(
"protocol data length error");
444 {
const char *eText =
"Request not signed";
448 <<
" verification failed; " <<eText);
449 SI->Bump(
SI->badSCnt);
450 return Link->setEtext(eText);
452 SI->Bump(
SI->aokSCnt);
458 <<
" unneeded signature discarded.");
460 {
eDest.Emsg(
"Protocol",
"Client is needlessly signing requests.");
463 SI->Bump(
SI->ignSCnt);
471 switch(
Request.header.requestid)
476 "Invalid request; user not logged in");
477 return Link->setEtext(
"request without login");
486 switch(
Request.header.requestid)
503 if (!
Request.header.dlen)
return do_Truncate();
513 switch(
Request.header.requestid)
525 "Invalid request; user not authenticated");
538 switch(
Request.header.requestid)
555 if (
Link->AddrInfo()->isPrivate())
rdType = 1;
563 {doRdr =
true;
break;}
575 switch(
Request.header.requestid)
583 SI->Bump(
SI->miscCnt);
587 switch(
Request.header.requestid)
593 case kXR_mv:
return do_Mv();
596 case kXR_rm:
return do_Rm();
632 return Link->setEtext(
"signature data length error");
658#define TRACELINK Link
675 if (reason && strcmp(reason,
"hangup"))
676 {snprintf(buff,
sizeof(buff),
"%s (%s)", ctbuff, reason);
678 }
else sfxp = ctbuff;
738 if (
Response.isOurs()) ProtStack.Push(&ProtLink);
745int XrdXrootdProtocol::StatGen(
struct stat &buf,
char *xxBuff,
int xxLen,
748 const mode_t isReadable = (S_IRUSR | S_IRGRP | S_IROTH);
749 const mode_t isWritable = (S_IWUSR | S_IWGRP | S_IWOTH);
750 const mode_t isExecable = (S_IXUSR | S_IXGRP | S_IXOTH);
753 union {
long long uuid;
struct {
int hi;
int lo;} id;} Dev;
764 Dev.id.lo = buf.st_ino;
765 Dev.id.hi = buf.st_dev;
769 if (buf.st_mode & isReadable
770 &&((buf.st_mode & S_IRUSR && theuid == buf.st_uid)
771 || (buf.st_mode & S_IRGRP && thegid == buf.st_gid)
776 if (buf.st_mode & isWritable
777 &&((buf.st_mode & S_IWUSR && theuid == buf.st_uid)
778 || (buf.st_mode & S_IWGRP && thegid == buf.st_gid)
783 if (buf.st_mode & isExecable
784 &&((buf.st_mode & S_IXUSR && theuid == buf.st_uid)
785 || (buf.st_mode & S_IXGRP && thegid == buf.st_gid)
786 || buf.st_mode & S_IXOTH)) flags |=
kXR_xset;
791 if (S_ISDIR(buf.st_mode)) flags |=
kXR_isDir;
792 else if (!S_ISREG(buf.st_mode)) flags |=
kXR_other;
801 fsz =
static_cast<long long>(buf.st_size);
805 m = snprintf(xxBuff, xxLen,
"%lld %lld %d %lld",
806 Dev.uuid, fsz, flags, (
long long) buf.st_mtime);
812 char *origP = xxBuff;
813 char *nullP = xxBuff + m++;
814 xxBuff += m; xxLen -= m;
815 n = snprintf(xxBuff, xxLen,
"%lld %lld %04o ",
816 (
long long) buf.st_ctime, (
long long) buf.st_atime,
818 if (n >= xxLen)
return m;
819 xxBuff += n; xxLen -= n;
823 if (buf.st_uid ==
myUID)
824 {
if (
myUNLen >= xxLen)
return m;
836 if (buf.st_gid ==
myGID)
837 {
if (
myGNLen >= xxLen)
return m;
848 return xxBuff - origP;
860 {
SI->statsMutex.Lock();
878 SI->statsMutex.UnLock();
883 return SI->Stats(buff, blen, do_sync);
898 {
if (bsz) *bsz = xbP->
bsize;
915 if (minasz >=
argp->bsize || datasz >=
argp->bsize/2)
return Swap(buff);
957 else if (buff !=
argp->buff) errno = EINVAL;
963 if (
argp)
return oldBP;
1008int XrdXrootdProtocol::CheckSum(
XrdOucStream *Stream,
char **argv,
int argc)
1015 {
Stream->PutLine(
"Internal error; not enough checksum args!");
1022 XrdOucErrInfo myInfo(argv[3], &myEnv);
1026 if (argc > 4 && *argv[4]) myEnv.
Put(
"request.name", argv[4]);
1034 Stream->PutLine(myInfo.getErrText(ecode));
1035 if (rc) {
SI->errorCnt++;
1036 if (ecode) rc = ecode;
1045void XrdXrootdProtocol::Cleanup()
1061 {
Stream[i]->Stream[0] = 0;
1063 Stream[i]->Recycle(0, 0, 0);
1101 SI->statsMutex.Lock();
1103 SI->statsMutex.UnLock();
1136 {
if (rlen != -ENOMSG)
return Link->setEtext(
"link read error");
1141 TRACEP(REQ, dtype <<
" timeout; read " <<rlen <<
" of " <<blen <<
" bytes");
1150 const char *dtype,
char *buff,
int blen)
1157 gdCtl.BuffLen = blen;
1158 gdCtl.Buffer = buff;
1159 gdCtl.CallBack= cbP;
1160 gdCtl.ioDType = dtype;
1167 {
gdCtl.useCB =
true;
1170 return getDataCont();
1174namespace {
int consumed = 0;}
1177 const char *dtype,
struct iovec *
iov,
int iovn)
1184 gdCtl.iovNum = iovn;
1186 gdCtl.CallBack= cbP;
1187 gdCtl.ioDType = dtype;
1194 {
gdCtl.useCB =
true;
1198 return getDataIovCont();
1205int XrdXrootdProtocol::getDataCont()
1212 return -EINPROGRESS;
1218 gdCtl.Buffer += rlen;
1219 gdCtl.BuffLen -= rlen;
1229 rlen =
gdCtl.CallBack->gdDone();
1230 if (rlen < 0)
break;
1243 if (
gdCtl.BuffLen < 0)
1244 {rlen =
Link->setEtext(
"link excessive read length error");
1252 Resume = &XrdXrootdProtocol::getDataCont;
1260 TRACEP(REQ,
gdCtl.ioDType <<
" timeout; read " <<rlen <<
" bytes "
1261 <<
gdCtl.BuffLen <<
" remaining");
1268 if (rlen != -ENOMSG)
return Link->setEtext(
"link read error");
1274 gdCtl.CallBack->gdFail();
1284int XrdXrootdProtocol::getDataIovCont()
1291 return -EINPROGRESS;
1296do{
struct iovec *ioV =
gdCtl.iovVec;
1297 int i, rlen, iovN =
gdCtl.iovNum, iNow =
gdCtl.iovNow;
1304 {
if (rlen != -ENOMSG)
Link->setEtext(
"link read error");
1306 {ioV[iNow].iov_base = ((
char *)ioV[iNow].iov_base) -
gdCtl.iovAdj;
1307 ioV[iNow].iov_len +=
gdCtl.iovAdj;
1316 for (i = iNow; i < iovN && (int)ioV[i].iov_len <= rlen; i++)
1317 rlen -= ioV[i].iov_len;
1321 if (i != iNow &&
gdCtl.iovAdj)
1322 {ioV[iNow].iov_base = ((
char *)ioV[iNow].iov_base) -
gdCtl.iovAdj;
1323 ioV[iNow].iov_len +=
gdCtl.iovAdj;
1337 rc =
gdCtl.CallBack->gdDone();
1347 rc =
Link->setEtext(
"link iov read length error");
1357 gdCtl.iovAdj += rlen;
1358 ioV[i].iov_base = ((
char *)ioV[i].iov_base) + rlen;
1359 ioV[i].iov_len -= rlen;
1365 Resume = &XrdXrootdProtocol::getDataIovCont;
1373 TRACEP(REQ,
gdCtl.ioDType<<
" read timeout; "<<iovN-i<<
" of "
1374 <<
gdCtl.iovNum <<
" iov elements left");
1383 gdCtl.CallBack->gdFail();
1400 gdCtl.DumpLen = dlen;
1401 gdCtl.ioDType = dtype;
1407 return (inCB ? 1 : getDumpCont());
1414int XrdXrootdProtocol::getDumpCont()
1416 int rlen = 0, rwant;
1423 while(
gdCtl.DumpLen > 0)
1424 {
if (
gdCtl.DumpLen <= (
int)
sizeof(buff)) rwant =
gdCtl.DumpLen;
1425 else rwant =
sizeof(buff);
1427 gdCtl.DumpLen -= rlen;
1432 if (rlen < 0 ||
gdCtl.DumpLen < 0)
1433 {
if (
gdCtl.DumpLen < 0)
Link->setEtext(
"link read overrun error");
1434 else if (rlen != -ENOMSG)
Link->setEtext(
"link read error");
1441 if (
gdCtl.DumpLen == 0)
1448 Resume = &XrdXrootdProtocol::getDumpCont;
1452 <<
" bytes left to discard");
1460void XrdXrootdProtocol::Reset()
1474 memset(&
IO, 0,
sizeof(
IO));
#define kXR_PROTOCOLVSTRING
#define kXR_PROTOCOLVERSION
XrdProtocol * XrdgetProtocol(const char *pname, char *parms, XrdProtocol_Config *pi)
int XrdgetProtocolPort(const char *pname, char *parms, XrdProtocol_Config *pi)
#define NEED2SECURE(protP)
This class implements the XRootD protocol security protection.
static const dev_t XRDSFS_HASBKUP
static const dev_t XRDSFS_RDVMASK
static const dev_t XRDSFS_OFFLINE
class XrdBuffer * XrdSfsXioHandle
XrdSys::RAtomic< int > RAtomic_int
const kXR_char XROOTD_MON_BOUNDP
const kXR_char XROOTD_MON_FORCED
XrdVERSIONINFO(XrdgetProtocol, xrootd)
XrdProtocol * XrdgetProtocol(const char *pname, char *parms, XrdProtocol_Config *pi)
XrdSysTrace XrdXrootdTrace("Xrootd")
int XrdgetProtocolPort(const char *pname, char *parms, XrdProtocol_Config *pi)
static const char * reqName(kXR_unt16 reqCode)
int setEtext(const char *text)
int Peek(char *buff, int blen, int timeout=-1)
int Recv(char *buff, int blen)
XrdNetAddrInfo * AddrInfo()
char * ID
Pointer to the client's link identity.
int Send(const char *buff, int blen)
const char * Host() const
static bool InDomain(XrdNetAddrInfo *epaddr)
void Put(const char *varname, const char *value)
static int GidName(gid_t gID, char *gName, int gNsz, time_t keepT=0)
static int UidName(uid_t uID, char *uName, int uNsz, time_t keepT=0)
XrdProtocol(const char *jname)
XrdNetAddrInfo * addrInfo
Entity's connection details.
char prot[XrdSecPROTOIDSIZE]
Auth protocol used (e.g. krb5).
char * host
Entity's host name dnr dependent.
XrdSfsXio(XrdSfsXioImpl &xioimpl)
void Say(const char *text1, const char *text2=0, const char *txt3=0, const char *text4=0, const char *text5=0, const char *txt6=0)
static char * s2hms(int sec, char *buff, int blen)
static void ClearErrorQueue()
Clear the SSL error queue for the calling thread.
static void Disc(unsigned int usrID)
static XrdXrootdStats * SI
static const char * myInst
XrdXrootdProtocol * VerifyStream(int &rc, int pID, bool lok=true)
static XrdSfsFileSystem * digFS
XrdNetPMark::Handle * pmHandle
static XrdNetPMark * PMark
XrdXrootdProtocol * Stream[maxStreams]
static short as_okstutter
static XrdXrootdXPath RPList
static XrdNetSocket * AdminSock
XrdProtocol * Match(XrdLink *lp) override
struct XrdXrootdProtocol::GetDataCtl gdCtl
static void Reclaim(XrdSfsXioHandle h)
XrdXrootdFileTable * FTab
static XrdXrootdJob * JobCKS
static XrdSysError & eDest
static unsigned int getSID()
XrdSecProtocol * AuthProt
int getData(gdCallBack *gdcbP, const char *dtype, char *buff, int blen)
XrdSfsXioHandle Claim(const char *buff, int datasz, int minasz=0) override
XrdXrootdMonitor::User Monitor
static XrdXrootdRedirPI * RedirPI
static const char * myCName
XrdSfsXioHandle Swap(const char *buff, XrdSfsXioHandle h=0) override
static XrdXrootdFileLock * Locker
int(XrdXrootdProtocol::* Resume)()
static XrdTlsContext * tlsCtx
static XrdXrootdXPath XPList
static XrdScheduler * Sched
static struct XrdXrootdProtocol::RC_Table RouteClient
int Process(XrdLink *lp) override
void Recycle(XrdLink *lp, int consec, const char *reason) override
static char * Buffer(XrdSfsXioHandle h, int *bsz)
static const char * myUName
XrdXrootdResponse Response
int(XrdXrootdProtocol::* ResumePio)()
static const char * TraceID
static int Configure(char *parms, XrdProtocol_Config *pi)
int Stats(char *buff, int blen, int do_sync=0) override
static const int maxStreams
int getDump(const char *dtype, int dlen)
static XrdOucTList * JobCKTLST
static XrdXrootdXPath RQList
static struct XrdXrootdProtocol::RD_Table Route[RD_Num]
static XrdSecProtector * DHS
static XrdBuffManager * BPool
static XrdSecService * CIA
static RAtomic_int srvrAioOps
static const char * myGName
static uint64_t fsFeatures
static XrdOucReqID * PrepID
static XrdSfsFileSystem * osFS
static const uint64_t hasCACH
Feature: Implements a data cache.
static const int Terminate
static const int inCallBk
static const int inDataIov