dwrote 0.1.3

Lightweight binding to DirectWrite.
/* This Source Code Form is subject to the terms of the Mozilla Public
 * License, v. 2.0. If a copy of the MPL was not distributed with this
 * file, You can obtain one at http://mozilla.org/MPL/2.0/. */

use std::ops::{Deref, DerefMut};
use std::ptr;
use winapi::{IUnknown, REFIID};

#[derive(Debug)]
pub struct ComPtr<T> {
    ptr: *mut T,
}

impl<T> ComPtr<T> {
    pub fn new() -> Self {
        ComPtr { ptr: ptr::null_mut() }
    }

    pub fn already_addrefed(ptr: *mut T) -> Self {
        ComPtr { ptr: ptr }
    }

    pub fn getter_addrefs<Q>(&mut self) -> *mut *mut Q {
        self.release();
        return &mut self.ptr as *mut *mut _ as *mut *mut Q;
    }

    pub fn as_ptr(&self) -> *mut T {
        self.ptr
    }

    pub fn query_interface<Q>(&self, _: REFIID) -> Option<ComPtr<Q>> {
        panic!("Here's where the code goes when you need it");
    }

    pub fn addref(&self) {
        unsafe {
            assert!(!self.ptr.is_null());
            (*(self.ptr as *mut IUnknown)).AddRef();
        }
    }

    pub fn release(&self) {
        unsafe {
            if !self.ptr.is_null() {
                (*(self.ptr as *mut IUnknown)).Release();
            }
        }
    }

    pub fn forget(&mut self) -> *mut T {
        let ptr = self.ptr;
        self.ptr = ptr::null_mut();
        ptr
    }
}

impl<T> Clone for ComPtr<T> {
    fn clone(&self) -> Self {
        if !self.ptr.is_null() {
            self.addref();
        }
        ComPtr { ptr: self.ptr }
    }
}

impl<T> Deref for ComPtr<T> {
    type Target = T;
    fn deref(&self) -> &T {
        assert!(!self.ptr.is_null());
        unsafe { &mut *self.ptr }
    }
}

impl<T> DerefMut for ComPtr<T> {
    fn deref_mut(&mut self) -> &mut T {
        assert!(!self.ptr.is_null());
        unsafe { &mut *self.ptr }
    }
}

impl<T> PartialEq for ComPtr<T> {
    fn eq(&self, other: &ComPtr<T>) -> bool {
        self.ptr == other.ptr
    }
}

impl<T> Drop for ComPtr<T> {
    fn drop(&mut self) {
        self.release();
    }
}