Source code of Windows XP (NT5)
You can not select more than 25 topics Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.

1684 lines
46 KiB

  1. // Copyright (c) 1996-1999 Microsoft Corporation
  2. //+-------------------------------------------------------------------------
  3. //
  4. // Microsoft Windows
  5. //
  6. // File: trksvr.hxx
  7. //
  8. // Contents: Definitions local to this directory.
  9. // Put includes of files liable to change often in this file.
  10. //
  11. // Classes: CTrkSvrSvc
  12. // CTrkSvrConfiguration
  13. // CDbConnection
  14. // CIntraDomainTable
  15. // CCrossDomainTable
  16. // CVolumeTable
  17. // CDenialChecker
  18. // CGhostTable - DBG only
  19. // CQuotaTable
  20. //
  21. // Functions:
  22. //
  23. //
  24. //
  25. // History: 18-Nov-96 BillMo Created.
  26. // 18-Nov-97 WeiruC Added CQuotaTable class.
  27. //
  28. // Notes: Put includes of files that don't change often into pch.cxx.
  29. //
  30. // Codework: Put in services.exe : _hDllReference
  31. //
  32. //--------------------------------------------------------------------------
  33. #include "trklib.hxx"
  34. #include <trkwks.h>
  35. #include <trksvr.h>
  36. #include "resource.h"
  37. //+-------------------------------------------------------------------------
  38. //
  39. // Strings
  40. //
  41. //--------------------------------------------------------------------------
  42. const extern TCHAR tszValueNameSeq[] INIT( TEXT("seqRefresh") );
  43. #ifndef EVENT_SOURCE_DEFINED
  44. #define EVENT_SOURCE_DEFINED
  45. const extern TCHAR* g_ptszEventSource INIT( TEXT("Distributed Link Tracking Server") );
  46. #endif
  47. //
  48. // LDAP classes and container names
  49. //
  50. const extern TCHAR s_VolumeTableRDN[] INIT( TEXT("CN=VolumeTable,CN=FileLinks,CN=System,") );
  51. const extern TCHAR s_ObjectMoveTableRDN[] INIT( TEXT("CN=ObjectMoveTable,CN=FileLinks,CN=System,") );
  52. const extern TCHAR s_objectClass[] INIT( TEXT("objectClass") );
  53. const extern TCHAR s_linkTrackVolumeTable[] INIT( TEXT("linkTrackVolumeTable") );
  54. const extern TCHAR s_linkTrackObjectMoveTable[] INIT( TEXT("linkTrackObjectMoveTable") );
  55. const extern TCHAR s_linkTrackVolEntry[] INIT( TEXT("linkTrackVolEntry") );
  56. const extern TCHAR s_linkTrackOMTEntry[] INIT( TEXT("linkTrackOMTEntry") );
  57. const extern TCHAR s_Cn[] INIT( TEXT("CN") );
  58. //
  59. // Tables
  60. //
  61. // Volume Table (voltab.cxx)
  62. //
  63. // Design VOLUMEID -> MACHINEID SECRET TIME NotificationSequence
  64. // Logical DN currMachineId volumeSecret timeVolChange seqNotification
  65. //
  66. // DN is stringized volume id
  67. const extern TCHAR s_currMachineId[] INIT( TEXT("volTableIdxGUID") ); // was "currMachineId"
  68. const extern TCHAR s_volumeSecret[] INIT( TEXT("linkTrackSecret") );
  69. const extern TCHAR s_objectCount[] INIT( TEXT("objectCount") );
  70. const extern TCHAR s_seqNotification[] INIT( TEXT("seqNotification") );
  71. const extern TCHAR s_timeRefresh[] INIT( TEXT("timeRefresh") );
  72. const extern TCHAR s_timeVolChange[] INIT( TEXT("timeVolChange") );
  73. const extern TCHAR s_volTableGUID[] INIT( TEXT("volTableGUID") );
  74. //const extern TCHAR s_volTableIdxGUID[] INIT( TEXT("volTableIdxGUID") ); // used for currMachineId
  75. #ifdef VOL_REPL
  76. const extern TCHAR s_timeVolChangeSearch[] INIT( TEXT("(timeVolChange;binary>=") );
  77. #endif
  78. const extern TCHAR s_currMachineIdSearch[] INIT( TEXT("(volTableIdxGUID;binary=") );
  79. // Object Move Table (idt_ldap.cxx)
  80. //
  81. // Design Birth/Current -> New Birth RefreshSequence
  82. // Design VOLUMEID:OBJID -> VOLUMEID:OBJID VOLUMEID:BIRTHID ULONG
  83. // Logical DN currVolumeId:currObjectId birthVolumeId:birthObjectId seqRefresh
  84. // Actual str(VOLUMEID:OBJID) currMachineId:objId birthMachineId:currReplsetid linkTrackSecret
  85. //
  86. // DN is stringized Birth/Current
  87. const extern TCHAR s_birthLocation[] INIT( TEXT("birthLocation") );
  88. const extern TCHAR s_currentLocation[] INIT( TEXT("currentLocation") );
  89. const extern TCHAR s_oMTGuid[] INIT( TEXT("oMTGuid") );
  90. const extern TCHAR s_oMTIndxGuid[] INIT( TEXT("oMTIndxGuid") );
  91. const extern TCHAR s_RestoreTime[] INIT( TEXT("System\\CurrentControlSet\\Services\\NtDs\\Parameters\\RestoreTime") );
  92. const extern TCHAR s_RestoreBegin[] INIT( TEXT("RestoreBegin") );
  93. const extern TCHAR s_rIDManagerReference[] INIT( TEXT("rIDManagerReference") );
  94. const extern TCHAR s_fSMORoleOwner[] INIT( TEXT("fSMORoleOwner") );
  95. //--------------------------------------------------------------------------
  96. //
  97. // General defines
  98. //
  99. //--------------------------------------------------------------------------
  100. #define MAX_SHORTENABLE_SEGMENTS (NUM_VOLUMES)
  101. #define CCH_UINT32 10 // Max chars for a DWORD value
  102. #define CCH_UINT64 20 // Max chars for a QUADWORD value
  103. #define MAX_WAIT_FOR_DS_STARTUP_SECONDS 360
  104. //--------------------------------------------------------------------------
  105. //
  106. // Forward declarations
  107. //
  108. //--------------------------------------------------------------------------
  109. class CTrkSvrConfiguration;
  110. class CTrkSvrSvc;
  111. //-------------------------------------------------------------------//
  112. // //
  113. // Global variables //
  114. // //
  115. //-------------------------------------------------------------------//
  116. EXTERN CTrkSvrSvc * g_ptrksvr; // For RPC stubs to call service
  117. EXTERN LONG g_ctrksvr INIT(0); // Used to protect against multiple service instances
  118. //--------------------------------------------------------------------------
  119. //
  120. // Ldap enumeration
  121. //
  122. //--------------------------------------------------------------------------
  123. typedef enum
  124. {
  125. ENUM_DELETE_ENTRY,
  126. ENUM_KEEP_ENTRY,
  127. ENUM_DELETE_QUOTAFLAGS,
  128. ENUM_ABORT
  129. } ENUM_ACTION;
  130. typedef ENUM_ACTION (*PFN_LDAP_ENUMERATE_CALLBACK)(
  131. LDAP * pLdap,
  132. LDAPMessage * pResult,
  133. void* UserParam1,
  134. void* UserParam2
  135. );
  136. ENUM_ACTION
  137. GcEnumerateCallback(
  138. LDAP * pLdap,
  139. LDAPMessage * pResult,
  140. PVOID pvContext,
  141. void*
  142. );
  143. BOOL // FALSE if aborted (ENUM_ABORT returned by callback)
  144. LdapEnumerate(
  145. LDAP * pLdap,
  146. TCHAR * ptszBaseDn,
  147. ULONG Scope,
  148. TCHAR * Filter,
  149. TCHAR * Attributes[],
  150. PFN_LDAP_ENUMERATE_CALLBACK pCallback,
  151. PVOID pvContext,
  152. void* UserParam2 = NULL
  153. );
  154. //--------------------------------------------------------------------------
  155. //
  156. // Garbage collection
  157. //
  158. //--------------------------------------------------------------------------
  159. class CQuotaTable;
  160. // This structure holds context during enumeration
  161. // in a GC
  162. typedef struct
  163. {
  164. // All entries with lower sequence numbers
  165. // should be deleted.
  166. SequenceNumber seqOldestToKeep;
  167. // The highest sequence number we should see.
  168. SequenceNumber seqCurrent;
  169. // If true, abort (happens on service stop).
  170. const BOOL *pfAbort;
  171. // How long to sleep between non-noop iterations
  172. DWORD dwRepetitiveTaskDelay;
  173. // The quota table
  174. CQuotaTable *pqtable;
  175. // The number of entries that have been deleted (otherwise).
  176. ULONG cEntries;
  177. } GC_ENUM_CONTEXT;
  178. typedef struct
  179. {
  180. LONG cDelta;
  181. DWORD dwRepetitiveTaskDelay;
  182. DWORD dwPass;
  183. BOOL fCountAll;
  184. enum EPass
  185. {
  186. FIRST_PASS, SECOND_PASS
  187. };
  188. } TRUE_COUNT_ENUM_CONTEXT;
  189. //--------------------------------------------------------------------------
  190. //
  191. // Quota table
  192. //
  193. //--------------------------------------------------------------------------
  194. #define QFLAG_UNCOUNTED 0x8
  195. #define QFLAG_DELETED 0x4
  196. //+-------------------------------------------------------------------------
  197. //
  198. // Class: CDbConnection
  199. //
  200. // Purpose: Handles shared connection to database
  201. //
  202. // Notes:
  203. //
  204. //--------------------------------------------------------------------------
  205. class CDbConnection
  206. {
  207. public:
  208. // --------------------> indent 24 characters
  209. inline CDbConnection();
  210. inline ~CDbConnection();
  211. void Initialize(CSvcCtrlInterface * psvc, OPTIONAL const TCHAR *ptszHostName = NULL );
  212. void UnInitialize();
  213. inline const TCHAR *GetBaseDn() { Ldap(); return _pszBaseDn; }
  214. LDAP * Ldap();
  215. private:
  216. BOOL _fInitializeCalled:1;
  217. CCriticalSection _cs;
  218. LDAP * _pldap;
  219. TCHAR * _pszBaseDn;
  220. };
  221. inline // put all types/attributes on same line as inline
  222. CDbConnection::CDbConnection() :
  223. _fInitializeCalled(FALSE)
  224. {
  225. // Normally critsecs are initialized in the Initialize method.
  226. // But for this class the Initialize method doesn't get called right
  227. // away, so we'll try to initialize here and deal with it in Ldap()
  228. // if this fails.
  229. __try
  230. {
  231. _cs.Initialize();
  232. }
  233. __except( EXCEPTION_EXECUTE_HANDLER )
  234. {
  235. }
  236. }
  237. inline
  238. CDbConnection::~CDbConnection()
  239. {
  240. UnInitialize();
  241. _cs.UnInitialize();
  242. }
  243. //+-------------------------------------------------------------------------
  244. //
  245. // Class: CCrossDomainTable
  246. //
  247. // Purpose: Contains cross domain links
  248. //
  249. // Notes: Columns
  250. // -------
  251. //
  252. // prev_location new_location birth_id
  253. //
  254. //--------------------------------------------------------------------------
  255. class CCrossDomainTable
  256. {
  257. public:
  258. inline CCrossDomainTable(CDbConnection & dbc);
  259. inline ~CCrossDomainTable();
  260. void Initialize();
  261. void UnInitialize();
  262. private:
  263. BOOL _fInitializeCalled:1;
  264. const CDbConnection &
  265. _dbc;
  266. };
  267. inline
  268. CCrossDomainTable::CCrossDomainTable(CDbConnection & dbc) :
  269. _fInitializeCalled(FALSE),
  270. _dbc(dbc)
  271. {
  272. }
  273. inline
  274. CCrossDomainTable::~CCrossDomainTable()
  275. {
  276. UnInitialize();
  277. }
  278. //+-------------------------------------------------------------------------
  279. //
  280. // Class: CRefreshSequenceStorage
  281. //
  282. // Purpose: Stores the refresh sequence number in the DS's restorable
  283. // table data.
  284. //
  285. //--------------------------------------------------------------------------
  286. class CVolumeTable;
  287. class CRefreshSequenceStorage
  288. {
  289. public:
  290. inline CRefreshSequenceStorage( CVolumeTable * pVolTab, CQuotaTable *pQuotaTab, CTrkSvrConfiguration *psvrconfig );
  291. inline ~CRefreshSequenceStorage();
  292. inline void Initialize();
  293. public:
  294. SequenceNumber GetSequenceNumber();
  295. void IncrementSequenceNumber();
  296. private:
  297. SequenceNumber _seq;
  298. CCriticalSection _cs;
  299. CVolumeTable * _pVolTab;
  300. CQuotaTable * _pQuotaTab;
  301. CTrkSvrConfiguration * _psvrconfig;
  302. CFILETIME _cftLastRead;
  303. };
  304. inline
  305. CRefreshSequenceStorage::CRefreshSequenceStorage( CVolumeTable * pVolTab,
  306. CQuotaTable *pQuotaTab,
  307. CTrkSvrConfiguration *psvrconfig ) :
  308. _pVolTab(pVolTab),
  309. _pQuotaTab(pQuotaTab),
  310. _cftLastRead(0),
  311. _psvrconfig(psvrconfig)
  312. {
  313. }
  314. inline void
  315. CRefreshSequenceStorage::Initialize()
  316. {
  317. _cs.Initialize();
  318. }
  319. inline
  320. CRefreshSequenceStorage::~CRefreshSequenceStorage()
  321. {
  322. _cs.UnInitialize();
  323. }
  324. //+----------------------------------------------------------------------------
  325. //
  326. // Class CAbbreviatedIDString
  327. //
  328. // Take an ID (e.g. a Droid) and create an abbreviated string for
  329. // dbg outs.
  330. //
  331. //+----------------------------------------------------------------------------
  332. class CAbbreviatedIDString
  333. {
  334. private:
  335. TCHAR _tsz[ 8 ]; // e.g. {01:02}
  336. public:
  337. inline CAbbreviatedIDString( const CDomainRelativeObjId &droid );
  338. inline operator const TCHAR *() const;
  339. };
  340. inline
  341. CAbbreviatedIDString::CAbbreviatedIDString( const CDomainRelativeObjId &droid )
  342. {
  343. GUID guidObjId, guidVolId;
  344. droid.GetVolumeId().SerializeRaw( reinterpret_cast<BYTE*>(&guidVolId) );
  345. droid.GetObjId().SerializeRaw( reinterpret_cast<BYTE*>(&guidObjId) );
  346. _stprintf( _tsz, TEXT("{%02x:%02x}"),
  347. guidVolId.Data1 & 0xFF, guidObjId.Data1 & 0xFF );
  348. }
  349. inline
  350. CAbbreviatedIDString::operator const TCHAR *() const
  351. {
  352. return( _tsz );
  353. }
  354. //+-------------------------------------------------------------------------
  355. //
  356. // Class: CIntraDomainTable (a.k.a. Object Move Table)
  357. //
  358. // Purpose: Handles updates and queries of the location database.
  359. //
  360. // Notes: Columns
  361. // -------
  362. //
  363. // prev_location(indexed) new_location birth_id
  364. //
  365. //--------------------------------------------------------------------------
  366. struct IDT_ENTRY;
  367. class CQuotaTable;
  368. class CIntraDomainTable
  369. {
  370. public:
  371. inline CIntraDomainTable(
  372. CDbConnection & dbc,
  373. CRefreshSequenceStorage * pseqRefresh
  374. );
  375. inline ~CIntraDomainTable();
  376. void Initialize( CTrkSvrConfiguration *pTrkSvrConfiguration, CQuotaTable* pqtable );
  377. void UnInitialize();
  378. public:
  379. BOOL Add(const CDomainRelativeObjId &droidKey,
  380. const CDomainRelativeObjId &droidNew,
  381. const CDomainRelativeObjId &droidBirth,
  382. BOOL *pfQuotaExceeded = NULL OPTIONAL
  383. );
  384. //-> indent 4 chars
  385. BOOL Delete(const CDomainRelativeObjId &droidKey);
  386. ULONG GarbageCollect( SequenceNumber seqCurrent,
  387. SequenceNumber seqOldestToKeep,
  388. const BOOL * pfAbort );
  389. BOOL Modify(const CDomainRelativeObjId &droidKey,
  390. const CDomainRelativeObjId &droidNew,
  391. const CDomainRelativeObjId &droidBirth
  392. );
  393. BOOL Query(const CDomainRelativeObjId &droidKey,
  394. CDomainRelativeObjId *pdroidNew,
  395. CDomainRelativeObjId *pdroidBirth,
  396. BOOL *pfDeleted = NULL OPTIONAL,
  397. BOOL *pfCounted = NULL OPTIONAL );
  398. BOOL Query( LDAP* pLdap,
  399. LDAPMessage *pEntry,
  400. const CDomainRelativeObjId ldKey,
  401. CDomainRelativeObjId *pldNew,
  402. CDomainRelativeObjId *pldBirth,
  403. BOOL *pfDeleted = NULL OPTIONAL,
  404. BOOL *pfCounted = NULL OPTIONAL );
  405. BOOL Touch(const CDomainRelativeObjId &droidKey);
  406. #if DBG
  407. void PurgeAll();
  408. #endif
  409. private:
  410. inline const TCHAR *GetBaseDn() { return _dbc.GetBaseDn(); }
  411. inline LDAP * Ldap() { return _dbc.Ldap(); }
  412. private:
  413. BOOL _fInitializeCalled:1;
  414. CDbConnection & _dbc;
  415. CRegBoolParameter _QuotaReported;
  416. CRefreshSequenceStorage *
  417. _pRefreshSequenceStorage;
  418. CQuotaTable* _pqtable;
  419. CTrkSvrConfiguration
  420. *_pTrkSvrConfiguration;
  421. };
  422. inline
  423. CIntraDomainTable::CIntraDomainTable(CDbConnection & dbc,
  424. CRefreshSequenceStorage * pRefreshSequenceStorage ) :
  425. _fInitializeCalled(FALSE),
  426. _dbc(dbc),
  427. _pRefreshSequenceStorage( pRefreshSequenceStorage ),
  428. _QuotaReported( TEXT("IntraDomainTableQuotaReported") )
  429. {
  430. }
  431. inline
  432. CIntraDomainTable::~CIntraDomainTable()
  433. {
  434. UnInitialize();
  435. }
  436. // Objects to make constructing quota table keys easy.
  437. class CLdapSeqNum
  438. {
  439. public:
  440. CLdapSeqNum()
  441. {
  442. _tcscpy( _tszSeq, TEXT("0") );
  443. }
  444. CLdapSeqNum(SequenceNumber seq)
  445. {
  446. _stprintf( _tszSeq, TEXT("%lu"), seq );
  447. }
  448. operator TCHAR*()
  449. {
  450. return( _tszSeq );
  451. }
  452. TCHAR _tszSeq[ CCH_UINT32 ];
  453. };
  454. class CLdapQuotaKeyDn
  455. {
  456. public:
  457. CLdapQuotaKeyDn(const TCHAR* ptszBaseDn, const CMachineId& mcid)
  458. {
  459. TCHAR* ptsz;
  460. _tcscpy(_tszDn, TEXT("CN="));
  461. _tcscat(_tszDn, TEXT("QTDC_"));
  462. ptsz = _tszDn + _tcslen(_tszDn);
  463. mcid.Stringize(ptsz);
  464. *ptsz++ = TEXT(',');
  465. _tcscpy(ptsz, s_VolumeTableRDN);
  466. _tcscat(_tszDn, ptszBaseDn);
  467. TrkAssert(_tcslen(_tszDn) < ELEMENTS(_tszDn));
  468. }
  469. inline operator TCHAR* () { return _tszDn; }
  470. private:
  471. TCHAR _tszDn[MAX_PATH];
  472. };
  473. class CLdapQuotaCounterKeyDn
  474. {
  475. public:
  476. CLdapQuotaCounterKeyDn(const TCHAR* ptszBaseDn)
  477. {
  478. TCHAR* ptsz;
  479. _tcscpy(_tszDn, TEXT("CN="));
  480. _tcscat(_tszDn, TEXT("QT_Counter"));
  481. ptsz = _tszDn + _tcslen(_tszDn);
  482. *ptsz++ = TEXT(',');
  483. _tcscpy(ptsz, s_VolumeTableRDN);
  484. _tcscat(_tszDn, ptszBaseDn);
  485. TrkAssert(_tcslen(_tszDn) < ELEMENTS(_tszDn));
  486. }
  487. inline operator TCHAR* () { return _tszDn; }
  488. private:
  489. TCHAR _tszDn[MAX_PATH];
  490. };
  491. //+------------------------------------------------------------------------
  492. //
  493. // Class: CQuotaTable
  494. //
  495. // Purpose: Encapsulate functionalities that enforce quota on the
  496. // tables in the DC (move table, volume table).
  497. //
  498. //-------------------------------------------------------------------------
  499. class CQuotaTable : public PTimerCallback
  500. {
  501. public:
  502. // constructor/destructor
  503. CQuotaTable(CDbConnection& dbc);
  504. ~CQuotaTable();
  505. // initialization/uninitialization
  506. void Initialize(CVolumeTable *pvoltab,
  507. CIntraDomainTable *pidt,
  508. CTrkSvrSvc *psvrsvc,
  509. CTrkSvrConfiguration *pcfgsvr );
  510. void UnInitialize();
  511. BOOL IsDesignatedDc( BOOL fRaiseOnError = FALSE );
  512. // timer call back
  513. virtual TimerContinuation Timer(ULONG);
  514. // checking quota
  515. BOOL IsMoveQuotaExceeded();
  516. BOOL IsVolumeQuotaExceeded(const CMachineId& mcid, ULONG cUncountedVolumes);
  517. BOOL UpdateFlags(LDAP* pLdap, TCHAR* dnKey, BYTE bFlags);
  518. void DeleteFlags(LDAP* pLdap, TCHAR* dnKey);
  519. void OnServiceStopRequest();
  520. void IncrementMoveCountCache() { InterlockedIncrement(&_lCachedMoveTableCount); }
  521. void DecrementMoveCountCache() { InterlockedDecrement(&_lCachedMoveTableCount); }
  522. inline void IncrementVolumeCountCache();
  523. inline void DecrementVolumeCountCache();
  524. void InvalidateCache()
  525. {
  526. _cftCacheLastUpdated = 0;
  527. DeleteCounter();
  528. }
  529. void Statistics( TRKSVR_STATISTICS *pTrkSvrStatistics );
  530. void OnMoveTableGcComplete( ULONG cEntriesDeleted );
  531. private:
  532. enum EBackgroundForegroundTask
  533. {
  534. BACKGROUND = 1,
  535. FOREGROUND
  536. };
  537. private:
  538. void Lock();
  539. void Unlock();
  540. DWORD CalculateMoveLimit(); // Doesn't raise
  541. inline LDAP* Ldap() const { return _dbc.Ldap(); }
  542. inline const TCHAR *GetBaseDn() { return _dbc.GetBaseDn(); }
  543. BOOL ReadCounter(DWORD* dwCounter);
  544. HRESULT DeleteCounter();
  545. void GetTrueCount( DWORD* pdwTrueCount,
  546. EBackgroundForegroundTask eBackgroundForegroundTask );
  547. void WriteCounter(DWORD);
  548. void ValidateCache( BOOL fForceHint = FALSE );
  549. HRESULT ReadFlags(LDAP* pLdap, TCHAR* dnKey, BYTE* bFlags);
  550. BOOL DeleteOrphanedEntries( const CDomainRelativeObjId rgdroidList[], ULONG cSegments,
  551. const CDomainRelativeObjId &droidBirth,
  552. const CDomainRelativeObjId &droidCurrent );
  553. void ShortenString( LDAP* pLdap, LDAPMessage* pMessage, BYTE *pbFlags,
  554. const CDomainRelativeObjId &droidCurrent );
  555. TCHAR * GetFirstCN( LDAP *pLdap, LDAPMessage *pMessage );
  556. static ENUM_ACTION
  557. MoveTableEnumCallback( LDAP * pLdap,
  558. LDAPMessage * pResult,
  559. void* cEntries,
  560. void* pvThis );
  561. static ENUM_ACTION
  562. VolumeTableEnumCallback( LDAP * pLdap,
  563. LDAPMessage * pResult,
  564. void* pcEntries,
  565. void* pvThis );
  566. private:
  567. enum
  568. {
  569. QUOTA_TIMER = 0,
  570. QUOTA_CLEANUP_TIMER = 1
  571. } EQuotaTimer;
  572. BOOL _fInitializeCalled:1;
  573. BOOL _fIsDesignatedDc:1;
  574. BOOL _fStopping:1;
  575. CFILETIME _cftDesignatedDc;
  576. CTrkSvrConfiguration *_pcfgsvr;
  577. CDbConnection& _dbc;
  578. DWORD _dwMoveLimit;
  579. CMachineId _mcid;
  580. LONG _lCachedMoveTableCount;
  581. LONG _lCachedVolumeTableCount;
  582. CFILETIME _cftCacheLastUpdated;
  583. CVolumeTable *_pvoltab;
  584. CIntraDomainTable *_pidt;
  585. CTrkSvrSvc *_psvrsvc;
  586. CNewTimer _timer;
  587. CCriticalSection _cs;
  588. #if DBG
  589. LONG _cLocks;
  590. #endif
  591. };
  592. inline void
  593. CQuotaTable::OnServiceStopRequest()
  594. {
  595. _fStopping = TRUE;
  596. }
  597. inline void
  598. CQuotaTable::Lock()
  599. {
  600. _cs.Enter();
  601. #if DBG
  602. _cLocks++;
  603. #endif
  604. }
  605. inline void
  606. CQuotaTable::Unlock()
  607. {
  608. TrkAssert( 0 < _cLocks );
  609. _cs.Leave();
  610. #if DBG
  611. _cLocks--;
  612. #endif
  613. }
  614. inline void
  615. CQuotaTable::IncrementVolumeCountCache()
  616. {
  617. Lock();
  618. _lCachedVolumeTableCount++;
  619. _dwMoveLimit = CalculateMoveLimit(); // Doesn't raise
  620. Unlock();
  621. }
  622. inline void
  623. CQuotaTable::DecrementVolumeCountCache()
  624. {
  625. Lock();
  626. --_lCachedVolumeTableCount;
  627. _dwMoveLimit = CalculateMoveLimit(); // Doesn't raise
  628. Unlock();
  629. }
  630. //+-------------------------------------------------------------------------
  631. //
  632. // Class: CVolumeTable
  633. //
  634. // Purpose: Table of volumes
  635. //
  636. // Notes:
  637. //
  638. //--------------------------------------------------------------------------
  639. class CVolumeTable
  640. #ifdef VOL_REPL
  641. : PTimerCallback
  642. #endif
  643. {
  644. public: // Construction/destruction
  645. inline CVolumeTable(
  646. CDbConnection & dbc,
  647. CRefreshSequenceStorage * pRefreshSequenceStorage
  648. );
  649. inline ~CVolumeTable();
  650. void Initialize(CTrkSvrConfiguration *pconfigSvr,
  651. CQuotaTable *pqtable);
  652. void UnInitialize();
  653. public: // Public RPC level interface - must be parameter checked
  654. HRESULT ClaimVolume( const CMachineId & mcidClient,
  655. const CVolumeId & volume,
  656. const CVolumeSecret & secretOld,
  657. const CVolumeSecret & secretNew,
  658. SequenceNumber * pseq,
  659. FILETIME * pftLastRefresh );
  660. HRESULT PreCreateVolume( const CMachineId & mcidClient,
  661. const CVolumeSecret & secret,
  662. ULONG cUncountedVolumes,
  663. CVolumeId * pvolume );
  664. HRESULT DeleteVolume( const CMachineId & mcidClient,
  665. const CVolumeId & volume );
  666. HRESULT FindVolume( const CVolumeId & volume,
  667. CMachineId * pmcid );
  668. HRESULT QueryVolume( const CMachineId & mcidClient,
  669. const CVolumeId & volume,
  670. SequenceNumber * pseq,
  671. FILETIME * pftLastRefresh );
  672. //-> indent 4 chars
  673. BOOL Touch( const CVolumeId & volume );
  674. public:
  675. HRESULT SetSequenceNumber(const CVolumeId & volume, SequenceNumber seq);
  676. HRESULT SetSecret( const CVolumeId & volume, const CVolumeSecret & secret );
  677. int GetAvailableNotificationQuota( const CVolumeId & volid );
  678. DWORD CountVolumes( const CMachineId & mcidClient );
  679. ULONG GarbageCollect( SequenceNumber seqCurrent, SequenceNumber seqOldestToKeep, const BOOL * pfAbort );
  680. HRESULT GetVolumeInfo( const CVolumeId & volume,
  681. CMachineId * pmcid,
  682. CVolumeSecret * psecret,
  683. SequenceNumber * pseq,
  684. CFILETIME *pcftRefresh );
  685. HRESULT GetMachine( const CVolumeId & volume, CMachineId * pmcid );
  686. HRESULT SetMachine( const CVolumeId & volume, const CMachineId & mcid );
  687. HRESULT SetMachineAndSecret(const CVolumeId & volume, const CMachineId & mcid,
  688. const CVolumeSecret & secret);
  689. inline const TCHAR *GetBaseDn() { return _dbc.GetBaseDn(); }
  690. inline LDAP * Ldap() const { return _dbc.Ldap(); }
  691. #if DBG
  692. void PurgeCache( );
  693. #endif
  694. HRESULT AddVolidToTable( const CVolumeId & volume,
  695. const CMachineId & mcidClient,
  696. const CVolumeSecret & secret );
  697. void PurgeAll();
  698. #ifdef VOL_REPL
  699. void QueryVolumeChanges( const CFILETIME &ftFirstChange, CVolumeMap * pVolMap );
  700. void SimpleTimer( DWORD dwTimerId );
  701. #endif
  702. private:
  703. HRESULT MapResult(int ldap_err) const;
  704. #ifdef VOL_REPL
  705. void _QueryVolumeChanges( const CFILETIME &ftFirstChange, CVolumeMap * pVolMap );
  706. #endif
  707. private:
  708. BOOL _fInitializeCalled:1;
  709. CDbConnection & _dbc;
  710. CRefreshSequenceStorage *
  711. _pRefreshSequenceStorage;
  712. CTrkSvrConfiguration *
  713. _pconfigSvr;
  714. // State for caching the volume table so that
  715. // synchronizing the workstation's volume tables is efficient (normally the workstation
  716. // queries will hit the cached query.)
  717. CQuotaTable* _pqtable;
  718. #ifdef VOL_REPL
  719. CCritcalSection _csQueryCache;
  720. CVolumeMap _VolMap; // keeps cache of query since we have many machines
  721. // getting the updates
  722. CFILETIME _cftCacheLowest;
  723. CSimpleTimer _timerQueryCache;
  724. DWORD _SecondsPreviousToNow;
  725. #endif
  726. };
  727. inline
  728. CVolumeTable::CVolumeTable( CDbConnection & dbc,
  729. CRefreshSequenceStorage * pRefreshSequenceStorage
  730. ) :
  731. _dbc(dbc),
  732. _fInitializeCalled(FALSE),
  733. _pRefreshSequenceStorage( pRefreshSequenceStorage ),
  734. _pconfigSvr(NULL)
  735. {
  736. }
  737. inline
  738. CVolumeTable::~CVolumeTable()
  739. {
  740. UnInitialize();
  741. }
  742. //+-------------------------------------------------------------------------
  743. //
  744. // Class: CDenialChecker
  745. //
  746. // Purpose: Checks for denial of service attacks
  747. //
  748. // Notes:
  749. //
  750. //--------------------------------------------------------------------------
  751. struct ACTIVECLIENT;
  752. class CDenialChecker : public PTimerCallback
  753. {
  754. public:
  755. inline CDenialChecker();
  756. inline ~CDenialChecker();
  757. void Initialize( ULONG ulHistoryPeriod );
  758. void UnInitialize();
  759. void CheckClient(const CMachineId &mcidClient);
  760. virtual TimerContinuation Timer( ULONG ulTimerId ); // timeout for clearing out old ACTIVECLIENTs
  761. private:
  762. BOOL _fInitializeCalled;
  763. CNewTimer _timer;
  764. ACTIVECLIENT * _pListHead;
  765. CCriticalSection _cs;
  766. #if DBG
  767. LONG _lAllocs;
  768. #endif
  769. };
  770. inline
  771. CDenialChecker::CDenialChecker() :
  772. _fInitializeCalled(FALSE)
  773. {
  774. }
  775. inline
  776. CDenialChecker::~CDenialChecker()
  777. {
  778. UnInitialize();
  779. }
  780. class CTrkSvrRpcServer : public CRpcServer
  781. {
  782. public:
  783. void Initialize( SVCHOST_GLOBAL_DATA * pSvcsGlobalData, CTrkSvrConfiguration *pTrkSvrConfig );
  784. void UnInitialize( SVCHOST_GLOBAL_DATA * pSvcsGlobalData );
  785. };
  786. //+-------------------------------------------------------------------------
  787. //
  788. // Class: CTrkSvrSvc
  789. //
  790. // Purpose: Tracking (Server) Service
  791. //
  792. // Notes:
  793. //
  794. //--------------------------------------------------------------------------
  795. #define MAX_SVR_THREADS 10
  796. class CTrkSvrSvc : public PTimerCallback,
  797. public IServiceHandler
  798. {
  799. public: // Initialization
  800. inline CTrkSvrSvc();
  801. inline ~CTrkSvrSvc();
  802. void Initialize(SVCHOST_GLOBAL_DATA * pSvcsGlobalData );
  803. void UnInitialize(HRESULT hr);
  804. public: // RPC server methods
  805. void DeleteNotify(
  806. const CMachineId & mcidClient,
  807. TRKSVR_CALL_DELETE * pDelete
  808. );
  809. HRESULT MoveNotify(
  810. const CMachineId & mcidClient,
  811. TRKSVR_CALL_MOVE_NOTIFICATION * pMove
  812. );
  813. void Search(
  814. TRK_FILE_TRACKING_INFORMATION * pSearches
  815. );
  816. void old_Search(
  817. old_TRK_FILE_TRACKING_INFORMATION * pSearches
  818. );
  819. HRESULT SyncVolume(
  820. const CMachineId & mcidClient,
  821. TRKSVR_SYNC_VOLUME * pSyncVolume,
  822. ULONG cUncountedCreates
  823. );
  824. void Refresh(
  825. const CMachineId & mcidClient,
  826. TRKSVR_CALL_REFRESH * pRefresh
  827. );
  828. void Statistics(
  829. TRKSVR_STATISTICS *pTrkSvrStatistics
  830. );
  831. public: // General publics
  832. HRESULT SvrMessage( handle_t IDL_handle, TRKSVR_MESSAGE_UNION * pMsg );
  833. inline void CheckClient( const CMachineId & mcid );
  834. BOOL CountPrioritizedThread( const TRKSVR_MESSAGE_UNION * pMsg );
  835. void ReleasePrioritizedThread( );
  836. void DoRefresh( const CRpcClientBinding & rc );
  837. inline DWORD GetState();
  838. virtual TimerContinuation Timer( ULONG ulTimerContext ); // refresh timer
  839. DWORD ServiceHandler(DWORD dwControl, // IServiceHandler implementation
  840. DWORD dwEventType,
  841. PVOID EventData,
  842. PVOID pData);
  843. inline const BOOL * GetStopFlagAddress();
  844. BOOL RequireSecureRPC();
  845. HRESULT CreateVolume(const CMachineId & mcidClient, const TRKSVR_SYNC_VOLUME& pSyncVolume);
  846. inline void SetLastError( HRESULT hr );
  847. void OnRequestStart( TRKSVR_MESSAGE_TYPE MsgType );
  848. void OnRequestEnd( TRKSVR_MESSAGE_UNION * pMsg, const CMachineId &mcid, HRESULT hr );
  849. void RaiseIfStopped();
  850. void Scan(
  851. IN const CDomainRelativeObjId * pdroidNotificationCurrent, OPTIONAL
  852. IN const CDomainRelativeObjId * pdroidNotificationNew, OPTIONAL
  853. IN const CDomainRelativeObjId & droidBirth,
  854. OUT CDomainRelativeObjId * pdroidList,
  855. IN int cdroidList,
  856. OUT int * pcSegments,
  857. IN OUT CDomainRelativeObjId * pdroidScan,
  858. OUT BOOL *pfStringDeleted
  859. );
  860. public: // testing
  861. friend class CTestSvrSvc;
  862. void TestRestore( const TCHAR * ptszMachine );
  863. private:
  864. void MoveNotify(const CDomainRelativeObjId &droidCurrent,
  865. const CDomainRelativeObjId &droidBirth,
  866. const CDomainRelativeObjId &droidNew,
  867. BOOL *pfQuotaExceeded );
  868. SequenceNumber GetSequenceNumber( const CMachineId & mcidClient,
  869. const CVolumeId & volume);
  870. void SetSequenceNumber( const CVolumeId & volume, // must ensure that validation already done for volume
  871. SequenceNumber seq );
  872. inline BOOL IncrementAndCheckWritesPerHour();
  873. inline void IncrementWritesPerHour();
  874. BOOL CheckWritesPerHour();
  875. inline ULONG NumWritesThisHour() const;
  876. BOOL VerifyMachineOwnsVolume( const CMachineId &mcid, const CVolumeId & volid );
  877. private: // data members
  878. BOOL _fInitializeCalled:1;
  879. // Indicates if we're hesitating before doing a GC
  880. // (see DoWork)
  881. BOOL _fHesitatingBeforeGC:1;
  882. // Indicates that service_stop/shutdown has been received
  883. BOOL _fStopping:1;
  884. // If true, and we're in a GC phase, GC the volume table
  885. // rather than the move table.
  886. BOOL _fGCVolumeTable:1;
  887. SVCHOST_GLOBAL_DATA * _pSvcsGlobalData;
  888. // Protect against denail-of-service attacks.
  889. CDenialChecker _denial;
  890. // LDAP manager
  891. CDbConnection _dbc;
  892. CRefreshSequenceStorage
  893. _refreshSequence;
  894. CCrossDomainTable _cdt;
  895. CIntraDomainTable _idt;
  896. CVolumeTable _voltab;
  897. CRegDwordParameter _MoveCounterReset;
  898. CNewTimer _timerGC;
  899. LONG _cGarbageCollectionsRunning;
  900. CQuotaTable _qtable;
  901. CTrkSvrRpcServer _rpc;
  902. CSvcCtrlInterface _svcctrl;
  903. CTrkSvrConfiguration
  904. _configSvr;
  905. LONG _cAvailableThreads, _cLowestAvailableThreads;
  906. // The following values track the number of writes (to the DS)
  907. // in an hour. If this gets too high, trksvr disallows writes
  908. // for the rest of the hour.
  909. CFILETIME _cftWritesPerHour;
  910. ULONG _cWritesPerHour;
  911. CCriticalSection _csWritesPerHour;
  912. // Statistics
  913. struct tagStats
  914. {
  915. ULONG cSyncVolumeRequests, cSyncVolumeErrors, cSyncVolumeThreads;
  916. ULONG cCreateVolumeRequests, cCreateVolumeErrors;
  917. ULONG cClaimVolumeRequests, cClaimVolumeErrors;
  918. ULONG cQueryVolumeRequests, cQueryVolumeErrors;
  919. ULONG cFindVolumeRequests, cFindVolumeErrors;
  920. ULONG cTestVolumeRequests, cTestVolumeErrors;
  921. ULONG cSearchRequests, cSearchErrors, cSearchThreads;
  922. ULONG cMoveNotifyRequests, cMoveNotifyErrors, cMoveNotifyThreads;
  923. ULONG cRefreshRequests, cRefreshErrors, cRefreshThreads;
  924. ULONG cDeleteNotifyRequests, cDeleteNotifyErrors, cDeleteNotifyThreads;
  925. //ULONG ulGCIterationPeriod;
  926. //SHORT cEntriesToGC;
  927. SHORT cEntriesGCed;
  928. SHORT cMaxDsWriteEvents;
  929. SHORT cCurrentFailedWrites;
  930. CFILETIME cftLastSuccessfulRequest;
  931. CFILETIME cftServiceStartTime;
  932. HRESULT hrLastError;
  933. } _Stats;
  934. public:
  935. COperationLog _OperationLog;
  936. };
  937. inline
  938. CTrkSvrSvc::CTrkSvrSvc() :
  939. _fInitializeCalled(FALSE),
  940. _fHesitatingBeforeGC(FALSE),
  941. _fStopping(FALSE),
  942. _idt(_dbc, &_refreshSequence),
  943. _cdt(_dbc),
  944. _voltab(_dbc, &_refreshSequence),
  945. _qtable(_dbc),
  946. _cWritesPerHour(0),
  947. _cGarbageCollectionsRunning(0),
  948. _fGCVolumeTable(FALSE),
  949. _MoveCounterReset( TEXT("MoveCounterReset") ),
  950. _refreshSequence( &_voltab, &_qtable, &_configSvr ) // refreshSequence uses voltab to store its state
  951. {
  952. // Statistics
  953. memset( &_Stats, 0, sizeof(_Stats) );
  954. _Stats.cftLastSuccessfulRequest = 0;
  955. _Stats.cftServiceStartTime = CFILETIME();
  956. }
  957. inline
  958. CTrkSvrSvc::~CTrkSvrSvc()
  959. {
  960. TrkAssert(!_fInitializeCalled);
  961. }
  962. inline const BOOL *
  963. CTrkSvrSvc::GetStopFlagAddress()
  964. {
  965. return(_svcctrl.GetStopFlagAddress());
  966. }
  967. inline BOOL
  968. CTrkSvrSvc::RequireSecureRPC()
  969. {
  970. return( _rpc.RpcSecurityEnabled() );
  971. }
  972. inline void
  973. CTrkSvrSvc::SetLastError( HRESULT hr )
  974. {
  975. _Stats.hrLastError = hr;
  976. }
  977. inline void
  978. CTrkSvrSvc::CheckClient( const CMachineId & mcid )
  979. {
  980. _denial.CheckClient( mcid );
  981. }
  982. inline DWORD
  983. CTrkSvrSvc::GetState()
  984. {
  985. return( _svcctrl.GetState() );
  986. }
  987. inline void
  988. CTrkSvrSvc::IncrementWritesPerHour()
  989. {
  990. #if DBG
  991. LONG l = InterlockedIncrement( (LONG*)&_cWritesPerHour );
  992. TrkLog(( TRKDBG_SVR, TEXT("WritesPerHour: %d"), l ));
  993. #else
  994. InterlockedIncrement( (LONG*)&_cWritesPerHour );
  995. #endif
  996. }
  997. inline BOOL
  998. CTrkSvrSvc::IncrementAndCheckWritesPerHour()
  999. {
  1000. IncrementWritesPerHour();
  1001. return CheckWritesPerHour();
  1002. }
  1003. inline ULONG
  1004. CTrkSvrSvc::NumWritesThisHour() const
  1005. {
  1006. return _cWritesPerHour;
  1007. }
  1008. //+-------------------------------------------------------------------------
  1009. //
  1010. // Class: CLdapBinaryMod
  1011. //
  1012. // Purpose: Wrapper for all the structures used by LDAP for binary
  1013. // attribute modifications
  1014. //
  1015. // Notes:
  1016. //
  1017. //--------------------------------------------------------------------------
  1018. class CLdapBinaryMod
  1019. {
  1020. public:
  1021. inline CLdapBinaryMod();
  1022. // note: the pszAttrName is not copied!
  1023. inline CLdapBinaryMod(const TCHAR * pszAttrName,
  1024. PCCH bv_val,
  1025. ULONG bv_len,
  1026. int mod_op);
  1027. // note: the pszAttrName is not copied!
  1028. void Init(const TCHAR * pszAttrName,
  1029. PCCH bv_val,
  1030. ULONG bv_len,
  1031. int mod_op);
  1032. public:
  1033. LDAPMod _mod;
  1034. private:
  1035. LDAP_BERVAL _BerVal;
  1036. LDAP_BERVAL * _apBerVals[2];
  1037. };
  1038. inline
  1039. CLdapBinaryMod::CLdapBinaryMod()
  1040. {
  1041. }
  1042. inline
  1043. CLdapBinaryMod::CLdapBinaryMod(const TCHAR * pszAttrName,
  1044. PCCH bv_val,
  1045. ULONG bv_len,
  1046. int mod_op)
  1047. {
  1048. Init(pszAttrName, bv_val, bv_len, mod_op);
  1049. }
  1050. inline void
  1051. CLdapBinaryMod::Init(const TCHAR * pszAttrName, PCCH bv_val, ULONG bv_len, int mod_op)
  1052. {
  1053. _mod.mod_op = LDAP_MOD_BVALUES | mod_op;
  1054. _mod.mod_type = (TCHAR*)pszAttrName;
  1055. // bv_val might be null if this is a delete.
  1056. if( NULL != bv_val )
  1057. {
  1058. _BerVal.bv_len = bv_len;
  1059. _BerVal.bv_val = const_cast<PCHAR>(bv_val);
  1060. _apBerVals[0] = &_BerVal;
  1061. _apBerVals[1] = NULL;
  1062. _mod.mod_bvalues = _apBerVals;
  1063. }
  1064. else
  1065. _mod.mod_bvalues = NULL;
  1066. }
  1067. //+-------------------------------------------------------------------------
  1068. //
  1069. // Class: CLdapStringMod
  1070. //
  1071. // Purpose: Wrapper for all the structures used by LDAP for string
  1072. // attribute modifications
  1073. //
  1074. // Notes:
  1075. //
  1076. //--------------------------------------------------------------------------
  1077. class CLdapStringMod
  1078. {
  1079. public:
  1080. inline CLdapStringMod();
  1081. // note: the pszAttrName is not copied!
  1082. inline CLdapStringMod(const TCHAR * pszAttrName,
  1083. const TCHAR * str_val,
  1084. int mod_op);
  1085. // note: the pszAttrName is not copied!
  1086. void Init(const TCHAR * pszAttrName,
  1087. const TCHAR * str_val,
  1088. int mod_op);
  1089. public:
  1090. LDAPMod _mod;
  1091. private:
  1092. TCHAR * _apsz[2];
  1093. };
  1094. inline
  1095. CLdapStringMod::CLdapStringMod()
  1096. {
  1097. }
  1098. inline
  1099. CLdapStringMod::CLdapStringMod(const TCHAR * pszAttrName,
  1100. const TCHAR * str_val,
  1101. int mod_op)
  1102. {
  1103. // if we're specifying the objectClass we don't
  1104. TrkAssert(_tcscmp(pszAttrName, s_objectClass) != 0 ||
  1105. mod_op == 0);
  1106. Init(pszAttrName, str_val, mod_op);
  1107. }
  1108. inline void
  1109. CLdapStringMod::Init(const TCHAR * pszAttrName,
  1110. const TCHAR * str_val,
  1111. int mod_op)
  1112. {
  1113. _mod.mod_op = mod_op;
  1114. _mod.mod_type = (TCHAR*)pszAttrName;
  1115. _mod.mod_vals.modv_strvals = _apsz;
  1116. _apsz[0] = (TCHAR*)str_val;
  1117. _apsz[1] = NULL;
  1118. }
  1119. //+-------------------------------------------------------------------------
  1120. //
  1121. // Class: CLdapTimeValue
  1122. //
  1123. //--------------------------------------------------------------------------
  1124. class CLdapTimeValue
  1125. {
  1126. public:
  1127. CLdapTimeValue()
  1128. {
  1129. //memset(_abPad,0,sizeof(_abPad));
  1130. _stprintf( _tszTime, TEXT("%I64u"), (LONGLONG)CFILETIME() );
  1131. }
  1132. CLdapTimeValue(const CFILETIME &cft)
  1133. {
  1134. _stprintf( _tszTime, TEXT("%I64u"), (LONGLONG) cft );
  1135. }
  1136. // mikehill: is this still necessary?
  1137. void Swap();
  1138. inline BYTE & Byte(int i)
  1139. {
  1140. return( ((BYTE*)this)[i] );
  1141. }
  1142. operator TCHAR*()
  1143. {
  1144. return( _tszTime );
  1145. }
  1146. TCHAR _tszTime[ CCH_UINT64 ];
  1147. //BYTE _abPad[sizeof(GUID) - sizeof(CFILETIME)];
  1148. };
  1149. inline void
  1150. CLdapTimeValue::Swap()
  1151. {
  1152. BYTE b;
  1153. for (int i=0; i<sizeof(*this)/2; i++)
  1154. {
  1155. b = Byte(i);
  1156. Byte(i) = Byte(sizeof(*this)-i-1);
  1157. Byte(sizeof(*this)-i-1) = b;
  1158. }
  1159. }
  1160. //+-------------------------------------------------------------------------
  1161. //
  1162. // Class: CLdapRefreshSeq
  1163. //
  1164. //--------------------------------------------------------------------------
  1165. class CLdapRefresh
  1166. {
  1167. public:
  1168. CLdapRefresh( SequenceNumber seq )
  1169. {
  1170. _stprintf( _tszSeq, TEXT("%u"), seq );
  1171. }
  1172. operator TCHAR*()
  1173. {
  1174. return( _tszSeq );
  1175. }
  1176. TCHAR _tszSeq[ CCH_UINT64 ];
  1177. };
  1178. //+-------------------------------------------------------------------------
  1179. //
  1180. // Class: CLdapOMTAddModify
  1181. //
  1182. // Purpose: Wrapper for all the structures used to set a linkTrackOMTEntry
  1183. // (an entry in the Object Move Table)
  1184. //
  1185. //
  1186. // Notes:
  1187. //
  1188. //--------------------------------------------------------------------------
  1189. class CLdapOMTAddModify
  1190. {
  1191. public:
  1192. CLdapOMTAddModify(
  1193. const CDomainRelativeObjId & droidKey,
  1194. const CDomainRelativeObjId & droidNew,
  1195. const CDomainRelativeObjId & droidBirth,
  1196. const ULONG & seqRefresh,
  1197. BYTE bFlags, // Only used on LDAP_MOD_ADD
  1198. int mod_op
  1199. );
  1200. CLdapStringMod _lsmClass;
  1201. CLdapBinaryMod _lbmCurrentLocation;
  1202. CLdapBinaryMod _lbmBirthLocation;
  1203. CLdapStringMod _lsmRefresh;
  1204. CLdapBinaryMod _lbmFlags;
  1205. // We need to keep a CLdapTimeValue as a member in order to guarantee that
  1206. // it will remain in memory (_lsmRefresh will point to this data).
  1207. CLdapRefresh _ltvRefresh;
  1208. LDAPMod * _mods[7];
  1209. };
  1210. class CLdapVolumeKeyDn
  1211. {
  1212. public:
  1213. // specific volume
  1214. CLdapVolumeKeyDn( const TCHAR * ptszBaseDn, const CVolumeId & volume )
  1215. {
  1216. // Compose, e.g., the following DN:
  1217. // "CN=e3d954b2-d0a7-11d0-8cb6-00c04fd90f85,CN=VolumeTable,CN=FileLinks,DC=TRKDOM"
  1218. TCHAR *psz = _szDn;
  1219. _tcscpy(_szDn, TEXT("CN="));
  1220. psz += _tcslen(_szDn);
  1221. volume.Stringize(psz);
  1222. *psz++ = TEXT(',');
  1223. _tcscpy(psz, s_VolumeTableRDN); // The vol table, relative to the base
  1224. _tcscat(psz, ptszBaseDn);
  1225. TrkAssert(_tcslen(_szDn) < ELEMENTS(_szDn));
  1226. }
  1227. // all volumes
  1228. CLdapVolumeKeyDn( const TCHAR * ptszBaseDn )
  1229. {
  1230. _tcscpy(_szDn, s_VolumeTableRDN);
  1231. _tcscat(_szDn, ptszBaseDn);
  1232. TrkAssert(_tcslen(_szDn) < ELEMENTS(_szDn));
  1233. }
  1234. inline operator TCHAR * () { return _szDn; }
  1235. private:
  1236. TCHAR _szDn[MAX_PATH];
  1237. };
  1238. //+-------------------------------------------------------------------------
  1239. //
  1240. // Class: CLdapIdtKeyDn
  1241. //
  1242. // Purpose: Generates distinguished name for a CDomainRelativeObjId
  1243. //
  1244. // Notes:
  1245. //
  1246. //--------------------------------------------------------------------------
  1247. class CLdapIdtKeyDn
  1248. {
  1249. public:
  1250. inline CLdapIdtKeyDn(const TCHAR * pszBaseDn, const CDomainRelativeObjId &ld);
  1251. inline CLdapIdtKeyDn(const TCHAR * pszBaseDn );
  1252. inline operator TCHAR * ();
  1253. private:
  1254. TCHAR _szDn[MAX_PATH];
  1255. };
  1256. inline CLdapIdtKeyDn::CLdapIdtKeyDn(const TCHAR * pszBaseDn, const CDomainRelativeObjId
  1257. &ld)
  1258. {
  1259. // puts in something like "CN=v23487...65239238o23487...48652398"
  1260. ld.FillLdapIdtKeyBuffer(_szDn, sizeof(_szDn)/sizeof(_szDn[0]));
  1261. // make sure room for that string and the table name and the base dn
  1262. if (_tcslen(_szDn) +
  1263. 1 +
  1264. _tcslen(s_ObjectMoveTableRDN) +
  1265. _tcslen(pszBaseDn) + 1 >
  1266. sizeof(_szDn)/sizeof(_szDn[0]))
  1267. {
  1268. TrkRaiseException(TRK_E_DN_TOO_LONG);
  1269. }
  1270. _tcscat(_szDn, TEXT(","));
  1271. _tcscat(_szDn, s_ObjectMoveTableRDN);
  1272. _tcscat(_szDn, pszBaseDn);
  1273. TrkAssert(_tcslen(_szDn) < sizeof(_szDn)/sizeof(_szDn[0]));
  1274. }
  1275. inline CLdapIdtKeyDn::CLdapIdtKeyDn(const TCHAR * pszBaseDn)
  1276. {
  1277. // make sure room for that string and the table name and the base dn
  1278. if (1 +
  1279. _tcslen(s_ObjectMoveTableRDN) +
  1280. _tcslen(pszBaseDn) + 1 >
  1281. sizeof(_szDn)/sizeof(_szDn[0]))
  1282. {
  1283. TrkRaiseException(TRK_E_DN_TOO_LONG);
  1284. }
  1285. _tcscpy(_szDn, s_ObjectMoveTableRDN);
  1286. _tcscat(_szDn, pszBaseDn);
  1287. TrkAssert(_tcslen(_szDn) < sizeof(_szDn)/sizeof(_szDn[0]));
  1288. }
  1289. inline CLdapIdtKeyDn::operator TCHAR * ()
  1290. {
  1291. return(_szDn);
  1292. }