xref: /aosp_15_r20/hardware/interfaces/camera/common/default/Exif.cpp (revision 4d7e907c777eeecc4c5bd7cf640a754fac206ff7)
1*4d7e907cSAndroid Build Coastguard Worker /*
2*4d7e907cSAndroid Build Coastguard Worker  * Copyright (C) 2018 The Android Open Source Project
3*4d7e907cSAndroid Build Coastguard Worker  *
4*4d7e907cSAndroid Build Coastguard Worker  * Licensed under the Apache License, Version 2.0 (the "License");
5*4d7e907cSAndroid Build Coastguard Worker  * you may not use this file except in compliance with the License.
6*4d7e907cSAndroid Build Coastguard Worker  * You may obtain a copy of the License at
7*4d7e907cSAndroid Build Coastguard Worker  *
8*4d7e907cSAndroid Build Coastguard Worker  *      http://www.apache.org/licenses/LICENSE-2.0
9*4d7e907cSAndroid Build Coastguard Worker  *
10*4d7e907cSAndroid Build Coastguard Worker  * Unless required by applicable law or agreed to in writing, software
11*4d7e907cSAndroid Build Coastguard Worker  * distributed under the License is distributed on an "AS IS" BASIS,
12*4d7e907cSAndroid Build Coastguard Worker  * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
13*4d7e907cSAndroid Build Coastguard Worker  * See the License for the specific language governing permissions and
14*4d7e907cSAndroid Build Coastguard Worker  * limitations under the License.
15*4d7e907cSAndroid Build Coastguard Worker  */
16*4d7e907cSAndroid Build Coastguard Worker 
17*4d7e907cSAndroid Build Coastguard Worker #define LOG_TAG "CamComm1.0-Exif"
18*4d7e907cSAndroid Build Coastguard Worker #define ATRACE_TAG ATRACE_TAG_CAMERA
19*4d7e907cSAndroid Build Coastguard Worker // #define LOG_NDEBUG 0
20*4d7e907cSAndroid Build Coastguard Worker 
21*4d7e907cSAndroid Build Coastguard Worker #include <android/log.h>
22*4d7e907cSAndroid Build Coastguard Worker 
23*4d7e907cSAndroid Build Coastguard Worker #include <inttypes.h>
24*4d7e907cSAndroid Build Coastguard Worker #include <math.h>
25*4d7e907cSAndroid Build Coastguard Worker #include <stdint.h>
26*4d7e907cSAndroid Build Coastguard Worker #include <string>
27*4d7e907cSAndroid Build Coastguard Worker #include <vector>
28*4d7e907cSAndroid Build Coastguard Worker 
29*4d7e907cSAndroid Build Coastguard Worker #include "Exif.h"
30*4d7e907cSAndroid Build Coastguard Worker 
31*4d7e907cSAndroid Build Coastguard Worker extern "C" {
32*4d7e907cSAndroid Build Coastguard Worker #include <libexif/exif-data.h>
33*4d7e907cSAndroid Build Coastguard Worker }
34*4d7e907cSAndroid Build Coastguard Worker 
35*4d7e907cSAndroid Build Coastguard Worker namespace std {
36*4d7e907cSAndroid Build Coastguard Worker 
37*4d7e907cSAndroid Build Coastguard Worker template <>
38*4d7e907cSAndroid Build Coastguard Worker struct default_delete<ExifEntry> {
operator ()std::default_delete39*4d7e907cSAndroid Build Coastguard Worker     inline void operator()(ExifEntry* entry) const { exif_entry_unref(entry); }
40*4d7e907cSAndroid Build Coastguard Worker };
41*4d7e907cSAndroid Build Coastguard Worker 
42*4d7e907cSAndroid Build Coastguard Worker }  // namespace std
43*4d7e907cSAndroid Build Coastguard Worker 
44*4d7e907cSAndroid Build Coastguard Worker namespace android {
45*4d7e907cSAndroid Build Coastguard Worker namespace hardware {
46*4d7e907cSAndroid Build Coastguard Worker namespace camera {
47*4d7e907cSAndroid Build Coastguard Worker namespace common {
48*4d7e907cSAndroid Build Coastguard Worker namespace helper {
49*4d7e907cSAndroid Build Coastguard Worker 
50*4d7e907cSAndroid Build Coastguard Worker class ExifUtilsImpl : public ExifUtils {
51*4d7e907cSAndroid Build Coastguard Worker   public:
52*4d7e907cSAndroid Build Coastguard Worker     ExifUtilsImpl();
53*4d7e907cSAndroid Build Coastguard Worker 
54*4d7e907cSAndroid Build Coastguard Worker     virtual ~ExifUtilsImpl();
55*4d7e907cSAndroid Build Coastguard Worker 
56*4d7e907cSAndroid Build Coastguard Worker     // Initialize() can be called multiple times. The setting of Exif tags will be
57*4d7e907cSAndroid Build Coastguard Worker     // cleared.
58*4d7e907cSAndroid Build Coastguard Worker     virtual bool initialize();
59*4d7e907cSAndroid Build Coastguard Worker 
60*4d7e907cSAndroid Build Coastguard Worker     // set all known fields from a metadata structure
61*4d7e907cSAndroid Build Coastguard Worker     virtual bool setFromMetadata(const CameraMetadata& metadata, const size_t imageWidth,
62*4d7e907cSAndroid Build Coastguard Worker                                  const size_t imageHeight);
63*4d7e907cSAndroid Build Coastguard Worker 
64*4d7e907cSAndroid Build Coastguard Worker     // sets the len aperture.
65*4d7e907cSAndroid Build Coastguard Worker     // Returns false if memory allocation fails.
66*4d7e907cSAndroid Build Coastguard Worker     virtual bool setAperture(uint32_t numerator, uint32_t denominator);
67*4d7e907cSAndroid Build Coastguard Worker 
68*4d7e907cSAndroid Build Coastguard Worker     // sets the value of brightness.
69*4d7e907cSAndroid Build Coastguard Worker     // Returns false if memory allocation fails.
70*4d7e907cSAndroid Build Coastguard Worker     virtual bool setBrightness(int32_t numerator, int32_t denominator);
71*4d7e907cSAndroid Build Coastguard Worker 
72*4d7e907cSAndroid Build Coastguard Worker     // sets the color space.
73*4d7e907cSAndroid Build Coastguard Worker     // Returns false if memory allocation fails.
74*4d7e907cSAndroid Build Coastguard Worker     virtual bool setColorSpace(uint16_t color_space);
75*4d7e907cSAndroid Build Coastguard Worker 
76*4d7e907cSAndroid Build Coastguard Worker     // sets the information to compressed data.
77*4d7e907cSAndroid Build Coastguard Worker     // Returns false if memory allocation fails.
78*4d7e907cSAndroid Build Coastguard Worker     virtual bool setComponentsConfiguration(const std::string& components_configuration);
79*4d7e907cSAndroid Build Coastguard Worker 
80*4d7e907cSAndroid Build Coastguard Worker     // sets the compression scheme used for the image data.
81*4d7e907cSAndroid Build Coastguard Worker     // Returns false if memory allocation fails.
82*4d7e907cSAndroid Build Coastguard Worker     virtual bool setCompression(uint16_t compression);
83*4d7e907cSAndroid Build Coastguard Worker 
84*4d7e907cSAndroid Build Coastguard Worker     // sets image contrast.
85*4d7e907cSAndroid Build Coastguard Worker     // Returns false if memory allocation fails.
86*4d7e907cSAndroid Build Coastguard Worker     virtual bool setContrast(uint16_t contrast);
87*4d7e907cSAndroid Build Coastguard Worker 
88*4d7e907cSAndroid Build Coastguard Worker     // sets the date and time of image last modified. It takes local time. The
89*4d7e907cSAndroid Build Coastguard Worker     // name of the tag is DateTime in IFD0.
90*4d7e907cSAndroid Build Coastguard Worker     // Returns false if memory allocation fails.
91*4d7e907cSAndroid Build Coastguard Worker     virtual bool setDateTime(const struct tm& t);
92*4d7e907cSAndroid Build Coastguard Worker 
93*4d7e907cSAndroid Build Coastguard Worker     // sets the image description.
94*4d7e907cSAndroid Build Coastguard Worker     // Returns false if memory allocation fails.
95*4d7e907cSAndroid Build Coastguard Worker     virtual bool setDescription(const std::string& description);
96*4d7e907cSAndroid Build Coastguard Worker 
97*4d7e907cSAndroid Build Coastguard Worker     // sets the digital zoom ratio. If the numerator is 0, it means digital zoom
98*4d7e907cSAndroid Build Coastguard Worker     // was not used.
99*4d7e907cSAndroid Build Coastguard Worker     // Returns false if memory allocation fails.
100*4d7e907cSAndroid Build Coastguard Worker     virtual bool setDigitalZoomRatio(uint32_t numerator, uint32_t denominator);
101*4d7e907cSAndroid Build Coastguard Worker 
102*4d7e907cSAndroid Build Coastguard Worker     // sets the exposure bias.
103*4d7e907cSAndroid Build Coastguard Worker     // Returns false if memory allocation fails.
104*4d7e907cSAndroid Build Coastguard Worker     virtual bool setExposureBias(int32_t numerator, int32_t denominator);
105*4d7e907cSAndroid Build Coastguard Worker 
106*4d7e907cSAndroid Build Coastguard Worker     // sets the exposure mode set when the image was shot.
107*4d7e907cSAndroid Build Coastguard Worker     // Returns false if memory allocation fails.
108*4d7e907cSAndroid Build Coastguard Worker     virtual bool setExposureMode(uint16_t exposure_mode);
109*4d7e907cSAndroid Build Coastguard Worker 
110*4d7e907cSAndroid Build Coastguard Worker     // sets the program used by the camera to set exposure when the picture is
111*4d7e907cSAndroid Build Coastguard Worker     // taken.
112*4d7e907cSAndroid Build Coastguard Worker     // Returns false if memory allocation fails.
113*4d7e907cSAndroid Build Coastguard Worker     virtual bool setExposureProgram(uint16_t exposure_program);
114*4d7e907cSAndroid Build Coastguard Worker 
115*4d7e907cSAndroid Build Coastguard Worker     // sets the exposure time, given in seconds.
116*4d7e907cSAndroid Build Coastguard Worker     // Returns false if memory allocation fails.
117*4d7e907cSAndroid Build Coastguard Worker     virtual bool setExposureTime(uint32_t numerator, uint32_t denominator);
118*4d7e907cSAndroid Build Coastguard Worker 
119*4d7e907cSAndroid Build Coastguard Worker     // sets the status of flash.
120*4d7e907cSAndroid Build Coastguard Worker     // Returns false if memory allocation fails.
121*4d7e907cSAndroid Build Coastguard Worker     virtual bool setFlash(uint16_t flash);
122*4d7e907cSAndroid Build Coastguard Worker 
123*4d7e907cSAndroid Build Coastguard Worker     // sets the F number.
124*4d7e907cSAndroid Build Coastguard Worker     // Returns false if memory allocation fails.
125*4d7e907cSAndroid Build Coastguard Worker     virtual bool setFNumber(uint32_t numerator, uint32_t denominator);
126*4d7e907cSAndroid Build Coastguard Worker 
127*4d7e907cSAndroid Build Coastguard Worker     // sets the focal length of lens used to take the image in millimeters.
128*4d7e907cSAndroid Build Coastguard Worker     // Returns false if memory allocation fails.
129*4d7e907cSAndroid Build Coastguard Worker     virtual bool setFocalLength(uint32_t numerator, uint32_t denominator);
130*4d7e907cSAndroid Build Coastguard Worker 
131*4d7e907cSAndroid Build Coastguard Worker     // sets the degree of overall image gain adjustment.
132*4d7e907cSAndroid Build Coastguard Worker     // Returns false if memory allocation fails.
133*4d7e907cSAndroid Build Coastguard Worker     virtual bool setGainControl(uint16_t gain_control);
134*4d7e907cSAndroid Build Coastguard Worker 
135*4d7e907cSAndroid Build Coastguard Worker     // sets the altitude in meters.
136*4d7e907cSAndroid Build Coastguard Worker     // Returns false if memory allocation fails.
137*4d7e907cSAndroid Build Coastguard Worker     virtual bool setGpsAltitude(double altitude);
138*4d7e907cSAndroid Build Coastguard Worker 
139*4d7e907cSAndroid Build Coastguard Worker     // sets the latitude with degrees minutes seconds format.
140*4d7e907cSAndroid Build Coastguard Worker     // Returns false if memory allocation fails.
141*4d7e907cSAndroid Build Coastguard Worker     virtual bool setGpsLatitude(double latitude);
142*4d7e907cSAndroid Build Coastguard Worker 
143*4d7e907cSAndroid Build Coastguard Worker     // sets the longitude with degrees minutes seconds format.
144*4d7e907cSAndroid Build Coastguard Worker     // Returns false if memory allocation fails.
145*4d7e907cSAndroid Build Coastguard Worker     virtual bool setGpsLongitude(double longitude);
146*4d7e907cSAndroid Build Coastguard Worker 
147*4d7e907cSAndroid Build Coastguard Worker     // sets GPS processing method.
148*4d7e907cSAndroid Build Coastguard Worker     // Returns false if memory allocation fails.
149*4d7e907cSAndroid Build Coastguard Worker     virtual bool setGpsProcessingMethod(const std::string& method);
150*4d7e907cSAndroid Build Coastguard Worker 
151*4d7e907cSAndroid Build Coastguard Worker     // sets GPS date stamp and time stamp (atomic clock). It takes UTC time.
152*4d7e907cSAndroid Build Coastguard Worker     // Returns false if memory allocation fails.
153*4d7e907cSAndroid Build Coastguard Worker     virtual bool setGpsTimestamp(const struct tm& t);
154*4d7e907cSAndroid Build Coastguard Worker 
155*4d7e907cSAndroid Build Coastguard Worker     // sets the length (number of rows) of main image.
156*4d7e907cSAndroid Build Coastguard Worker     // Returns false if memory allocation fails.
157*4d7e907cSAndroid Build Coastguard Worker     virtual bool setImageHeight(uint32_t length);
158*4d7e907cSAndroid Build Coastguard Worker 
159*4d7e907cSAndroid Build Coastguard Worker     // sets the width (number of columes) of main image.
160*4d7e907cSAndroid Build Coastguard Worker     // Returns false if memory allocation fails.
161*4d7e907cSAndroid Build Coastguard Worker     virtual bool setImageWidth(uint32_t width);
162*4d7e907cSAndroid Build Coastguard Worker 
163*4d7e907cSAndroid Build Coastguard Worker     // sets the ISO speed.
164*4d7e907cSAndroid Build Coastguard Worker     // Returns false if memory allocation fails.
165*4d7e907cSAndroid Build Coastguard Worker     virtual bool setIsoSpeedRating(uint16_t iso_speed_ratings);
166*4d7e907cSAndroid Build Coastguard Worker 
167*4d7e907cSAndroid Build Coastguard Worker     // sets the kind of light source.
168*4d7e907cSAndroid Build Coastguard Worker     // Returns false if memory allocation fails.
169*4d7e907cSAndroid Build Coastguard Worker     virtual bool setLightSource(uint16_t light_source);
170*4d7e907cSAndroid Build Coastguard Worker 
171*4d7e907cSAndroid Build Coastguard Worker     // sets the smallest F number of the lens.
172*4d7e907cSAndroid Build Coastguard Worker     // Returns false if memory allocation fails.
173*4d7e907cSAndroid Build Coastguard Worker     virtual bool setMaxAperture(uint32_t numerator, uint32_t denominator);
174*4d7e907cSAndroid Build Coastguard Worker 
175*4d7e907cSAndroid Build Coastguard Worker     // sets the metering mode.
176*4d7e907cSAndroid Build Coastguard Worker     // Returns false if memory allocation fails.
177*4d7e907cSAndroid Build Coastguard Worker     virtual bool setMeteringMode(uint16_t metering_mode);
178*4d7e907cSAndroid Build Coastguard Worker 
179*4d7e907cSAndroid Build Coastguard Worker     // sets image orientation.
180*4d7e907cSAndroid Build Coastguard Worker     // Returns false if memory allocation fails.
181*4d7e907cSAndroid Build Coastguard Worker     virtual bool setOrientation(uint16_t orientation);
182*4d7e907cSAndroid Build Coastguard Worker 
183*4d7e907cSAndroid Build Coastguard Worker     // sets the unit for measuring XResolution and YResolution.
184*4d7e907cSAndroid Build Coastguard Worker     // Returns false if memory allocation fails.
185*4d7e907cSAndroid Build Coastguard Worker     virtual bool setResolutionUnit(uint16_t resolution_unit);
186*4d7e907cSAndroid Build Coastguard Worker 
187*4d7e907cSAndroid Build Coastguard Worker     // sets image saturation.
188*4d7e907cSAndroid Build Coastguard Worker     // Returns false if memory allocation fails.
189*4d7e907cSAndroid Build Coastguard Worker     virtual bool setSaturation(uint16_t saturation);
190*4d7e907cSAndroid Build Coastguard Worker 
191*4d7e907cSAndroid Build Coastguard Worker     // sets the type of scene that was shot.
192*4d7e907cSAndroid Build Coastguard Worker     // Returns false if memory allocation fails.
193*4d7e907cSAndroid Build Coastguard Worker     virtual bool setSceneCaptureType(uint16_t type);
194*4d7e907cSAndroid Build Coastguard Worker 
195*4d7e907cSAndroid Build Coastguard Worker     // sets image sharpness.
196*4d7e907cSAndroid Build Coastguard Worker     // Returns false if memory allocation fails.
197*4d7e907cSAndroid Build Coastguard Worker     virtual bool setSharpness(uint16_t sharpness);
198*4d7e907cSAndroid Build Coastguard Worker 
199*4d7e907cSAndroid Build Coastguard Worker     // sets the shutter speed.
200*4d7e907cSAndroid Build Coastguard Worker     // Returns false if memory allocation fails.
201*4d7e907cSAndroid Build Coastguard Worker     virtual bool setShutterSpeed(int32_t numerator, int32_t denominator);
202*4d7e907cSAndroid Build Coastguard Worker 
203*4d7e907cSAndroid Build Coastguard Worker     // sets the distance to the subject, given in meters.
204*4d7e907cSAndroid Build Coastguard Worker     // Returns false if memory allocation fails.
205*4d7e907cSAndroid Build Coastguard Worker     virtual bool setSubjectDistance(uint32_t numerator, uint32_t denominator);
206*4d7e907cSAndroid Build Coastguard Worker 
207*4d7e907cSAndroid Build Coastguard Worker     // sets the fractions of seconds for the <DateTime> tag.
208*4d7e907cSAndroid Build Coastguard Worker     // Returns false if memory allocation fails.
209*4d7e907cSAndroid Build Coastguard Worker     virtual bool setSubsecTime(const std::string& subsec_time);
210*4d7e907cSAndroid Build Coastguard Worker 
211*4d7e907cSAndroid Build Coastguard Worker     // sets the white balance mode set when the image was shot.
212*4d7e907cSAndroid Build Coastguard Worker     // Returns false if memory allocation fails.
213*4d7e907cSAndroid Build Coastguard Worker     virtual bool setWhiteBalance(uint16_t white_balance);
214*4d7e907cSAndroid Build Coastguard Worker 
215*4d7e907cSAndroid Build Coastguard Worker     // sets the number of pixels per resolution unit in the image width.
216*4d7e907cSAndroid Build Coastguard Worker     // Returns false if memory allocation fails.
217*4d7e907cSAndroid Build Coastguard Worker     virtual bool setXResolution(uint32_t numerator, uint32_t denominator);
218*4d7e907cSAndroid Build Coastguard Worker 
219*4d7e907cSAndroid Build Coastguard Worker     // sets the position of chrominance components in relation to the luminance
220*4d7e907cSAndroid Build Coastguard Worker     // component.
221*4d7e907cSAndroid Build Coastguard Worker     // Returns false if memory allocation fails.
222*4d7e907cSAndroid Build Coastguard Worker     virtual bool setYCbCrPositioning(uint16_t ycbcr_positioning);
223*4d7e907cSAndroid Build Coastguard Worker 
224*4d7e907cSAndroid Build Coastguard Worker     // sets the number of pixels per resolution unit in the image length.
225*4d7e907cSAndroid Build Coastguard Worker     // Returns false if memory allocation fails.
226*4d7e907cSAndroid Build Coastguard Worker     virtual bool setYResolution(uint32_t numerator, uint32_t denominator);
227*4d7e907cSAndroid Build Coastguard Worker 
228*4d7e907cSAndroid Build Coastguard Worker     // sets the manufacturer of camera.
229*4d7e907cSAndroid Build Coastguard Worker     // Returns false if memory allocation fails.
230*4d7e907cSAndroid Build Coastguard Worker     virtual bool setMake(const std::string& make);
231*4d7e907cSAndroid Build Coastguard Worker 
232*4d7e907cSAndroid Build Coastguard Worker     // sets the model number of camera.
233*4d7e907cSAndroid Build Coastguard Worker     // Returns false if memory allocation fails.
234*4d7e907cSAndroid Build Coastguard Worker     virtual bool setModel(const std::string& model);
235*4d7e907cSAndroid Build Coastguard Worker 
236*4d7e907cSAndroid Build Coastguard Worker     // Generates APP1 segment.
237*4d7e907cSAndroid Build Coastguard Worker     // Returns false if generating APP1 segment fails.
238*4d7e907cSAndroid Build Coastguard Worker     virtual bool generateApp1(const void* thumbnail_buffer, uint32_t size);
239*4d7e907cSAndroid Build Coastguard Worker 
240*4d7e907cSAndroid Build Coastguard Worker     // Gets buffer of APP1 segment. This method must be called only after calling
241*4d7e907cSAndroid Build Coastguard Worker     // GenerateAPP1().
242*4d7e907cSAndroid Build Coastguard Worker     virtual const uint8_t* getApp1Buffer();
243*4d7e907cSAndroid Build Coastguard Worker 
244*4d7e907cSAndroid Build Coastguard Worker     // Gets length of APP1 segment. This method must be called only after calling
245*4d7e907cSAndroid Build Coastguard Worker     // GenerateAPP1().
246*4d7e907cSAndroid Build Coastguard Worker     virtual unsigned int getApp1Length();
247*4d7e907cSAndroid Build Coastguard Worker 
248*4d7e907cSAndroid Build Coastguard Worker   protected:
249*4d7e907cSAndroid Build Coastguard Worker     // sets the version of this standard supported.
250*4d7e907cSAndroid Build Coastguard Worker     // Returns false if memory allocation fails.
251*4d7e907cSAndroid Build Coastguard Worker     virtual bool setExifVersion(const std::string& exif_version);
252*4d7e907cSAndroid Build Coastguard Worker 
253*4d7e907cSAndroid Build Coastguard Worker     // Resets the pointers and memories.
254*4d7e907cSAndroid Build Coastguard Worker     virtual void reset();
255*4d7e907cSAndroid Build Coastguard Worker 
256*4d7e907cSAndroid Build Coastguard Worker     // Adds a variable length tag to |exif_data_|. It will remove the original one
257*4d7e907cSAndroid Build Coastguard Worker     // if the tag exists.
258*4d7e907cSAndroid Build Coastguard Worker     // Returns the entry of the tag. The reference count of returned ExifEntry is
259*4d7e907cSAndroid Build Coastguard Worker     // two.
260*4d7e907cSAndroid Build Coastguard Worker     virtual std::unique_ptr<ExifEntry> addVariableLengthEntry(ExifIfd ifd, ExifTag tag,
261*4d7e907cSAndroid Build Coastguard Worker                                                               ExifFormat format,
262*4d7e907cSAndroid Build Coastguard Worker                                                               uint64_t components,
263*4d7e907cSAndroid Build Coastguard Worker                                                               unsigned int size);
264*4d7e907cSAndroid Build Coastguard Worker 
265*4d7e907cSAndroid Build Coastguard Worker     // Adds a entry of |tag| in |exif_data_|. It won't remove the original one if
266*4d7e907cSAndroid Build Coastguard Worker     // the tag exists.
267*4d7e907cSAndroid Build Coastguard Worker     // Returns the entry of the tag. It adds one reference count to returned
268*4d7e907cSAndroid Build Coastguard Worker     // ExifEntry.
269*4d7e907cSAndroid Build Coastguard Worker     virtual std::unique_ptr<ExifEntry> addEntry(ExifIfd ifd, ExifTag tag);
270*4d7e907cSAndroid Build Coastguard Worker 
271*4d7e907cSAndroid Build Coastguard Worker     // Helpe functions to add exif data with different types.
272*4d7e907cSAndroid Build Coastguard Worker     virtual bool setShort(ExifIfd ifd, ExifTag tag, uint16_t value, const std::string& msg);
273*4d7e907cSAndroid Build Coastguard Worker 
274*4d7e907cSAndroid Build Coastguard Worker     virtual bool setLong(ExifIfd ifd, ExifTag tag, uint32_t value, const std::string& msg);
275*4d7e907cSAndroid Build Coastguard Worker 
276*4d7e907cSAndroid Build Coastguard Worker     virtual bool setRational(ExifIfd ifd, ExifTag tag, uint32_t numerator, uint32_t denominator,
277*4d7e907cSAndroid Build Coastguard Worker                              const std::string& msg);
278*4d7e907cSAndroid Build Coastguard Worker 
279*4d7e907cSAndroid Build Coastguard Worker     virtual bool setSRational(ExifIfd ifd, ExifTag tag, int32_t numerator, int32_t denominator,
280*4d7e907cSAndroid Build Coastguard Worker                               const std::string& msg);
281*4d7e907cSAndroid Build Coastguard Worker 
282*4d7e907cSAndroid Build Coastguard Worker     virtual bool setString(ExifIfd ifd, ExifTag tag, ExifFormat format, const std::string& buffer,
283*4d7e907cSAndroid Build Coastguard Worker                            const std::string& msg);
284*4d7e907cSAndroid Build Coastguard Worker 
285*4d7e907cSAndroid Build Coastguard Worker     // Destroys the buffer of APP1 segment if exists.
286*4d7e907cSAndroid Build Coastguard Worker     virtual void destroyApp1();
287*4d7e907cSAndroid Build Coastguard Worker 
288*4d7e907cSAndroid Build Coastguard Worker     // The Exif data (APP1). Owned by this class.
289*4d7e907cSAndroid Build Coastguard Worker     ExifData* exif_data_;
290*4d7e907cSAndroid Build Coastguard Worker     // The raw data of APP1 segment. It's allocated by ExifMem in |exif_data_| but
291*4d7e907cSAndroid Build Coastguard Worker     // owned by this class.
292*4d7e907cSAndroid Build Coastguard Worker     uint8_t* app1_buffer_;
293*4d7e907cSAndroid Build Coastguard Worker     // The length of |app1_buffer_|.
294*4d7e907cSAndroid Build Coastguard Worker     unsigned int app1_length_;
295*4d7e907cSAndroid Build Coastguard Worker };
296*4d7e907cSAndroid Build Coastguard Worker 
297*4d7e907cSAndroid Build Coastguard Worker #define SET_SHORT(ifd, tag, value)                                  \
298*4d7e907cSAndroid Build Coastguard Worker     do {                                                            \
299*4d7e907cSAndroid Build Coastguard Worker         if (setShort(ifd, tag, value, #tag) == false) return false; \
300*4d7e907cSAndroid Build Coastguard Worker     } while (0);
301*4d7e907cSAndroid Build Coastguard Worker 
302*4d7e907cSAndroid Build Coastguard Worker #define SET_LONG(ifd, tag, value)                                  \
303*4d7e907cSAndroid Build Coastguard Worker     do {                                                           \
304*4d7e907cSAndroid Build Coastguard Worker         if (setLong(ifd, tag, value, #tag) == false) return false; \
305*4d7e907cSAndroid Build Coastguard Worker     } while (0);
306*4d7e907cSAndroid Build Coastguard Worker 
307*4d7e907cSAndroid Build Coastguard Worker #define SET_RATIONAL(ifd, tag, numerator, denominator)                                  \
308*4d7e907cSAndroid Build Coastguard Worker     do {                                                                                \
309*4d7e907cSAndroid Build Coastguard Worker         if (setRational(ifd, tag, numerator, denominator, #tag) == false) return false; \
310*4d7e907cSAndroid Build Coastguard Worker     } while (0);
311*4d7e907cSAndroid Build Coastguard Worker 
312*4d7e907cSAndroid Build Coastguard Worker #define SET_SRATIONAL(ifd, tag, numerator, denominator)                                  \
313*4d7e907cSAndroid Build Coastguard Worker     do {                                                                                 \
314*4d7e907cSAndroid Build Coastguard Worker         if (setSRational(ifd, tag, numerator, denominator, #tag) == false) return false; \
315*4d7e907cSAndroid Build Coastguard Worker     } while (0);
316*4d7e907cSAndroid Build Coastguard Worker 
317*4d7e907cSAndroid Build Coastguard Worker #define SET_STRING(ifd, tag, format, buffer)                                  \
318*4d7e907cSAndroid Build Coastguard Worker     do {                                                                      \
319*4d7e907cSAndroid Build Coastguard Worker         if (setString(ifd, tag, format, buffer, #tag) == false) return false; \
320*4d7e907cSAndroid Build Coastguard Worker     } while (0);
321*4d7e907cSAndroid Build Coastguard Worker 
322*4d7e907cSAndroid Build Coastguard Worker // This comes from the Exif Version 2.2 standard table 6.
323*4d7e907cSAndroid Build Coastguard Worker const char gExifAsciiPrefix[] = {0x41, 0x53, 0x43, 0x49, 0x49, 0x0, 0x0, 0x0};
324*4d7e907cSAndroid Build Coastguard Worker 
setLatitudeOrLongitudeData(unsigned char * data,double num)325*4d7e907cSAndroid Build Coastguard Worker static void setLatitudeOrLongitudeData(unsigned char* data, double num) {
326*4d7e907cSAndroid Build Coastguard Worker     // Take the integer part of |num|.
327*4d7e907cSAndroid Build Coastguard Worker     ExifLong degrees = static_cast<ExifLong>(num);
328*4d7e907cSAndroid Build Coastguard Worker     ExifLong minutes = static_cast<ExifLong>(60 * (num - degrees));
329*4d7e907cSAndroid Build Coastguard Worker     ExifLong microseconds = static_cast<ExifLong>(3600000000u * (num - degrees - minutes / 60.0));
330*4d7e907cSAndroid Build Coastguard Worker     exif_set_rational(data, EXIF_BYTE_ORDER_INTEL, {degrees, 1});
331*4d7e907cSAndroid Build Coastguard Worker     exif_set_rational(data + sizeof(ExifRational), EXIF_BYTE_ORDER_INTEL, {minutes, 1});
332*4d7e907cSAndroid Build Coastguard Worker     exif_set_rational(data + 2 * sizeof(ExifRational), EXIF_BYTE_ORDER_INTEL,
333*4d7e907cSAndroid Build Coastguard Worker                       {microseconds, 1000000});
334*4d7e907cSAndroid Build Coastguard Worker }
335*4d7e907cSAndroid Build Coastguard Worker 
create()336*4d7e907cSAndroid Build Coastguard Worker ExifUtils* ExifUtils::create() {
337*4d7e907cSAndroid Build Coastguard Worker     return new ExifUtilsImpl();
338*4d7e907cSAndroid Build Coastguard Worker }
339*4d7e907cSAndroid Build Coastguard Worker 
~ExifUtils()340*4d7e907cSAndroid Build Coastguard Worker ExifUtils::~ExifUtils() {}
341*4d7e907cSAndroid Build Coastguard Worker 
ExifUtilsImpl()342*4d7e907cSAndroid Build Coastguard Worker ExifUtilsImpl::ExifUtilsImpl() : exif_data_(nullptr), app1_buffer_(nullptr), app1_length_(0) {}
343*4d7e907cSAndroid Build Coastguard Worker 
~ExifUtilsImpl()344*4d7e907cSAndroid Build Coastguard Worker ExifUtilsImpl::~ExifUtilsImpl() {
345*4d7e907cSAndroid Build Coastguard Worker     reset();
346*4d7e907cSAndroid Build Coastguard Worker }
347*4d7e907cSAndroid Build Coastguard Worker 
initialize()348*4d7e907cSAndroid Build Coastguard Worker bool ExifUtilsImpl::initialize() {
349*4d7e907cSAndroid Build Coastguard Worker     reset();
350*4d7e907cSAndroid Build Coastguard Worker     exif_data_ = exif_data_new();
351*4d7e907cSAndroid Build Coastguard Worker     if (exif_data_ == nullptr) {
352*4d7e907cSAndroid Build Coastguard Worker         ALOGE("%s: allocate memory for exif_data_ failed", __FUNCTION__);
353*4d7e907cSAndroid Build Coastguard Worker         return false;
354*4d7e907cSAndroid Build Coastguard Worker     }
355*4d7e907cSAndroid Build Coastguard Worker     // set the image options.
356*4d7e907cSAndroid Build Coastguard Worker     exif_data_set_option(exif_data_, EXIF_DATA_OPTION_FOLLOW_SPECIFICATION);
357*4d7e907cSAndroid Build Coastguard Worker     exif_data_set_data_type(exif_data_, EXIF_DATA_TYPE_COMPRESSED);
358*4d7e907cSAndroid Build Coastguard Worker     exif_data_set_byte_order(exif_data_, EXIF_BYTE_ORDER_INTEL);
359*4d7e907cSAndroid Build Coastguard Worker 
360*4d7e907cSAndroid Build Coastguard Worker     // set exif version to 2.2.
361*4d7e907cSAndroid Build Coastguard Worker     if (!setExifVersion("0220")) {
362*4d7e907cSAndroid Build Coastguard Worker         return false;
363*4d7e907cSAndroid Build Coastguard Worker     }
364*4d7e907cSAndroid Build Coastguard Worker 
365*4d7e907cSAndroid Build Coastguard Worker     return true;
366*4d7e907cSAndroid Build Coastguard Worker }
367*4d7e907cSAndroid Build Coastguard Worker 
setAperture(uint32_t numerator,uint32_t denominator)368*4d7e907cSAndroid Build Coastguard Worker bool ExifUtilsImpl::setAperture(uint32_t numerator, uint32_t denominator) {
369*4d7e907cSAndroid Build Coastguard Worker     SET_RATIONAL(EXIF_IFD_EXIF, EXIF_TAG_APERTURE_VALUE, numerator, denominator);
370*4d7e907cSAndroid Build Coastguard Worker     return true;
371*4d7e907cSAndroid Build Coastguard Worker }
372*4d7e907cSAndroid Build Coastguard Worker 
setBrightness(int32_t numerator,int32_t denominator)373*4d7e907cSAndroid Build Coastguard Worker bool ExifUtilsImpl::setBrightness(int32_t numerator, int32_t denominator) {
374*4d7e907cSAndroid Build Coastguard Worker     SET_SRATIONAL(EXIF_IFD_EXIF, EXIF_TAG_BRIGHTNESS_VALUE, numerator, denominator);
375*4d7e907cSAndroid Build Coastguard Worker     return true;
376*4d7e907cSAndroid Build Coastguard Worker }
377*4d7e907cSAndroid Build Coastguard Worker 
setColorSpace(uint16_t color_space)378*4d7e907cSAndroid Build Coastguard Worker bool ExifUtilsImpl::setColorSpace(uint16_t color_space) {
379*4d7e907cSAndroid Build Coastguard Worker     SET_SHORT(EXIF_IFD_EXIF, EXIF_TAG_COLOR_SPACE, color_space);
380*4d7e907cSAndroid Build Coastguard Worker     return true;
381*4d7e907cSAndroid Build Coastguard Worker }
382*4d7e907cSAndroid Build Coastguard Worker 
setComponentsConfiguration(const std::string & components_configuration)383*4d7e907cSAndroid Build Coastguard Worker bool ExifUtilsImpl::setComponentsConfiguration(const std::string& components_configuration) {
384*4d7e907cSAndroid Build Coastguard Worker     SET_STRING(EXIF_IFD_EXIF, EXIF_TAG_COMPONENTS_CONFIGURATION, EXIF_FORMAT_UNDEFINED,
385*4d7e907cSAndroid Build Coastguard Worker                components_configuration);
386*4d7e907cSAndroid Build Coastguard Worker     return true;
387*4d7e907cSAndroid Build Coastguard Worker }
388*4d7e907cSAndroid Build Coastguard Worker 
setCompression(uint16_t compression)389*4d7e907cSAndroid Build Coastguard Worker bool ExifUtilsImpl::setCompression(uint16_t compression) {
390*4d7e907cSAndroid Build Coastguard Worker     SET_SHORT(EXIF_IFD_0, EXIF_TAG_COMPRESSION, compression);
391*4d7e907cSAndroid Build Coastguard Worker     return true;
392*4d7e907cSAndroid Build Coastguard Worker }
393*4d7e907cSAndroid Build Coastguard Worker 
setContrast(uint16_t contrast)394*4d7e907cSAndroid Build Coastguard Worker bool ExifUtilsImpl::setContrast(uint16_t contrast) {
395*4d7e907cSAndroid Build Coastguard Worker     SET_SHORT(EXIF_IFD_EXIF, EXIF_TAG_CONTRAST, contrast);
396*4d7e907cSAndroid Build Coastguard Worker     return true;
397*4d7e907cSAndroid Build Coastguard Worker }
398*4d7e907cSAndroid Build Coastguard Worker 
setDateTime(const struct tm & t)399*4d7e907cSAndroid Build Coastguard Worker bool ExifUtilsImpl::setDateTime(const struct tm& t) {
400*4d7e907cSAndroid Build Coastguard Worker     // The length is 20 bytes including NULL for termination in Exif standard.
401*4d7e907cSAndroid Build Coastguard Worker     char str[20];
402*4d7e907cSAndroid Build Coastguard Worker     int result = snprintf(str, sizeof(str), "%04i:%02i:%02i %02i:%02i:%02i", t.tm_year + 1900,
403*4d7e907cSAndroid Build Coastguard Worker                           t.tm_mon + 1, t.tm_mday, t.tm_hour, t.tm_min, t.tm_sec);
404*4d7e907cSAndroid Build Coastguard Worker     if (result != sizeof(str) - 1) {
405*4d7e907cSAndroid Build Coastguard Worker         ALOGW("%s: Input time is invalid", __FUNCTION__);
406*4d7e907cSAndroid Build Coastguard Worker         return false;
407*4d7e907cSAndroid Build Coastguard Worker     }
408*4d7e907cSAndroid Build Coastguard Worker     std::string buffer(str);
409*4d7e907cSAndroid Build Coastguard Worker     SET_STRING(EXIF_IFD_0, EXIF_TAG_DATE_TIME, EXIF_FORMAT_ASCII, buffer);
410*4d7e907cSAndroid Build Coastguard Worker     SET_STRING(EXIF_IFD_EXIF, EXIF_TAG_DATE_TIME_ORIGINAL, EXIF_FORMAT_ASCII, buffer);
411*4d7e907cSAndroid Build Coastguard Worker     SET_STRING(EXIF_IFD_EXIF, EXIF_TAG_DATE_TIME_DIGITIZED, EXIF_FORMAT_ASCII, buffer);
412*4d7e907cSAndroid Build Coastguard Worker     return true;
413*4d7e907cSAndroid Build Coastguard Worker }
414*4d7e907cSAndroid Build Coastguard Worker 
setDescription(const std::string & description)415*4d7e907cSAndroid Build Coastguard Worker bool ExifUtilsImpl::setDescription(const std::string& description) {
416*4d7e907cSAndroid Build Coastguard Worker     SET_STRING(EXIF_IFD_0, EXIF_TAG_IMAGE_DESCRIPTION, EXIF_FORMAT_ASCII, description);
417*4d7e907cSAndroid Build Coastguard Worker     return true;
418*4d7e907cSAndroid Build Coastguard Worker }
419*4d7e907cSAndroid Build Coastguard Worker 
setDigitalZoomRatio(uint32_t numerator,uint32_t denominator)420*4d7e907cSAndroid Build Coastguard Worker bool ExifUtilsImpl::setDigitalZoomRatio(uint32_t numerator, uint32_t denominator) {
421*4d7e907cSAndroid Build Coastguard Worker     SET_RATIONAL(EXIF_IFD_EXIF, EXIF_TAG_DIGITAL_ZOOM_RATIO, numerator, denominator);
422*4d7e907cSAndroid Build Coastguard Worker     return true;
423*4d7e907cSAndroid Build Coastguard Worker }
424*4d7e907cSAndroid Build Coastguard Worker 
setExposureBias(int32_t numerator,int32_t denominator)425*4d7e907cSAndroid Build Coastguard Worker bool ExifUtilsImpl::setExposureBias(int32_t numerator, int32_t denominator) {
426*4d7e907cSAndroid Build Coastguard Worker     SET_SRATIONAL(EXIF_IFD_EXIF, EXIF_TAG_EXPOSURE_BIAS_VALUE, numerator, denominator);
427*4d7e907cSAndroid Build Coastguard Worker     return true;
428*4d7e907cSAndroid Build Coastguard Worker }
429*4d7e907cSAndroid Build Coastguard Worker 
setExposureMode(uint16_t exposure_mode)430*4d7e907cSAndroid Build Coastguard Worker bool ExifUtilsImpl::setExposureMode(uint16_t exposure_mode) {
431*4d7e907cSAndroid Build Coastguard Worker     SET_SHORT(EXIF_IFD_EXIF, EXIF_TAG_EXPOSURE_MODE, exposure_mode);
432*4d7e907cSAndroid Build Coastguard Worker     return true;
433*4d7e907cSAndroid Build Coastguard Worker }
434*4d7e907cSAndroid Build Coastguard Worker 
setExposureProgram(uint16_t exposure_program)435*4d7e907cSAndroid Build Coastguard Worker bool ExifUtilsImpl::setExposureProgram(uint16_t exposure_program) {
436*4d7e907cSAndroid Build Coastguard Worker     SET_SHORT(EXIF_IFD_EXIF, EXIF_TAG_EXPOSURE_PROGRAM, exposure_program);
437*4d7e907cSAndroid Build Coastguard Worker     return true;
438*4d7e907cSAndroid Build Coastguard Worker }
439*4d7e907cSAndroid Build Coastguard Worker 
setExposureTime(uint32_t numerator,uint32_t denominator)440*4d7e907cSAndroid Build Coastguard Worker bool ExifUtilsImpl::setExposureTime(uint32_t numerator, uint32_t denominator) {
441*4d7e907cSAndroid Build Coastguard Worker     SET_RATIONAL(EXIF_IFD_EXIF, EXIF_TAG_EXPOSURE_TIME, numerator, denominator);
442*4d7e907cSAndroid Build Coastguard Worker     return true;
443*4d7e907cSAndroid Build Coastguard Worker }
444*4d7e907cSAndroid Build Coastguard Worker 
setFlash(uint16_t flash)445*4d7e907cSAndroid Build Coastguard Worker bool ExifUtilsImpl::setFlash(uint16_t flash) {
446*4d7e907cSAndroid Build Coastguard Worker     SET_SHORT(EXIF_IFD_EXIF, EXIF_TAG_FLASH, flash);
447*4d7e907cSAndroid Build Coastguard Worker     return true;
448*4d7e907cSAndroid Build Coastguard Worker }
449*4d7e907cSAndroid Build Coastguard Worker 
setFNumber(uint32_t numerator,uint32_t denominator)450*4d7e907cSAndroid Build Coastguard Worker bool ExifUtilsImpl::setFNumber(uint32_t numerator, uint32_t denominator) {
451*4d7e907cSAndroid Build Coastguard Worker     SET_RATIONAL(EXIF_IFD_EXIF, EXIF_TAG_FNUMBER, numerator, denominator);
452*4d7e907cSAndroid Build Coastguard Worker     return true;
453*4d7e907cSAndroid Build Coastguard Worker }
454*4d7e907cSAndroid Build Coastguard Worker 
setFocalLength(uint32_t numerator,uint32_t denominator)455*4d7e907cSAndroid Build Coastguard Worker bool ExifUtilsImpl::setFocalLength(uint32_t numerator, uint32_t denominator) {
456*4d7e907cSAndroid Build Coastguard Worker     SET_RATIONAL(EXIF_IFD_EXIF, EXIF_TAG_FOCAL_LENGTH, numerator, denominator);
457*4d7e907cSAndroid Build Coastguard Worker     return true;
458*4d7e907cSAndroid Build Coastguard Worker }
459*4d7e907cSAndroid Build Coastguard Worker 
setGainControl(uint16_t gain_control)460*4d7e907cSAndroid Build Coastguard Worker bool ExifUtilsImpl::setGainControl(uint16_t gain_control) {
461*4d7e907cSAndroid Build Coastguard Worker     SET_SHORT(EXIF_IFD_EXIF, EXIF_TAG_GAIN_CONTROL, gain_control);
462*4d7e907cSAndroid Build Coastguard Worker     return true;
463*4d7e907cSAndroid Build Coastguard Worker }
464*4d7e907cSAndroid Build Coastguard Worker 
setGpsAltitude(double altitude)465*4d7e907cSAndroid Build Coastguard Worker bool ExifUtilsImpl::setGpsAltitude(double altitude) {
466*4d7e907cSAndroid Build Coastguard Worker     ExifTag refTag = static_cast<ExifTag>(EXIF_TAG_GPS_ALTITUDE_REF);
467*4d7e907cSAndroid Build Coastguard Worker     std::unique_ptr<ExifEntry> refEntry =
468*4d7e907cSAndroid Build Coastguard Worker             addVariableLengthEntry(EXIF_IFD_GPS, refTag, EXIF_FORMAT_BYTE, 1, 1);
469*4d7e907cSAndroid Build Coastguard Worker     if (!refEntry) {
470*4d7e907cSAndroid Build Coastguard Worker         ALOGE("%s: Adding GPSAltitudeRef exif entry failed", __FUNCTION__);
471*4d7e907cSAndroid Build Coastguard Worker         return false;
472*4d7e907cSAndroid Build Coastguard Worker     }
473*4d7e907cSAndroid Build Coastguard Worker     if (altitude >= 0) {
474*4d7e907cSAndroid Build Coastguard Worker         *refEntry->data = 0;
475*4d7e907cSAndroid Build Coastguard Worker     } else {
476*4d7e907cSAndroid Build Coastguard Worker         *refEntry->data = 1;
477*4d7e907cSAndroid Build Coastguard Worker         altitude *= -1;
478*4d7e907cSAndroid Build Coastguard Worker     }
479*4d7e907cSAndroid Build Coastguard Worker 
480*4d7e907cSAndroid Build Coastguard Worker     ExifTag tag = static_cast<ExifTag>(EXIF_TAG_GPS_ALTITUDE);
481*4d7e907cSAndroid Build Coastguard Worker     std::unique_ptr<ExifEntry> entry = addVariableLengthEntry(
482*4d7e907cSAndroid Build Coastguard Worker             EXIF_IFD_GPS, tag, EXIF_FORMAT_RATIONAL, 1, sizeof(ExifRational));
483*4d7e907cSAndroid Build Coastguard Worker     if (!entry) {
484*4d7e907cSAndroid Build Coastguard Worker         exif_content_remove_entry(exif_data_->ifd[EXIF_IFD_GPS], refEntry.get());
485*4d7e907cSAndroid Build Coastguard Worker         ALOGE("%s: Adding GPSAltitude exif entry failed", __FUNCTION__);
486*4d7e907cSAndroid Build Coastguard Worker         return false;
487*4d7e907cSAndroid Build Coastguard Worker     }
488*4d7e907cSAndroid Build Coastguard Worker     exif_set_rational(entry->data, EXIF_BYTE_ORDER_INTEL,
489*4d7e907cSAndroid Build Coastguard Worker                       {static_cast<ExifLong>(altitude * 1000), 1000});
490*4d7e907cSAndroid Build Coastguard Worker 
491*4d7e907cSAndroid Build Coastguard Worker     return true;
492*4d7e907cSAndroid Build Coastguard Worker }
493*4d7e907cSAndroid Build Coastguard Worker 
setGpsLatitude(double latitude)494*4d7e907cSAndroid Build Coastguard Worker bool ExifUtilsImpl::setGpsLatitude(double latitude) {
495*4d7e907cSAndroid Build Coastguard Worker     const ExifTag refTag = static_cast<ExifTag>(EXIF_TAG_GPS_LATITUDE_REF);
496*4d7e907cSAndroid Build Coastguard Worker     std::unique_ptr<ExifEntry> refEntry =
497*4d7e907cSAndroid Build Coastguard Worker             addVariableLengthEntry(EXIF_IFD_GPS, refTag, EXIF_FORMAT_ASCII, 2, 2);
498*4d7e907cSAndroid Build Coastguard Worker     if (!refEntry) {
499*4d7e907cSAndroid Build Coastguard Worker         ALOGE("%s: Adding GPSLatitudeRef exif entry failed", __FUNCTION__);
500*4d7e907cSAndroid Build Coastguard Worker         return false;
501*4d7e907cSAndroid Build Coastguard Worker     }
502*4d7e907cSAndroid Build Coastguard Worker     if (latitude >= 0) {
503*4d7e907cSAndroid Build Coastguard Worker         memcpy(refEntry->data, "N", sizeof("N"));
504*4d7e907cSAndroid Build Coastguard Worker     } else {
505*4d7e907cSAndroid Build Coastguard Worker         memcpy(refEntry->data, "S", sizeof("S"));
506*4d7e907cSAndroid Build Coastguard Worker         latitude *= -1;
507*4d7e907cSAndroid Build Coastguard Worker     }
508*4d7e907cSAndroid Build Coastguard Worker 
509*4d7e907cSAndroid Build Coastguard Worker     const ExifTag tag = static_cast<ExifTag>(EXIF_TAG_GPS_LATITUDE);
510*4d7e907cSAndroid Build Coastguard Worker     std::unique_ptr<ExifEntry> entry = addVariableLengthEntry(
511*4d7e907cSAndroid Build Coastguard Worker             EXIF_IFD_GPS, tag, EXIF_FORMAT_RATIONAL, 3, 3 * sizeof(ExifRational));
512*4d7e907cSAndroid Build Coastguard Worker     if (!entry) {
513*4d7e907cSAndroid Build Coastguard Worker         exif_content_remove_entry(exif_data_->ifd[EXIF_IFD_GPS], refEntry.get());
514*4d7e907cSAndroid Build Coastguard Worker         ALOGE("%s: Adding GPSLatitude exif entry failed", __FUNCTION__);
515*4d7e907cSAndroid Build Coastguard Worker         return false;
516*4d7e907cSAndroid Build Coastguard Worker     }
517*4d7e907cSAndroid Build Coastguard Worker     setLatitudeOrLongitudeData(entry->data, latitude);
518*4d7e907cSAndroid Build Coastguard Worker 
519*4d7e907cSAndroid Build Coastguard Worker     return true;
520*4d7e907cSAndroid Build Coastguard Worker }
521*4d7e907cSAndroid Build Coastguard Worker 
setGpsLongitude(double longitude)522*4d7e907cSAndroid Build Coastguard Worker bool ExifUtilsImpl::setGpsLongitude(double longitude) {
523*4d7e907cSAndroid Build Coastguard Worker     ExifTag refTag = static_cast<ExifTag>(EXIF_TAG_GPS_LONGITUDE_REF);
524*4d7e907cSAndroid Build Coastguard Worker     std::unique_ptr<ExifEntry> refEntry =
525*4d7e907cSAndroid Build Coastguard Worker             addVariableLengthEntry(EXIF_IFD_GPS, refTag, EXIF_FORMAT_ASCII, 2, 2);
526*4d7e907cSAndroid Build Coastguard Worker     if (!refEntry) {
527*4d7e907cSAndroid Build Coastguard Worker         ALOGE("%s: Adding GPSLongitudeRef exif entry failed", __FUNCTION__);
528*4d7e907cSAndroid Build Coastguard Worker         return false;
529*4d7e907cSAndroid Build Coastguard Worker     }
530*4d7e907cSAndroid Build Coastguard Worker     if (longitude >= 0) {
531*4d7e907cSAndroid Build Coastguard Worker         memcpy(refEntry->data, "E", sizeof("E"));
532*4d7e907cSAndroid Build Coastguard Worker     } else {
533*4d7e907cSAndroid Build Coastguard Worker         memcpy(refEntry->data, "W", sizeof("W"));
534*4d7e907cSAndroid Build Coastguard Worker         longitude *= -1;
535*4d7e907cSAndroid Build Coastguard Worker     }
536*4d7e907cSAndroid Build Coastguard Worker 
537*4d7e907cSAndroid Build Coastguard Worker     ExifTag tag = static_cast<ExifTag>(EXIF_TAG_GPS_LONGITUDE);
538*4d7e907cSAndroid Build Coastguard Worker     std::unique_ptr<ExifEntry> entry = addVariableLengthEntry(
539*4d7e907cSAndroid Build Coastguard Worker             EXIF_IFD_GPS, tag, EXIF_FORMAT_RATIONAL, 3, 3 * sizeof(ExifRational));
540*4d7e907cSAndroid Build Coastguard Worker     if (!entry) {
541*4d7e907cSAndroid Build Coastguard Worker         exif_content_remove_entry(exif_data_->ifd[EXIF_IFD_GPS], refEntry.get());
542*4d7e907cSAndroid Build Coastguard Worker         ALOGE("%s: Adding GPSLongitude exif entry failed", __FUNCTION__);
543*4d7e907cSAndroid Build Coastguard Worker         return false;
544*4d7e907cSAndroid Build Coastguard Worker     }
545*4d7e907cSAndroid Build Coastguard Worker     setLatitudeOrLongitudeData(entry->data, longitude);
546*4d7e907cSAndroid Build Coastguard Worker 
547*4d7e907cSAndroid Build Coastguard Worker     return true;
548*4d7e907cSAndroid Build Coastguard Worker }
549*4d7e907cSAndroid Build Coastguard Worker 
setGpsProcessingMethod(const std::string & method)550*4d7e907cSAndroid Build Coastguard Worker bool ExifUtilsImpl::setGpsProcessingMethod(const std::string& method) {
551*4d7e907cSAndroid Build Coastguard Worker     std::string buffer = std::string(gExifAsciiPrefix, sizeof(gExifAsciiPrefix)) + method;
552*4d7e907cSAndroid Build Coastguard Worker     SET_STRING(EXIF_IFD_GPS, static_cast<ExifTag>(EXIF_TAG_GPS_PROCESSING_METHOD),
553*4d7e907cSAndroid Build Coastguard Worker                EXIF_FORMAT_UNDEFINED, buffer);
554*4d7e907cSAndroid Build Coastguard Worker     return true;
555*4d7e907cSAndroid Build Coastguard Worker }
556*4d7e907cSAndroid Build Coastguard Worker 
setGpsTimestamp(const struct tm & t)557*4d7e907cSAndroid Build Coastguard Worker bool ExifUtilsImpl::setGpsTimestamp(const struct tm& t) {
558*4d7e907cSAndroid Build Coastguard Worker     const ExifTag dateTag = static_cast<ExifTag>(EXIF_TAG_GPS_DATE_STAMP);
559*4d7e907cSAndroid Build Coastguard Worker     const size_t kGpsDateStampSize = 11;
560*4d7e907cSAndroid Build Coastguard Worker     std::unique_ptr<ExifEntry> entry = addVariableLengthEntry(
561*4d7e907cSAndroid Build Coastguard Worker             EXIF_IFD_GPS, dateTag, EXIF_FORMAT_ASCII, kGpsDateStampSize, kGpsDateStampSize);
562*4d7e907cSAndroid Build Coastguard Worker     if (!entry) {
563*4d7e907cSAndroid Build Coastguard Worker         ALOGE("%s: Adding GPSDateStamp exif entry failed", __FUNCTION__);
564*4d7e907cSAndroid Build Coastguard Worker         return false;
565*4d7e907cSAndroid Build Coastguard Worker     }
566*4d7e907cSAndroid Build Coastguard Worker     int result = snprintf(reinterpret_cast<char*>(entry->data), kGpsDateStampSize, "%04i:%02i:%02i",
567*4d7e907cSAndroid Build Coastguard Worker                           t.tm_year + 1900, t.tm_mon + 1, t.tm_mday);
568*4d7e907cSAndroid Build Coastguard Worker     if (result != kGpsDateStampSize - 1) {
569*4d7e907cSAndroid Build Coastguard Worker         ALOGW("%s: Input time is invalid", __FUNCTION__);
570*4d7e907cSAndroid Build Coastguard Worker         return false;
571*4d7e907cSAndroid Build Coastguard Worker     }
572*4d7e907cSAndroid Build Coastguard Worker 
573*4d7e907cSAndroid Build Coastguard Worker     const ExifTag timeTag = static_cast<ExifTag>(EXIF_TAG_GPS_TIME_STAMP);
574*4d7e907cSAndroid Build Coastguard Worker     entry = addVariableLengthEntry(EXIF_IFD_GPS, timeTag, EXIF_FORMAT_RATIONAL, 3,
575*4d7e907cSAndroid Build Coastguard Worker                                    3 * sizeof(ExifRational));
576*4d7e907cSAndroid Build Coastguard Worker     if (!entry) {
577*4d7e907cSAndroid Build Coastguard Worker         ALOGE("%s: Adding GPSTimeStamp exif entry failed", __FUNCTION__);
578*4d7e907cSAndroid Build Coastguard Worker         return false;
579*4d7e907cSAndroid Build Coastguard Worker     }
580*4d7e907cSAndroid Build Coastguard Worker     exif_set_rational(entry->data, EXIF_BYTE_ORDER_INTEL, {static_cast<ExifLong>(t.tm_hour), 1});
581*4d7e907cSAndroid Build Coastguard Worker     exif_set_rational(entry->data + sizeof(ExifRational), EXIF_BYTE_ORDER_INTEL,
582*4d7e907cSAndroid Build Coastguard Worker                       {static_cast<ExifLong>(t.tm_min), 1});
583*4d7e907cSAndroid Build Coastguard Worker     exif_set_rational(entry->data + 2 * sizeof(ExifRational), EXIF_BYTE_ORDER_INTEL,
584*4d7e907cSAndroid Build Coastguard Worker                       {static_cast<ExifLong>(t.tm_sec), 1});
585*4d7e907cSAndroid Build Coastguard Worker 
586*4d7e907cSAndroid Build Coastguard Worker     return true;
587*4d7e907cSAndroid Build Coastguard Worker }
588*4d7e907cSAndroid Build Coastguard Worker 
setImageHeight(uint32_t length)589*4d7e907cSAndroid Build Coastguard Worker bool ExifUtilsImpl::setImageHeight(uint32_t length) {
590*4d7e907cSAndroid Build Coastguard Worker     SET_SHORT(EXIF_IFD_0, EXIF_TAG_IMAGE_LENGTH, length);
591*4d7e907cSAndroid Build Coastguard Worker     SET_LONG(EXIF_IFD_EXIF, EXIF_TAG_PIXEL_Y_DIMENSION, length);
592*4d7e907cSAndroid Build Coastguard Worker     return true;
593*4d7e907cSAndroid Build Coastguard Worker }
594*4d7e907cSAndroid Build Coastguard Worker 
setImageWidth(uint32_t width)595*4d7e907cSAndroid Build Coastguard Worker bool ExifUtilsImpl::setImageWidth(uint32_t width) {
596*4d7e907cSAndroid Build Coastguard Worker     SET_SHORT(EXIF_IFD_0, EXIF_TAG_IMAGE_WIDTH, width);
597*4d7e907cSAndroid Build Coastguard Worker     SET_LONG(EXIF_IFD_EXIF, EXIF_TAG_PIXEL_X_DIMENSION, width);
598*4d7e907cSAndroid Build Coastguard Worker     return true;
599*4d7e907cSAndroid Build Coastguard Worker }
600*4d7e907cSAndroid Build Coastguard Worker 
setIsoSpeedRating(uint16_t iso_speed_ratings)601*4d7e907cSAndroid Build Coastguard Worker bool ExifUtilsImpl::setIsoSpeedRating(uint16_t iso_speed_ratings) {
602*4d7e907cSAndroid Build Coastguard Worker     SET_SHORT(EXIF_IFD_EXIF, EXIF_TAG_ISO_SPEED_RATINGS, iso_speed_ratings);
603*4d7e907cSAndroid Build Coastguard Worker     return true;
604*4d7e907cSAndroid Build Coastguard Worker }
605*4d7e907cSAndroid Build Coastguard Worker 
setLightSource(uint16_t light_source)606*4d7e907cSAndroid Build Coastguard Worker bool ExifUtilsImpl::setLightSource(uint16_t light_source) {
607*4d7e907cSAndroid Build Coastguard Worker     SET_SHORT(EXIF_IFD_EXIF, EXIF_TAG_LIGHT_SOURCE, light_source);
608*4d7e907cSAndroid Build Coastguard Worker     return true;
609*4d7e907cSAndroid Build Coastguard Worker }
610*4d7e907cSAndroid Build Coastguard Worker 
setMaxAperture(uint32_t numerator,uint32_t denominator)611*4d7e907cSAndroid Build Coastguard Worker bool ExifUtilsImpl::setMaxAperture(uint32_t numerator, uint32_t denominator) {
612*4d7e907cSAndroid Build Coastguard Worker     SET_RATIONAL(EXIF_IFD_EXIF, EXIF_TAG_MAX_APERTURE_VALUE, numerator, denominator);
613*4d7e907cSAndroid Build Coastguard Worker     return true;
614*4d7e907cSAndroid Build Coastguard Worker }
615*4d7e907cSAndroid Build Coastguard Worker 
setMeteringMode(uint16_t metering_mode)616*4d7e907cSAndroid Build Coastguard Worker bool ExifUtilsImpl::setMeteringMode(uint16_t metering_mode) {
617*4d7e907cSAndroid Build Coastguard Worker     SET_SHORT(EXIF_IFD_EXIF, EXIF_TAG_METERING_MODE, metering_mode);
618*4d7e907cSAndroid Build Coastguard Worker     return true;
619*4d7e907cSAndroid Build Coastguard Worker }
620*4d7e907cSAndroid Build Coastguard Worker 
setOrientation(uint16_t orientation)621*4d7e907cSAndroid Build Coastguard Worker bool ExifUtilsImpl::setOrientation(uint16_t orientation) {
622*4d7e907cSAndroid Build Coastguard Worker     /*
623*4d7e907cSAndroid Build Coastguard Worker      * Orientation value:
624*4d7e907cSAndroid Build Coastguard Worker      *  1      2      3      4      5          6          7          8
625*4d7e907cSAndroid Build Coastguard Worker      *
626*4d7e907cSAndroid Build Coastguard Worker      *  888888 888888     88 88     8888888888 88                 88 8888888888
627*4d7e907cSAndroid Build Coastguard Worker      *  88         88     88 88     88  88     88  88         88  88     88  88
628*4d7e907cSAndroid Build Coastguard Worker      *  8888     8888   8888 8888   88         8888888888 8888888888         88
629*4d7e907cSAndroid Build Coastguard Worker      *  88         88     88 88
630*4d7e907cSAndroid Build Coastguard Worker      *  88         88 888888 888888
631*4d7e907cSAndroid Build Coastguard Worker      */
632*4d7e907cSAndroid Build Coastguard Worker     int value = 1;
633*4d7e907cSAndroid Build Coastguard Worker     switch (orientation) {
634*4d7e907cSAndroid Build Coastguard Worker         case 90:
635*4d7e907cSAndroid Build Coastguard Worker             value = 6;
636*4d7e907cSAndroid Build Coastguard Worker             break;
637*4d7e907cSAndroid Build Coastguard Worker         case 180:
638*4d7e907cSAndroid Build Coastguard Worker             value = 3;
639*4d7e907cSAndroid Build Coastguard Worker             break;
640*4d7e907cSAndroid Build Coastguard Worker         case 270:
641*4d7e907cSAndroid Build Coastguard Worker             value = 8;
642*4d7e907cSAndroid Build Coastguard Worker             break;
643*4d7e907cSAndroid Build Coastguard Worker         default:
644*4d7e907cSAndroid Build Coastguard Worker             break;
645*4d7e907cSAndroid Build Coastguard Worker     }
646*4d7e907cSAndroid Build Coastguard Worker     SET_SHORT(EXIF_IFD_0, EXIF_TAG_ORIENTATION, value);
647*4d7e907cSAndroid Build Coastguard Worker     return true;
648*4d7e907cSAndroid Build Coastguard Worker }
649*4d7e907cSAndroid Build Coastguard Worker 
setResolutionUnit(uint16_t resolution_unit)650*4d7e907cSAndroid Build Coastguard Worker bool ExifUtilsImpl::setResolutionUnit(uint16_t resolution_unit) {
651*4d7e907cSAndroid Build Coastguard Worker     SET_SHORT(EXIF_IFD_EXIF, EXIF_TAG_RESOLUTION_UNIT, resolution_unit);
652*4d7e907cSAndroid Build Coastguard Worker     return true;
653*4d7e907cSAndroid Build Coastguard Worker }
654*4d7e907cSAndroid Build Coastguard Worker 
setSaturation(uint16_t saturation)655*4d7e907cSAndroid Build Coastguard Worker bool ExifUtilsImpl::setSaturation(uint16_t saturation) {
656*4d7e907cSAndroid Build Coastguard Worker     SET_SHORT(EXIF_IFD_EXIF, EXIF_TAG_SATURATION, saturation);
657*4d7e907cSAndroid Build Coastguard Worker     return true;
658*4d7e907cSAndroid Build Coastguard Worker }
659*4d7e907cSAndroid Build Coastguard Worker 
setSceneCaptureType(uint16_t type)660*4d7e907cSAndroid Build Coastguard Worker bool ExifUtilsImpl::setSceneCaptureType(uint16_t type) {
661*4d7e907cSAndroid Build Coastguard Worker     SET_SHORT(EXIF_IFD_EXIF, EXIF_TAG_SCENE_CAPTURE_TYPE, type);
662*4d7e907cSAndroid Build Coastguard Worker     return true;
663*4d7e907cSAndroid Build Coastguard Worker }
664*4d7e907cSAndroid Build Coastguard Worker 
setSharpness(uint16_t sharpness)665*4d7e907cSAndroid Build Coastguard Worker bool ExifUtilsImpl::setSharpness(uint16_t sharpness) {
666*4d7e907cSAndroid Build Coastguard Worker     SET_SHORT(EXIF_IFD_EXIF, EXIF_TAG_SHARPNESS, sharpness);
667*4d7e907cSAndroid Build Coastguard Worker     return true;
668*4d7e907cSAndroid Build Coastguard Worker }
669*4d7e907cSAndroid Build Coastguard Worker 
setShutterSpeed(int32_t numerator,int32_t denominator)670*4d7e907cSAndroid Build Coastguard Worker bool ExifUtilsImpl::setShutterSpeed(int32_t numerator, int32_t denominator) {
671*4d7e907cSAndroid Build Coastguard Worker     SET_SRATIONAL(EXIF_IFD_EXIF, EXIF_TAG_SHUTTER_SPEED_VALUE, numerator, denominator);
672*4d7e907cSAndroid Build Coastguard Worker     return true;
673*4d7e907cSAndroid Build Coastguard Worker }
674*4d7e907cSAndroid Build Coastguard Worker 
setSubjectDistance(uint32_t numerator,uint32_t denominator)675*4d7e907cSAndroid Build Coastguard Worker bool ExifUtilsImpl::setSubjectDistance(uint32_t numerator, uint32_t denominator) {
676*4d7e907cSAndroid Build Coastguard Worker     SET_RATIONAL(EXIF_IFD_EXIF, EXIF_TAG_SUBJECT_DISTANCE, numerator, denominator);
677*4d7e907cSAndroid Build Coastguard Worker     return true;
678*4d7e907cSAndroid Build Coastguard Worker }
679*4d7e907cSAndroid Build Coastguard Worker 
setSubsecTime(const std::string & subsec_time)680*4d7e907cSAndroid Build Coastguard Worker bool ExifUtilsImpl::setSubsecTime(const std::string& subsec_time) {
681*4d7e907cSAndroid Build Coastguard Worker     SET_STRING(EXIF_IFD_EXIF, EXIF_TAG_SUB_SEC_TIME, EXIF_FORMAT_ASCII, subsec_time);
682*4d7e907cSAndroid Build Coastguard Worker     SET_STRING(EXIF_IFD_EXIF, EXIF_TAG_SUB_SEC_TIME_ORIGINAL, EXIF_FORMAT_ASCII, subsec_time);
683*4d7e907cSAndroid Build Coastguard Worker     SET_STRING(EXIF_IFD_EXIF, EXIF_TAG_SUB_SEC_TIME_DIGITIZED, EXIF_FORMAT_ASCII, subsec_time);
684*4d7e907cSAndroid Build Coastguard Worker     return true;
685*4d7e907cSAndroid Build Coastguard Worker }
686*4d7e907cSAndroid Build Coastguard Worker 
setWhiteBalance(uint16_t white_balance)687*4d7e907cSAndroid Build Coastguard Worker bool ExifUtilsImpl::setWhiteBalance(uint16_t white_balance) {
688*4d7e907cSAndroid Build Coastguard Worker     SET_SHORT(EXIF_IFD_EXIF, EXIF_TAG_WHITE_BALANCE, white_balance);
689*4d7e907cSAndroid Build Coastguard Worker     return true;
690*4d7e907cSAndroid Build Coastguard Worker }
691*4d7e907cSAndroid Build Coastguard Worker 
setXResolution(uint32_t numerator,uint32_t denominator)692*4d7e907cSAndroid Build Coastguard Worker bool ExifUtilsImpl::setXResolution(uint32_t numerator, uint32_t denominator) {
693*4d7e907cSAndroid Build Coastguard Worker     SET_RATIONAL(EXIF_IFD_EXIF, EXIF_TAG_X_RESOLUTION, numerator, denominator);
694*4d7e907cSAndroid Build Coastguard Worker     return true;
695*4d7e907cSAndroid Build Coastguard Worker }
696*4d7e907cSAndroid Build Coastguard Worker 
setYCbCrPositioning(uint16_t ycbcr_positioning)697*4d7e907cSAndroid Build Coastguard Worker bool ExifUtilsImpl::setYCbCrPositioning(uint16_t ycbcr_positioning) {
698*4d7e907cSAndroid Build Coastguard Worker     SET_SHORT(EXIF_IFD_0, EXIF_TAG_YCBCR_POSITIONING, ycbcr_positioning);
699*4d7e907cSAndroid Build Coastguard Worker     return true;
700*4d7e907cSAndroid Build Coastguard Worker }
701*4d7e907cSAndroid Build Coastguard Worker 
setYResolution(uint32_t numerator,uint32_t denominator)702*4d7e907cSAndroid Build Coastguard Worker bool ExifUtilsImpl::setYResolution(uint32_t numerator, uint32_t denominator) {
703*4d7e907cSAndroid Build Coastguard Worker     SET_RATIONAL(EXIF_IFD_EXIF, EXIF_TAG_Y_RESOLUTION, numerator, denominator);
704*4d7e907cSAndroid Build Coastguard Worker     return true;
705*4d7e907cSAndroid Build Coastguard Worker }
706*4d7e907cSAndroid Build Coastguard Worker 
generateApp1(const void * thumbnail_buffer,uint32_t size)707*4d7e907cSAndroid Build Coastguard Worker bool ExifUtilsImpl::generateApp1(const void* thumbnail_buffer, uint32_t size) {
708*4d7e907cSAndroid Build Coastguard Worker     destroyApp1();
709*4d7e907cSAndroid Build Coastguard Worker     exif_data_->data = const_cast<uint8_t*>(static_cast<const uint8_t*>(thumbnail_buffer));
710*4d7e907cSAndroid Build Coastguard Worker     exif_data_->size = size;
711*4d7e907cSAndroid Build Coastguard Worker     // Save the result into |app1_buffer_|.
712*4d7e907cSAndroid Build Coastguard Worker     exif_data_save_data(exif_data_, &app1_buffer_, &app1_length_);
713*4d7e907cSAndroid Build Coastguard Worker     if (!app1_length_) {
714*4d7e907cSAndroid Build Coastguard Worker         ALOGE("%s: Allocate memory for app1_buffer_ failed", __FUNCTION__);
715*4d7e907cSAndroid Build Coastguard Worker         return false;
716*4d7e907cSAndroid Build Coastguard Worker     }
717*4d7e907cSAndroid Build Coastguard Worker     /*
718*4d7e907cSAndroid Build Coastguard Worker      * The JPEG segment size is 16 bits in spec. The size of APP1 segment should
719*4d7e907cSAndroid Build Coastguard Worker      * be smaller than 65533 because there are two bytes for segment size field.
720*4d7e907cSAndroid Build Coastguard Worker      */
721*4d7e907cSAndroid Build Coastguard Worker     if (app1_length_ > 65533) {
722*4d7e907cSAndroid Build Coastguard Worker         destroyApp1();
723*4d7e907cSAndroid Build Coastguard Worker         ALOGE("%s: The size of APP1 segment is too large", __FUNCTION__);
724*4d7e907cSAndroid Build Coastguard Worker         return false;
725*4d7e907cSAndroid Build Coastguard Worker     }
726*4d7e907cSAndroid Build Coastguard Worker     return true;
727*4d7e907cSAndroid Build Coastguard Worker }
728*4d7e907cSAndroid Build Coastguard Worker 
getApp1Buffer()729*4d7e907cSAndroid Build Coastguard Worker const uint8_t* ExifUtilsImpl::getApp1Buffer() {
730*4d7e907cSAndroid Build Coastguard Worker     return app1_buffer_;
731*4d7e907cSAndroid Build Coastguard Worker }
732*4d7e907cSAndroid Build Coastguard Worker 
getApp1Length()733*4d7e907cSAndroid Build Coastguard Worker unsigned int ExifUtilsImpl::getApp1Length() {
734*4d7e907cSAndroid Build Coastguard Worker     return app1_length_;
735*4d7e907cSAndroid Build Coastguard Worker }
736*4d7e907cSAndroid Build Coastguard Worker 
setExifVersion(const std::string & exif_version)737*4d7e907cSAndroid Build Coastguard Worker bool ExifUtilsImpl::setExifVersion(const std::string& exif_version) {
738*4d7e907cSAndroid Build Coastguard Worker     SET_STRING(EXIF_IFD_EXIF, EXIF_TAG_EXIF_VERSION, EXIF_FORMAT_UNDEFINED, exif_version);
739*4d7e907cSAndroid Build Coastguard Worker     return true;
740*4d7e907cSAndroid Build Coastguard Worker }
741*4d7e907cSAndroid Build Coastguard Worker 
setMake(const std::string & make)742*4d7e907cSAndroid Build Coastguard Worker bool ExifUtilsImpl::setMake(const std::string& make) {
743*4d7e907cSAndroid Build Coastguard Worker     SET_STRING(EXIF_IFD_0, EXIF_TAG_MAKE, EXIF_FORMAT_ASCII, make);
744*4d7e907cSAndroid Build Coastguard Worker     return true;
745*4d7e907cSAndroid Build Coastguard Worker }
746*4d7e907cSAndroid Build Coastguard Worker 
setModel(const std::string & model)747*4d7e907cSAndroid Build Coastguard Worker bool ExifUtilsImpl::setModel(const std::string& model) {
748*4d7e907cSAndroid Build Coastguard Worker     SET_STRING(EXIF_IFD_0, EXIF_TAG_MODEL, EXIF_FORMAT_ASCII, model);
749*4d7e907cSAndroid Build Coastguard Worker     return true;
750*4d7e907cSAndroid Build Coastguard Worker }
751*4d7e907cSAndroid Build Coastguard Worker 
reset()752*4d7e907cSAndroid Build Coastguard Worker void ExifUtilsImpl::reset() {
753*4d7e907cSAndroid Build Coastguard Worker     destroyApp1();
754*4d7e907cSAndroid Build Coastguard Worker     if (exif_data_) {
755*4d7e907cSAndroid Build Coastguard Worker         /*
756*4d7e907cSAndroid Build Coastguard Worker          * Since we decided to ignore the original APP1, we are sure that there is
757*4d7e907cSAndroid Build Coastguard Worker          * no thumbnail allocated by libexif. |exif_data_->data| is actually
758*4d7e907cSAndroid Build Coastguard Worker          * allocated by JpegCompressor. sets |exif_data_->data| to nullptr to
759*4d7e907cSAndroid Build Coastguard Worker          * prevent exif_data_unref() destroy it incorrectly.
760*4d7e907cSAndroid Build Coastguard Worker          */
761*4d7e907cSAndroid Build Coastguard Worker         exif_data_->data = nullptr;
762*4d7e907cSAndroid Build Coastguard Worker         exif_data_->size = 0;
763*4d7e907cSAndroid Build Coastguard Worker         exif_data_unref(exif_data_);
764*4d7e907cSAndroid Build Coastguard Worker         exif_data_ = nullptr;
765*4d7e907cSAndroid Build Coastguard Worker     }
766*4d7e907cSAndroid Build Coastguard Worker }
767*4d7e907cSAndroid Build Coastguard Worker 
addVariableLengthEntry(ExifIfd ifd,ExifTag tag,ExifFormat format,uint64_t components,unsigned int size)768*4d7e907cSAndroid Build Coastguard Worker std::unique_ptr<ExifEntry> ExifUtilsImpl::addVariableLengthEntry(ExifIfd ifd, ExifTag tag,
769*4d7e907cSAndroid Build Coastguard Worker                                                                  ExifFormat format,
770*4d7e907cSAndroid Build Coastguard Worker                                                                  uint64_t components,
771*4d7e907cSAndroid Build Coastguard Worker                                                                  unsigned int size) {
772*4d7e907cSAndroid Build Coastguard Worker     // Remove old entry if exists.
773*4d7e907cSAndroid Build Coastguard Worker     exif_content_remove_entry(exif_data_->ifd[ifd],
774*4d7e907cSAndroid Build Coastguard Worker                               exif_content_get_entry(exif_data_->ifd[ifd], tag));
775*4d7e907cSAndroid Build Coastguard Worker     ExifMem* mem = exif_mem_new_default();
776*4d7e907cSAndroid Build Coastguard Worker     if (!mem) {
777*4d7e907cSAndroid Build Coastguard Worker         ALOGE("%s: Allocate memory for exif entry failed", __FUNCTION__);
778*4d7e907cSAndroid Build Coastguard Worker         return nullptr;
779*4d7e907cSAndroid Build Coastguard Worker     }
780*4d7e907cSAndroid Build Coastguard Worker     std::unique_ptr<ExifEntry> entry(exif_entry_new_mem(mem));
781*4d7e907cSAndroid Build Coastguard Worker     if (!entry) {
782*4d7e907cSAndroid Build Coastguard Worker         ALOGE("%s: Allocate memory for exif entry failed", __FUNCTION__);
783*4d7e907cSAndroid Build Coastguard Worker         exif_mem_unref(mem);
784*4d7e907cSAndroid Build Coastguard Worker         return nullptr;
785*4d7e907cSAndroid Build Coastguard Worker     }
786*4d7e907cSAndroid Build Coastguard Worker     void* tmpBuffer = exif_mem_alloc(mem, size);
787*4d7e907cSAndroid Build Coastguard Worker     if (!tmpBuffer) {
788*4d7e907cSAndroid Build Coastguard Worker         ALOGE("%s: Allocate memory for exif entry failed", __FUNCTION__);
789*4d7e907cSAndroid Build Coastguard Worker         exif_mem_unref(mem);
790*4d7e907cSAndroid Build Coastguard Worker         return nullptr;
791*4d7e907cSAndroid Build Coastguard Worker     }
792*4d7e907cSAndroid Build Coastguard Worker 
793*4d7e907cSAndroid Build Coastguard Worker     entry->data = static_cast<unsigned char*>(tmpBuffer);
794*4d7e907cSAndroid Build Coastguard Worker     entry->tag = tag;
795*4d7e907cSAndroid Build Coastguard Worker     entry->format = format;
796*4d7e907cSAndroid Build Coastguard Worker     entry->components = components;
797*4d7e907cSAndroid Build Coastguard Worker     entry->size = size;
798*4d7e907cSAndroid Build Coastguard Worker 
799*4d7e907cSAndroid Build Coastguard Worker     exif_content_add_entry(exif_data_->ifd[ifd], entry.get());
800*4d7e907cSAndroid Build Coastguard Worker     exif_mem_unref(mem);
801*4d7e907cSAndroid Build Coastguard Worker 
802*4d7e907cSAndroid Build Coastguard Worker     return entry;
803*4d7e907cSAndroid Build Coastguard Worker }
804*4d7e907cSAndroid Build Coastguard Worker 
addEntry(ExifIfd ifd,ExifTag tag)805*4d7e907cSAndroid Build Coastguard Worker std::unique_ptr<ExifEntry> ExifUtilsImpl::addEntry(ExifIfd ifd, ExifTag tag) {
806*4d7e907cSAndroid Build Coastguard Worker     std::unique_ptr<ExifEntry> entry(exif_content_get_entry(exif_data_->ifd[ifd], tag));
807*4d7e907cSAndroid Build Coastguard Worker     if (entry) {
808*4d7e907cSAndroid Build Coastguard Worker         // exif_content_get_entry() won't ref the entry, so we ref here.
809*4d7e907cSAndroid Build Coastguard Worker         exif_entry_ref(entry.get());
810*4d7e907cSAndroid Build Coastguard Worker         return entry;
811*4d7e907cSAndroid Build Coastguard Worker     }
812*4d7e907cSAndroid Build Coastguard Worker     entry.reset(exif_entry_new());
813*4d7e907cSAndroid Build Coastguard Worker     if (!entry) {
814*4d7e907cSAndroid Build Coastguard Worker         ALOGE("%s: Allocate memory for exif entry failed", __FUNCTION__);
815*4d7e907cSAndroid Build Coastguard Worker         return nullptr;
816*4d7e907cSAndroid Build Coastguard Worker     }
817*4d7e907cSAndroid Build Coastguard Worker     entry->tag = tag;
818*4d7e907cSAndroid Build Coastguard Worker     exif_content_add_entry(exif_data_->ifd[ifd], entry.get());
819*4d7e907cSAndroid Build Coastguard Worker     exif_entry_initialize(entry.get(), tag);
820*4d7e907cSAndroid Build Coastguard Worker     return entry;
821*4d7e907cSAndroid Build Coastguard Worker }
822*4d7e907cSAndroid Build Coastguard Worker 
setShort(ExifIfd ifd,ExifTag tag,uint16_t value,const std::string & msg)823*4d7e907cSAndroid Build Coastguard Worker bool ExifUtilsImpl::setShort(ExifIfd ifd, ExifTag tag, uint16_t value, const std::string& msg) {
824*4d7e907cSAndroid Build Coastguard Worker     std::unique_ptr<ExifEntry> entry = addEntry(ifd, tag);
825*4d7e907cSAndroid Build Coastguard Worker     if (!entry) {
826*4d7e907cSAndroid Build Coastguard Worker         ALOGE("%s: Adding '%s' entry failed", __FUNCTION__, msg.c_str());
827*4d7e907cSAndroid Build Coastguard Worker         return false;
828*4d7e907cSAndroid Build Coastguard Worker     }
829*4d7e907cSAndroid Build Coastguard Worker     exif_set_short(entry->data, EXIF_BYTE_ORDER_INTEL, value);
830*4d7e907cSAndroid Build Coastguard Worker     return true;
831*4d7e907cSAndroid Build Coastguard Worker }
832*4d7e907cSAndroid Build Coastguard Worker 
setLong(ExifIfd ifd,ExifTag tag,uint32_t value,const std::string & msg)833*4d7e907cSAndroid Build Coastguard Worker bool ExifUtilsImpl::setLong(ExifIfd ifd, ExifTag tag, uint32_t value, const std::string& msg) {
834*4d7e907cSAndroid Build Coastguard Worker     std::unique_ptr<ExifEntry> entry = addEntry(ifd, tag);
835*4d7e907cSAndroid Build Coastguard Worker     if (!entry) {
836*4d7e907cSAndroid Build Coastguard Worker         ALOGE("%s: Adding '%s' entry failed", __FUNCTION__, msg.c_str());
837*4d7e907cSAndroid Build Coastguard Worker         return false;
838*4d7e907cSAndroid Build Coastguard Worker     }
839*4d7e907cSAndroid Build Coastguard Worker     exif_set_long(entry->data, EXIF_BYTE_ORDER_INTEL, value);
840*4d7e907cSAndroid Build Coastguard Worker     return true;
841*4d7e907cSAndroid Build Coastguard Worker }
842*4d7e907cSAndroid Build Coastguard Worker 
setRational(ExifIfd ifd,ExifTag tag,uint32_t numerator,uint32_t denominator,const std::string & msg)843*4d7e907cSAndroid Build Coastguard Worker bool ExifUtilsImpl::setRational(ExifIfd ifd, ExifTag tag, uint32_t numerator, uint32_t denominator,
844*4d7e907cSAndroid Build Coastguard Worker                                 const std::string& msg) {
845*4d7e907cSAndroid Build Coastguard Worker     std::unique_ptr<ExifEntry> entry = addEntry(ifd, tag);
846*4d7e907cSAndroid Build Coastguard Worker     if (!entry) {
847*4d7e907cSAndroid Build Coastguard Worker         ALOGE("%s: Adding '%s' entry failed", __FUNCTION__, msg.c_str());
848*4d7e907cSAndroid Build Coastguard Worker         return false;
849*4d7e907cSAndroid Build Coastguard Worker     }
850*4d7e907cSAndroid Build Coastguard Worker     exif_set_rational(entry->data, EXIF_BYTE_ORDER_INTEL, {numerator, denominator});
851*4d7e907cSAndroid Build Coastguard Worker     return true;
852*4d7e907cSAndroid Build Coastguard Worker }
853*4d7e907cSAndroid Build Coastguard Worker 
setSRational(ExifIfd ifd,ExifTag tag,int32_t numerator,int32_t denominator,const std::string & msg)854*4d7e907cSAndroid Build Coastguard Worker bool ExifUtilsImpl::setSRational(ExifIfd ifd, ExifTag tag, int32_t numerator, int32_t denominator,
855*4d7e907cSAndroid Build Coastguard Worker                                  const std::string& msg) {
856*4d7e907cSAndroid Build Coastguard Worker     std::unique_ptr<ExifEntry> entry = addEntry(ifd, tag);
857*4d7e907cSAndroid Build Coastguard Worker     if (!entry) {
858*4d7e907cSAndroid Build Coastguard Worker         ALOGE("%s: Adding '%s' entry failed", __FUNCTION__, msg.c_str());
859*4d7e907cSAndroid Build Coastguard Worker         return false;
860*4d7e907cSAndroid Build Coastguard Worker     }
861*4d7e907cSAndroid Build Coastguard Worker     exif_set_srational(entry->data, EXIF_BYTE_ORDER_INTEL, {numerator, denominator});
862*4d7e907cSAndroid Build Coastguard Worker     return true;
863*4d7e907cSAndroid Build Coastguard Worker }
864*4d7e907cSAndroid Build Coastguard Worker 
setString(ExifIfd ifd,ExifTag tag,ExifFormat format,const std::string & buffer,const std::string & msg)865*4d7e907cSAndroid Build Coastguard Worker bool ExifUtilsImpl::setString(ExifIfd ifd, ExifTag tag, ExifFormat format,
866*4d7e907cSAndroid Build Coastguard Worker                               const std::string& buffer, const std::string& msg) {
867*4d7e907cSAndroid Build Coastguard Worker     size_t entry_size = buffer.length();
868*4d7e907cSAndroid Build Coastguard Worker     // Since the exif format is undefined, NULL termination is not necessary.
869*4d7e907cSAndroid Build Coastguard Worker     if (format == EXIF_FORMAT_ASCII) {
870*4d7e907cSAndroid Build Coastguard Worker         entry_size++;
871*4d7e907cSAndroid Build Coastguard Worker     }
872*4d7e907cSAndroid Build Coastguard Worker     std::unique_ptr<ExifEntry> entry =
873*4d7e907cSAndroid Build Coastguard Worker             addVariableLengthEntry(ifd, tag, format, entry_size, entry_size);
874*4d7e907cSAndroid Build Coastguard Worker     if (!entry) {
875*4d7e907cSAndroid Build Coastguard Worker         ALOGE("%s: Adding '%s' entry failed", __FUNCTION__, msg.c_str());
876*4d7e907cSAndroid Build Coastguard Worker         return false;
877*4d7e907cSAndroid Build Coastguard Worker     }
878*4d7e907cSAndroid Build Coastguard Worker     memcpy(entry->data, buffer.c_str(), entry_size);
879*4d7e907cSAndroid Build Coastguard Worker     return true;
880*4d7e907cSAndroid Build Coastguard Worker }
881*4d7e907cSAndroid Build Coastguard Worker 
destroyApp1()882*4d7e907cSAndroid Build Coastguard Worker void ExifUtilsImpl::destroyApp1() {
883*4d7e907cSAndroid Build Coastguard Worker     /*
884*4d7e907cSAndroid Build Coastguard Worker      * Since there is no API to access ExifMem in ExifData->priv, we use free
885*4d7e907cSAndroid Build Coastguard Worker      * here, which is the default free function in libexif. See
886*4d7e907cSAndroid Build Coastguard Worker      * exif_data_save_data() for detail.
887*4d7e907cSAndroid Build Coastguard Worker      */
888*4d7e907cSAndroid Build Coastguard Worker     free(app1_buffer_);
889*4d7e907cSAndroid Build Coastguard Worker     app1_buffer_ = nullptr;
890*4d7e907cSAndroid Build Coastguard Worker     app1_length_ = 0;
891*4d7e907cSAndroid Build Coastguard Worker }
892*4d7e907cSAndroid Build Coastguard Worker 
setFromMetadata(const CameraMetadata & metadata,const size_t imageWidth,const size_t imageHeight)893*4d7e907cSAndroid Build Coastguard Worker bool ExifUtilsImpl::setFromMetadata(const CameraMetadata& metadata, const size_t imageWidth,
894*4d7e907cSAndroid Build Coastguard Worker                                     const size_t imageHeight) {
895*4d7e907cSAndroid Build Coastguard Worker     // How precise the float-to-rational conversion for EXIF tags would be.
896*4d7e907cSAndroid Build Coastguard Worker     constexpr int kRationalPrecision = 10000;
897*4d7e907cSAndroid Build Coastguard Worker     if (!setImageWidth(imageWidth) || !setImageHeight(imageHeight)) {
898*4d7e907cSAndroid Build Coastguard Worker         ALOGE("%s: setting image resolution failed.", __FUNCTION__);
899*4d7e907cSAndroid Build Coastguard Worker         return false;
900*4d7e907cSAndroid Build Coastguard Worker     }
901*4d7e907cSAndroid Build Coastguard Worker 
902*4d7e907cSAndroid Build Coastguard Worker     struct timespec tp;
903*4d7e907cSAndroid Build Coastguard Worker     struct tm time_info;
904*4d7e907cSAndroid Build Coastguard Worker     bool time_available = clock_gettime(CLOCK_REALTIME, &tp) != -1;
905*4d7e907cSAndroid Build Coastguard Worker     localtime_r(&tp.tv_sec, &time_info);
906*4d7e907cSAndroid Build Coastguard Worker     if (!setDateTime(time_info)) {
907*4d7e907cSAndroid Build Coastguard Worker         ALOGE("%s: setting data time failed.", __FUNCTION__);
908*4d7e907cSAndroid Build Coastguard Worker         return false;
909*4d7e907cSAndroid Build Coastguard Worker     }
910*4d7e907cSAndroid Build Coastguard Worker 
911*4d7e907cSAndroid Build Coastguard Worker     float focal_length;
912*4d7e907cSAndroid Build Coastguard Worker     camera_metadata_ro_entry entry = metadata.find(ANDROID_LENS_FOCAL_LENGTH);
913*4d7e907cSAndroid Build Coastguard Worker     if (entry.count) {
914*4d7e907cSAndroid Build Coastguard Worker         focal_length = entry.data.f[0];
915*4d7e907cSAndroid Build Coastguard Worker 
916*4d7e907cSAndroid Build Coastguard Worker         if (!setFocalLength(static_cast<uint32_t>(focal_length * kRationalPrecision),
917*4d7e907cSAndroid Build Coastguard Worker                             kRationalPrecision)) {
918*4d7e907cSAndroid Build Coastguard Worker             ALOGE("%s: setting focal length failed.", __FUNCTION__);
919*4d7e907cSAndroid Build Coastguard Worker             return false;
920*4d7e907cSAndroid Build Coastguard Worker         }
921*4d7e907cSAndroid Build Coastguard Worker     } else {
922*4d7e907cSAndroid Build Coastguard Worker         ALOGV("%s: Cannot find focal length in metadata.", __FUNCTION__);
923*4d7e907cSAndroid Build Coastguard Worker     }
924*4d7e907cSAndroid Build Coastguard Worker 
925*4d7e907cSAndroid Build Coastguard Worker     if (metadata.exists(ANDROID_JPEG_GPS_COORDINATES)) {
926*4d7e907cSAndroid Build Coastguard Worker         entry = metadata.find(ANDROID_JPEG_GPS_COORDINATES);
927*4d7e907cSAndroid Build Coastguard Worker         if (entry.count < 3) {
928*4d7e907cSAndroid Build Coastguard Worker             ALOGE("%s: Gps coordinates in metadata is not complete.", __FUNCTION__);
929*4d7e907cSAndroid Build Coastguard Worker             return false;
930*4d7e907cSAndroid Build Coastguard Worker         }
931*4d7e907cSAndroid Build Coastguard Worker         if (!setGpsLatitude(entry.data.d[0])) {
932*4d7e907cSAndroid Build Coastguard Worker             ALOGE("%s: setting gps latitude failed.", __FUNCTION__);
933*4d7e907cSAndroid Build Coastguard Worker             return false;
934*4d7e907cSAndroid Build Coastguard Worker         }
935*4d7e907cSAndroid Build Coastguard Worker         if (!setGpsLongitude(entry.data.d[1])) {
936*4d7e907cSAndroid Build Coastguard Worker             ALOGE("%s: setting gps longitude failed.", __FUNCTION__);
937*4d7e907cSAndroid Build Coastguard Worker             return false;
938*4d7e907cSAndroid Build Coastguard Worker         }
939*4d7e907cSAndroid Build Coastguard Worker         if (!setGpsAltitude(entry.data.d[2])) {
940*4d7e907cSAndroid Build Coastguard Worker             ALOGE("%s: setting gps altitude failed.", __FUNCTION__);
941*4d7e907cSAndroid Build Coastguard Worker             return false;
942*4d7e907cSAndroid Build Coastguard Worker         }
943*4d7e907cSAndroid Build Coastguard Worker     }
944*4d7e907cSAndroid Build Coastguard Worker 
945*4d7e907cSAndroid Build Coastguard Worker     if (metadata.exists(ANDROID_JPEG_GPS_PROCESSING_METHOD)) {
946*4d7e907cSAndroid Build Coastguard Worker         entry = metadata.find(ANDROID_JPEG_GPS_PROCESSING_METHOD);
947*4d7e907cSAndroid Build Coastguard Worker         std::string method_str(reinterpret_cast<const char*>(entry.data.u8));
948*4d7e907cSAndroid Build Coastguard Worker         if (!setGpsProcessingMethod(method_str)) {
949*4d7e907cSAndroid Build Coastguard Worker             ALOGE("%s: setting gps processing method failed.", __FUNCTION__);
950*4d7e907cSAndroid Build Coastguard Worker             return false;
951*4d7e907cSAndroid Build Coastguard Worker         }
952*4d7e907cSAndroid Build Coastguard Worker     }
953*4d7e907cSAndroid Build Coastguard Worker 
954*4d7e907cSAndroid Build Coastguard Worker     if (time_available && metadata.exists(ANDROID_JPEG_GPS_TIMESTAMP)) {
955*4d7e907cSAndroid Build Coastguard Worker         entry = metadata.find(ANDROID_JPEG_GPS_TIMESTAMP);
956*4d7e907cSAndroid Build Coastguard Worker         time_t timestamp = static_cast<time_t>(entry.data.i64[0]);
957*4d7e907cSAndroid Build Coastguard Worker         if (gmtime_r(&timestamp, &time_info)) {
958*4d7e907cSAndroid Build Coastguard Worker             if (!setGpsTimestamp(time_info)) {
959*4d7e907cSAndroid Build Coastguard Worker                 ALOGE("%s: setting gps timestamp failed.", __FUNCTION__);
960*4d7e907cSAndroid Build Coastguard Worker                 return false;
961*4d7e907cSAndroid Build Coastguard Worker             }
962*4d7e907cSAndroid Build Coastguard Worker         } else {
963*4d7e907cSAndroid Build Coastguard Worker             ALOGE("%s: Time tranformation failed.", __FUNCTION__);
964*4d7e907cSAndroid Build Coastguard Worker             return false;
965*4d7e907cSAndroid Build Coastguard Worker         }
966*4d7e907cSAndroid Build Coastguard Worker     }
967*4d7e907cSAndroid Build Coastguard Worker 
968*4d7e907cSAndroid Build Coastguard Worker     if (metadata.exists(ANDROID_JPEG_ORIENTATION)) {
969*4d7e907cSAndroid Build Coastguard Worker         entry = metadata.find(ANDROID_JPEG_ORIENTATION);
970*4d7e907cSAndroid Build Coastguard Worker         if (!setOrientation(entry.data.i32[0])) {
971*4d7e907cSAndroid Build Coastguard Worker             ALOGE("%s: setting orientation failed.", __FUNCTION__);
972*4d7e907cSAndroid Build Coastguard Worker             return false;
973*4d7e907cSAndroid Build Coastguard Worker         }
974*4d7e907cSAndroid Build Coastguard Worker     }
975*4d7e907cSAndroid Build Coastguard Worker 
976*4d7e907cSAndroid Build Coastguard Worker     if (metadata.exists(ANDROID_SENSOR_EXPOSURE_TIME)) {
977*4d7e907cSAndroid Build Coastguard Worker         entry = metadata.find(ANDROID_SENSOR_EXPOSURE_TIME);
978*4d7e907cSAndroid Build Coastguard Worker         // int64_t of nanoseconds
979*4d7e907cSAndroid Build Coastguard Worker         if (!setExposureTime(entry.data.i64[0], 1000000000u)) {
980*4d7e907cSAndroid Build Coastguard Worker             ALOGE("%s: setting exposure time failed.", __FUNCTION__);
981*4d7e907cSAndroid Build Coastguard Worker             return false;
982*4d7e907cSAndroid Build Coastguard Worker         }
983*4d7e907cSAndroid Build Coastguard Worker     }
984*4d7e907cSAndroid Build Coastguard Worker 
985*4d7e907cSAndroid Build Coastguard Worker     if (metadata.exists(ANDROID_LENS_APERTURE)) {
986*4d7e907cSAndroid Build Coastguard Worker         const int kAperturePrecision = 10000;
987*4d7e907cSAndroid Build Coastguard Worker         entry = metadata.find(ANDROID_LENS_APERTURE);
988*4d7e907cSAndroid Build Coastguard Worker         if (!setFNumber(entry.data.f[0] * kAperturePrecision, kAperturePrecision)) {
989*4d7e907cSAndroid Build Coastguard Worker             ALOGE("%s: setting F number failed.", __FUNCTION__);
990*4d7e907cSAndroid Build Coastguard Worker             return false;
991*4d7e907cSAndroid Build Coastguard Worker         }
992*4d7e907cSAndroid Build Coastguard Worker     }
993*4d7e907cSAndroid Build Coastguard Worker 
994*4d7e907cSAndroid Build Coastguard Worker     if (metadata.exists(ANDROID_FLASH_INFO_AVAILABLE)) {
995*4d7e907cSAndroid Build Coastguard Worker         entry = metadata.find(ANDROID_FLASH_INFO_AVAILABLE);
996*4d7e907cSAndroid Build Coastguard Worker         if (entry.data.u8[0] == ANDROID_FLASH_INFO_AVAILABLE_FALSE) {
997*4d7e907cSAndroid Build Coastguard Worker             const uint32_t kNoFlashFunction = 0x20;
998*4d7e907cSAndroid Build Coastguard Worker             if (!setFlash(kNoFlashFunction)) {
999*4d7e907cSAndroid Build Coastguard Worker                 ALOGE("%s: setting flash failed.", __FUNCTION__);
1000*4d7e907cSAndroid Build Coastguard Worker                 return false;
1001*4d7e907cSAndroid Build Coastguard Worker             }
1002*4d7e907cSAndroid Build Coastguard Worker         } else {
1003*4d7e907cSAndroid Build Coastguard Worker             ALOGE("%s: Unsupported flash info: %d", __FUNCTION__, entry.data.u8[0]);
1004*4d7e907cSAndroid Build Coastguard Worker             return false;
1005*4d7e907cSAndroid Build Coastguard Worker         }
1006*4d7e907cSAndroid Build Coastguard Worker     }
1007*4d7e907cSAndroid Build Coastguard Worker 
1008*4d7e907cSAndroid Build Coastguard Worker     if (metadata.exists(ANDROID_CONTROL_AWB_MODE)) {
1009*4d7e907cSAndroid Build Coastguard Worker         entry = metadata.find(ANDROID_CONTROL_AWB_MODE);
1010*4d7e907cSAndroid Build Coastguard Worker         if (entry.data.u8[0] == ANDROID_CONTROL_AWB_MODE_AUTO) {
1011*4d7e907cSAndroid Build Coastguard Worker             const uint16_t kAutoWhiteBalance = 0;
1012*4d7e907cSAndroid Build Coastguard Worker             if (!setWhiteBalance(kAutoWhiteBalance)) {
1013*4d7e907cSAndroid Build Coastguard Worker                 ALOGE("%s: setting white balance failed.", __FUNCTION__);
1014*4d7e907cSAndroid Build Coastguard Worker                 return false;
1015*4d7e907cSAndroid Build Coastguard Worker             }
1016*4d7e907cSAndroid Build Coastguard Worker         } else {
1017*4d7e907cSAndroid Build Coastguard Worker             ALOGE("%s: Unsupported awb mode: %d", __FUNCTION__, entry.data.u8[0]);
1018*4d7e907cSAndroid Build Coastguard Worker             return false;
1019*4d7e907cSAndroid Build Coastguard Worker         }
1020*4d7e907cSAndroid Build Coastguard Worker     }
1021*4d7e907cSAndroid Build Coastguard Worker 
1022*4d7e907cSAndroid Build Coastguard Worker     if (time_available) {
1023*4d7e907cSAndroid Build Coastguard Worker         char str[4];
1024*4d7e907cSAndroid Build Coastguard Worker         if (snprintf(str, sizeof(str), "%03ld", tp.tv_nsec / 1000000) < 0) {
1025*4d7e907cSAndroid Build Coastguard Worker             ALOGE("%s: Subsec is invalid: %ld", __FUNCTION__, tp.tv_nsec);
1026*4d7e907cSAndroid Build Coastguard Worker             return false;
1027*4d7e907cSAndroid Build Coastguard Worker         }
1028*4d7e907cSAndroid Build Coastguard Worker         if (!setSubsecTime(std::string(str))) {
1029*4d7e907cSAndroid Build Coastguard Worker             ALOGE("%s: setting subsec time failed.", __FUNCTION__);
1030*4d7e907cSAndroid Build Coastguard Worker             return false;
1031*4d7e907cSAndroid Build Coastguard Worker         }
1032*4d7e907cSAndroid Build Coastguard Worker     }
1033*4d7e907cSAndroid Build Coastguard Worker 
1034*4d7e907cSAndroid Build Coastguard Worker     return true;
1035*4d7e907cSAndroid Build Coastguard Worker }
1036*4d7e907cSAndroid Build Coastguard Worker 
1037*4d7e907cSAndroid Build Coastguard Worker }  // namespace helper
1038*4d7e907cSAndroid Build Coastguard Worker }  // namespace common
1039*4d7e907cSAndroid Build Coastguard Worker }  // namespace camera
1040*4d7e907cSAndroid Build Coastguard Worker }  // namespace hardware
1041*4d7e907cSAndroid Build Coastguard Worker }  // namespace android
1042