Commit 1196aa19 authored by claes's avatar claes

Changes in network data conversion to handle attributes of class-type

parent 76a0cab1
......@@ -78,7 +78,8 @@ cmvolc_GetCachedClass (
const gdb_sVolume *vp,
mvol_sAttribute *ap,
pwr_tBoolean *equal, /**< set if classes are equal then NULL is returned */
pwr_tBoolean *fetched /**< true if the class has been fected from the remote node */
pwr_tBoolean *fetched, /**< true if the class has been fected from the remote node */
gdb_sClass *cp
)
{
qcom_sQid tgt;
......@@ -93,6 +94,8 @@ cmvolc_GetCachedClass (
pwr_tUInt32 size;
pwr_tUInt32 nextIdx = 0;
pwr_tUInt32 stopIdx;
gdb_sObject *cop;
pwr_tTime time;
gdb_AssumeLocked;
......@@ -103,16 +106,20 @@ cmvolc_GetCachedClass (
/* Handle nodes that don't support cached classes */
if (!np->cclassSupport) {
*equal = 1;
if (cp == NULL) {
ap->op->u.c.flags.b.classChecked = 1;
ap->op->u.c.flags.b.classEqual = 1;
}
pwr_Return(NULL, sts, GDH__SUCCESS);
}
/** @todo Check vp->u.c.equalClasses first (when implemented) */
ccvKey.nid = np->nid;
if (cp == NULL)
ccvKey.vid = ap->op->g.cid >> 16; /* Class Id to Class Volume Id */
else
ccvKey.vid = cp->cid >> 16;
ccvp = hash_Search(sts, gdbroot->ccvol_ht, &ccvKey);
......@@ -121,28 +128,40 @@ cmvolc_GetCachedClass (
/** @todo Verify that this bugcheck is valid */
}
if (ccvp->equalClasses) {
/* If cp is not NULL then we always fetch the class */
if (ccvp->equalClasses && cp == NULL) {
*equal = 1;
ap->op->u.c.flags.b.classChecked = 1;
ap->op->u.c.flags.b.classEqual = 1;
pwr_Return(NULL, sts, GDH__SUCCESS);
}
if (cp != NULL) {
ccKey.cid = cp->cid;
cop = pool_Address(NULL, gdbroot->pool, cp->cor);
time = cop->u.n.time;
}
else {
ccKey.cid = ap->op->g.cid;
time = ap->cop->u.n.time;
}
ccKey.ccvoltime = ccvp->time;
ccp = hash_Search(sts, gdbroot->cclass_ht, &ccKey);
if (ccp != NULL) {
cmvolc_TouchClass(ccp);
if (time_Acomp(&ap->cop->u.n.time, &ccp->time) == 0) {
if (time_Acomp(&time, &ccp->time) == 0 && cp == NULL) {
*equal = 1;
ap->op->u.c.flags.b.classChecked = 1;
ap->op->u.c.flags.b.classEqual = 1;
pwr_Return(NULL, sts, GDH__SUCCESS);
} else {
if (cp == NULL) {
ap->op->u.c.flags.b.classChecked = 1;
ap->op->u.c.flags.b.classEqual = 0;
}
pwr_Return(ccp, sts, GDH__SUCCESS);
}
}
......@@ -151,6 +170,36 @@ cmvolc_GetCachedClass (
*fetched = 1;
/* If classes equal, create cached class */
if (ccvp->equalClasses && cp != NULL) {
*fetched = 0;
size = sizeof(*ccp) + (cp->acount - 1) * sizeof(ccp->attr[0]);
ccp = pool_Alloc(sts, gdbroot->pool, size);
if (ccp == NULL) {
return NULL;
}
*equal = 1;
ccp->size = cp->size;
ccp->time = time;
for (i = 0; i < cp->acount; i++) {
ccp->attr[i].aix = cp->attr[i].aix;
ccp->attr[i].flags = cp->attr[i].flags;
ccp->attr[i].type = cp->attr[i].type;
ccp->attr[i].offs = cp->attr[i].offs;
ccp->attr[i].size = cp->attr[i].size;
ccp->attr[i].elem = cp->attr[i].elem;
ccp->attr[i].moffset = cp->attr[i].moffset;
}
ccp->acount = cp->acount;
} else {
do {
gdb_Unlock;
......@@ -162,7 +211,7 @@ cmvolc_GetCachedClass (
tgt.qix = net_cProcHandler;
smp->ver = net_cVersion;
smp->cid = ccKey.cid;
smp->time = ap->cop->u.n.time;
smp->time = time;
smp->aidx = nextIdx;
rmp = net_Request(sts, &tgt, &put, NULL, net_eMsg_getCclassR);
......@@ -172,8 +221,10 @@ cmvolc_GetCachedClass (
gdb_Lock;
if (ccp == NULL) {
if (rmp->equal)
if (rmp->equal && cp == NULL)
size = sizeof(*ccp);
else if (rmp->equal && cp != NULL)
size = sizeof(*ccp) + (cp->acount - 1) * sizeof(ccp->attr[0]);
else
size = sizeof(*ccp) + (rmp->cclass.acount - 1) * sizeof(ccp->attr[0]);
......@@ -183,21 +234,37 @@ cmvolc_GetCachedClass (
return NULL;
}
if (rmp->equal) {
if (rmp->equal && cp == NULL) {
ccp->size = 0;
ccp->time = ap->cop->u.n.time;
ccp->time = time;
} else if (rmp->equal && cp != NULL) {
ccp->size = cp->size;
ccp->time = time;
} else {
ccp->size = rmp->cclass.size;
ccp->time = rmp->cclass.time;
}
}
if (rmp->equal) {
if (rmp->equal && cp == NULL) {
*equal = 1;
ccp->acount = 0;
} else if (rmp->equal && cp != NULL) {
*equal = 1;
for (i = 0; i < cp->acount; i++) {
ccp->attr[i].aix = cp->attr[i].aix;
ccp->attr[i].flags = cp->attr[i].flags;
ccp->attr[i].type = cp->attr[i].type;
ccp->attr[i].offs = cp->attr[i].offs;
ccp->attr[i].size = cp->attr[i].size;
ccp->attr[i].elem = cp->attr[i].elem;
ccp->attr[i].moffset = cp->attr[i].moffset;
}
ccp->acount = cp->acount;
} else {
nextIdx = rmp->naidx;
stopIdx = nextIdx != ULONG_MAX ? nextIdx : rmp->cclass.acount;
......@@ -212,16 +279,19 @@ cmvolc_GetCachedClass (
rmp = NULL;
} while(!*equal && nextIdx != ULONG_MAX);
}
ccp->key = ccKey;
ccp->flags.b.equal = *equal;
ccp = linkCclass(sts, ccp);
if (*equal)
if (*equal && cp == NULL)
ccp = NULL;
if (cp != NULL)
*equal = 0;
return ccp;
......
......@@ -31,7 +31,8 @@ cmvolc_GetCachedClass (
const gdb_sVolume *vp,
mvol_sAttribute *ap,
pwr_tBoolean *equal, /**< set if classes are equal then NULL is returned */
pwr_tBoolean *fetched /**< true if the class has been fected from the remote node */
pwr_tBoolean *fetched, /**< true if the class has been fected from the remote node */
gdb_sClass *cp
);
void
......
......@@ -109,6 +109,9 @@ cvolc_GetObjectInfo (
net_sGetObjectInfoR *rmp; /* Receive message. */
pwr_tBoolean equal;
ndc_sRemoteToNative *tbl = NULL;
pwr_tBoolean first = 1;
int rsize;
cdh_uTypeId cid;
gdb_AssumeUnlocked;
pwr_Assert(sts != NULL);
......@@ -137,7 +140,13 @@ cvolc_GetObjectInfo (
if (ODD(rmp->sts)) {
size = MIN(arp->Size, size);
if (ccp == NULL || equal) {
ndc_ConvertData(sts, np, arp, p, rmp->info, size, ndc_eOp_decode);
gdb_sClass *cp;
cid.pwr = arp->Body;
cid.c.bix = 0; /* To get the class id. */
cp = hash_Search(sts, gdbroot->cid_ht, &cid.pwr);
if (cp != NULL) {
ndc_ConvertData(sts, np, cp, arp, p, rmp->info, &size, ndc_eOp_decode, arp->Offset, 0);
}
} else {
if (!ccp->flags.b.rnConv && ap->aop == NULL) { /* whole object */
......@@ -152,7 +161,7 @@ cvolc_GetObjectInfo (
if (tbl == NULL)
break;
ndc_UpdateRemoteToNativeTable(sts, tbl, ap->cp->acount, ap->cp, ccp);
ndc_UpdateRemoteToNativeTable(sts, tbl, ap->cp->acount, ap->cp, ccp, np->nid);
if (ODD(*sts)) {
ccp->rnConv = pool_Reference(NULL, gdbroot->pool, tbl);
ccp->flags.b.rnConv = 1;
......@@ -165,7 +174,8 @@ cvolc_GetObjectInfo (
} gdb_ScopeUnlock;
}
ndc_ConvertRemoteData(sts, np, ccp, rarp, rmp->info, rmp->info, rarp->Size, ndc_eOp_decode);
rsize = rarp->Size;
ndc_ConvertRemoteData(sts, np, ccp, rarp, rmp->info, rmp->info, &rsize, ndc_eOp_decode, rarp->Offset, 0);
if (ODD(*sts)) {
if (ccp->flags.b.rnConv) {
if (tbl == NULL) {
......@@ -173,13 +183,16 @@ cvolc_GetObjectInfo (
if (tbl == NULL)
errh_Bugcheck(*sts, "failed getting address for conversion table");
}
gdb_ScopeLock {
ndc_ConvertRemoteToNativeTable(sts, ccp, tbl, rarp,
arp, p, rmp->info, size);
arp, p, rmp->info, &size,
arp->Offset, 0, 0, &first, np->nid);
} gdb_ScopeUnlock;
} else {
/* The object pointer may be invalid after gdb has been open, reset it */
ap->op = NULL;
ndc_ConvertRemoteToNativeData(sts, ccp, ridx, ap, rarp, arp, p, rmp->info, size);
ndc_ConvertRemoteToNativeData(sts, ccp, ridx, ap, rarp, arp, p, rmp->info, &size, arp->Offset, 0, 0, np->nid);
}
}
}
......@@ -364,6 +377,8 @@ cvolc_SetObjectInfo (
int msize;
net_sSetObjectInfoR *rmp;
pwr_tBoolean equal;
int rsize;
cdh_uTypeId cid;
gdb_AssumeUnlocked;
......@@ -389,14 +404,22 @@ cvolc_SetObjectInfo (
smp->size = rarp->Size;
smp->aref = *rarp;
if (equal)
ndc_ConvertData(sts, np, arp, smp->info, p, size, ndc_eOp_encode);
else {
if (equal) {
gdb_sClass *cp;
cid.pwr = arp->Body;
cid.c.bix = 0; /* To get the class id. */
cp = hash_Search(sts, gdbroot->cid_ht, &cid.pwr);
if (cp != NULL) {
ndc_ConvertData(sts, np, cp, arp, smp->info, p, &size, ndc_eOp_encode, arp->Offset, 0);
}
} else {
/* The object pointer may be invalid after gdb has been open, reset it */
ap->op = NULL;
ndc_ConvertNativeToRemoteData(sts, ccp, ridx, ap, rarp, arp, smp->info, p, rarp->Size);
rsize = rarp->Size;
ndc_ConvertNativeToRemoteData(sts, ccp, ridx, ap, rarp, arp, smp->info, p, &rsize, rarp->Offset, 0, 0, np->nid);
if (ODD(*sts)) {
ndc_ConvertRemoteData(sts, np, ccp, rarp, smp->info, smp->info, rarp->Size, ndc_eOp_encode);
rsize = rarp->Size;
ndc_ConvertRemoteData(sts, np, ccp, rarp, smp->info, smp->info, &rsize, ndc_eOp_encode, rarp->Offset, 0);
}
if (EVEN(*sts)) {
net_Free(NULL, smp);
......
......@@ -227,6 +227,8 @@ cvolsm_GetObjectInfo (
net_sGetObjectInfo *mp = get->data;
qcom_sPut put;
gdb_sNode *np;
cdh_uTypeId cid;
gdb_sClass *cp;
gdb_AssumeUnlocked;
......@@ -247,9 +249,14 @@ cvolsm_GetObjectInfo (
} gdb_ScopeUnlock;
if (p != NULL)
ndc_ConvertData(&sts, np, &mp->aref, rmp->info, p, mp->aref.Size, ndc_eOp_encode);
if (p != NULL) {
size = mp->aref.Size;
cid.pwr = mp->aref.Body;
cid.c.bix = 0; /* To get the class id. */
cp = hash_Search(&sts, gdbroot->cid_ht, &cid.pwr);
if (cp != NULL)
ndc_ConvertData(&sts, np, cp, &mp->aref, rmp->info, p, &size, ndc_eOp_encode, mp->aref.Offset, 0);
}
rmp->aref = mp->aref;
rmp->sts = sts;
rmp->size = mp->aref.Size;
......@@ -349,6 +356,9 @@ cvolsm_SetObjectInfo (
qcom_sPut put;
net_sSetObjectInfo *mp = get->data;
gdb_sNode *np;
int size;
cdh_uTypeId cid;
gdb_sClass *cp;
gdb_AssumeUnlocked;
......@@ -367,8 +377,14 @@ cvolsm_SetObjectInfo (
} gdb_ScopeUnlock;
if (p != NULL)
ndc_ConvertData(&sts, np, &mp->aref, p, mp->info, mp->aref.Size, ndc_eOp_decode);
if (p != NULL) {
size = mp->aref.Size;
cid.pwr = mp->aref.Body;
cid.c.bix = 0; /* To get the class id. */
cp = hash_Search(&sts, gdbroot->cid_ht, &cid.pwr);
if (cp != NULL)
ndc_ConvertData(&sts, np, cp, &mp->aref, p, mp->info, &size, ndc_eOp_decode, mp->aref.Offset, 0);
}
rmp->aref = mp->aref;
rmp->sts = sts;
......
......@@ -605,7 +605,7 @@ gdh_GetObjectInfo (
ccp = NULL;
/* Get cached class if needed */
if (!ap->op->u.c.flags.b.classChecked || !ap->op->u.c.flags.b.classEqual) {
ccp = cmvolc_GetCachedClass(&sts, np, vp, ap, &equal, &fetched);
ccp = cmvolc_GetCachedClass(&sts, np, vp, ap, &equal, &fetched, NULL);
if (EVEN(sts)) {
np = NULL;
break;
......@@ -630,7 +630,7 @@ gdh_GetObjectInfo (
if (equal)
break;
rarp = ndc_NarefToRaref(&sts, ap, arp, ccp, &ridx, &raref, &equal);
rarp = ndc_NarefToRaref(&sts, ap, arp, ccp, &ridx, &raref, &equal, pn, ccpLocked, vp, np);
}
break;
} while (1);
......@@ -717,7 +717,7 @@ gdh_GetObjectInfoAttrref (
ccp = NULL;
/* Get cached class if needed */
if (!ap->op->u.c.flags.b.classChecked || !ap->op->u.c.flags.b.classEqual) {
ccp = cmvolc_GetCachedClass(&sts, np, vp, ap, &equal, &fetched);
ccp = cmvolc_GetCachedClass(&sts, np, vp, ap, &equal, &fetched, NULL);
if (EVEN(sts)) {
np = NULL;
break;
......@@ -740,7 +740,7 @@ gdh_GetObjectInfoAttrref (
if (equal)
break;
rarp = ndc_NarefToRaref(&sts, ap, arp, ccp, &ridx, &raref, &equal);
rarp = ndc_NarefToRaref(&sts, ap, arp, ccp, &ridx, &raref, &equal, NULL, ccpLocked, vp, np);
}
break;
......@@ -2062,7 +2062,7 @@ gdh_SetObjectInfo (
ccp = NULL;
/* Get cached class if needed */
if (!ap->op->u.c.flags.b.classChecked || !ap->op->u.c.flags.b.classEqual) {
ccp = cmvolc_GetCachedClass(&sts, np, vp, ap, &equal, &fetched);
ccp = cmvolc_GetCachedClass(&sts, np, vp, ap, &equal, &fetched, NULL);
if (EVEN(sts)) {
np = NULL;
break;
......@@ -2087,7 +2087,7 @@ gdh_SetObjectInfo (
if (equal)
break;
rarp = ndc_NarefToRaref(&sts, ap, arp, ccp, &ridx, &raref, &equal);
rarp = ndc_NarefToRaref(&sts, ap, arp, ccp, &ridx, &raref, &equal, pn, ccpLocked, vp, np);
}
break;
......@@ -2175,7 +2175,7 @@ gdh_SetObjectInfoAttrref (
ccp = NULL;
/* Get cached class if needed */
if (!ap->op->u.c.flags.b.classChecked || !ap->op->u.c.flags.b.classEqual) {
ccp = cmvolc_GetCachedClass(&sts, np, vp, ap, &equal, &fetched);
ccp = cmvolc_GetCachedClass(&sts, np, vp, ap, &equal, &fetched, NULL);
if (EVEN(sts)) {
np = NULL;
break;
......@@ -2198,7 +2198,7 @@ gdh_SetObjectInfoAttrref (
if (equal)
break;
rarp = ndc_NarefToRaref(&sts, ap, arp, ccp, &ridx, &raref, &equal);
rarp = ndc_NarefToRaref(&sts, ap, arp, ccp, &ridx, &raref, &equal, NULL, ccpLocked, vp, np);
}
break;
......@@ -2510,6 +2510,8 @@ gdh_SubData (
if (cp->cclass != pool_cNRef) {
gdb_sCclass *ccp;
ndc_sRemoteToNative *tbl;
pwr_tUInt32 size;
pwr_tBoolean first = 1;
ccp = pool_Address(NULL, gdbroot->pool, cp->cclass);
if (ccp == NULL) errh_Bugcheck(GDH__WEIRD, "gdh_SubData, get cached class address");
......@@ -2519,7 +2521,8 @@ gdh_SubData (
tbl = pool_Address(NULL, gdbroot->pool, ccp->rnConv);
if (tbl == NULL)errh_Bugcheck(GDH__WEIRD, "gdh_SubData, get cached class address");
ndc_ConvertRemoteToNativeTable(&sts, ccp, tbl, &cp->raref, &cp->aref, bp, p, MIN(bsize, cp->aref.Size));
size = MIN(bsize, cp->aref.Size);
ndc_ConvertRemoteToNativeTable(&sts, ccp, tbl, &cp->raref, &cp->aref, bp, p, &size, cp->aref.Offset, 0, 0, &first, cp->nid);
} else
memcpy(bp, p, MIN(bsize, cp->aref.Size));
......
......@@ -193,6 +193,7 @@ mvol_AnameToAttribute (
ap->adef = NULL;
ap->idx = ULONG_MAX;
} else {
acp = ap->cp;
abop = ap->bop;
acp = ap->cp;
for ( i = 0; i < pn->nAttribute; i++) {
......
......@@ -23,6 +23,7 @@
#include "rt_cvolc.h"
#include "rt_ndc.h"
#include "rt_ndc_msg.h"
#include "rt_cmvolc.h"
/* Vax f-float on a little endian machine.
......@@ -159,6 +160,7 @@ union i3e_s_be {
# define ENDIAN_SWAP_BOOL(t, s) ENDIAN_SWAP_INT(t, s)
#endif
#define touchObject(op) if (op != NULL && op->l.flags.b.isCached) cvolc_TouchObject(op)
#ifndef __powerpc__
......@@ -632,18 +634,21 @@ pwr_tBoolean
ndc_ConvertData (
pwr_tStatus *sts,
const gdb_sNode *np,
gdb_sClass *cp,
const pwr_sAttrRef *arp,
void *tp, /* Address of target. */
const void *sp, /* Address of source. */
pwr_tUInt32 size, /* Size of source. */
ndc_eOp op
pwr_tUInt32 *size, /* Size of source. */
ndc_eOp op,
pwr_tUInt32 offset,
pwr_tUInt32 offs
)
{
gdb_sClass *cp;
cdh_uTypeId cid;
int i;
int base;
gdb_sAttribute *ap;
pwr_tUInt32 aoffs;
pwr_tUInt32 count;
/* The new way, convert if different co_mFormat
* The old way, always convert if different OS
......@@ -653,44 +658,69 @@ ndc_ConvertData (
|| (np->netver < net_cFirstCclassVersion &&
np->os == gdbroot->my_node->os && np->fm.b.bo == gdbroot->my_node->fm.b.bo)) {
if (tp != sp)
memcpy(tp, sp, size);
memcpy(tp, sp, *size);
return TRUE;
}
cid.pwr = arp->Body;
cid.c.bix = 0; /* To get the class id. */
cp = hash_Search(sts, gdbroot->cid_ht, &cid.pwr);
if (cp == NULL) return FALSE;
/* Find attribute. */
for (i = 0, ap = cp->attr; i < cp->acount; i++, ap++)
if (arp->Offset <= ap->moffset)
if (offset <= ap->moffset)
break;
if (i >= cp->acount) pwr_Return(NO, sts, NDC__OFFSET);
if (arp->Offset == 0)
if (offset == 0)
base = 0;
else
base = ap->offs;
switch (op) {
case ndc_eOp_encode:
for (; i < cp->acount && size > 0; i++, ap++) {
if(!encode[pwr_Tix(ap->type)](ap->elem, ap->size, (char *)tp + (ap->offs - base),
(char *)sp + (ap->offs - base), &size))
for (; i < cp->acount && *size > 0; i++, ap++) {
if (ap->flags.b.isclass) {
gdb_sClass *lcp;
lcp = hash_Search(sts, gdbroot->cid_ht, &ap->tid);
/* Single attribute or array */
/* Loop n:o element */
aoffs = 0; /* Attribute offset - source */
for (count = ap->elem; count > 0 && *size > 0; count--) {
ndc_ConvertData(sts, np, lcp, arp, tp, sp, size, op, (offset - ap->offs) % (ap->size / ap->elem), ap->offs - base + aoffs + offs);
aoffs += ap->size / ap->elem;
}
} else {
if(!encode[pwr_Tix(ap->type)](ap->elem, ap->size, (char *)tp + (ap->offs - base + offs),
(char *)sp + (ap->offs - base + offs), size))
pwr_Return(NO, sts, NDC__CONVERT);
}
}
break;
case ndc_eOp_decode:
for (; i < cp->acount && size > 0; i++, ap++) {
if(!decode[pwr_Tix(ap->type)](ap->elem, ap->size, (char *)tp + (ap->offs - base),
(char *)sp + (ap->offs - base), &size))
for (; i < cp->acount && *size > 0; i++, ap++) {
if (ap->flags.b.isclass) {
gdb_sClass *lcp;
lcp = hash_Search(sts, gdbroot->cid_ht, &ap->tid);
/* Single attribute or array */
/* Loop n:o element */
aoffs = 0; /* Attribute offset - source */
for (count = ap->elem; count > 0 && *size > 0; count--) {
ndc_ConvertData(sts, np, lcp, arp, tp, sp, size, op, (offset - ap->offs) % (ap->size / ap->elem), ap->offs - base + aoffs + offs);
aoffs += ap->size / ap->elem;
}
} else {
if(!decode[pwr_Tix(ap->type)](ap->elem, ap->size, (char *)tp + (ap->offs - base + offs),
(char *)sp + (ap->offs - base + offs), size))
pwr_Return(NO, sts, NDC__CONVERT);
}
}
break;
default:
{
......@@ -718,7 +748,11 @@ ndc_ConvertNativeToRemoteData (
const pwr_sAttrRef *narp, /**< Native attribute reference */
void *tp, /**< Address of target. */
const void *sp, /**< Address of source. */
pwr_tUInt32 size /**< Size of target buffer. */
pwr_tUInt32 *size, /**< Size of target buffer. */
pwr_tUInt32 offset,
pwr_tUInt32 toffs,
pwr_tUInt32 soffs,
pwr_tNodeId nid
)
{
const net_sCattribute *cap;
......@@ -726,22 +760,57 @@ ndc_ConvertNativeToRemoteData (
int i,j;
int tasize,sasize;
pwr_mAdef adef;
gdb_AssumeUnlocked;
if (nap->aop == NULL) { /* whole object */
int tcount;
int scount;
int zsize;
pwr_tUInt32 asoffs;
pwr_tUInt32 atoffs;
const gdb_sClass *cp;
const gdb_sAttribute *ap;
gdb_AssumeUnlocked;
if (offset == 0) { /* from start */
cp = hash_Search(sts, gdbroot->cid_ht, &ccp->key.cid);
if (cp == NULL)
errh_Bugcheck(GDH__WEIRD, "can't find native class");
for (i = 0, cap = ccp->attr; i < ccp->acount && size > 0; i++, cap++) {
for (i = 0, cap = ccp->attr; i < ccp->acount && *size > 0; i++, cap++) {
for (j = 0, ap = cp->attr; j < cp->acount; j++, ap++) {
if (ap->aix == cap->aix) {
if (ap->flags.b.isclass) {
gdb_sCcVolKey ccvKey;
gdb_sCclassKey ccKey;
gdb_sCclass *lccp;
gdb_sCclassVolume *ccvp;
ccvKey.nid = nid;
ccvKey.vid = ap->tid >> 16;
ccvp = hash_Search(sts, gdbroot->ccvol_ht, &ccvKey);
ccKey.cid = ap->tid;
ccKey.ccvoltime = ccvp->time;
lccp = hash_Search(sts, gdbroot->cclass_ht, &ccKey);
if (lccp == NULL) errh_Bugcheck(GDH__WEIRD, "can't get class");
atoffs = 0; /* Attribute offset - target */
asoffs = 0; /* Attribute offset - source */
for (scount = ap->elem, tcount = cap->elem; tcount > 0 && *size > 0; tcount--, scount--) {
if (scount > 0) {
ndc_ConvertNativeToRemoteData(sts, lccp, ridx, nap, rarp, narp, tp, sp, size, 0,
toffs + atoffs, soffs + asoffs, nid);
asoffs += ap->size / ap->elem;
} else {
memset(tp + toffs + atoffs, 0, cap->size / cap->elem);
*size -= cap->size / cap->elem;
}
atoffs += cap->size / cap->elem;
}
} else {
tasize = cap->size / cap->elem;
sasize = ap->size / ap->elem;
......@@ -756,32 +825,77 @@ ndc_ConvertNativeToRemoteData (
adef.b.privatepointer = 1;
if (!conv_Fctn[cidx](cap->elem, tasize, (char *)tp + cap->offs, (int *)&size, ap->elem, sasize, (const char *)sp + ap->offs, adef)) {
if (!conv_Fctn[cidx](cap->elem, tasize, (char *)tp + cap->offs + toffs, (int *) size,
ap->elem, sasize, (const char *)sp + ap->offs + soffs, adef)) {
pwr_Return(NO, sts, NDC__CONVERT);
}
}
break;
}
}
if (j >= cp->acount) {/* the remote attribute doesn't exist locally */
if (!conv_Fctn[conv_eIdx_zero](cap->elem, tasize, (char *)tp + cap->offs, (int *)&size, 0, 0, 0, adef)) {
pwr_Return(NO, sts, NDC__CONVERT);
}
zsize = MIN(*size, cap->size);
memset((char *) tp + (cap->offs + toffs), 0, zsize);
*size -= zsize;
}
}
} else { /* single attribute */
/* Find attribute. */
cp = hash_Search(sts, gdbroot->cid_ht, &ccp->key.cid);
if (cp == NULL)
return FALSE;
for (i = 0, cap = ccp->attr; i < ccp->acount; i++, cap++)
if (offset <= cap->moffset)
break;
if (i >= ccp->acount) pwr_Return(NO, sts, NDC__OFFSET);
for (j = 0, ap = cp->attr; j < cp->acount; j++, ap++) {
if (ap->aix == cap->aix) {
if (ap->flags.b.isclass) {
gdb_sCcVolKey ccvKey;
gdb_sCclassKey ccKey;
gdb_sCclass *lccp;
gdb_sCclassVolume *ccvp;
ccvKey.nid = nid;
ccvKey.vid = ap->tid >> 16;
ccvp = hash_Search(sts, gdbroot->ccvol_ht, &ccvKey);
ccKey.cid = ap->tid;
ccKey.ccvoltime = ccvp->time;
lccp = hash_Search(sts, gdbroot->cclass_ht, &ccKey);
if (lccp == NULL) errh_Bugcheck(GDH__WEIRD, "can't get class");
atoffs = 0; /* Attribute offset - target */
asoffs = 0; /* Attribute offset - source */
for (tcount = cap->elem, scount = ap->elem; tcount > 0 && *size > 0; tcount--, scount--) {
if (scount > 0) {
ndc_ConvertRemoteToNativeData(sts, lccp, ridx, nap, rarp, narp, tp, sp, size, (offset - cap->offs) % (cap->size / cap->elem),
toffs + atoffs, soffs + asoffs, nid);
asoffs += ap->size / ap->elem;
} else {
memset(tp + toffs + atoffs, 0, cap->size / cap->elem);
*size -= cap->size / cap->elem;
}
atoffs += cap->size / cap->elem;
}
} else {
pwr_Assert(nap->adef != NULL);
pwr_Assert(ridx < ccp->acount);
cap = &ccp->attr[ridx];
sasize = nap->size / nap->elem;
tasize = cap->size / cap->elem;
cidx = conv_GetIdx(nap->adef->Info.Type, cap->type);
cidx = conv_GetIdx(ap->type, cap->type);
if (cidx == conv_eIdx_invalid) {
pwr_Return(NO, sts, NDC__NOCONV);
}
......@@ -790,16 +904,20 @@ ndc_ConvertNativeToRemoteData (
* If we are unlucky we can get a floating point exception.
*/
adef.m = nap->adef->Info.Flags;
if (adef.b.array && size > cap->size/cap->elem)
if (adef.b.array && *size > cap->size/cap->elem)
adef.b.privatepointer = 1;
if (!conv_Fctn[cidx](cap->elem, tasize, tp, (int *)&size,
nap->elem, sasize, sp, adef)
) {
if (!conv_Fctn[cidx](cap->elem, tasize, tp, (int *) size,
nap->elem, sasize, sp, adef)) {
pwr_Return(NO, sts, NDC__CONVERT);
}
}
}
}
if (j >= cp->acount) {/* the remote attribute doesn't exist locally */
memset((char *) tp, 0, *size);
*size = 0;
}
}
pwr_Return(YES, sts, NDC__SUCCESS);
......@@ -818,49 +936,103 @@ ndc_ConvertRemoteData (
const pwr_sAttrRef *arp,
void *tp, /* Address of target. */
const void *sp, /* Address of source. */
pwr_tUInt32 size, /* Size of source. */
ndc_eOp op
pwr_tUInt32 *size, /* Size of source. */
ndc_eOp op,
pwr_tUInt32 offset,
pwr_tUInt32 offs
)
{
int i;
int base;
const net_sCattribute *cap;
pwr_tUInt32 aoffs;
pwr_tUInt32 count;
if (np->fm.m == gdbroot->my_node->fm.m) {
if (tp != sp)
memcpy(tp, sp, size);
memcpy(tp, sp, *size);
return TRUE;
}
/* Find attribute. */
for (i = 0, cap = ccp->attr; i < ccp->acount; i++, cap++)
if (arp->Offset <= cap->offs + cap->size - 1) /* maximum attribute offset */
if (offset <= cap->offs + cap->size - 1) /* maximum attribute offset */
break;
if (i >= ccp->acount) pwr_Return(NO, sts, NDC__OFFSET);
if (arp->Offset == 0)
if (offset == 0)
base = 0;
else
base = cap->offs;
switch (op) {
case ndc_eOp_encode:
for (; i < ccp->acount && size > 0; i++, cap++) {
if(!encode[pwr_Tix(cap->type)](cap->elem, cap->size, (char *)tp + (cap->offs - base),
(char *)sp + (cap->offs - base), &size))
for (; i < ccp->acount && *size > 0; i++, cap++) {
if (cap->flags.b.isclass) {
gdb_sCcVolKey ccvKey;
gdb_sCclassKey ccKey;
gdb_sCclass *lccp;
gdb_sCclassVolume *ccvp;
ccvKey.nid = np->nid;
ccvKey.vid = cap->type >> 16;
ccvp = hash_Search(sts, gdbroot->ccvol_ht, &ccvKey);
ccKey.cid = cap->type;
ccKey.ccvoltime = ccvp->time;
lccp = hash_Search(sts, gdbroot->cclass_ht, &ccKey);
/* Single attribute or array */
/* Loop n:o element */
aoffs = 0; /* Attribute offset - source */
for (count = cap->elem; count > 0 && *size > 0; count--) {
ndc_ConvertRemoteData(sts, np, lccp, arp, tp, sp, size, op, (offset - cap->offs) % (cap->size / cap->elem), cap->offs - base + aoffs + offs);
aoffs += cap->size / cap->elem;
}
} else {
if(!encode[pwr_Tix(cap->type)](cap->elem, cap->size, (char *)tp + (cap->offs - base + offs),
(char *)sp + (cap->offs - base + offs), size))
pwr_Return(NO, sts, NDC__CONVERT);
}
}
break;
case ndc_eOp_decode:
for (; i < ccp->acount && size > 0; i++, cap++) {
if(!decode[pwr_Tix(cap->type)](cap->elem, cap->size, (char *)tp + (cap->offs - base),
(char *)sp + (cap->offs - base), &size))
for (; i < ccp->acount && *size > 0; i++, cap++) {
if (cap->flags.b.isclass) {
gdb_sCcVolKey ccvKey;
gdb_sCclassKey ccKey;
gdb_sCclass *lccp;
gdb_sCclassVolume *ccvp;
ccvKey.nid = np->nid;
ccvKey.vid = cap->type >> 16;
ccvp = hash_Search(sts, gdbroot->ccvol_ht, &ccvKey);
ccKey.cid = cap->type;
ccKey.ccvoltime = ccvp->time;
lccp = hash_Search(sts, gdbroot->cclass_ht, &ccKey);
/* Single attribute or array */
/* Loop n:o element */
aoffs = 0; /* Attribute offset - source */
for (count = cap->elem; count > 0 && *size > 0; count--) {
ndc_ConvertRemoteData(sts, np, lccp, arp, tp, sp, size, op, (offset - cap->offs) % (cap->size / cap->elem), cap->offs - base + aoffs + offs);
aoffs += cap->size / cap->elem;
}
} else {
if(!decode[pwr_Tix(cap->type)](cap->elem, cap->size, (char *)tp + (cap->offs - base + offs),
(char *)sp + (cap->offs - base + offs), size))
pwr_Return(NO, sts, NDC__CONVERT);
}
}
break;
default:
{
......@@ -888,7 +1060,11 @@ ndc_ConvertRemoteToNativeData (
const pwr_sAttrRef *narp, /**< Native attribute reference */
void *tp, /**< Address of target. */
const void *sp, /**< Address of source. */
pwr_tUInt32 size /**< Size of target buffer. */
pwr_tUInt32 *size, /**< Size of target buffer. */
pwr_tUInt32 offset,
pwr_tUInt32 toffs,
pwr_tUInt32 soffs,
pwr_tNodeId nid
)
{
const net_sCattribute *cap;
......@@ -896,15 +1072,20 @@ ndc_ConvertRemoteToNativeData (
int i,j;
int tasize,sasize;
pwr_mAdef adef;
int tcount;
int scount;
int zsize;
pwr_tUInt32 asoffs;
pwr_tUInt32 atoffs;
const gdb_sAttribute *ap;
const gdb_sClass *cp;
gdb_AssumeUnlocked;
if (nap->aop == NULL) { /* whole object */
const gdb_sClass *cp;
const gdb_sAttribute *ap;
if (offset == 0) { /* from start */
pwr_Assert(narp->Offset == 0);
// pwr_Assert(narp->Offset == 0);
cp = hash_Search(sts, gdbroot->cid_ht, &ccp->key.cid);
if (cp == NULL)
......@@ -912,9 +1093,41 @@ ndc_ConvertRemoteToNativeData (
ap = cp->attr;
for (i = 0; i < cp->acount && size > 0; i++, ap++) {
for (i = 0; i < cp->acount && *size > 0; i++, ap++) {
for (j = 0, cap = ccp->attr; j < ccp->acount; j++, cap++) {
if (ap->aix == cap->aix) {
if (ap->flags.b.isclass) {
gdb_sCcVolKey ccvKey;
gdb_sCclassKey ccKey;
gdb_sCclass *lccp;
gdb_sCclassVolume *ccvp;
ccvKey.nid = nid;
ccvKey.vid = ap->tid >> 16;
ccvp = hash_Search(sts, gdbroot->ccvol_ht, &ccvKey);
ccKey.cid = ap->tid;
ccKey.ccvoltime = ccvp->time;
lccp = hash_Search(sts, gdbroot->cclass_ht, &ccKey);
if (lccp == NULL) errh_Bugcheck(GDH__WEIRD, "can't get class");
atoffs = 0; /* Attribute offset - target */
asoffs = 0; /* Attribute offset - source */
for (tcount = ap->elem, scount = cap->elem; tcount > 0 && *size > 0; tcount--, scount--) {
if (scount > 0) {
ndc_ConvertRemoteToNativeData(sts, lccp, ridx, nap, rarp, narp, tp, sp, size, 0,
toffs + atoffs, soffs + asoffs, nid);
asoffs += cap->size / cap->elem;
} else {
memset(tp + toffs + atoffs, 0, ap->size / ap->elem);
*size -= ap->size / ap->elem;
}
atoffs += ap->size / ap->elem;
}
} else {
tasize = ap->size / ap->elem;
sasize = cap->size / cap->elem;
......@@ -928,31 +1141,75 @@ ndc_ConvertRemoteToNativeData (
adef.m = cap->flags.m;
adef.b.privatepointer = 1;
if (!conv_Fctn[cidx](ap->elem, tasize, tp, (int *)&size, cap->elem, sasize, sp, adef)) {
if (!conv_Fctn[cidx](ap->elem, tasize, tp + ap->offs + toffs, (int *) size, cap->elem, sasize, sp + cap->offs + soffs, adef)) {
pwr_Return(NO, sts, NDC__CONVERT);
}
}
break;
}
}
if (j >= ccp->acount) {/* the native attribute doesn't exist remotely */
if (!conv_Fctn[cidx](ap->elem, tasize, tp, (int *)&size, 0, 0, 0, adef)) {
pwr_Return(NO, sts, NDC__CONVERT);
}
zsize = MIN(*size, ap->size);
memset((char *) tp + (ap->offs + toffs), 0, zsize);
*size -= zsize;
}
}
} else { /* single attribute */
/* Find attribute. */
cp = hash_Search(sts, gdbroot->cid_ht, &ccp->key.cid);
if (cp == NULL)
return FALSE;
for (i = 0, ap = cp->attr; i < cp->acount; i++, ap++)
if (offset <= ap->moffset)
break;
if (i >= cp->acount) pwr_Return(NO, sts, NDC__OFFSET);
for (j = 0, cap = ccp->attr; j < ccp->acount; j++, cap++) {
if (ap->aix == cap->aix) {
if (ap->flags.b.isclass) {
gdb_sCcVolKey ccvKey;
gdb_sCclassKey ccKey;
gdb_sCclass *lccp;
gdb_sCclassVolume *ccvp;
ccvKey.nid = nid;
ccvKey.vid = ap->tid >> 16;
ccvp = hash_Search(sts, gdbroot->ccvol_ht, &ccvKey);
ccKey.cid = ap->tid;
ccKey.ccvoltime = ccvp->time;
lccp = hash_Search(sts, gdbroot->cclass_ht, &ccKey);
if (lccp == NULL) errh_Bugcheck(GDH__WEIRD, "can't get class");
atoffs = 0; /* Attribute offset - target */
asoffs = 0; /* Attribute offset - source */
for (tcount = ap->elem, scount = cap->elem; tcount > 0 && *size > 0; tcount--, scount--) {
if (scount > 0) {
ndc_ConvertRemoteToNativeData(sts, lccp, ridx, nap, rarp, narp, tp, sp, size, (offset - ap->offs) % (ap->size / ap->elem),
toffs + atoffs, soffs + asoffs, nid);
asoffs += cap->size / cap->elem;
} else {
memset(tp + toffs + atoffs, 0, ap->size / ap->elem);
*size -= ap->size / ap->elem;
}
atoffs += ap->size / ap->elem;
}
} else {
pwr_Assert(nap->adef != NULL);
pwr_Assert(ridx < ccp->acount);
cap = &ccp->attr[ridx];
tasize = nap->size / nap->elem;
sasize = cap->size / cap->elem;
cidx = conv_GetIdx(cap->type, nap->adef->Info.Type);
cidx = conv_GetIdx(cap->type, ap->type);
if (cidx == conv_eIdx_invalid) {
pwr_Return(NO, sts, NDC__NOCONV);
}
......@@ -960,18 +1217,249 @@ ndc_ConvertRemoteToNativeData (
/* Prevent conversion of pointers if it's not a single attribute.
* If we are unlucky we can get a floating point exception.
*/
size = MIN(size, nap->size);
*size = MIN(*size, nap->size);
adef = cap->flags;
if (adef.b.array && size > nap->size/nap->elem)
if (adef.b.array && *size > nap->size/nap->elem)
adef.b.privatepointer = 1;
if (!conv_Fctn[cidx](nap->elem, tasize, tp, (int *) size,
cap->elem, sasize, sp, adef)) {
pwr_Return(NO, sts, NDC__CONVERT);
}
}
}
}
if (j >= ccp->acount) {/* the native attribute doesn't exist remotely */
memset((char *) tp, 0, *size);
*size = 0;
}
}
pwr_Return(YES, sts, NDC__SUCCESS);
}
/**
* Converts remote data that has a different class version.
* The data has already been converted to native data format
*/
pwr_tBoolean
ndc_ConvertRemoteToNativeTable (
pwr_tStatus *sts,
const gdb_sCclass *ccp, /**< Cached class */
const ndc_sRemoteToNative *tbl,
const pwr_sAttrRef *rarp, /**< Remote attribute reference */
const pwr_sAttrRef *narp, /**< Native attribute reference */
void *tp, /**< Address of target. */
const void *sp, /**< Address of source. */
pwr_tUInt32 *size, /**< Size of target buffer. */
pwr_tUInt32 offset,
pwr_tUInt32 toffs,
pwr_tUInt32 soffs,
pwr_tBoolean *first,
pwr_tNodeId nid
)
{
const gdb_sClass *cp;
const gdb_sAttribute *ap;
const net_sCattribute *cap;
int i;
int base;
int zsize;
int idx;
int relem;
int tcount;
int scount;
pwr_tUInt32 asoffs;
pwr_tUInt32 atoffs;
pwr_tUInt32 roffs;
pwr_tUInt32 raidx;
conv_eIdx cidx;
pwr_mAdef adef;
gdb_AssumeLocked;
cp = hash_Search(sts, gdbroot->cid_ht, &ccp->key.cid);
if (cp == NULL)
return FALSE;
/* Find attribute. */
for (i = 0, ap = cp->attr; i < cp->acount; i++, ap++)
if (offset <= ap->moffset)
break;
if (i >= cp->acount) pwr_Return(NO, sts, NDC__OFFSET);
if (offset == 0)
base = 0;
else
base = ap->offs;
for (; i < cp->acount && *size > 0; i++, ap++) {
cidx = tbl[i].cidx;
raidx = tbl[i].raidx;
pwr_Assert(raidx < ccp->acount || raidx == ULONG_MAX);
if ( raidx == ULONG_MAX || cidx == conv_eIdx_invalid) {
/* Attribute doesn't exist on remote node or there is no valid conversion
* Zero the local attribute
*/
zsize = MIN(*size, ap->size);
memset((char *)tp + (ap->offs - base + toffs), 0, zsize);
*size -= zsize;
if (*first)
*first = 0;
}
else if (ap->flags.b.isclass) {
gdb_sCcVolKey ccvKey;
gdb_sCclassKey ccKey;
gdb_sCclass *lccp;
gdb_sCclassVolume *ccvp;
ndc_sRemoteToNative *ltbl;
ccvKey.nid = nid;
ccvKey.vid = ap->tid >> 16;
ccvp = hash_Search(sts, gdbroot->ccvol_ht, &ccvKey);
ccKey.cid = ap->tid;
ccKey.ccvoltime = ccvp->time;
if (!conv_Fctn[cidx](nap->elem, tasize, tp, (int *)&size,
cap->elem, sasize, sp, adef)
) {
lccp = hash_Search(sts, gdbroot->cclass_ht, &ccKey);
if (!lccp->flags.b.rnConv) {
const gdb_sClass *lcp = hash_Search(sts, gdbroot->cid_ht, &lccp->key.cid);
ltbl = pool_Alloc(sts, gdbroot->pool, sizeof(*ltbl) * lcp->acount);
ndc_UpdateRemoteToNativeTable(sts, ltbl, lcp->acount, lcp, lccp, nid);
if (ODD(*sts)) {
lccp->rnConv = pool_Reference(NULL, gdbroot->pool, ltbl);
lccp->flags.b.rnConv = 1;
} else {
pool_Free(NULL, gdbroot->pool, ltbl);
pwr_Return(NO, sts, NDC__CONVERT);
}
} else {
ltbl = pool_Address(NULL, gdbroot->pool, lccp->rnConv);
}
cap = &ccp->attr[raidx];
if (lccp == NULL) errh_Bugcheck(GDH__WEIRD, "can't get class");
/* It could either be: */
/* - Whole attribute, or ... */
/* - Single array-element, or ... */
/* - Atrribute in attribute. */
if (*first) {
if (base != 0 && offset > ap->offs) {
if (*size == ap->size / ap->elem) {
/* Single array-element */
/* Check if source element exist */
if ((offset - ap->offs) / (ap->size / ap->elem) < cap->elem) {
ndc_ConvertRemoteToNativeTable(sts, lccp, ltbl, NULL, NULL, tp, sp, size, 0,
toffs + ap->offs - base, soffs, first, nid);
} else {
memset(tp + toffs + ap->offs - base, 0, *size);
*size = 0;
}
} else {
/* Atrribute in attribute */
ndc_ConvertRemoteToNativeTable(sts, lccp, ltbl, NULL, NULL, tp, sp, size, (offset - ap->offs) % (ap->size/ap->elem),
toffs + ap->offs - base, soffs, first, nid);
}
} else {
/* Single attribute or array */
/* Loop n:o element */
/* Check boundaries */
atoffs = 0; /* Attribute offset - target */
asoffs = 0; /* Attribute offset - source */
for (tcount = ap->elem, scount = cap->elem; tcount > 0 && *size > 0; tcount--, scount--) {
if (scount > 0) {
ndc_ConvertRemoteToNativeTable(sts, lccp, ltbl, NULL, NULL, tp, sp, size, (offset - ap->offs) % (ap->size/ap->elem),
toffs + atoffs + ap->offs - base, soffs + asoffs, first, nid);
asoffs += cap->size / cap->elem;
} else {
memset(tp + toffs + atoffs + ap->offs - base, 0, ap->size / ap->elem);
*size -= ap->size / ap->elem;
}
atoffs += ap->size / ap->elem;
}
}
*first = 0;
} else {
/* Single attribute or array */
/* Loop n:o element */
/* Check boundaries */
atoffs = 0; /* Attribute offset - target */
asoffs = 0; /* Attribute offset - source */
for (tcount = ap->elem, scount = cap->elem; tcount > 0 && *size > 0; tcount--, scount--) {
if (scount > 0) {
ndc_ConvertRemoteToNativeTable(sts, lccp, ltbl, NULL, NULL, tp, sp, size, (offset - ap->offs) % (ap->size/ap->elem),
toffs + atoffs + ap->offs - base, cap->offs + soffs + asoffs, first, nid);
asoffs += cap->size / cap->elem;
} else {
memset(tp + toffs + atoffs + ap->offs - base, 0, ap->size / ap->elem);
*size -= ap->size / ap->elem;
}
atoffs += ap->size / ap->elem;
}
}
}
else {
cap = &ccp->attr[raidx];
/** @note Pointers are only handled correctly for a single pointer,
* not arrays. See, vol_AttributeToAddress. Set private for all other
* cases.
* It's quite tricky to find out if it's a single array element. Let's
* hope that the size has the exact size of one element. Maybe we should
* add a flag to the attribute reference that indicates single array element.
*/
adef = cap->flags;
if (!*first || (adef.b.array && *size > ap->size/ap->elem))
adef.b.privatepointer = 1; /* prevent floating point exceptions */
if (*first) {
*first = 0;
roffs = 0;
/* Check if the first attribute is an array element with index > 0
* and that the index exist in the remote attribute.
*/
if (base != 0 && offset > ap->offs) {
pwr_Assert(ap->elem > 1);
idx = (offset - ap->offs) / (ap->size/ap->elem);
/* Calm down, the convert routine will only use the source if relem > 0 */
relem = cap->elem - idx;
} else
relem = cap->elem;
} else {
roffs = cap->offs;
relem = cap->elem;
}
if(!conv_Fctn[cidx](ap->elem, ap->size/ap->elem, (char *)tp + (ap->offs - base) + toffs, (int *) size,
relem, cap->size/cap->elem, (const char *)sp + roffs + soffs, adef))
pwr_Return(NO, sts, NDC__CONVERT);
}
}
pwr_Return(YES, sts, NDC__SUCCESS);
......@@ -982,7 +1470,7 @@ ndc_ConvertRemoteToNativeData (
* The data has already been converted to native data format
*/
pwr_tBoolean
ndc_ConvertRemoteToNativeTable (
ndc_ConvertRemoteToNativeTableOld (
pwr_tStatus *sts,
const gdb_sCclass *ccp, /**< Cached class */
const ndc_sRemoteToNative *tbl,
......@@ -1099,17 +1587,29 @@ ndc_ConvertRemoteToNativeTable (
pwr_sAttrRef *
ndc_NarefToRaref(
pwr_tStatus *sts, /**< Status */
const mvol_sAttribute *ap, /**< Native mvol attribute */
const pwr_sAttrRef *narp, /**< Native attribute reference */
const gdb_sCclass *ccp, /**< Cached class */
mvol_sAttribute *ap, /**< Native mvol attribute */
pwr_sAttrRef *narp, /**< Native attribute reference */
gdb_sCclass *ccp, /**< Cached class */
pwr_tUInt32 *ridx, /**< Attribute index in ccp or UINT_LONG if whole object */
pwr_sAttrRef *rarp, /**< Remote attribute reference */
pwr_tBoolean *equal /**< Set if the attribute references are equal, not checked if whole object */
pwr_tBoolean *equal, /**< Set if the attribute references are equal, not checked if whole object */
cdh_sParseName *pn, /**< Not NULL if called from Get-/SetObjectInfo */
gdb_sCclass *ccpLocked,
gdb_sVolume *vp,
gdb_sNode *np
)
{
pwr_tUInt32 i;
pwr_tUInt32 i, j;
const net_sCattribute *cap;
gdb_sClass *acp;
int offset = 0;
int roffset = 0;
pwr_tBoolean fetched;
mvol_sAttribute attribute;
pwr_sAttrRef aref;
pwr_tClassId cid;
gdb_sCclass *l_ccp; /**< Cached class */
pwr_tBoolean l_equal;
gdb_AssumeLocked;
......@@ -1123,46 +1623,188 @@ ndc_NarefToRaref(
return rarp;
}
/* It's a single attribute */
pwr_Assert(ap->aix != ULONG_MAX);
for (i = 0, cap = ccp->attr; i < ccp->acount; i++, cap++) {
acp = ap->cp;
if (acp == NULL)
pwr_Return(NULL, sts, GDH__NOSUCHCLASS);
if (ap->aix == cap->aix) {
pwr_Assert(ap->adef != NULL);
*ridx = i;
l_ccp = ccp;
/* Loop until we get to the correct offset */
while (1) {
for (i = 0; i < acp->acount; i++) {
if (ap->offs <= (offset + acp->attr[i].moffset))
break;
}
if (i == acp->acount)
pwr_Return(NULL, sts, GDH__ATTRIBUTE);
offset += acp->attr[i].offs;
for (j = 0, cap = l_ccp->attr; j < l_ccp->acount; j++, cap++) {
if (acp->attr[i].aix == cap->aix) {
roffset += cap->offs;
*ridx = j;
break;
}
}
/* Attribute doesn't exist on remote node */
if (j == l_ccp->acount)
pwr_Return(NULL, sts, NDC__NRATTRIBUTE);
if (!acp->attr[i].flags.b.isclass) {
if (acp->attr[i].elem > 1) {
roffset += ((narp->Offset - offset) /
(acp->attr[i].size / acp->attr[i].elem)) *
(cap->size / cap->elem);
offset = narp->Offset;
}
break;
}
if (acp->attr[i].size == narp->Size) {
/* Fetch the class */
if (acp->attr[i].flags.b.isclass) {
cid = acp->attr[i].tid;
acp = hash_Search(sts, gdbroot->cid_ht, &cid);
if (acp == NULL)
pwr_Return(NULL, sts, GDH__NOSUCHCLASS);
l_ccp = cmvolc_GetCachedClass(sts, np, vp, NULL, equal, &fetched, acp);
if (EVEN(*sts)) {
np = NULL;
pwr_Return(NULL, sts, GDH__NOSUCHCLASS);
}
}
break;
}
if (acp->attr[i].flags.b.array) {
for (j = 0; j < acp->attr[i].elem; j++) {
if ( narp->Offset < (offset + acp->attr[i].size / acp->attr[i].elem))
break;
offset += acp->attr[i].size / acp->attr[i].elem;
roffset += cap->size / cap->elem;
}
if (acp->attr[i].size / acp->attr[i].elem == narp->Size) {
/* Fetch the class */
if (acp->attr[i].flags.b.isclass) {
cid = acp->attr[i].tid;
acp = hash_Search(sts, gdbroot->cid_ht, &cid);
if (acp == NULL)
pwr_Return(NULL, sts, GDH__NOSUCHCLASS);
l_ccp = cmvolc_GetCachedClass(sts, np, vp, NULL, equal, &fetched, acp);
if (EVEN(*sts)) {
np = NULL;
pwr_Return(NULL, sts, GDH__NOSUCHCLASS);
}
}
break;
}
}
if (l_ccp != ccp) {
cmvolc_UnlockClass(NULL, l_ccp);
}
// if (ccpLocked) {
// cmvolc_UnlockClass(NULL, ccpLocked);
// ccpLocked = NULL;
// }
/* we're not through yet, get next class */
cid = acp->attr[i].tid;
acp = hash_Search(sts, gdbroot->cid_ht, &cid);
if (acp == NULL)
pwr_Return(NULL, sts, GDH__NOSUCHCLASS);
l_ccp = cmvolc_GetCachedClass(sts, np, vp, NULL, equal, &fetched, acp);
if (EVEN(*sts)) {
np = NULL;
pwr_Return(NULL, sts, GDH__NOSUCHCLASS);
}
if (l_ccp != NULL) {
cmvolc_LockClass(NULL, l_ccp);
}
/* If gdb has been unlocked, refresh pointers (cmvolc_getCached.. unlocks) */
/** @todo Check if we can do it more efficient, eg. vol_ArefToAttribute */
/* I cannot explain why this must be done, maybe LW has the answer ?? */
if (fetched) {
memset(&attribute, 0, sizeof(attribute));
np = NULL;
if (pn) {
ap = vol_NameToAttribute(sts, &attribute, pn, gdb_mLo_global, vol_mTrans_all);
if ((ap == NULL) || (ap->op == NULL))
pwr_Return(NULL, sts, GDH__NOSUCHCLASS);
touchObject(ap->op);
narp = mvol_AttributeToAref(sts, ap, &aref);
if (narp == NULL)
pwr_Return(NULL, sts, GDH__NOSUCHCLASS);
}
else {
ap = vol_ArefToAttribute(sts, &attribute, narp, gdb_mLo_global, vol_mTrans_all);
if ((ap == NULL) || (ap->op == NULL))
pwr_Return(NULL, sts, GDH__NOSUCHCLASS);
touchObject(ap->op);
}
acp = hash_Search(sts, gdbroot->cid_ht, &cid);
if (acp == NULL)
pwr_Return(NULL, sts, GDH__NOSUCHCLASS);
vp = pool_Address(NULL, gdbroot->pool, ap->op->l.vr);
np = hash_Search(sts, gdbroot->nid_ht, &vp->g.nid);
if (np == NULL)
pwr_Return(NULL, sts, GDH__NOSUCHNODE);
l_ccp = cmvolc_GetCachedClass(sts, np, vp, NULL, equal, &fetched, acp);
if (EVEN(*sts)) {
np = NULL;
pwr_Return(NULL, sts, GDH__NOSUCHCLASS);
}
/* Refresh original cached class */
ccp = cmvolc_GetCachedClass(sts, np, vp, ap, &l_equal, &fetched, NULL);
if (EVEN(*sts)) {
np = NULL;
pwr_Return(NULL, sts, GDH__NOSUCHCLASS);
}
ccpLocked = ccp;
}
}
if (l_ccp != ccp) {
cmvolc_UnlockClass(NULL, l_ccp);
}
if (ap->idx == ULONG_MAX) {
if (ap->adef->Info.Type == cap->type &&
ap->offs == cap->offs &&
ap->size == cap->size &&
ap->elem == cap->elem &&
ap->flags.b.Indirect == (cap->flags.b.pointer && !cap->flags.b.privatepointer)
) {
*rarp = *narp;
*equal = 1;
} else {
rarp->Objid = narp->Objid;
rarp->Body = narp->Body;
rarp->Offset = cap->offs;
rarp->Offset = roffset;
rarp->Size = cap->size;
rarp->Flags.m = narp->Flags.m;
rarp->Flags.b.Indirect = (cap->flags.b.pointer && !cap->flags.b.privatepointer);
rarp->Flags.b.Array = cap->flags.b.array;
}
} else { /* It's an array element */
if (ap->adef->Info.Type == cap->type &&
ap->adef->Info.Offset == cap->offs &&
ap->adef->Info.Size == cap->size &&
ap->adef->Info.Elements == cap->elem &&
ap->flags.b.Indirect == (cap->flags.b.pointer && !cap->flags.b.privatepointer)
) {
*rarp = *narp;
*equal = 1;
} else {
else { /* It's an array element */
if (ap->idx >= cap->elem) {
*sts = NDC__NRELEM_IDX;
return NULL;
......@@ -1171,17 +1813,14 @@ ndc_NarefToRaref(
rarp->Objid = narp->Objid;
rarp->Body = narp->Body;
rarp->Size = cap->size / cap->elem;
rarp->Offset = cap->offs + rarp->Size * ap->idx;
rarp->Offset = roffset;
rarp->Flags.m = narp->Flags.m;
rarp->Flags.b.Indirect = (cap->flags.b.pointer && !cap->flags.b.privatepointer);
rarp->Flags.b.Array = cap->flags.b.array;
}
}
return rarp;
}
} /* for acount */
/* Attribute doesn't exist on remote node */
*sts = NDC__NRATTRIBUTE;
return NULL;
}
ndc_sRemoteToNative *
......@@ -1190,11 +1829,13 @@ ndc_UpdateRemoteToNativeTable(
ndc_sRemoteToNative *tbl,
pwr_tUInt32 tcnt, /**< # table entries */
const gdb_sClass *cp,
const gdb_sCclass *ccp
const gdb_sCclass *ccp,
pwr_tNodeId nid
)
{
const gdb_sAttribute *ap;
const net_sCattribute *cap;
int i,j;
......@@ -1203,13 +1844,51 @@ ndc_UpdateRemoteToNativeTable(
return NULL;
}
for (i = 0, ap = cp->attr; i < cp->acount; i++, ap++) {
for (j = 0, cap = ccp->attr; j < ccp->acount; j++, cap++) {
if (ap->aix == cap->aix) {
/* If attribute is class then continue with this one */
if (ap->flags.b.isclass) {
/* const gdb_sClass *lcp = hash_Search(sts, gdbroot->cid_ht, &ap->tid);
gdb_sCcVolKey ccvKey;
gdb_sCclassKey ccKey;
gdb_sCclass *lccp;
gdb_sCclassVolume *ccvp;
ccvKey.nid = nid;
ccvKey.vid = ap->tid >> 16;
ccvp = hash_Search(sts, gdbroot->ccvol_ht, &ccvKey);
ccKey.cid = ap->tid;
ccKey.ccvoltime = ccvp->time;
lccp = hash_Search(sts, gdbroot->cclass_ht, &ccKey);
if ((lcp == NULL) || (lccp == NULL)) errh_Bugcheck(GDH__WEIRD, "can't get class");
if (!lccp->flags.b.rnConv) {
ndc_sRemoteToNative *ltbl;
ltbl = pool_Alloc(sts, gdbroot->pool, sizeof(*tbl) * lcp->acount);
ndc_UpdateRemoteToNativeTable(sts, ltbl, lcp->acount, lcp, lccp, nid);
if (ODD(*sts)) {
lccp->rnConv = pool_Reference(NULL, gdbroot->pool, ltbl);
lccp->flags.b.rnConv = 1;
} else {
pool_Free(NULL, gdbroot->pool, ltbl);
return NULL;
}
} */
tbl[i].cidx = conv_eIdx_;
tbl[i].raidx = j;
}
else {
tbl[i].cidx = conv_GetIdx(cap->type, ap->type);
tbl[i].raidx = j;
}
break;
}
}
......
......@@ -50,11 +50,14 @@ pwr_tBoolean
ndc_ConvertData (
pwr_tStatus *sts,
const gdb_sNode *np,
gdb_sClass *cp,
const pwr_sAttrRef *arp,
void *tp, /* Address of target. */
const void *sp, /* Address of source. */
pwr_tUInt32 size, /* Size of source. */
ndc_eOp op
pwr_tUInt32 *size, /* Size of source. */
ndc_eOp op,
pwr_tUInt32 offset,
pwr_tUInt32 offs
);
/**
......@@ -71,7 +74,11 @@ ndc_ConvertNativeToRemoteData (
const pwr_sAttrRef *narp, /**< Native attribute reference */
void *tp, /**< Address of target. */
const void *sp, /**< Address of source. */
pwr_tUInt32 size /**< Size of target buffer. */
pwr_tUInt32 *size, /**< Size of target buffer. */
pwr_tUInt32 offset, /**< Offset in class */
pwr_tUInt32 toffs, /**< Offset i ntarget buffer */
pwr_tUInt32 soffs, /**< Offset i source buffer */
pwr_tNodeId nid /**< Node id */
);
/**
......@@ -85,8 +92,10 @@ ndc_ConvertRemoteData (
const pwr_sAttrRef *arp,
void *tp, /* Address of target. */
const void *sp, /* Address of source. */
pwr_tUInt32 size, /* Size of source. */
ndc_eOp op
pwr_tUInt32 *size, /* Size of source. */
ndc_eOp op,
pwr_tUInt32 offset, /**< Offset to attribute in class. */
pwr_tUInt32 offs /**< Offset in buffer. */
);
......@@ -105,7 +114,11 @@ ndc_ConvertRemoteToNativeData (
const pwr_sAttrRef *narp, /**< Native attribute reference */
void *tp, /**< Address of target. */
const void *sp, /**< Address of source. */
pwr_tUInt32 size /**< Size of target buffer. */
pwr_tUInt32 *size, /**< Size of target buffer. */
pwr_tUInt32 offset, /**< Offset in class */
pwr_tUInt32 toffs, /**< Offset i ntarget buffer */
pwr_tUInt32 soffs, /**< Offset i source buffer */
pwr_tNodeId nid /**< Node id */
);
/**
......@@ -121,7 +134,12 @@ ndc_ConvertRemoteToNativeTable (
const pwr_sAttrRef *narp, /**< Native attribute reference */
void *tp, /**< Address of target. */
const void *sp, /**< Address of source. */
pwr_tUInt32 size /**< Size of target buffer. */
pwr_tUInt32 *size, /**< Size of target buffer. */
pwr_tUInt32 offset, /**< Offset to attribute in class. */
pwr_tUInt32 toffs, /**< Offset in target buffer. */
pwr_tUInt32 soffs, /**< Offset in source buffer. */
pwr_tBoolean *first, /**< First scan. */
pwr_tNodeId nid /**< Node id */
);
......@@ -133,12 +151,16 @@ ndc_ConvertRemoteToNativeTable (
pwr_sAttrRef *
ndc_NarefToRaref(
pwr_tStatus *sts, /**< Status */
const mvol_sAttribute *ap, /**< Native mvol attribute */
const pwr_sAttrRef *narp, /**< Native attribute reference */
const gdb_sCclass *ccp, /**< Cached class */
pwr_tUInt32 *ridx, /**< Attribute index in ccp or UINT_LONG if whole object*/
mvol_sAttribute *ap, /**< Native mvol attribute */
pwr_sAttrRef *narp, /**< Native attribute reference */
gdb_sCclass *ccp, /**< Cached class */
pwr_tUInt32 *ridx, /**< Attribute index in ccp or UINT_LONG if whole object */
pwr_sAttrRef *rarp, /**< Remote attribute reference */
pwr_tBoolean *equal /**< Set if the attribute references are equal */
pwr_tBoolean *equal, /**< Set if the attribute references are equal, not checked if whole object */
cdh_sParseName *pn, /**< Not NULL if called from Get-/SetObjectInfo */
gdb_sCclass *ccpLocked,
gdb_sVolume *vp,
gdb_sNode *np
);
ndc_sRemoteToNative *
......@@ -147,7 +169,8 @@ ndc_UpdateRemoteToNativeTable(
ndc_sRemoteToNative *tbl,
pwr_tUInt32 tcnt, /**< # table entries */
const gdb_sClass *cp,
const gdb_sCclass *ccp
const gdb_sCclass *ccp,
pwr_tNodeId nid
);
......
......@@ -180,6 +180,7 @@ testClient (
pwr_sAttrRef *arp;
pwr_sAttrRef *rarp;
gdb_sCclass *ccp;
gdb_sCclass *ccpLocked;
pool_tRef ccr;
pwr_tUInt32 ridx;
pwr_tBoolean equal;
......@@ -217,7 +218,7 @@ testClient (
/* Get cached class if needed */
if (!op->u.c.flags.b.classChecked || !op->u.c.flags.b.classEqual) {
ccp = cmvolc_GetCachedClass(&lsts, np, vp, ap, &equal, &fetched);
ccp = cmvolc_GetCachedClass(&lsts, np, vp, ap, &equal, &fetched, NULL);
if (EVEN(lsts)) {
np = NULL;
op = NULL;
......@@ -298,7 +299,7 @@ testClient (
if (!op->u.c.flags.b.classChecked || !op->u.c.flags.b.classEqual) {
ap = vol_ArefToAttribute(&lsts, &attribute, &cp->aref, gdb_mLo_global, vol_mTrans_all);
if (ap == NULL) break;
ccp = cmvolc_GetCachedClass(&lsts, np, vp, ap, &equal, &fetched);
ccp = cmvolc_GetCachedClass(&lsts, np, vp, ap, &equal, &fetched, NULL);
if (EVEN(lsts)) {
np = NULL;
op = NULL;
......@@ -363,9 +364,10 @@ testClient (
np = pool_Address(NULL, gdbroot->pool, vp->l.nr);
if (!equal) {
rarp = ndc_NarefToRaref(sts, ap, arp, ccp, &ridx, &cp->raref, &equal);
ccpLocked = ccp;
rarp = ndc_NarefToRaref(sts, ap, arp, ccp, &ridx, &cp->raref, &equal, NULL, ccpLocked, vp, np );
if (rarp == NULL || equal) {
if (ccp->flags.b.cacheLock)
cmvolc_UnlockClass(NULL, ccp);
cp->cclass = pool_cNRef;
if (rarp == NULL)
......@@ -378,7 +380,7 @@ testClient (
tbl = pool_Alloc(sts, gdbroot->pool, sizeof(*tbl) * c->acount);
ndc_UpdateRemoteToNativeTable(sts, tbl, c->acount, c, ccp);
ndc_UpdateRemoteToNativeTable(sts, tbl, c->acount, c, ccp, np->nid);
if (ODD(*sts)) {
ccp->rnConv = pool_Reference(NULL, gdbroot->pool, tbl);
ccp->flags.b.rnConv = 1;
......
......@@ -111,6 +111,7 @@ subcm_Data (
qcom_sQid tgt;
gdb_sCclass *ccp;
ndc_sRemoteToNative *tbl;
int rsize;
clock_gettime(CLOCK_REALTIME, &curtim);
......@@ -180,15 +181,26 @@ subcm_Data (
refcount++;
if (1 || mp->msg.hdr.xdr) {
if (cp->cclass == pool_cNRef)
ndc_ConvertData(&sts, np, &cp->aref, dp->data, dp->data, dp->size, ndc_eOp_decode);
if (cp->cclass == pool_cNRef) {
gdb_sClass *classp;
cdh_uTypeId cid;
cid.pwr = cp->aref.Body;
cid.c.bix = 0; /* To get the class id. */
classp = hash_Search(&sts, gdbroot->cid_ht, &cid.pwr);
rsize = dp->size;
if (classp != NULL)
ndc_ConvertData(&sts, np, classp, &cp->aref, dp->data, dp->data, &rsize, ndc_eOp_decode, cp->aref.Offset, 0);
}
else {
cp->old = TRUE;
ccp = pool_Address(&cp->sts, gdbroot->pool, cp->cclass);
if (ccp != NULL) {
tbl = pool_Address(&cp->sts, gdbroot->pool, ccp->rnConv);
if (tbl != NULL) {
ndc_ConvertRemoteData(&cp->sts, np, ccp, &cp->raref, dp->data, dp->data, dp->size, ndc_eOp_decode);
rsize = dp->size;
ndc_ConvertRemoteData(&cp->sts, np, ccp, &cp->raref, dp->data, dp->data, &rsize, ndc_eOp_decode, cp->raref.Offset, 0);
if (ODD(cp->sts))
cp->old = FALSE;
}
......@@ -207,7 +219,10 @@ subcm_Data (
if (cp->cclass == pool_cNRef)
memcpy(adrs, dp->data, MIN(dp->size, cp->usersize));
else if (!cp->old) {
ndc_ConvertRemoteToNativeTable(&cp->sts, ccp, tbl, &cp->raref, &cp->aref, adrs, dp->data, cp->usersize);
pwr_tUInt32 size = cp->usersize;
pwr_tBoolean first = 1;
ndc_ConvertRemoteToNativeTable(&cp->sts, ccp, tbl, &cp->raref, &cp->aref, adrs, dp->data, &size,
cp->aref.Offset, 0, 0, &first, np->nid);
if (EVEN(cp->sts))
cp->old = TRUE;
}
......
......@@ -284,6 +284,8 @@ subsm_SendBuffer (
pwr_tBoolean remote = (bp->nid != gdbroot->db->nid);
pool_sQlink *sl;
void *data;
gdb_sClass *classp;
cdh_uTypeId cid;
gdb_AssumeLocked;
......@@ -363,7 +365,12 @@ subsm_SendBuffer (
dp->sts = sp->sts;
if (ODD(sp->sts)) {
ndc_ConvertData(&dp->sts, np, &sp->aref, dp->data, data, sp->aref.Size, ndc_eOp_encode);
size = sp->aref.Size;
cid.pwr = sp->aref.Body;
cid.c.bix = 0; /* To get the class id. */
classp = hash_Search(&sts, gdbroot->cid_ht, &cid.pwr);
if (classp == NULL)
ndc_ConvertData(&dp->sts, np, classp, &sp->aref, dp->data, data, &size, ndc_eOp_encode, sp->aref.Offset, 0);
sp->count++;
mp->count++;
}
......
Markdown is supported
0%
or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment