• Home
  • Line#
  • Scopes#
  • Navigate#
  • Raw
  • Download
1 // Copyright (c) 2012 The Chromium Authors. All rights reserved.
2 // Use of this source code is governed by a BSD-style license that can be
3 // found in the LICENSE file.
4 
5 #ifndef PPAPI_SHARED_IMPL_ARRAY_WRITER_H_
6 #define PPAPI_SHARED_IMPL_ARRAY_WRITER_H_
7 
8 #include <string.h>
9 
10 #include <vector>
11 
12 #include "base/memory/ref_counted.h"
13 #include "ppapi/c/pp_array_output.h"
14 #include "ppapi/c/pp_resource.h"
15 #include "ppapi/c/pp_var.h"
16 #include "ppapi/shared_impl/ppapi_shared_export.h"
17 
18 namespace ppapi {
19 
20 class Resource;
21 class Var;
22 
23 // Holds a PP_ArrayWriter and provides helper functions for writing arrays
24 // to it. It also handles 0-initialization of the raw C struct and attempts
25 // to prevent you from writing the array twice.
26 class PPAPI_SHARED_EXPORT ArrayWriter {
27  public:
28   ArrayWriter();  // Creates an is_null() object
29   ArrayWriter(const PP_ArrayOutput& output);
30   ~ArrayWriter();
31 
is_valid()32   bool is_valid() const { return !!pp_array_output_.GetDataBuffer; }
is_null()33   bool is_null() const { return !is_valid(); }
34 
set_pp_array_output(const PP_ArrayOutput & output)35   void set_pp_array_output(const PP_ArrayOutput& output) {
36     pp_array_output_ = output;
37   }
38 
39   // Sets the array output back to its is_null() state.
40   void Reset();
41 
42   // StoreArray() and StoreVector() copy the given array/vector of data to the
43   // plugin output array.
44   //
45   // Returns true on success, false if the plugin reported allocation failure.
46   // In either case, the object will become is_null() immediately after the
47   // call since one output function should only be issued once.
48   //
49   // THIS IS DESIGNED FOR POD ONLY. For the case of resources, for example, we
50   // want to transfer a reference only on success. Likewise, if you have a
51   // structure of PP_Vars or a struct that contains a PP_Resource, we need to
52   // make sure that the right thing happens with the ref on success and failure.
53   template <typename T>
StoreArray(const T * input,uint32_t count)54   bool StoreArray(const T* input, uint32_t count) {
55     // Always call the alloc function, even on 0 array size.
56     void* dest = pp_array_output_.GetDataBuffer(
57         pp_array_output_.user_data, count, sizeof(T));
58 
59     // Regardless of success, we clear the output to prevent future calls on
60     // this same output object.
61     Reset();
62 
63     if (count == 0)
64       return true;  // Allow plugin to return NULL on 0 elements.
65     if (!dest)
66       return false;
67 
68     if (input)
69       memcpy(dest, input, sizeof(T) * count);
70     return true;
71   }
72 
73   // Copies the given array/vector of data to the plugin output array.  See
74   // comment of StoreArray() for detail.
75   template <typename T>
StoreVector(const std::vector<T> & input)76   bool StoreVector(const std::vector<T>& input) {
77     return StoreArray(input.size() ? &input[0] : NULL, input.size());
78   }
79 
80   // Stores the given vector of resources as PP_Resources to the output vector,
81   // adding one reference to each.
82   //
83   // On failure this returns false, nothing will be copied, and the resource
84   // refcounts will be unchanged. In either case, the object will become
85   // is_null() immediately after the call since one output function should only
86   // be issued once.
87   //
88   // Note: potentially this could be a template in case you have a vector of
89   // FileRef objects, for example. However, this saves code since there's only
90   // one instantiation and is sufficient for now.
91   bool StoreResourceVector(const std::vector<scoped_refptr<Resource> >& input);
92 
93   // Like the above version but takes an array of AddRef'ed PP_Resources. On
94   // storage failure, this will release each resource.
95   bool StoreResourceVector(const std::vector<PP_Resource>& input);
96 
97   // Stores the given vector of vars as PP_Vars to the output vector,
98   // adding one reference to each.
99   //
100   // On failure this returns false, nothing will be copied, and the var
101   // refcounts will be unchanged. In either case, the object will become
102   // is_null() immediately after the call since one output function should only
103   // be issued once.
104   bool StoreVarVector(const std::vector<scoped_refptr<Var> >& input);
105 
106   // Like the above version but takes an array of AddRef'ed PP_Vars. On
107   // storage failure, this will release each var.
108   bool StoreVarVector(const std::vector<PP_Var>& input);
109 
110  private:
111   PP_ArrayOutput pp_array_output_;
112 
113   DISALLOW_COPY_AND_ASSIGN(ArrayWriter);
114 };
115 
116 }  // namespace ppapi
117 
118 #endif  // PPAPI_SHARED_IMPL_ARRAY_WRITER_H_
119