Uploaded image for project: 'Lustre'
  1. Lustre
  2. LU-2160

Implement ZFS dmu_tx_hold_append() declarations for llog

    XMLWordPrintable

Details

    • Improvement
    • Resolution: Unresolved
    • Minor
    • None
    • None
    • 5
    • 3
    • 2749

    Description

        llog records are written in append fashion and holes in llog files are not allowed. Lustre doesn't know until late into transaction life cycle if it will need to write llog record for this operation, at what offset and to what llog file. In other words at dmu_tx_assign() time Lustre only knows the size of a potential llog record it might write during this transaction but doesn't know the exact start offset and object id for this write into llog file (it also knows the maximum llog file size). But dmu_tx_ APIs assume that the caller knows precise offset of a future write before dmu_tx_assign(). This is needed to calculate precise amount of space that will be consumed by a given write and this amount may differ for the same write size at different offsets (a second reason is that in debug mode dmu_tx_dirty_buf() verifies all writes are done only to offsets declared with dmu_tx_hold_write()). Changing Lustre code to calculate precise offset in llog before dmu_tx_assign() step doesn't seem to yield an efficient solution therefore we propose to add 2 new dmu_tx APIs to accommodate Lustre requirements.

      1)

      void
      dmu_tx_hold_append(dmu_tx_t *tx, uint64_t object, uint64_t startoff,
                         uint64_t obj_maxsize, uint64_t len, int bs, int ibs)
      

      This API will reserve the maximum possible amount of space needed to append len bytes to any plain file within tx->tx_objset objset. The caller guarantees the size of this file is at most obj_maxsize but the actual start offset for the write may be anywhere from startoff to max_objsize-len (assert(len <= obj_maxsize)). This API is passed an object as object may be unknown when it's called. bs and ibs specify direct and indirect blocksizes used for the object we'll write (this avoids assuming worst cases for block sizes). if bs and ibs is 0 worst case for those blocksizes should be assumed.

      dmu_tx_hold_append() should pick the worst range for len bytes write. This will be the range where start and end byte are covered by a different indirect block at every level of the tree (i.e. seems like it should be [obj_maxsize/2 - len/2, obj_maxsize/2 + len/2)). dmu_tx_hold_append() can assume all writes to data blocks will only create new data and metadata blocks even if there's no snapshot. Therefore only txh_space_towrite field needs to be calculated for txh structure created by dmu_tx_hold_append() (space from metadnode update should be added just as in case of dmu_tx_hold_write() when dn is NULL).

      While dmu_tx_hold_append() may overestimate the net space consumed on disk by always assuming there's a snapshot and all writes create new metadata (this is correct assumption for data since we only write in append fashion to llog) it will avoid over reservation done by dmu_tx_hold_write() in other respects. Specifically dmu_tx_count_write() (called by dmu_tx_hold_write()) assumes the object size can be up to 2^64-1 when either dnode is not specified or declared write extends beyond current dn_maxblkid. In this case dmu_tx_count_write() accounts for maximum number of indirect blocks (and for the first write when maxblkid is 0 it uses the worst possible overhead from indirect blocks by computing the number of indirect blocks based on smallest indir blksz but using max indir blkzs as block size).

      dmu_tx_hold_append() can avoid that by knowing max file size and actual indirect/direct blocks sizes the caller guarantees to use. Specifically for 4K direct and 16K indirect blocksizes and 512MB max_objsize dmu_tx_hold_append() for 40 byte write will reserve 5 indirect blocks and 2 direct blocks or 88KB excluding metadnode file write.

      In the same case dmu_tx_hold_write() will reserve from this code (used when we extend the object into new block):

      	/*
      	 * 'end' is the last thing we will access, not one past.
      	 * This way we won't overflow when accessing the last byte.
      	 */
      	start = P2ALIGN(off, 1ULL << max_bs);
      	end = P2ROUNDUP(off + len, 1ULL << max_bs) - 1;
      	txh->txh_space_towrite += end - start + 1;
      
      	start >>= min_bs;
      	end >>= min_bs;
      
      	epbs = min_ibs - SPA_BLKPTRSHIFT;
              /*
      	 * The object contains at most 2^(64 - min_bs) blocks,
      	 * and each indirect level maps 2^epbs.
      	 */
      	for (bits = 64 - min_bs; bits >= 0; bits -= epbs) {
      		start >>= epbs;
      		end >>= epbs;
      		ASSERT3U(end, >=, start);
      		txh->txh_space_towrite += (end - start + 1) << max_ibs;
      		if (start != 0) {
      			/*
      			 * We also need a new blkid=0 indirect block
      			 * to reference any existing file data.
      			 */
      			txh->txh_space_towrite += 1ULL << max_ibs;
      		}
      	}
      

      at least (52/7 + 1) * 16K + 16K + 4K = 148KB (I added extra 16K for all writes when start is > 64MB).

      That's greater than 88KB reserved by dmu_tx_hold_append() excluding metadnode reservation (metadnode reservation would be the same if we don't know object id upfront). And when dnode id is unknown (which is the actual case for llog writes) dmu_tx_count_write() would reserve much more (as it uses 3 not 7 for epbs above).

      Attachments

        Issue Links

          Activity

            People

              bzzz Alex Zhuravlev
              adilger Andreas Dilger
              Votes:
              1 Vote for this issue
              Watchers:
              6 Start watching this issue

              Dates

                Created:
                Updated: