Hi swift-corelibs-dev,

I recently found myself with a DispatchData and wanted to use writev(2). So 
what I need was basically an array of pointers and lengths, not necessarily 
contiguous. That's exactly what DispatchData is so I thought it should be quite 
straightforward.

Obviously I need a guarantee that the pointers to the storage of all the 
DispatchData 'chunks' are all valid until writev returns.

I first checked the C API because it's documented and dispatch_data_apply seems 
to give me the guarantees that I need:

<quote source="man dispatch_data_apply">
     The dispatch_data_apply() function provides read access to represented 
memory without requiring it to be mapped as a
     single contiguous region. It traverses the memory regions represented by 
the data argument in logical order, invokes
     the specified applier block for each region and returns a boolean 
indicating whether traversal completed success-
     fully. The applier block is passed the following arguments for each memory 
region and returns a boolean indicating
     whether traversal should continue:
           dispatch_data_t rgn  data object representing the region
           size_t offset        logical position of the region in data
           const void *loc      memory location of the region
           size_t size          extent of the region
     The rgn data object is released by the system when the applier block 
returns.  The associated memory location loc is
     valid only as long as rgn has not been deallocated; if loc is needed 
outside of the applier block, the rgn object
     must be retained in the block.
</quote>

Ie. to guarantee that all the `loc`s I get are still valid, I just need to make 
sure that all the `rgn`s I get are retained until writev returns. Easy!

Now I wanted to switch to Swift and unfortunately saw that 
DispatchData.enumerateBytes doesn't give me access to the region so I can't 
retain it :(.

So how can I achieve that, is it guaranteed that it's good enough to retain the 
overall DispatchData? Or in code, is this guaranteed to be correct?

extension DispatchData {
    func writeVector(fileDescriptor: Int32) -> ssize_t {
        var iobufs: [iovec] = []
        self.enumerateBytes { (buf, _, _) in
            var iobuf = iovec()
            iobuf.iov_base = UnsafeMutableRawPointer(mutating: buf.baseAddress!)
            iobuf.iov_len = buf.count
            iobufs.append(iobuf)
        }
        return writev(fileDescriptor, iobufs, Int32(iobufs.count))
    }
}


Many thanks,
  Johannes
_______________________________________________
swift-corelibs-dev mailing list
swift-corelibs-dev@swift.org
https://lists.swift.org/mailman/listinfo/swift-corelibs-dev

Reply via email to