• Home
  • Line#
  • Scopes#
  • Navigate#
  • Raw
  • Download
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, &degree, &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