Sha256: 8e972865b89b103206effdf5cd61c31bd5b9ac75d7bdb5a211183c37dc0628fc

Contents?: true

Size: 1.57 KB

Versions: 3

Compression:

Stored size: 1.57 KB

Contents

# frozen_string_literal: true

require "ffi"

module GDAL
  # A wrapper for the way GDAL does key/value pair options for methods.
  class Options
    # Shortcut for if you just want to build the options and get the pointer to
    # them.
    #
    # @param hash [Hash]
    # @return [FFI::MemoryPointer, nil]
    def self.pointer(hash)
      return if hash.empty?

      options_ptr = FFI::MemoryPointer.new(:pointer)
      options_ptr.autorelease = false

      hash.each do |key, value|
        # Note, we started off with a MemoryPointer above, but this returns a
        # new FFI::Pointer.
        options_ptr = FFI::CPL::String.CSLAddNameValue(options_ptr, key.to_s.upcase, value.to_s)
        options_ptr.autorelease = false
      end

      FFI::AutoPointer.new(options_ptr, Options.method(:release))
    end

    # Takes a GDAL options pointer and turns it into a Ruby Hash.
    #
    # @param pointer [FFI::Pointer]
    # @return [Hash]
    def self.to_hash(pointer)
      # Docs say passing a null pointer here is ok; will result in 0.
      count = FFI::CPL::String.CSLCount(pointer)

      count.times.each_with_object({}) do |i, o|
        # Docs say that the pointer returned from CSLGetField shouldn't be freed.
        key_and_value, ptr = FFI::CPL::String.CSLGetField(pointer, i)
        ptr.autorelease = false

        key, value = key_and_value.split("=")
        o[key.downcase.to_sym] = value
      end
    end

    # @param pointer [FFI::Pointer]
    def self.release(pointer)
      return if pointer.nil? || pointer.null?

      FFI::CPL::String.CSLDestroy(pointer)
    end
  end
end

Version data entries

3 entries across 3 versions & 1 rubygems

Version Path
ffi-gdal-1.1.0 lib/gdal/options.rb
ffi-gdal-1.0.4 lib/gdal/options.rb
ffi-gdal-1.0.3 lib/gdal/options.rb