1 /*
2 * Copyright (c) 2022 Huawei Device Co., Ltd.
3 * Licensed under the Apache License, Version 2.0 (the "License");
4 * you may not use this file except in compliance with the License.
5 * You may obtain a copy of the License at
6 *
7 * http://www.apache.org/licenses/LICENSE-2.0
8 *
9 * Unless required by applicable law or agreed to in writing, software
10 * distributed under the License is distributed on an "AS IS" BASIS,
11 * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
12 * See the License for the specific language governing permissions and
13 * limitations under the License.
14 */
15
16 #include "exif_utils.h"
17 #include <cassert>
18 #include <cstdio>
19 #include <cstdint>
20 #include <cmath>
21 #include <cstring>
22 #include <iostream>
23 #include <camera.h>
24 #include "securec.h"
25
26 namespace OHOS::Camera {
27 static const unsigned int IMAGE_DATA_OFFSET = 20;
28
29 // Raw exif header data
30 static const unsigned char EXIF_HEADER[] = {0xff, 0xd8, 0xff, 0xe1};
31
32 static const unsigned int EXIF_HEADER_LENGTH = sizeof(EXIF_HEADER);
33
34 #define FILE_BYTE_ORDER EXIF_BYTE_ORDER_INTEL
35
CreateTag(ExifData * exif,ExifIfd ifd,ExifTag tag,size_t len,ExifFormat format)36 static ExifEntry *CreateTag(ExifData *exif, ExifIfd ifd, ExifTag tag, size_t len, ExifFormat format)
37 {
38 void *buf = nullptr;
39 ExifEntry *entry = nullptr;
40
41 ExifMem *mem = exif_mem_new_default();
42 assert(mem != NULL);
43
44 entry = exif_entry_new_mem(mem);
45 assert(entry != nullptr);
46
47 buf = exif_mem_alloc(mem, len);
48 assert(buf != nullptr);
49
50 entry->data = static_cast<unsigned char*>(buf);
51 entry->size = len;
52 entry->tag = tag;
53 entry->components = len;
54 entry->format = format;
55
56 exif_content_add_entry(exif->ifd[ifd], entry);
57
58 exif_mem_unref(mem);
59 exif_entry_unref(entry);
60
61 return entry;
62 }
63
AddLatOrLongInfo(ExifData * exif,double number,LatOrLong latOrLongType)64 uint32_t ExifUtils::AddLatOrLongInfo(ExifData *exif,
65 double number, LatOrLong latOrLongType)
66 {
67 ExifEntry *entry = nullptr;
68 char gpsRef[2] = {0}; // Index
69 ExifRational gpsRational[3]; // Index
70 char north[2] = "N";
71 char south[2] = "S";
72 char east[2] = "E";
73 char west[2] = "W";
74 int32_t degree = 0;
75 int32_t minute = 0;
76 int32_t second = 0;
77
78 if (latOrLongType == LATITUDE_TYPE) {
79 if (number > 0) {
80 if (strncpy_s(gpsRef, sizeof(gpsRef), north, strlen(north)) != 0) {
81 CAMERA_LOGE("%{public}s exif strncpy_s failed.", __FUNCTION__);
82 return RC_ERROR;
83 }
84 } else {
85 number = abs(number);
86 if (strncpy_s(gpsRef, sizeof(gpsRef), south, strlen(south)) != 0) {
87 CAMERA_LOGE("%{public}s exif strncpy_s failed.", __FUNCTION__);
88 return RC_ERROR;
89 }
90 }
91 } else {
92 if (number > 0) {
93 if (strncpy_s(gpsRef, sizeof(gpsRef), east, strlen(east)) != 0) {
94 CAMERA_LOGE("%{public}s exif strncpy_s failed.", __FUNCTION__);
95 return RC_ERROR;
96 }
97 } else {
98 number = abs(number);
99 if (strncpy_s(gpsRef, sizeof(gpsRef), west, strlen(west)) != 0) {
100 CAMERA_LOGE("%{public}s exif strncpy_s failed.", __FUNCTION__);
101 return RC_ERROR;
102 }
103 }
104 }
105 ConvertGpsDataToDms(number, °ree, &minute, &second);
106 gpsRational[0].numerator = degree; // Index
107 gpsRational[0].denominator = 1;
108 gpsRational[1].numerator = minute; // Index
109 gpsRational[1].denominator = 1;
110 gpsRational[2].numerator = second; // Index
111 gpsRational[2].denominator = 1;
112
113 // LATITUDE_TYPE/LONGITUDE_TYPE reference
114 if (latOrLongType == LATITUDE_TYPE) {
115 entry = CreateTag(exif, EXIF_IFD_GPS, EXIF_TAG_GPS_LATITUDE_REF, sizeof(gpsRef), EXIF_FORMAT_ASCII);
116 } else {
117 entry = CreateTag(exif, EXIF_IFD_GPS, EXIF_TAG_GPS_LONGITUDE_REF, sizeof(gpsRef), EXIF_FORMAT_ASCII);
118 }
119 if (memcpy_s(entry->data, entry->size, gpsRef, sizeof(gpsRef)) != 0) {
120 CAMERA_LOGE("%{public}s exif memcpy_s failed.", __FUNCTION__);
121 return RC_ERROR;
122 }
123 // LATITUDE_TYPE/LONGITUDE_TYPE value
124 constexpr uint32_t gpsDmsCount = 3;
125 if (latOrLongType == LATITUDE_TYPE) {
126 entry = CreateTag(exif, EXIF_IFD_GPS, EXIF_TAG_GPS_LATITUDE,
127 gpsDmsCount * exif_format_get_size(EXIF_FORMAT_RATIONAL),
128 EXIF_FORMAT_RATIONAL);
129 } else {
130 entry = CreateTag(exif, EXIF_IFD_GPS, EXIF_TAG_GPS_LONGITUDE,
131 gpsDmsCount * exif_format_get_size(EXIF_FORMAT_RATIONAL),
132 EXIF_FORMAT_RATIONAL);
133 }
134 exif_set_rational(entry->data, FILE_BYTE_ORDER, gpsRational[0]);
135 exif_set_rational(entry->data + 8, FILE_BYTE_ORDER, gpsRational[1]); // 8bit
136 exif_set_rational(entry->data + 16, FILE_BYTE_ORDER, gpsRational[2]); // 16bit
137 return RC_OK;
138 }
139
AddAltitudeInfo(ExifData * exif,double altitude)140 uint32_t ExifUtils::AddAltitudeInfo(ExifData *exif, double altitude)
141 {
142 unsigned char seaLevelFlag = 0;
143 ExifEntry *entry = nullptr;
144 ExifRational gpsAltitudeRational;
145 exif_rational altitudeRational;
146
147 if (altitude > 0) {
148 seaLevelFlag = 0;
149 } else {
150 altitude = abs(altitude);
151 seaLevelFlag = 1;
152 }
153 ConvertAltitudeToRational(altitude, altitudeRational);
154 gpsAltitudeRational.numerator = altitudeRational.numerator;
155 gpsAltitudeRational.denominator = altitudeRational.denominator;
156 // Altitude reference
157 entry = CreateTag(exif, EXIF_IFD_GPS, EXIF_TAG_GPS_ALTITUDE_REF, sizeof(seaLevelFlag), EXIF_FORMAT_BYTE);
158 exif_set_short(entry->data, FILE_BYTE_ORDER, seaLevelFlag);
159
160 // Altitude value
161 entry = CreateTag(exif, EXIF_IFD_GPS, EXIF_TAG_GPS_ALTITUDE, exif_format_get_size(EXIF_FORMAT_RATIONAL),
162 EXIF_FORMAT_RATIONAL);
163 exif_set_rational(entry->data, FILE_BYTE_ORDER, gpsAltitudeRational);
164 return RC_OK;
165 }
166
IsJpegPicture(unsigned char * dataBuffer,int32_t dataBufferSize,void * address)167 uint32_t ExifUtils::IsJpegPicture(unsigned char *dataBuffer, int32_t dataBufferSize, void *address)
168 {
169 if (memcpy_s(dataBuffer, dataBufferSize, address, dataBufferSize) != 0) {
170 CAMERA_LOGE("%{public}s exif memcpy_s failed.", __FUNCTION__);
171 return RC_ERROR;
172 }
173
174 if (!(dataBuffer[0] == 0xFF && dataBuffer[1] == 0xD8)) {
175 CAMERA_LOGE("%{public}s not jpeg file,won't add exif for it.", __FUNCTION__);
176 return RC_ERROR;
177 }
178
179 if ((dataBuffer[6] == 'E' && dataBuffer[7] == 'x' && dataBuffer[8] == 'i' && dataBuffer[9] == 'f')) { // Index
180 CAMERA_LOGE("%{public}s already add exif, won't overwrite exif info.", __FUNCTION__);
181 return RC_ERROR;
182 }
183 return RC_OK;
184 }
185
PackageJpeg(unsigned char * tempBuffer,int32_t totalTempBufferSize,unsigned char * exifData,unsigned int exifDataLength,data_info sourceData)186 uint32_t ExifUtils::PackageJpeg(unsigned char *tempBuffer, int32_t totalTempBufferSize, unsigned char *exifData,
187 unsigned int exifDataLength, data_info sourceData)
188 {
189 unsigned char orderValue = 0;
190 unsigned char value = 0;
191 constexpr uint32_t exifBlockLength = 2;
192 orderValue = (exifDataLength + exifBlockLength) >> 8; // 8bit
193 value = (exifDataLength + exifBlockLength) & 0xff;
194 if (memcpy_s(tempBuffer, totalTempBufferSize, EXIF_HEADER, EXIF_HEADER_LENGTH) != 0) {
195 CAMERA_LOGE("%{public}s exif memcpy_s failed.", __FUNCTION__);
196 return RC_ERROR;
197 }
198 if (memcpy_s(tempBuffer + EXIF_HEADER_LENGTH, totalTempBufferSize, &orderValue,
199 sizeof(orderValue)) != 0) {
200 CAMERA_LOGE("%{public}s exif memcpy_s failed.", __FUNCTION__);
201 return RC_ERROR;
202 }
203 if (memcpy_s(tempBuffer + EXIF_HEADER_LENGTH + sizeof(orderValue), totalTempBufferSize, &value,
204 sizeof(value)) != 0) {
205 CAMERA_LOGE("%{public}s exif memcpy_s failed.", __FUNCTION__);
206 return RC_ERROR;
207 }
208 if (memcpy_s(tempBuffer + EXIF_HEADER_LENGTH + sizeof(orderValue) + sizeof(value), totalTempBufferSize,
209 exifData, exifDataLength) != 0) {
210 CAMERA_LOGE("%{public}s exif memcpy_s failed.", __FUNCTION__);
211 return RC_ERROR;
212 }
213 if (memcpy_s(tempBuffer + EXIF_HEADER_LENGTH + sizeof(orderValue) + sizeof(value) + exifDataLength,
214 totalTempBufferSize,
215 sourceData.dataBuffer + IMAGE_DATA_OFFSET,
216 sourceData.dataBufferSize - IMAGE_DATA_OFFSET) != 0) {
217 CAMERA_LOGE("%{public}s exif memcpy_s failed.", __FUNCTION__);
218 return RC_ERROR;
219 }
220 return RC_OK;
221 }
222
AddCustomExifInfo(exif_data info,void * address,int32_t & outPutSize)223 uint32_t ExifUtils::AddCustomExifInfo(exif_data info,
224 void *address, int32_t &outPutSize)
225 {
226 int32_t ret = RC_ERROR;
227 unsigned char *exifData = nullptr;
228 unsigned int exifDataLength = 0;
229 ExifData *exif = nullptr;
230 unsigned char *dataBuffer = nullptr;
231 unsigned char *tempBuffer = nullptr;
232 int32_t totalTempBufferSize = 0;
233 int32_t dataBufferSize = info.frame_size;
234 double latitudeNumber = info.latitude;
235 double longitudeNumber = info.longitude;
236 constexpr uint32_t exifBlockLength = 2;
237
238 exif = exif_data_new();
239 if (!exif) {
240 CAMERA_LOGE("%{public}s exif new failed.", __FUNCTION__);
241 return ret;
242 }
243 exif_data_set_option(exif, EXIF_DATA_OPTION_FOLLOW_SPECIFICATION);
244 exif_data_set_data_type(exif, EXIF_DATA_TYPE_COMPRESSED);
245 exif_data_set_byte_order(exif, FILE_BYTE_ORDER);
246 if (AddLatOrLongInfo(exif, latitudeNumber, LATITUDE_TYPE) != RC_OK) {
247 return RC_ERROR;
248 }
249 if (AddLatOrLongInfo(exif, longitudeNumber, LONGITUDE_TYPE) != RC_OK) {
250 return RC_ERROR;
251 }
252 if (AddAltitudeInfo(exif, info.altitude) != RC_OK) {
253 return RC_ERROR;
254 }
255 exif_data_save_data(exif, &exifData, &exifDataLength);
256
257 dataBuffer = static_cast<unsigned char *>(malloc(dataBufferSize));
258 if (!dataBuffer) {
259 CAMERA_LOGE("%{public}s Allocate data buf failed.", __FUNCTION__);
260 return ret;
261 }
262 data_info sourceData;
263 sourceData.dataBuffer = dataBuffer;
264 sourceData.dataBufferSize = dataBufferSize;
265
266 // Check buffer whether is valid
267 if (IsJpegPicture(dataBuffer, dataBufferSize, address) == RC_ERROR) {
268 goto error;
269 }
270 totalTempBufferSize = EXIF_HEADER_LENGTH + exifBlockLength + exifDataLength + (dataBufferSize - IMAGE_DATA_OFFSET);
271 tempBuffer = static_cast<unsigned char *>(malloc(totalTempBufferSize));
272 if (!tempBuffer) {
273 CAMERA_LOGE("%{public}s Allocate temp buf failed.", __FUNCTION__);
274 return ret;
275 }
276 ret = PackageJpeg(tempBuffer, totalTempBufferSize, exifData, exifDataLength, sourceData);
277 outPutSize = totalTempBufferSize;
278 if (memcpy_s(address, totalTempBufferSize, tempBuffer, totalTempBufferSize) != 0) {
279 CAMERA_LOGE("%{public}s exif memcpy_s failed.", __FUNCTION__);
280 return RC_ERROR;
281 }
282
283 error:
284 if (dataBuffer != nullptr) {
285 free(dataBuffer);
286 }
287 if (tempBuffer != nullptr) {
288 free(tempBuffer);
289 }
290 free(exifData);
291 exif_data_unref(exif);
292
293 return ret;
294 }
295
ConvertGpsDataToDms(double number,int32_t * degrees,int32_t * minutes,int32_t * seconds)296 void ExifUtils::ConvertGpsDataToDms(double number, int32_t *degrees, int32_t *minutes, int32_t *seconds)
297 {
298 double approximateNumber = 0.0;
299 constexpr uint32_t timePeriod = 60;
300 constexpr uint32_t roundingValue = 5;
301 constexpr uint32_t precision = 10;
302 int32_t hour = static_cast<int32_t>(number);
303 int32_t minute = static_cast<int32_t>((number - hour) * timePeriod);
304 int32_t second = static_cast<int32_t>(((number - hour) * timePeriod - minute) * timePeriod);
305
306 approximateNumber = ((number - hour) * timePeriod - minute) * timePeriod - second;
307 if (static_cast<int32_t>(approximateNumber * precision) >= roundingValue) {
308 second = second + 1;
309 }
310 if (second == timePeriod) {
311 second = 0;
312 minute = minute + 1;
313 }
314 if (minute == timePeriod) {
315 minute = 0;
316 hour = hour + 1;
317 }
318 *degrees = hour;
319 *minutes = minute;
320 *seconds = second;
321
322 return;
323 }
324
ConvertAltitudeToRational(double altitude,exif_rational & outPutAltitude)325 void ExifUtils::ConvertAltitudeToRational(double altitude, exif_rational &outPutAltitude)
326 {
327 long long numerator = 0;
328 long long denominator = 1;
329 bool isSeparator = false;
330 int count = 0;
331 std::string strData = "";
332 strData = std::to_string(altitude);
333 CAMERA_LOGI("%{public}s strData = %{public}s", __FUNCTION__, strData.c_str());
334
335 count = strData.length();
336 CAMERA_LOGI("%{public}s count = %{public}d", __FUNCTION__, count);
337 constexpr uint32_t digitPosition = 10;
338 for (int i = 0; i < count; i++) {
339 char character = strData[i];
340 if (character == '.') {
341 isSeparator = true;
342 } else {
343 numerator = numerator * digitPosition + (character - '0');
344 CAMERA_LOGI("%{public}s numerator = %{public}lld", __FUNCTION__, numerator);
345 if (isSeparator) {
346 denominator *= digitPosition;
347 CAMERA_LOGI("%{public}s denominator = %{public}lld", __FUNCTION__, denominator);
348 }
349 }
350 }
351 constexpr uint32_t commonDivisor = 2;
352 constexpr uint32_t resetValue = 1;
353 for (int i = commonDivisor; static_cast<long long>(i) < numerator; i++) {
354 if ((numerator % i == 0) && (denominator % i == 0)) {
355 numerator /= i;
356 denominator /= i;
357 i = resetValue;
358 }
359 }
360
361 outPutAltitude.numerator = numerator;
362 outPutAltitude.denominator = denominator;
363 CAMERA_LOGI("%{public}s outPutAltitude.numerator = %{public}d and outPutAltitude.denominator = %{public}d",
364 __FUNCTION__, outPutAltitude.numerator, outPutAltitude.denominator);
365 }
366 } // namespace OHOS::Camera
367