// Copyright (c) 2012 The Chromium Authors. All rights reserved.
// Use of this source code is governed by a BSD-style license that can be
// found in the LICENSE file.
#ifndef PPAPI_SHARED_IMPL_ARRAY_WRITER_H_
#define PPAPI_SHARED_IMPL_ARRAY_WRITER_H_
#include <string.h>
#include <vector>
#include "base/memory/ref_counted.h"
#include "ppapi/c/pp_array_output.h"
#include "ppapi/c/pp_resource.h"
#include "ppapi/c/pp_var.h"
#include "ppapi/shared_impl/ppapi_shared_export.h"
namespace ppapi {
class Resource;
class Var;
// Holds a PP_ArrayWriter and provides helper functions for writing arrays
// to it. It also handles 0-initialization of the raw C struct and attempts
// to prevent you from writing the array twice.
class PPAPI_SHARED_EXPORT ArrayWriter {
public:
ArrayWriter(); // Creates an is_null() object
ArrayWriter(const PP_ArrayOutput& output);
~ArrayWriter();
bool is_valid() const { return !!pp_array_output_.GetDataBuffer; }
bool is_null() const { return !is_valid(); }
void set_pp_array_output(const PP_ArrayOutput& output) {
pp_array_output_ = output;
}
// Sets the array output back to its is_null() state.
void Reset();
// StoreArray() and StoreVector() copy the given array/vector of data to the
// plugin output array.
//
// Returns true on success, false if the plugin reported allocation failure.
// In either case, the object will become is_null() immediately after the
// call since one output function should only be issued once.
//
// THIS IS DESIGNED FOR POD ONLY. For the case of resources, for example, we
// want to transfer a reference only on success. Likewise, if you have a
// structure of PP_Vars or a struct that contains a PP_Resource, we need to
// make sure that the right thing happens with the ref on success and failure.
template <typename T>
bool StoreArray(const T* input, uint32_t count) {
// Always call the alloc function, even on 0 array size.
void* dest = pp_array_output_.GetDataBuffer(
pp_array_output_.user_data,
count,
sizeof(T));
// Regardless of success, we clear the output to prevent future calls on
// this same output object.
Reset();
if (count == 0)
return true; // Allow plugin to return NULL on 0 elements.
if (!dest)
return false;
if (input)
memcpy(dest, input, sizeof(T) * count);
return true;
}
// Copies the given array/vector of data to the plugin output array. See
// comment of StoreArray() for detail.
template<typename T>
bool StoreVector(const std::vector<T>& input) {
return StoreArray(input.size() ? &input[0] : NULL, input.size());
}
// Stores the given vector of resources as PP_Resources to the output vector,
// adding one reference to each.
//
// On failure this returns false, nothing will be copied, and the resource
// refcounts will be unchanged. In either case, the object will become
// is_null() immediately after the call since one output function should only
// be issued once.
//
// Note: potentially this could be a template in case you have a vector of
// FileRef objects, for example. However, this saves code since there's only
// one instantiation and is sufficient for now.
bool StoreResourceVector(const std::vector< scoped_refptr<Resource> >& input);
// Like the above version but takes an array of AddRef'ed PP_Resources. On
// storage failure, this will release each resource.
bool StoreResourceVector(const std::vector<PP_Resource>& input);
// Stores the given vector of vars as PP_Vars to the output vector,
// adding one reference to each.
//
// On failure this returns false, nothing will be copied, and the var
// refcounts will be unchanged. In either case, the object will become
// is_null() immediately after the call since one output function should only
// be issued once.
bool StoreVarVector(const std::vector< scoped_refptr<Var> >& input);
// Like the above version but takes an array of AddRef'ed PP_Vars. On
// storage failure, this will release each var.
bool StoreVarVector(const std::vector<PP_Var>& input);
private:
PP_ArrayOutput pp_array_output_;
DISALLOW_COPY_AND_ASSIGN(ArrayWriter);
};
} // namespace ppapi
#endif // PPAPI_SHARED_IMPL_ARRAY_WRITER_H_