xref: /dokuwiki/inc/JpegMeta.php (revision 6c16a3a9aa602bb7e269fb6d5d18e1353e17f97f)
155efc227SAndreas Gohr<?php
255efc227SAndreas Gohr/**
355efc227SAndreas Gohr * JPEG metadata reader/writer
455efc227SAndreas Gohr *
52bd74074SAndreas Gohr * @license    BSD <http://www.opensource.org/licenses/bsd-license.php>
62bd74074SAndreas Gohr * @link       http://github.com/sd/jpeg-php
755efc227SAndreas Gohr * @author     Sebastian Delmont <sdelmont@zonageek.com>
855efc227SAndreas Gohr * @author     Andreas Gohr <andi@splitbrain.org>
9431c7fc8Shakan.sandell * @author     Hakan Sandell <hakan.sandell@mydata.se>
1036df6fa3SAndreas Gohr * @todo       Add support for Maker Notes, Extend for GIF and PNG metadata
1155efc227SAndreas Gohr */
1255efc227SAndreas Gohr
132bd74074SAndreas Gohr// Original copyright notice:
1455efc227SAndreas Gohr//
152bd74074SAndreas Gohr// Copyright (c) 2003 Sebastian Delmont <sdelmont@zonageek.com>
162bd74074SAndreas Gohr// All rights reserved.
172bd74074SAndreas Gohr//
182bd74074SAndreas Gohr// Redistribution and use in source and binary forms, with or without
192bd74074SAndreas Gohr// modification, are permitted provided that the following conditions
202bd74074SAndreas Gohr// are met:
212bd74074SAndreas Gohr// 1. Redistributions of source code must retain the above copyright
222bd74074SAndreas Gohr//    notice, this list of conditions and the following disclaimer.
232bd74074SAndreas Gohr// 2. Redistributions in binary form must reproduce the above copyright
242bd74074SAndreas Gohr//    notice, this list of conditions and the following disclaimer in the
252bd74074SAndreas Gohr//    documentation and/or other materials provided with the distribution.
262bd74074SAndreas Gohr// 3. Neither the name of the author nor the names of its contributors
272bd74074SAndreas Gohr//    may be used to endorse or promote products derived from this software
282bd74074SAndreas Gohr//    without specific prior written permission.
292bd74074SAndreas Gohr//
302bd74074SAndreas Gohr// THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS
312bd74074SAndreas Gohr// IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED
322bd74074SAndreas Gohr// TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A
332bd74074SAndreas Gohr// PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT
342bd74074SAndreas Gohr// HOLDER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL,
352bd74074SAndreas Gohr// SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED
362bd74074SAndreas Gohr// TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR
372bd74074SAndreas Gohr// PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF
382bd74074SAndreas Gohr// LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING
392bd74074SAndreas Gohr// NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS
402bd74074SAndreas Gohr// SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE
4155efc227SAndreas Gohr
420b17fdc6SAndreas Gohrclass JpegMeta {
4355efc227SAndreas Gohr    var $_fileName;
4455efc227SAndreas Gohr    var $_fp = null;
4559bc3b48SGerrit Uitslag    var $_fpout = null;
4655efc227SAndreas Gohr    var $_type = 'unknown';
4755efc227SAndreas Gohr
4855efc227SAndreas Gohr    var $_markers;
4955efc227SAndreas Gohr    var $_info;
5055efc227SAndreas Gohr
5155efc227SAndreas Gohr
5255efc227SAndreas Gohr    /**
5355efc227SAndreas Gohr     * Constructor
5455efc227SAndreas Gohr     *
5555efc227SAndreas Gohr     * @author Sebastian Delmont <sdelmont@zonageek.com>
56f50a239bSTakamura     *
57f50a239bSTakamura     * @param $fileName
5855efc227SAndreas Gohr     */
595c3b310dSChristopher Smith    function __construct($fileName) {
6055efc227SAndreas Gohr
6155efc227SAndreas Gohr        $this->_fileName = $fileName;
6255efc227SAndreas Gohr
6355efc227SAndreas Gohr        $this->_fp = null;
6455efc227SAndreas Gohr        $this->_type = 'unknown';
6555efc227SAndreas Gohr
6655efc227SAndreas Gohr        unset($this->_info);
6755efc227SAndreas Gohr        unset($this->_markers);
6855efc227SAndreas Gohr    }
6955efc227SAndreas Gohr
7055efc227SAndreas Gohr    /**
7155efc227SAndreas Gohr     * Returns all gathered info as multidim array
7255efc227SAndreas Gohr     *
7355efc227SAndreas Gohr     * @author Sebastian Delmont <sdelmont@zonageek.com>
7455efc227SAndreas Gohr     */
750b17fdc6SAndreas Gohr    function & getRawInfo() {
7655efc227SAndreas Gohr        $this->_parseAll();
7755efc227SAndreas Gohr
7855efc227SAndreas Gohr        if ($this->_markers == null) {
7955efc227SAndreas Gohr            return false;
8055efc227SAndreas Gohr        }
8155efc227SAndreas Gohr
8255efc227SAndreas Gohr        return $this->_info;
8355efc227SAndreas Gohr    }
8455efc227SAndreas Gohr
8555efc227SAndreas Gohr    /**
8655efc227SAndreas Gohr     * Returns basic image info
8755efc227SAndreas Gohr     *
8855efc227SAndreas Gohr     * @author Sebastian Delmont <sdelmont@zonageek.com>
8955efc227SAndreas Gohr     */
900b17fdc6SAndreas Gohr    function & getBasicInfo() {
9155efc227SAndreas Gohr        $this->_parseAll();
9255efc227SAndreas Gohr
9355efc227SAndreas Gohr        $info = array();
9455efc227SAndreas Gohr
9555efc227SAndreas Gohr        if ($this->_markers == null) {
9655efc227SAndreas Gohr            return false;
9755efc227SAndreas Gohr        }
9855efc227SAndreas Gohr
9955efc227SAndreas Gohr        $info['Name'] = $this->_info['file']['Name'];
10055efc227SAndreas Gohr        if (isset($this->_info['file']['Url'])) {
10155efc227SAndreas Gohr            $info['Url'] = $this->_info['file']['Url'];
10255efc227SAndreas Gohr            $info['NiceSize'] = "???KB";
1030b17fdc6SAndreas Gohr        } else {
10455efc227SAndreas Gohr            $info['Size'] = $this->_info['file']['Size'];
10555efc227SAndreas Gohr            $info['NiceSize'] = $this->_info['file']['NiceSize'];
10655efc227SAndreas Gohr        }
10755efc227SAndreas Gohr
10855efc227SAndreas Gohr        if (@isset($this->_info['sof']['Format'])) {
10955efc227SAndreas Gohr            $info['Format'] = $this->_info['sof']['Format'] . " JPEG";
1100b17fdc6SAndreas Gohr        } else {
11155efc227SAndreas Gohr            $info['Format'] = $this->_info['sof']['Format'] . " JPEG";
11255efc227SAndreas Gohr        }
11355efc227SAndreas Gohr
11455efc227SAndreas Gohr        if (@isset($this->_info['sof']['ColorChannels'])) {
11555efc227SAndreas Gohr            $info['ColorMode'] = ($this->_info['sof']['ColorChannels'] > 1) ? "Color" : "B&W";
11655efc227SAndreas Gohr        }
11755efc227SAndreas Gohr
11855efc227SAndreas Gohr        $info['Width'] = $this->getWidth();
11955efc227SAndreas Gohr        $info['Height'] = $this->getHeight();
12055efc227SAndreas Gohr        $info['DimStr'] = $this->getDimStr();
12155efc227SAndreas Gohr
12255efc227SAndreas Gohr        $dates = $this->getDates();
12355efc227SAndreas Gohr
12455efc227SAndreas Gohr        $info['DateTime'] = $dates['EarliestTime'];
12555efc227SAndreas Gohr        $info['DateTimeStr'] = $dates['EarliestTimeStr'];
12655efc227SAndreas Gohr
12755efc227SAndreas Gohr        $info['HasThumbnail'] = $this->hasThumbnail();
12855efc227SAndreas Gohr
12955efc227SAndreas Gohr        return $info;
13055efc227SAndreas Gohr    }
13155efc227SAndreas Gohr
13255efc227SAndreas Gohr
13355efc227SAndreas Gohr    /**
13455efc227SAndreas Gohr     * Convinience function to access nearly all available Data
13555efc227SAndreas Gohr     * through one function
13655efc227SAndreas Gohr     *
13755efc227SAndreas Gohr     * @author Andreas Gohr <andi@splitbrain.org>
13842ea7f44SGerrit Uitslag     *
13942ea7f44SGerrit Uitslag     * @param array|string $fields field name or array with field names
14042ea7f44SGerrit Uitslag     * @return bool|string
14155efc227SAndreas Gohr     */
1420b17fdc6SAndreas Gohr    function getField($fields) {
14355efc227SAndreas Gohr        if(!is_array($fields)) $fields = array($fields);
14455efc227SAndreas Gohr        $info = false;
14555efc227SAndreas Gohr        foreach($fields as $field){
146*6c16a3a9Sfiwswe            $lower_field = strtolower($field);
147*6c16a3a9Sfiwswe            if(str_starts_with($lower_field, 'iptc.')){
14855efc227SAndreas Gohr                $info = $this->getIPTCField(substr($field,5));
149*6c16a3a9Sfiwswe            }elseif(str_starts_with($lower_field, 'exif.')){
15055efc227SAndreas Gohr                $info = $this->getExifField(substr($field,5));
151*6c16a3a9Sfiwswe            }elseif(str_starts_with($lower_field, 'xmp.')){
152431c7fc8Shakan.sandell                $info = $this->getXmpField(substr($field,4));
153*6c16a3a9Sfiwswe            }elseif(str_starts_with($lower_field, 'file.')){
15455efc227SAndreas Gohr                $info = $this->getFileField(substr($field,5));
155*6c16a3a9Sfiwswe            }elseif(str_starts_with($lower_field, 'date.')){
15655efc227SAndreas Gohr                $info = $this->getDateField(substr($field,5));
157*6c16a3a9Sfiwswe            }elseif($lower_field == 'simple.camera'){
15855efc227SAndreas Gohr                $info = $this->getCamera();
159*6c16a3a9Sfiwswe            }elseif($lower_field == 'simple.raw'){
16055efc227SAndreas Gohr                return $this->getRawInfo();
161*6c16a3a9Sfiwswe            }elseif($lower_field == 'simple.title'){
16255efc227SAndreas Gohr                $info = $this->getTitle();
163*6c16a3a9Sfiwswe            }elseif($lower_field == 'simple.shutterspeed'){
1646db72d46SJoe Lapp                $info = $this->getShutterSpeed();
16555efc227SAndreas Gohr            }else{
16655efc227SAndreas Gohr                $info = $this->getExifField($field);
16755efc227SAndreas Gohr            }
16855efc227SAndreas Gohr            if($info != false) break;
16955efc227SAndreas Gohr        }
17055efc227SAndreas Gohr
1715aaca723SGerrit Uitslag        if($info === false)  $info = '';
17255efc227SAndreas Gohr        if(is_array($info)){
17355efc227SAndreas Gohr            if(isset($info['val'])){
17455efc227SAndreas Gohr                $info = $info['val'];
17555efc227SAndreas Gohr            }else{
176e3b89425Sasivery                $arr = array();
177e3b89425Sasivery                foreach($info as $part){
178e3b89425Sasivery                    if(is_array($part)){
179e3b89425Sasivery                        if(isset($part['val'])){
180e3b89425Sasivery                            $arr[] = $part['val'];
181e3b89425Sasivery                        }else{
182e3b89425Sasivery                            $arr[] = join(', ',$part);
183e3b89425Sasivery                        }
184e3b89425Sasivery                    }else{
185e3b89425Sasivery                        $arr[] = $part;
186e3b89425Sasivery                    }
187e3b89425Sasivery                }
188e3b89425Sasivery                $info = join(', ',$arr);
18955efc227SAndreas Gohr            }
19055efc227SAndreas Gohr        }
19155efc227SAndreas Gohr        return trim($info);
19255efc227SAndreas Gohr    }
19355efc227SAndreas Gohr
19455efc227SAndreas Gohr    /**
19536df6fa3SAndreas Gohr     * Convinience function to set nearly all available Data
19636df6fa3SAndreas Gohr     * through one function
19736df6fa3SAndreas Gohr     *
19836df6fa3SAndreas Gohr     * @author Andreas Gohr <andi@splitbrain.org>
19942ea7f44SGerrit Uitslag     *
20042ea7f44SGerrit Uitslag     * @param string $field field name
20142ea7f44SGerrit Uitslag     * @param string $value
20242ea7f44SGerrit Uitslag     * @return bool success or fail
20336df6fa3SAndreas Gohr     */
2040b17fdc6SAndreas Gohr    function setField($field, $value) {
205*6c16a3a9Sfiwswe        $lower_field = strtolower($field);
206*6c16a3a9Sfiwswe        if(str_starts_with($lower_field, 'iptc.')){
20736df6fa3SAndreas Gohr            return $this->setIPTCField(substr($field,5),$value);
208*6c16a3a9Sfiwswe        }elseif(str_starts_with($lower_field, 'exif.')){
20936df6fa3SAndreas Gohr            return $this->setExifField(substr($field,5),$value);
21036df6fa3SAndreas Gohr        }else{
21136df6fa3SAndreas Gohr            return $this->setExifField($field,$value);
21236df6fa3SAndreas Gohr        }
21336df6fa3SAndreas Gohr    }
21436df6fa3SAndreas Gohr
21536df6fa3SAndreas Gohr    /**
21636df6fa3SAndreas Gohr     * Convinience function to delete nearly all available Data
21736df6fa3SAndreas Gohr     * through one function
21836df6fa3SAndreas Gohr     *
21936df6fa3SAndreas Gohr     * @author Andreas Gohr <andi@splitbrain.org>
22042ea7f44SGerrit Uitslag     *
22142ea7f44SGerrit Uitslag     * @param string $field field name
22242ea7f44SGerrit Uitslag     * @return bool
22336df6fa3SAndreas Gohr     */
2240b17fdc6SAndreas Gohr    function deleteField($field) {
225*6c16a3a9Sfiwswe        $lower_field = strtolower($field);
226*6c16a3a9Sfiwswe        if(str_starts_with($lower_field, 'iptc.')){
22736df6fa3SAndreas Gohr            return $this->deleteIPTCField(substr($field,5));
228*6c16a3a9Sfiwswe        }elseif(str_starts_with($lower_field, 'exif.')){
22936df6fa3SAndreas Gohr            return $this->deleteExifField(substr($field,5));
23036df6fa3SAndreas Gohr        }else{
23136df6fa3SAndreas Gohr            return $this->deleteExifField($field);
23236df6fa3SAndreas Gohr        }
23336df6fa3SAndreas Gohr    }
23436df6fa3SAndreas Gohr
23536df6fa3SAndreas Gohr    /**
23655efc227SAndreas Gohr     * Return a date field
23755efc227SAndreas Gohr     *
23855efc227SAndreas Gohr     * @author Andreas Gohr <andi@splitbrain.org>
23942ea7f44SGerrit Uitslag     *
24042ea7f44SGerrit Uitslag     * @param string $field
24142ea7f44SGerrit Uitslag     * @return false|string
24255efc227SAndreas Gohr     */
2430b17fdc6SAndreas Gohr    function getDateField($field) {
24455efc227SAndreas Gohr        if (!isset($this->_info['dates'])) {
24555efc227SAndreas Gohr            $this->_info['dates'] = $this->getDates();
24655efc227SAndreas Gohr        }
24755efc227SAndreas Gohr
24855efc227SAndreas Gohr        if (isset($this->_info['dates'][$field])) {
24955efc227SAndreas Gohr            return $this->_info['dates'][$field];
25055efc227SAndreas Gohr        }
25155efc227SAndreas Gohr
25255efc227SAndreas Gohr        return false;
25355efc227SAndreas Gohr    }
25455efc227SAndreas Gohr
25555efc227SAndreas Gohr    /**
25655efc227SAndreas Gohr     * Return a file info field
25755efc227SAndreas Gohr     *
25855efc227SAndreas Gohr     * @author Andreas Gohr <andi@splitbrain.org>
25942ea7f44SGerrit Uitslag     *
26042ea7f44SGerrit Uitslag     * @param string $field field name
26142ea7f44SGerrit Uitslag     * @return false|string
26255efc227SAndreas Gohr     */
2630b17fdc6SAndreas Gohr    function getFileField($field) {
26455efc227SAndreas Gohr        if (!isset($this->_info['file'])) {
26555efc227SAndreas Gohr            $this->_parseFileInfo();
26655efc227SAndreas Gohr        }
26755efc227SAndreas Gohr
26855efc227SAndreas Gohr        if (isset($this->_info['file'][$field])) {
26955efc227SAndreas Gohr            return $this->_info['file'][$field];
27055efc227SAndreas Gohr        }
27155efc227SAndreas Gohr
27255efc227SAndreas Gohr        return false;
27355efc227SAndreas Gohr    }
27455efc227SAndreas Gohr
27555efc227SAndreas Gohr    /**
27655efc227SAndreas Gohr     * Return the camera info (Maker and Model)
27755efc227SAndreas Gohr     *
27855efc227SAndreas Gohr     * @author Andreas Gohr <andi@splitbrain.org>
27955efc227SAndreas Gohr     * @todo   handle makernotes
28042ea7f44SGerrit Uitslag     *
28142ea7f44SGerrit Uitslag     * @return false|string
28255efc227SAndreas Gohr     */
28355efc227SAndreas Gohr    function getCamera(){
28455efc227SAndreas Gohr        $make  = $this->getField(array('Exif.Make','Exif.TIFFMake'));
28555efc227SAndreas Gohr        $model = $this->getField(array('Exif.Model','Exif.TIFFModel'));
28655efc227SAndreas Gohr        $cam = trim("$make $model");
28755efc227SAndreas Gohr        if(empty($cam)) return false;
28855efc227SAndreas Gohr        return $cam;
28955efc227SAndreas Gohr    }
29055efc227SAndreas Gohr
29155efc227SAndreas Gohr    /**
2926db72d46SJoe Lapp     * Return shutter speed as a ratio
2936db72d46SJoe Lapp     *
2946db72d46SJoe Lapp     * @author Joe Lapp <joe.lapp@pobox.com>
29542ea7f44SGerrit Uitslag     *
29642ea7f44SGerrit Uitslag     * @return string
2976db72d46SJoe Lapp     */
2980b17fdc6SAndreas Gohr    function getShutterSpeed() {
2996db72d46SJoe Lapp        if (!isset($this->_info['exif'])) {
3006db72d46SJoe Lapp            $this->_parseMarkerExif();
3016db72d46SJoe Lapp        }
3026db72d46SJoe Lapp        if(!isset($this->_info['exif']['ExposureTime'])){
3036db72d46SJoe Lapp            return '';
3046db72d46SJoe Lapp        }
3056db72d46SJoe Lapp
3066db72d46SJoe Lapp        $field = $this->_info['exif']['ExposureTime'];
3076db72d46SJoe Lapp        if($field['den'] == 1) return $field['num'];
3086db72d46SJoe Lapp        return $field['num'].'/'.$field['den'];
3096db72d46SJoe Lapp    }
3106db72d46SJoe Lapp
3116db72d46SJoe Lapp    /**
31255efc227SAndreas Gohr     * Return an EXIF field
31355efc227SAndreas Gohr     *
31455efc227SAndreas Gohr     * @author Sebastian Delmont <sdelmont@zonageek.com>
31542ea7f44SGerrit Uitslag     *
31642ea7f44SGerrit Uitslag     * @param string $field field name
31742ea7f44SGerrit Uitslag     * @return false|string
31855efc227SAndreas Gohr     */
3190b17fdc6SAndreas Gohr    function getExifField($field) {
32055efc227SAndreas Gohr        if (!isset($this->_info['exif'])) {
32155efc227SAndreas Gohr            $this->_parseMarkerExif();
32255efc227SAndreas Gohr        }
32355efc227SAndreas Gohr
32455efc227SAndreas Gohr        if ($this->_markers == null) {
32555efc227SAndreas Gohr            return false;
32655efc227SAndreas Gohr        }
32755efc227SAndreas Gohr
32855efc227SAndreas Gohr        if (isset($this->_info['exif'][$field])) {
32955efc227SAndreas Gohr            return $this->_info['exif'][$field];
33055efc227SAndreas Gohr        }
33155efc227SAndreas Gohr
33255efc227SAndreas Gohr        return false;
33355efc227SAndreas Gohr    }
33455efc227SAndreas Gohr
33555efc227SAndreas Gohr    /**
336431c7fc8Shakan.sandell     * Return an XMP field
337431c7fc8Shakan.sandell     *
338431c7fc8Shakan.sandell     * @author Hakan Sandell <hakan.sandell@mydata.se>
33942ea7f44SGerrit Uitslag     *
34042ea7f44SGerrit Uitslag     * @param string $field field name
34142ea7f44SGerrit Uitslag     * @return false|string
342431c7fc8Shakan.sandell     */
3430b17fdc6SAndreas Gohr    function getXmpField($field) {
344431c7fc8Shakan.sandell        if (!isset($this->_info['xmp'])) {
345431c7fc8Shakan.sandell            $this->_parseMarkerXmp();
346431c7fc8Shakan.sandell        }
347431c7fc8Shakan.sandell
348431c7fc8Shakan.sandell        if ($this->_markers == null) {
349431c7fc8Shakan.sandell            return false;
350431c7fc8Shakan.sandell        }
351431c7fc8Shakan.sandell
352431c7fc8Shakan.sandell        if (isset($this->_info['xmp'][$field])) {
353431c7fc8Shakan.sandell            return $this->_info['xmp'][$field];
354431c7fc8Shakan.sandell        }
355431c7fc8Shakan.sandell
356431c7fc8Shakan.sandell        return false;
357431c7fc8Shakan.sandell    }
358431c7fc8Shakan.sandell
359431c7fc8Shakan.sandell    /**
36055efc227SAndreas Gohr     * Return an Adobe Field
36155efc227SAndreas Gohr     *
36255efc227SAndreas Gohr     * @author Sebastian Delmont <sdelmont@zonageek.com>
36342ea7f44SGerrit Uitslag     *
36442ea7f44SGerrit Uitslag     * @param string $field field name
36542ea7f44SGerrit Uitslag     * @return false|string
36655efc227SAndreas Gohr     */
3670b17fdc6SAndreas Gohr    function getAdobeField($field) {
36855efc227SAndreas Gohr        if (!isset($this->_info['adobe'])) {
36955efc227SAndreas Gohr            $this->_parseMarkerAdobe();
37055efc227SAndreas Gohr        }
37155efc227SAndreas Gohr
37255efc227SAndreas Gohr        if ($this->_markers == null) {
37355efc227SAndreas Gohr            return false;
37455efc227SAndreas Gohr        }
37555efc227SAndreas Gohr
37655efc227SAndreas Gohr        if (isset($this->_info['adobe'][$field])) {
37755efc227SAndreas Gohr            return $this->_info['adobe'][$field];
37855efc227SAndreas Gohr        }
37955efc227SAndreas Gohr
38055efc227SAndreas Gohr        return false;
38155efc227SAndreas Gohr    }
38255efc227SAndreas Gohr
38355efc227SAndreas Gohr    /**
38455efc227SAndreas Gohr     * Return an IPTC field
38555efc227SAndreas Gohr     *
38655efc227SAndreas Gohr     * @author Sebastian Delmont <sdelmont@zonageek.com>
38742ea7f44SGerrit Uitslag     *
38842ea7f44SGerrit Uitslag     * @param string $field field name
38942ea7f44SGerrit Uitslag     * @return false|string
39055efc227SAndreas Gohr     */
3910b17fdc6SAndreas Gohr    function getIPTCField($field) {
39255efc227SAndreas Gohr        if (!isset($this->_info['iptc'])) {
39355efc227SAndreas Gohr            $this->_parseMarkerAdobe();
39455efc227SAndreas Gohr        }
39555efc227SAndreas Gohr
39655efc227SAndreas Gohr        if ($this->_markers == null) {
39755efc227SAndreas Gohr            return false;
39855efc227SAndreas Gohr        }
39955efc227SAndreas Gohr
40055efc227SAndreas Gohr        if (isset($this->_info['iptc'][$field])) {
40155efc227SAndreas Gohr            return $this->_info['iptc'][$field];
40255efc227SAndreas Gohr        }
40355efc227SAndreas Gohr
40455efc227SAndreas Gohr        return false;
40555efc227SAndreas Gohr    }
40655efc227SAndreas Gohr
40755efc227SAndreas Gohr    /**
40855efc227SAndreas Gohr     * Set an EXIF field
40955efc227SAndreas Gohr     *
41055efc227SAndreas Gohr     * @author Sebastian Delmont <sdelmont@zonageek.com>
411b5a81756SJoe Lapp     * @author Joe Lapp <joe.lapp@pobox.com>
41242ea7f44SGerrit Uitslag     *
41342ea7f44SGerrit Uitslag     * @param string $field field name
41442ea7f44SGerrit Uitslag     * @param string $value
41542ea7f44SGerrit Uitslag     * @return bool
41655efc227SAndreas Gohr     */
4170b17fdc6SAndreas Gohr    function setExifField($field, $value) {
41855efc227SAndreas Gohr        if (!isset($this->_info['exif'])) {
41955efc227SAndreas Gohr            $this->_parseMarkerExif();
42055efc227SAndreas Gohr        }
42155efc227SAndreas Gohr
42255efc227SAndreas Gohr        if ($this->_markers == null) {
42355efc227SAndreas Gohr            return false;
42455efc227SAndreas Gohr        }
42555efc227SAndreas Gohr
42655efc227SAndreas Gohr        if ($this->_info['exif'] == false) {
42755efc227SAndreas Gohr            $this->_info['exif'] = array();
42855efc227SAndreas Gohr        }
42955efc227SAndreas Gohr
430b5a81756SJoe Lapp        // make sure datetimes are in correct format
431*6c16a3a9Sfiwswe        if(strlen($field) >= 8 && str_starts_with(strtolower($field), 'datetime')) {
4322401f18dSSyntaxseed            if(strlen($value) < 8 || $value[4] != ':' || $value[7] != ':') {
433b5a81756SJoe Lapp                $value = date('Y:m:d H:i:s', strtotime($value));
434b5a81756SJoe Lapp            }
435b5a81756SJoe Lapp        }
436b5a81756SJoe Lapp
43755efc227SAndreas Gohr        $this->_info['exif'][$field] = $value;
43855efc227SAndreas Gohr
43955efc227SAndreas Gohr        return true;
44055efc227SAndreas Gohr    }
44155efc227SAndreas Gohr
44255efc227SAndreas Gohr    /**
44355efc227SAndreas Gohr     * Set an Adobe Field
44455efc227SAndreas Gohr     *
44555efc227SAndreas Gohr     * @author Sebastian Delmont <sdelmont@zonageek.com>
44642ea7f44SGerrit Uitslag     *
44742ea7f44SGerrit Uitslag     * @param string $field field name
44842ea7f44SGerrit Uitslag     * @param string $value
44942ea7f44SGerrit Uitslag     * @return bool
45055efc227SAndreas Gohr     */
4510b17fdc6SAndreas Gohr    function setAdobeField($field, $value) {
45255efc227SAndreas Gohr        if (!isset($this->_info['adobe'])) {
45355efc227SAndreas Gohr            $this->_parseMarkerAdobe();
45455efc227SAndreas Gohr        }
45555efc227SAndreas Gohr
45655efc227SAndreas Gohr        if ($this->_markers == null) {
45755efc227SAndreas Gohr            return false;
45855efc227SAndreas Gohr        }
45955efc227SAndreas Gohr
46055efc227SAndreas Gohr        if ($this->_info['adobe'] == false) {
46155efc227SAndreas Gohr            $this->_info['adobe'] = array();
46255efc227SAndreas Gohr        }
46355efc227SAndreas Gohr
46455efc227SAndreas Gohr        $this->_info['adobe'][$field] = $value;
46555efc227SAndreas Gohr
46655efc227SAndreas Gohr        return true;
46755efc227SAndreas Gohr    }
46855efc227SAndreas Gohr
46955efc227SAndreas Gohr    /**
47023a34783SAndreas Gohr     * Calculates the multiplier needed to resize the image to the given
47123a34783SAndreas Gohr     * dimensions
47223a34783SAndreas Gohr     *
47323a34783SAndreas Gohr     * @author Andreas Gohr <andi@splitbrain.org>
47442ea7f44SGerrit Uitslag     *
47542ea7f44SGerrit Uitslag     * @param int $maxwidth
47642ea7f44SGerrit Uitslag     * @param int $maxheight
47742ea7f44SGerrit Uitslag     * @return float|int
47823a34783SAndreas Gohr     */
47923a34783SAndreas Gohr    function getResizeRatio($maxwidth,$maxheight=0){
48023a34783SAndreas Gohr        if(!$maxheight) $maxheight = $maxwidth;
48123a34783SAndreas Gohr
48223a34783SAndreas Gohr        $w = $this->getField('File.Width');
48323a34783SAndreas Gohr        $h = $this->getField('File.Height');
48423a34783SAndreas Gohr
48523a34783SAndreas Gohr        $ratio = 1;
48623a34783SAndreas Gohr        if($w >= $h){
48723a34783SAndreas Gohr            if($w >= $maxwidth){
48823a34783SAndreas Gohr                $ratio = $maxwidth/$w;
48923a34783SAndreas Gohr            }elseif($h > $maxheight){
49023a34783SAndreas Gohr                $ratio = $maxheight/$h;
49123a34783SAndreas Gohr            }
49223a34783SAndreas Gohr        }else{
49323a34783SAndreas Gohr            if($h >= $maxheight){
49423a34783SAndreas Gohr                $ratio = $maxheight/$h;
49523a34783SAndreas Gohr            }elseif($w > $maxwidth){
49623a34783SAndreas Gohr                $ratio = $maxwidth/$w;
49723a34783SAndreas Gohr            }
49823a34783SAndreas Gohr        }
49923a34783SAndreas Gohr        return $ratio;
50023a34783SAndreas Gohr    }
50123a34783SAndreas Gohr
50223a34783SAndreas Gohr
50323a34783SAndreas Gohr    /**
50455efc227SAndreas Gohr     * Set an IPTC field
50555efc227SAndreas Gohr     *
50655efc227SAndreas Gohr     * @author Sebastian Delmont <sdelmont@zonageek.com>
50742ea7f44SGerrit Uitslag     *
50842ea7f44SGerrit Uitslag     * @param string $field field name
50942ea7f44SGerrit Uitslag     * @param string $value
51042ea7f44SGerrit Uitslag     * @return bool
51155efc227SAndreas Gohr     */
5120b17fdc6SAndreas Gohr    function setIPTCField($field, $value) {
51355efc227SAndreas Gohr        if (!isset($this->_info['iptc'])) {
51455efc227SAndreas Gohr            $this->_parseMarkerAdobe();
51555efc227SAndreas Gohr        }
51655efc227SAndreas Gohr
51755efc227SAndreas Gohr        if ($this->_markers == null) {
51855efc227SAndreas Gohr            return false;
51955efc227SAndreas Gohr        }
52055efc227SAndreas Gohr
52155efc227SAndreas Gohr        if ($this->_info['iptc'] == false) {
52255efc227SAndreas Gohr            $this->_info['iptc'] = array();
52355efc227SAndreas Gohr        }
52455efc227SAndreas Gohr
52555efc227SAndreas Gohr        $this->_info['iptc'][$field] = $value;
52655efc227SAndreas Gohr
52755efc227SAndreas Gohr        return true;
52855efc227SAndreas Gohr    }
52955efc227SAndreas Gohr
53055efc227SAndreas Gohr    /**
53155efc227SAndreas Gohr     * Delete an EXIF field
53255efc227SAndreas Gohr     *
53355efc227SAndreas Gohr     * @author Sebastian Delmont <sdelmont@zonageek.com>
53442ea7f44SGerrit Uitslag     *
53542ea7f44SGerrit Uitslag     * @param string $field field name
53642ea7f44SGerrit Uitslag     * @return bool
53755efc227SAndreas Gohr     */
5380b17fdc6SAndreas Gohr    function deleteExifField($field) {
53955efc227SAndreas Gohr        if (!isset($this->_info['exif'])) {
54055efc227SAndreas Gohr            $this->_parseMarkerAdobe();
54155efc227SAndreas Gohr        }
54255efc227SAndreas Gohr
54355efc227SAndreas Gohr        if ($this->_markers == null) {
54455efc227SAndreas Gohr            return false;
54555efc227SAndreas Gohr        }
54655efc227SAndreas Gohr
54755efc227SAndreas Gohr        if ($this->_info['exif'] != false) {
54855efc227SAndreas Gohr            unset($this->_info['exif'][$field]);
54955efc227SAndreas Gohr        }
55055efc227SAndreas Gohr
55155efc227SAndreas Gohr        return true;
55255efc227SAndreas Gohr    }
55355efc227SAndreas Gohr
55455efc227SAndreas Gohr    /**
55555efc227SAndreas Gohr     * Delete an Adobe field
55655efc227SAndreas Gohr     *
55755efc227SAndreas Gohr     * @author Sebastian Delmont <sdelmont@zonageek.com>
55842ea7f44SGerrit Uitslag     *
55942ea7f44SGerrit Uitslag     * @param string $field field name
56042ea7f44SGerrit Uitslag     * @return bool
56155efc227SAndreas Gohr     */
5620b17fdc6SAndreas Gohr    function deleteAdobeField($field) {
56355efc227SAndreas Gohr        if (!isset($this->_info['adobe'])) {
56455efc227SAndreas Gohr            $this->_parseMarkerAdobe();
56555efc227SAndreas Gohr        }
56655efc227SAndreas Gohr
56755efc227SAndreas Gohr        if ($this->_markers == null) {
56855efc227SAndreas Gohr            return false;
56955efc227SAndreas Gohr        }
57055efc227SAndreas Gohr
57155efc227SAndreas Gohr        if ($this->_info['adobe'] != false) {
57255efc227SAndreas Gohr            unset($this->_info['adobe'][$field]);
57355efc227SAndreas Gohr        }
57455efc227SAndreas Gohr
57555efc227SAndreas Gohr        return true;
57655efc227SAndreas Gohr    }
57755efc227SAndreas Gohr
57855efc227SAndreas Gohr    /**
57955efc227SAndreas Gohr     * Delete an IPTC field
58055efc227SAndreas Gohr     *
58155efc227SAndreas Gohr     * @author Sebastian Delmont <sdelmont@zonageek.com>
58242ea7f44SGerrit Uitslag     *
58342ea7f44SGerrit Uitslag     * @param string $field field name
58442ea7f44SGerrit Uitslag     * @return bool
58555efc227SAndreas Gohr     */
5860b17fdc6SAndreas Gohr    function deleteIPTCField($field) {
58755efc227SAndreas Gohr        if (!isset($this->_info['iptc'])) {
58855efc227SAndreas Gohr            $this->_parseMarkerAdobe();
58955efc227SAndreas Gohr        }
59055efc227SAndreas Gohr
59155efc227SAndreas Gohr        if ($this->_markers == null) {
59255efc227SAndreas Gohr            return false;
59355efc227SAndreas Gohr        }
59455efc227SAndreas Gohr
59555efc227SAndreas Gohr        if ($this->_info['iptc'] != false) {
59655efc227SAndreas Gohr            unset($this->_info['iptc'][$field]);
59755efc227SAndreas Gohr        }
59855efc227SAndreas Gohr
59955efc227SAndreas Gohr        return true;
60055efc227SAndreas Gohr    }
60155efc227SAndreas Gohr
60255efc227SAndreas Gohr    /**
60355efc227SAndreas Gohr     * Get the image's title, tries various fields
60455efc227SAndreas Gohr     *
60555efc227SAndreas Gohr     * @param int $max maximum number chars (keeps words)
60642ea7f44SGerrit Uitslag     * @return false|string
60759bc3b48SGerrit Uitslag     *
60855efc227SAndreas Gohr     * @author Andreas Gohr <andi@splitbrain.org>
60955efc227SAndreas Gohr     */
61055efc227SAndreas Gohr    function getTitle($max=80){
61155efc227SAndreas Gohr        // try various fields
61255efc227SAndreas Gohr        $cap = $this->getField(array('Iptc.Headline',
61355efc227SAndreas Gohr                    'Iptc.Caption',
614431c7fc8Shakan.sandell                    'Xmp.dc:title',
61555efc227SAndreas Gohr                    'Exif.UserComment',
61655efc227SAndreas Gohr                    'Exif.TIFFUserComment',
6172684e50aSAndreas Gohr                    'Exif.TIFFImageDescription',
6182684e50aSAndreas Gohr                    'File.Name'));
61955efc227SAndreas Gohr        if (empty($cap)) return false;
62055efc227SAndreas Gohr
62155efc227SAndreas Gohr        if(!$max) return $cap;
62255efc227SAndreas Gohr        // Shorten to 80 chars (keeping words)
62355efc227SAndreas Gohr        $new = preg_replace('/\n.+$/','',wordwrap($cap, $max));
62455efc227SAndreas Gohr        if($new != $cap) $new .= '...';
62555efc227SAndreas Gohr
62655efc227SAndreas Gohr        return $new;
62755efc227SAndreas Gohr    }
62855efc227SAndreas Gohr
62955efc227SAndreas Gohr    /**
63055efc227SAndreas Gohr     * Gather various date fields
63155efc227SAndreas Gohr     *
63255efc227SAndreas Gohr     * @author Sebastian Delmont <sdelmont@zonageek.com>
63342ea7f44SGerrit Uitslag     *
63442ea7f44SGerrit Uitslag     * @return array|bool
63555efc227SAndreas Gohr     */
6360b17fdc6SAndreas Gohr    function getDates() {
63755efc227SAndreas Gohr        $this->_parseAll();
63855efc227SAndreas Gohr        if ($this->_markers == null) {
639a73b5b7eSAndreas Gohr            if (@isset($this->_info['file']['UnixTime'])) {
64059bc3b48SGerrit Uitslag                $dates = array();
641a73b5b7eSAndreas Gohr                $dates['FileModified'] = $this->_info['file']['UnixTime'];
642a73b5b7eSAndreas Gohr                $dates['Time'] = $this->_info['file']['UnixTime'];
643a73b5b7eSAndreas Gohr                $dates['TimeSource'] = 'FileModified';
644a73b5b7eSAndreas Gohr                $dates['TimeStr'] = date("Y-m-d H:i:s", $this->_info['file']['UnixTime']);
645a73b5b7eSAndreas Gohr                $dates['EarliestTime'] = $this->_info['file']['UnixTime'];
646a73b5b7eSAndreas Gohr                $dates['EarliestTimeSource'] = 'FileModified';
647a73b5b7eSAndreas Gohr                $dates['EarliestTimeStr'] = date("Y-m-d H:i:s", $this->_info['file']['UnixTime']);
648a73b5b7eSAndreas Gohr                $dates['LatestTime'] = $this->_info['file']['UnixTime'];
649a73b5b7eSAndreas Gohr                $dates['LatestTimeSource'] = 'FileModified';
650a73b5b7eSAndreas Gohr                $dates['LatestTimeStr'] = date("Y-m-d H:i:s", $this->_info['file']['UnixTime']);
651a73b5b7eSAndreas Gohr                return $dates;
652a73b5b7eSAndreas Gohr            }
65355efc227SAndreas Gohr            return false;
65455efc227SAndreas Gohr        }
65555efc227SAndreas Gohr
65655efc227SAndreas Gohr        $dates = array();
65755efc227SAndreas Gohr
65855efc227SAndreas Gohr        $latestTime = 0;
65955efc227SAndreas Gohr        $latestTimeSource = "";
66055efc227SAndreas Gohr        $earliestTime = time();
66155efc227SAndreas Gohr        $earliestTimeSource = "";
66255efc227SAndreas Gohr
66355efc227SAndreas Gohr        if (@isset($this->_info['exif']['DateTime'])) {
66455efc227SAndreas Gohr            $dates['ExifDateTime'] = $this->_info['exif']['DateTime'];
66555efc227SAndreas Gohr
66655efc227SAndreas Gohr            $aux = $this->_info['exif']['DateTime'];
6672401f18dSSyntaxseed            $aux[4] = "-";
6682401f18dSSyntaxseed            $aux[7] = "-";
66955efc227SAndreas Gohr            $t = strtotime($aux);
67055efc227SAndreas Gohr
6712114dafdSAndreas Gohr            if ($t && $t > $latestTime) {
67255efc227SAndreas Gohr                $latestTime = $t;
67355efc227SAndreas Gohr                $latestTimeSource = "ExifDateTime";
67455efc227SAndreas Gohr            }
67555efc227SAndreas Gohr
6762114dafdSAndreas Gohr            if ($t && $t < $earliestTime) {
67755efc227SAndreas Gohr                $earliestTime = $t;
67855efc227SAndreas Gohr                $earliestTimeSource = "ExifDateTime";
67955efc227SAndreas Gohr            }
68055efc227SAndreas Gohr        }
68155efc227SAndreas Gohr
68255efc227SAndreas Gohr        if (@isset($this->_info['exif']['DateTimeOriginal'])) {
683f1892550SStephan Bauer            $dates['ExifDateTimeOriginal'] = $this->_info['exif']['DateTimeOriginal'];
68455efc227SAndreas Gohr
68555efc227SAndreas Gohr            $aux = $this->_info['exif']['DateTimeOriginal'];
6862401f18dSSyntaxseed            $aux[4] = "-";
6872401f18dSSyntaxseed            $aux[7] = "-";
68855efc227SAndreas Gohr            $t = strtotime($aux);
68955efc227SAndreas Gohr
6902114dafdSAndreas Gohr            if ($t && $t > $latestTime) {
69155efc227SAndreas Gohr                $latestTime = $t;
69255efc227SAndreas Gohr                $latestTimeSource = "ExifDateTimeOriginal";
69355efc227SAndreas Gohr            }
69455efc227SAndreas Gohr
6952114dafdSAndreas Gohr            if ($t && $t < $earliestTime) {
69655efc227SAndreas Gohr                $earliestTime = $t;
69755efc227SAndreas Gohr                $earliestTimeSource = "ExifDateTimeOriginal";
69855efc227SAndreas Gohr            }
69955efc227SAndreas Gohr        }
70055efc227SAndreas Gohr
70155efc227SAndreas Gohr        if (@isset($this->_info['exif']['DateTimeDigitized'])) {
702f1892550SStephan Bauer            $dates['ExifDateTimeDigitized'] = $this->_info['exif']['DateTimeDigitized'];
70355efc227SAndreas Gohr
70455efc227SAndreas Gohr            $aux = $this->_info['exif']['DateTimeDigitized'];
7052401f18dSSyntaxseed            $aux[4] = "-";
7062401f18dSSyntaxseed            $aux[7] = "-";
70755efc227SAndreas Gohr            $t = strtotime($aux);
70855efc227SAndreas Gohr
7092114dafdSAndreas Gohr            if ($t && $t > $latestTime) {
71055efc227SAndreas Gohr                $latestTime = $t;
71155efc227SAndreas Gohr                $latestTimeSource = "ExifDateTimeDigitized";
71255efc227SAndreas Gohr            }
71355efc227SAndreas Gohr
7142114dafdSAndreas Gohr            if ($t && $t < $earliestTime) {
71555efc227SAndreas Gohr                $earliestTime = $t;
71655efc227SAndreas Gohr                $earliestTimeSource = "ExifDateTimeDigitized";
71755efc227SAndreas Gohr            }
71855efc227SAndreas Gohr        }
71955efc227SAndreas Gohr
72055efc227SAndreas Gohr        if (@isset($this->_info['iptc']['DateCreated'])) {
72155efc227SAndreas Gohr            $dates['IPTCDateCreated'] = $this->_info['iptc']['DateCreated'];
72255efc227SAndreas Gohr
72355efc227SAndreas Gohr            $aux = $this->_info['iptc']['DateCreated'];
72455efc227SAndreas Gohr            $aux = substr($aux, 0, 4) . "-" . substr($aux, 4, 2) . "-" . substr($aux, 6, 2);
72555efc227SAndreas Gohr            $t = strtotime($aux);
72655efc227SAndreas Gohr
7272114dafdSAndreas Gohr            if ($t && $t > $latestTime) {
72855efc227SAndreas Gohr                $latestTime = $t;
72955efc227SAndreas Gohr                $latestTimeSource = "IPTCDateCreated";
73055efc227SAndreas Gohr            }
73155efc227SAndreas Gohr
7322114dafdSAndreas Gohr            if ($t && $t < $earliestTime) {
73355efc227SAndreas Gohr                $earliestTime = $t;
73455efc227SAndreas Gohr                $earliestTimeSource = "IPTCDateCreated";
73555efc227SAndreas Gohr            }
73655efc227SAndreas Gohr        }
73755efc227SAndreas Gohr
73855efc227SAndreas Gohr        if (@isset($this->_info['file']['UnixTime'])) {
73955efc227SAndreas Gohr            $dates['FileModified'] = $this->_info['file']['UnixTime'];
74055efc227SAndreas Gohr
74155efc227SAndreas Gohr            $t = $this->_info['file']['UnixTime'];
74255efc227SAndreas Gohr
7432114dafdSAndreas Gohr            if ($t && $t > $latestTime) {
74455efc227SAndreas Gohr                $latestTime = $t;
74555efc227SAndreas Gohr                $latestTimeSource = "FileModified";
74655efc227SAndreas Gohr            }
74755efc227SAndreas Gohr
7482114dafdSAndreas Gohr            if ($t && $t < $earliestTime) {
74955efc227SAndreas Gohr                $earliestTime = $t;
75055efc227SAndreas Gohr                $earliestTimeSource = "FileModified";
75155efc227SAndreas Gohr            }
75255efc227SAndreas Gohr        }
75355efc227SAndreas Gohr
75455efc227SAndreas Gohr        $dates['Time'] = $earliestTime;
75555efc227SAndreas Gohr        $dates['TimeSource'] = $earliestTimeSource;
75655efc227SAndreas Gohr        $dates['TimeStr'] = date("Y-m-d H:i:s", $earliestTime);
75755efc227SAndreas Gohr        $dates['EarliestTime'] = $earliestTime;
75855efc227SAndreas Gohr        $dates['EarliestTimeSource'] = $earliestTimeSource;
75955efc227SAndreas Gohr        $dates['EarliestTimeStr'] = date("Y-m-d H:i:s", $earliestTime);
76055efc227SAndreas Gohr        $dates['LatestTime'] = $latestTime;
76155efc227SAndreas Gohr        $dates['LatestTimeSource'] = $latestTimeSource;
76255efc227SAndreas Gohr        $dates['LatestTimeStr'] = date("Y-m-d H:i:s", $latestTime);
76355efc227SAndreas Gohr
76455efc227SAndreas Gohr        return $dates;
76555efc227SAndreas Gohr    }
76655efc227SAndreas Gohr
76755efc227SAndreas Gohr    /**
76855efc227SAndreas Gohr     * Get the image width, tries various fields
76955efc227SAndreas Gohr     *
77055efc227SAndreas Gohr     * @author Sebastian Delmont <sdelmont@zonageek.com>
77142ea7f44SGerrit Uitslag     *
77242ea7f44SGerrit Uitslag     * @return false|string
77355efc227SAndreas Gohr     */
7740b17fdc6SAndreas Gohr    function getWidth() {
77555efc227SAndreas Gohr        if (!isset($this->_info['sof'])) {
77655efc227SAndreas Gohr            $this->_parseMarkerSOF();
77755efc227SAndreas Gohr        }
77855efc227SAndreas Gohr
77955efc227SAndreas Gohr        if ($this->_markers == null) {
78055efc227SAndreas Gohr            return false;
78155efc227SAndreas Gohr        }
78255efc227SAndreas Gohr
78355efc227SAndreas Gohr        if (isset($this->_info['sof']['ImageWidth'])) {
78455efc227SAndreas Gohr            return $this->_info['sof']['ImageWidth'];
78555efc227SAndreas Gohr        }
78655efc227SAndreas Gohr
78755efc227SAndreas Gohr        if (!isset($this->_info['exif'])) {
78855efc227SAndreas Gohr            $this->_parseMarkerExif();
78955efc227SAndreas Gohr        }
79055efc227SAndreas Gohr
79155efc227SAndreas Gohr        if (isset($this->_info['exif']['PixelXDimension'])) {
79255efc227SAndreas Gohr            return $this->_info['exif']['PixelXDimension'];
79355efc227SAndreas Gohr        }
79455efc227SAndreas Gohr
79555efc227SAndreas Gohr        return false;
79655efc227SAndreas Gohr    }
79755efc227SAndreas Gohr
79855efc227SAndreas Gohr    /**
79955efc227SAndreas Gohr     * Get the image height, tries various fields
80055efc227SAndreas Gohr     *
80155efc227SAndreas Gohr     * @author Sebastian Delmont <sdelmont@zonageek.com>
80242ea7f44SGerrit Uitslag     *
80342ea7f44SGerrit Uitslag     * @return false|string
80455efc227SAndreas Gohr     */
8050b17fdc6SAndreas Gohr    function getHeight() {
80655efc227SAndreas Gohr        if (!isset($this->_info['sof'])) {
80755efc227SAndreas Gohr            $this->_parseMarkerSOF();
80855efc227SAndreas Gohr        }
80955efc227SAndreas Gohr
81055efc227SAndreas Gohr        if ($this->_markers == null) {
81155efc227SAndreas Gohr            return false;
81255efc227SAndreas Gohr        }
81355efc227SAndreas Gohr
81455efc227SAndreas Gohr        if (isset($this->_info['sof']['ImageHeight'])) {
81555efc227SAndreas Gohr            return $this->_info['sof']['ImageHeight'];
81655efc227SAndreas Gohr        }
81755efc227SAndreas Gohr
81855efc227SAndreas Gohr        if (!isset($this->_info['exif'])) {
81955efc227SAndreas Gohr            $this->_parseMarkerExif();
82055efc227SAndreas Gohr        }
82155efc227SAndreas Gohr
82255efc227SAndreas Gohr        if (isset($this->_info['exif']['PixelYDimension'])) {
82355efc227SAndreas Gohr            return $this->_info['exif']['PixelYDimension'];
82455efc227SAndreas Gohr        }
82555efc227SAndreas Gohr
82655efc227SAndreas Gohr        return false;
82755efc227SAndreas Gohr    }
82855efc227SAndreas Gohr
82955efc227SAndreas Gohr    /**
83055efc227SAndreas Gohr     * Get an dimension string for use in img tag
83155efc227SAndreas Gohr     *
83255efc227SAndreas Gohr     * @author Sebastian Delmont <sdelmont@zonageek.com>
83342ea7f44SGerrit Uitslag     *
83442ea7f44SGerrit Uitslag     * @return false|string
83555efc227SAndreas Gohr     */
8360b17fdc6SAndreas Gohr    function getDimStr() {
83755efc227SAndreas Gohr        if ($this->_markers == null) {
83855efc227SAndreas Gohr            return false;
83955efc227SAndreas Gohr        }
84055efc227SAndreas Gohr
84155efc227SAndreas Gohr        $w = $this->getWidth();
84255efc227SAndreas Gohr        $h = $this->getHeight();
84355efc227SAndreas Gohr
84455efc227SAndreas Gohr        return "width='" . $w . "' height='" . $h . "'";
84555efc227SAndreas Gohr    }
84655efc227SAndreas Gohr
84755efc227SAndreas Gohr    /**
84855efc227SAndreas Gohr     * Checks for an embedded thumbnail
84955efc227SAndreas Gohr     *
85055efc227SAndreas Gohr     * @author Sebastian Delmont <sdelmont@zonageek.com>
85142ea7f44SGerrit Uitslag     *
85242ea7f44SGerrit Uitslag     * @param string $which possible values: 'any', 'exif' or 'adobe'
85342ea7f44SGerrit Uitslag     * @return false|string
85455efc227SAndreas Gohr     */
8550b17fdc6SAndreas Gohr    function hasThumbnail($which = 'any') {
85655efc227SAndreas Gohr        if (($which == 'any') || ($which == 'exif')) {
85755efc227SAndreas Gohr            if (!isset($this->_info['exif'])) {
85855efc227SAndreas Gohr                $this->_parseMarkerExif();
85955efc227SAndreas Gohr            }
86055efc227SAndreas Gohr
86155efc227SAndreas Gohr            if ($this->_markers == null) {
86255efc227SAndreas Gohr                return false;
86355efc227SAndreas Gohr            }
86455efc227SAndreas Gohr
86555efc227SAndreas Gohr            if (isset($this->_info['exif']) && is_array($this->_info['exif'])) {
86655efc227SAndreas Gohr                if (isset($this->_info['exif']['JFIFThumbnail'])) {
86755efc227SAndreas Gohr                    return 'exif';
86855efc227SAndreas Gohr                }
86955efc227SAndreas Gohr            }
87055efc227SAndreas Gohr        }
87155efc227SAndreas Gohr
87255efc227SAndreas Gohr        if ($which == 'adobe') {
87355efc227SAndreas Gohr            if (!isset($this->_info['adobe'])) {
87455efc227SAndreas Gohr                $this->_parseMarkerAdobe();
87555efc227SAndreas Gohr            }
87655efc227SAndreas Gohr
87755efc227SAndreas Gohr            if ($this->_markers == null) {
87855efc227SAndreas Gohr                return false;
87955efc227SAndreas Gohr            }
88055efc227SAndreas Gohr
88155efc227SAndreas Gohr            if (isset($this->_info['adobe']) && is_array($this->_info['adobe'])) {
88255efc227SAndreas Gohr                if (isset($this->_info['adobe']['ThumbnailData'])) {
88355efc227SAndreas Gohr                    return 'exif';
88455efc227SAndreas Gohr                }
88555efc227SAndreas Gohr            }
88655efc227SAndreas Gohr        }
88755efc227SAndreas Gohr
88855efc227SAndreas Gohr        return false;
88955efc227SAndreas Gohr    }
89055efc227SAndreas Gohr
89155efc227SAndreas Gohr    /**
89255efc227SAndreas Gohr     * Send embedded thumbnail to browser
89355efc227SAndreas Gohr     *
89455efc227SAndreas Gohr     * @author Sebastian Delmont <sdelmont@zonageek.com>
89542ea7f44SGerrit Uitslag     *
89642ea7f44SGerrit Uitslag     * @param string $which possible values: 'any', 'exif' or 'adobe'
89742ea7f44SGerrit Uitslag     * @return bool
89855efc227SAndreas Gohr     */
8990b17fdc6SAndreas Gohr    function sendThumbnail($which = 'any') {
90055efc227SAndreas Gohr        $data = null;
90155efc227SAndreas Gohr
90255efc227SAndreas Gohr        if (($which == 'any') || ($which == 'exif')) {
90355efc227SAndreas Gohr            if (!isset($this->_info['exif'])) {
90455efc227SAndreas Gohr                $this->_parseMarkerExif();
90555efc227SAndreas Gohr            }
90655efc227SAndreas Gohr
90755efc227SAndreas Gohr            if ($this->_markers == null) {
90855efc227SAndreas Gohr                return false;
90955efc227SAndreas Gohr            }
91055efc227SAndreas Gohr
91155efc227SAndreas Gohr            if (isset($this->_info['exif']) && is_array($this->_info['exif'])) {
91255efc227SAndreas Gohr                if (isset($this->_info['exif']['JFIFThumbnail'])) {
91355efc227SAndreas Gohr                    $data =& $this->_info['exif']['JFIFThumbnail'];
91455efc227SAndreas Gohr                }
91555efc227SAndreas Gohr            }
91655efc227SAndreas Gohr        }
91755efc227SAndreas Gohr
91855efc227SAndreas Gohr        if (($which == 'adobe') || ($data == null)){
91955efc227SAndreas Gohr            if (!isset($this->_info['adobe'])) {
92055efc227SAndreas Gohr                $this->_parseMarkerAdobe();
92155efc227SAndreas Gohr            }
92255efc227SAndreas Gohr
92355efc227SAndreas Gohr            if ($this->_markers == null) {
92455efc227SAndreas Gohr                return false;
92555efc227SAndreas Gohr            }
92655efc227SAndreas Gohr
92755efc227SAndreas Gohr            if (isset($this->_info['adobe']) && is_array($this->_info['adobe'])) {
92855efc227SAndreas Gohr                if (isset($this->_info['adobe']['ThumbnailData'])) {
92955efc227SAndreas Gohr                    $data =& $this->_info['adobe']['ThumbnailData'];
93055efc227SAndreas Gohr                }
93155efc227SAndreas Gohr            }
93255efc227SAndreas Gohr        }
93355efc227SAndreas Gohr
93455efc227SAndreas Gohr        if ($data != null) {
93555efc227SAndreas Gohr            header("Content-type: image/jpeg");
93655efc227SAndreas Gohr            echo $data;
93755efc227SAndreas Gohr            return true;
93855efc227SAndreas Gohr        }
93955efc227SAndreas Gohr
94055efc227SAndreas Gohr        return false;
94155efc227SAndreas Gohr    }
94255efc227SAndreas Gohr
94355efc227SAndreas Gohr    /**
94455efc227SAndreas Gohr     * Save changed Metadata
94555efc227SAndreas Gohr     *
94655efc227SAndreas Gohr     * @author Sebastian Delmont <sdelmont@zonageek.com>
94736df6fa3SAndreas Gohr     * @author Andreas Gohr <andi@splitbrain.org>
94842ea7f44SGerrit Uitslag     *
94942ea7f44SGerrit Uitslag     * @param string $fileName file name or empty string for a random name
95042ea7f44SGerrit Uitslag     * @return bool
95155efc227SAndreas Gohr     */
95255efc227SAndreas Gohr    function save($fileName = "") {
95355efc227SAndreas Gohr        if ($fileName == "") {
95436df6fa3SAndreas Gohr            $tmpName = tempnam(dirname($this->_fileName),'_metatemp_');
95555efc227SAndreas Gohr            $this->_writeJPEG($tmpName);
95679e79377SAndreas Gohr            if (file_exists($tmpName)) {
957bf5e5a5bSAndreas Gohr                return io_rename($tmpName, $this->_fileName);
95855efc227SAndreas Gohr            }
95936df6fa3SAndreas Gohr        } else {
96036df6fa3SAndreas Gohr            return $this->_writeJPEG($fileName);
96155efc227SAndreas Gohr        }
96236df6fa3SAndreas Gohr        return false;
96355efc227SAndreas Gohr    }
96455efc227SAndreas Gohr
96555efc227SAndreas Gohr    /*************************************************************/
96655efc227SAndreas Gohr    /* PRIVATE FUNCTIONS (Internal Use Only!)                    */
96755efc227SAndreas Gohr    /*************************************************************/
96855efc227SAndreas Gohr
96955efc227SAndreas Gohr    /*************************************************************/
9705aaca723SGerrit Uitslag    function _dispose($fileName = "") {
97155efc227SAndreas Gohr        $this->_fileName = $fileName;
97255efc227SAndreas Gohr
97355efc227SAndreas Gohr        $this->_fp = null;
97455efc227SAndreas Gohr        $this->_type = 'unknown';
97555efc227SAndreas Gohr
97655efc227SAndreas Gohr        unset($this->_markers);
97755efc227SAndreas Gohr        unset($this->_info);
97855efc227SAndreas Gohr    }
97955efc227SAndreas Gohr
98055efc227SAndreas Gohr    /*************************************************************/
9810b17fdc6SAndreas Gohr    function _readJPEG() {
98255efc227SAndreas Gohr        unset($this->_markers);
983a73b5b7eSAndreas Gohr        //unset($this->_info);
98455efc227SAndreas Gohr        $this->_markers = array();
985a73b5b7eSAndreas Gohr        //$this->_info = array();
98655efc227SAndreas Gohr
98755efc227SAndreas Gohr        $this->_fp = @fopen($this->_fileName, 'rb');
98855efc227SAndreas Gohr        if ($this->_fp) {
98955efc227SAndreas Gohr            if (file_exists($this->_fileName)) {
99055efc227SAndreas Gohr                $this->_type = 'file';
99155efc227SAndreas Gohr            }
99255efc227SAndreas Gohr            else {
99355efc227SAndreas Gohr                $this->_type = 'url';
99455efc227SAndreas Gohr            }
9950b17fdc6SAndreas Gohr        } else {
99655efc227SAndreas Gohr            $this->_fp = null;
99755efc227SAndreas Gohr            return false;  // ERROR: Can't open file
99855efc227SAndreas Gohr        }
99955efc227SAndreas Gohr
100055efc227SAndreas Gohr        // Check for the JPEG signature
100155efc227SAndreas Gohr        $c1 = ord(fgetc($this->_fp));
100255efc227SAndreas Gohr        $c2 = ord(fgetc($this->_fp));
100355efc227SAndreas Gohr
100455efc227SAndreas Gohr        if ($c1 != 0xFF || $c2 != 0xD8) {   // (0xFF + SOI)
100555efc227SAndreas Gohr            $this->_markers = null;
100655efc227SAndreas Gohr            return false;  // ERROR: File is not a JPEG
100755efc227SAndreas Gohr        }
100855efc227SAndreas Gohr
100955efc227SAndreas Gohr        $count = 0;
101055efc227SAndreas Gohr
101155efc227SAndreas Gohr        $done = false;
101255efc227SAndreas Gohr        $ok = true;
101355efc227SAndreas Gohr
101455efc227SAndreas Gohr        while (!$done) {
101555efc227SAndreas Gohr            $capture = false;
101655efc227SAndreas Gohr
101755efc227SAndreas Gohr            // First, skip any non 0xFF bytes
101855efc227SAndreas Gohr            $discarded = 0;
101955efc227SAndreas Gohr            $c = ord(fgetc($this->_fp));
102055efc227SAndreas Gohr            while (!feof($this->_fp) && ($c != 0xFF)) {
102155efc227SAndreas Gohr                $discarded++;
102255efc227SAndreas Gohr                $c = ord(fgetc($this->_fp));
102355efc227SAndreas Gohr            }
102455efc227SAndreas Gohr            // Then skip all 0xFF until the marker byte
102555efc227SAndreas Gohr            do {
102655efc227SAndreas Gohr                $marker = ord(fgetc($this->_fp));
102755efc227SAndreas Gohr            } while (!feof($this->_fp) && ($marker == 0xFF));
102855efc227SAndreas Gohr
102955efc227SAndreas Gohr            if (feof($this->_fp)) {
103055efc227SAndreas Gohr                return false; // ERROR: Unexpected EOF
103155efc227SAndreas Gohr            }
103255efc227SAndreas Gohr            if ($discarded != 0) {
103355efc227SAndreas Gohr                return false; // ERROR: Extraneous data
103455efc227SAndreas Gohr            }
103555efc227SAndreas Gohr
103655efc227SAndreas Gohr            $length = ord(fgetc($this->_fp)) * 256 + ord(fgetc($this->_fp));
103755efc227SAndreas Gohr            if (feof($this->_fp)) {
103855efc227SAndreas Gohr                return false; // ERROR: Unexpected EOF
103955efc227SAndreas Gohr            }
104055efc227SAndreas Gohr            if ($length < 2) {
104155efc227SAndreas Gohr                return false; // ERROR: Extraneous data
104255efc227SAndreas Gohr            }
104355efc227SAndreas Gohr            $length = $length - 2; // The length we got counts itself
104455efc227SAndreas Gohr
104555efc227SAndreas Gohr            switch ($marker) {
104655efc227SAndreas Gohr                case 0xC0:    // SOF0
104755efc227SAndreas Gohr                case 0xC1:    // SOF1
104855efc227SAndreas Gohr                case 0xC2:    // SOF2
104955efc227SAndreas Gohr                case 0xC9:    // SOF9
105055efc227SAndreas Gohr                case 0xE0:    // APP0: JFIF data
1051431c7fc8Shakan.sandell                case 0xE1:    // APP1: EXIF or XMP data
105255efc227SAndreas Gohr                case 0xED:    // APP13: IPTC / Photoshop data
105355efc227SAndreas Gohr                    $capture = true;
105455efc227SAndreas Gohr                    break;
105555efc227SAndreas Gohr                case 0xDA:    // SOS: Start of scan... the image itself and the last block on the file
105655efc227SAndreas Gohr                    $capture = false;
105755efc227SAndreas Gohr                    $length = -1;  // This field has no length... it includes all data until EOF
105855efc227SAndreas Gohr                    $done = true;
105955efc227SAndreas Gohr                    break;
106055efc227SAndreas Gohr                default:
106155efc227SAndreas Gohr                    $capture = true;//false;
106255efc227SAndreas Gohr                    break;
106355efc227SAndreas Gohr            }
106455efc227SAndreas Gohr
106555efc227SAndreas Gohr            $this->_markers[$count] = array();
106655efc227SAndreas Gohr            $this->_markers[$count]['marker'] = $marker;
106755efc227SAndreas Gohr            $this->_markers[$count]['length'] = $length;
106855efc227SAndreas Gohr
106955efc227SAndreas Gohr            if ($capture) {
1070ed3655c4STom N Harris                if ($length)
10710ea5ced2SGerrit Uitslag                    $this->_markers[$count]['data'] = fread($this->_fp, $length);
1072ed3655c4STom N Harris                else
1073ed3655c4STom N Harris                    $this->_markers[$count]['data'] = "";
107455efc227SAndreas Gohr            }
107555efc227SAndreas Gohr            elseif (!$done) {
107655efc227SAndreas Gohr                $result = @fseek($this->_fp, $length, SEEK_CUR);
107755efc227SAndreas Gohr                // fseek doesn't seem to like HTTP 'files', but fgetc has no problem
107855efc227SAndreas Gohr                if (!($result === 0)) {
107955efc227SAndreas Gohr                    for ($i = 0; $i < $length; $i++) {
108055efc227SAndreas Gohr                        fgetc($this->_fp);
108155efc227SAndreas Gohr                    }
108255efc227SAndreas Gohr                }
108355efc227SAndreas Gohr            }
108455efc227SAndreas Gohr            $count++;
108555efc227SAndreas Gohr        }
108655efc227SAndreas Gohr
108755efc227SAndreas Gohr        if ($this->_fp) {
108855efc227SAndreas Gohr            fclose($this->_fp);
108955efc227SAndreas Gohr            $this->_fp = null;
109055efc227SAndreas Gohr        }
109155efc227SAndreas Gohr
109255efc227SAndreas Gohr        return $ok;
109355efc227SAndreas Gohr    }
109455efc227SAndreas Gohr
109555efc227SAndreas Gohr    /*************************************************************/
10960b17fdc6SAndreas Gohr    function _parseAll() {
10971017ae2eSAndreas Gohr        if (!isset($this->_info['file'])) {
10981017ae2eSAndreas Gohr            $this->_parseFileInfo();
10991017ae2eSAndreas Gohr        }
110055efc227SAndreas Gohr        if (!isset($this->_markers)) {
110155efc227SAndreas Gohr            $this->_readJPEG();
110255efc227SAndreas Gohr        }
110355efc227SAndreas Gohr
110455efc227SAndreas Gohr        if ($this->_markers == null) {
110555efc227SAndreas Gohr            return false;
110655efc227SAndreas Gohr        }
110755efc227SAndreas Gohr
110855efc227SAndreas Gohr        if (!isset($this->_info['jfif'])) {
110955efc227SAndreas Gohr            $this->_parseMarkerJFIF();
111055efc227SAndreas Gohr        }
111155efc227SAndreas Gohr        if (!isset($this->_info['jpeg'])) {
111255efc227SAndreas Gohr            $this->_parseMarkerSOF();
111355efc227SAndreas Gohr        }
111455efc227SAndreas Gohr        if (!isset($this->_info['exif'])) {
111555efc227SAndreas Gohr            $this->_parseMarkerExif();
111655efc227SAndreas Gohr        }
1117431c7fc8Shakan.sandell        if (!isset($this->_info['xmp'])) {
1118431c7fc8Shakan.sandell            $this->_parseMarkerXmp();
1119431c7fc8Shakan.sandell        }
112055efc227SAndreas Gohr        if (!isset($this->_info['adobe'])) {
112155efc227SAndreas Gohr            $this->_parseMarkerAdobe();
112255efc227SAndreas Gohr        }
112355efc227SAndreas Gohr    }
112455efc227SAndreas Gohr
112555efc227SAndreas Gohr    /*************************************************************/
1126276820f7SScrutinizer Auto-Fixer
1127276820f7SScrutinizer Auto-Fixer    /**
1128276820f7SScrutinizer Auto-Fixer     * @param string $outputName
1129f50a239bSTakamura     *
1130f50a239bSTakamura     * @return bool
1131276820f7SScrutinizer Auto-Fixer     */
11320b17fdc6SAndreas Gohr    function _writeJPEG($outputName) {
113355efc227SAndreas Gohr        $this->_parseAll();
113455efc227SAndreas Gohr
113555efc227SAndreas Gohr        $wroteEXIF = false;
113655efc227SAndreas Gohr        $wroteAdobe = false;
113755efc227SAndreas Gohr
113855efc227SAndreas Gohr        $this->_fp = @fopen($this->_fileName, 'r');
113955efc227SAndreas Gohr        if ($this->_fp) {
114055efc227SAndreas Gohr            if (file_exists($this->_fileName)) {
114155efc227SAndreas Gohr                $this->_type = 'file';
114255efc227SAndreas Gohr            }
114355efc227SAndreas Gohr            else {
114455efc227SAndreas Gohr                $this->_type = 'url';
114555efc227SAndreas Gohr            }
11460b17fdc6SAndreas Gohr        } else {
114755efc227SAndreas Gohr            $this->_fp = null;
114855efc227SAndreas Gohr            return false;  // ERROR: Can't open file
114955efc227SAndreas Gohr        }
115055efc227SAndreas Gohr
115155efc227SAndreas Gohr        $this->_fpout = fopen($outputName, 'wb');
11520b17fdc6SAndreas Gohr        if (!$this->_fpout) {
115355efc227SAndreas Gohr            $this->_fpout = null;
115455efc227SAndreas Gohr            fclose($this->_fp);
115555efc227SAndreas Gohr            $this->_fp = null;
115655efc227SAndreas Gohr            return false;  // ERROR: Can't open output file
115755efc227SAndreas Gohr        }
115855efc227SAndreas Gohr
115955efc227SAndreas Gohr        // Check for the JPEG signature
116055efc227SAndreas Gohr        $c1 = ord(fgetc($this->_fp));
116155efc227SAndreas Gohr        $c2 = ord(fgetc($this->_fp));
116255efc227SAndreas Gohr
116355efc227SAndreas Gohr        if ($c1 != 0xFF || $c2 != 0xD8) {   // (0xFF + SOI)
116455efc227SAndreas Gohr            return false;  // ERROR: File is not a JPEG
116555efc227SAndreas Gohr        }
116655efc227SAndreas Gohr
116755efc227SAndreas Gohr        fputs($this->_fpout, chr(0xFF), 1);
116855efc227SAndreas Gohr        fputs($this->_fpout, chr(0xD8), 1); // (0xFF + SOI)
116955efc227SAndreas Gohr
117055efc227SAndreas Gohr        $count = 0;
117155efc227SAndreas Gohr
117255efc227SAndreas Gohr        $done = false;
117355efc227SAndreas Gohr        $ok = true;
117455efc227SAndreas Gohr
117555efc227SAndreas Gohr        while (!$done) {
117655efc227SAndreas Gohr            // First, skip any non 0xFF bytes
117755efc227SAndreas Gohr            $discarded = 0;
117855efc227SAndreas Gohr            $c = ord(fgetc($this->_fp));
117955efc227SAndreas Gohr            while (!feof($this->_fp) && ($c != 0xFF)) {
118055efc227SAndreas Gohr                $discarded++;
118155efc227SAndreas Gohr                $c = ord(fgetc($this->_fp));
118255efc227SAndreas Gohr            }
118355efc227SAndreas Gohr            // Then skip all 0xFF until the marker byte
118455efc227SAndreas Gohr            do {
118555efc227SAndreas Gohr                $marker = ord(fgetc($this->_fp));
118655efc227SAndreas Gohr            } while (!feof($this->_fp) && ($marker == 0xFF));
118755efc227SAndreas Gohr
118855efc227SAndreas Gohr            if (feof($this->_fp)) {
118955efc227SAndreas Gohr                $ok = false;
119055efc227SAndreas Gohr                break; // ERROR: Unexpected EOF
119155efc227SAndreas Gohr            }
119255efc227SAndreas Gohr            if ($discarded != 0) {
119355efc227SAndreas Gohr                $ok = false;
119455efc227SAndreas Gohr                break; // ERROR: Extraneous data
119555efc227SAndreas Gohr            }
119655efc227SAndreas Gohr
119755efc227SAndreas Gohr            $length = ord(fgetc($this->_fp)) * 256 + ord(fgetc($this->_fp));
119855efc227SAndreas Gohr            if (feof($this->_fp)) {
119955efc227SAndreas Gohr                $ok = false;
120055efc227SAndreas Gohr                break; // ERROR: Unexpected EOF
120155efc227SAndreas Gohr            }
120255efc227SAndreas Gohr            if ($length < 2) {
120355efc227SAndreas Gohr                $ok = false;
120455efc227SAndreas Gohr                break; // ERROR: Extraneous data
120555efc227SAndreas Gohr            }
120655efc227SAndreas Gohr            $length = $length - 2; // The length we got counts itself
120755efc227SAndreas Gohr
120855efc227SAndreas Gohr            unset($data);
120955efc227SAndreas Gohr            if ($marker == 0xE1) { // APP1: EXIF data
121055efc227SAndreas Gohr                $data =& $this->_createMarkerEXIF();
121155efc227SAndreas Gohr                $wroteEXIF = true;
121255efc227SAndreas Gohr            }
121355efc227SAndreas Gohr            elseif ($marker == 0xED) { // APP13: IPTC / Photoshop data
121455efc227SAndreas Gohr                $data =& $this->_createMarkerAdobe();
121555efc227SAndreas Gohr                $wroteAdobe = true;
121655efc227SAndreas Gohr            }
121755efc227SAndreas Gohr            elseif ($marker == 0xDA) { // SOS: Start of scan... the image itself and the last block on the file
121855efc227SAndreas Gohr                $done = true;
121955efc227SAndreas Gohr            }
122055efc227SAndreas Gohr
122155efc227SAndreas Gohr            if (!$wroteEXIF && (($marker < 0xE0) || ($marker > 0xEF))) {
122255efc227SAndreas Gohr                if (isset($this->_info['exif']) && is_array($this->_info['exif'])) {
122355efc227SAndreas Gohr                    $exif =& $this->_createMarkerEXIF();
122455efc227SAndreas Gohr                    $this->_writeJPEGMarker(0xE1, strlen($exif), $exif, 0);
122555efc227SAndreas Gohr                    unset($exif);
122655efc227SAndreas Gohr                }
122755efc227SAndreas Gohr                $wroteEXIF = true;
122855efc227SAndreas Gohr            }
122955efc227SAndreas Gohr
123055efc227SAndreas Gohr            if (!$wroteAdobe && (($marker < 0xE0) || ($marker > 0xEF))) {
123155efc227SAndreas Gohr                if ((isset($this->_info['adobe']) && is_array($this->_info['adobe']))
123255efc227SAndreas Gohr                        || (isset($this->_info['iptc']) && is_array($this->_info['iptc']))) {
123355efc227SAndreas Gohr                    $adobe =& $this->_createMarkerAdobe();
123455efc227SAndreas Gohr                    $this->_writeJPEGMarker(0xED, strlen($adobe), $adobe, 0);
123555efc227SAndreas Gohr                    unset($adobe);
123655efc227SAndreas Gohr                }
123755efc227SAndreas Gohr                $wroteAdobe = true;
123855efc227SAndreas Gohr            }
123955efc227SAndreas Gohr
124055efc227SAndreas Gohr            $origLength = $length;
124155efc227SAndreas Gohr            if (isset($data)) {
124255efc227SAndreas Gohr                $length = strlen($data);
124355efc227SAndreas Gohr            }
124455efc227SAndreas Gohr
124555efc227SAndreas Gohr            if ($marker != -1) {
124655efc227SAndreas Gohr                $this->_writeJPEGMarker($marker, $length, $data, $origLength);
124755efc227SAndreas Gohr            }
124855efc227SAndreas Gohr        }
124955efc227SAndreas Gohr
125055efc227SAndreas Gohr        if ($this->_fp) {
125155efc227SAndreas Gohr            fclose($this->_fp);
125255efc227SAndreas Gohr            $this->_fp = null;
125355efc227SAndreas Gohr        }
125455efc227SAndreas Gohr
125555efc227SAndreas Gohr        if ($this->_fpout) {
125655efc227SAndreas Gohr            fclose($this->_fpout);
125755efc227SAndreas Gohr            $this->_fpout = null;
125855efc227SAndreas Gohr        }
125955efc227SAndreas Gohr
126055efc227SAndreas Gohr        return $ok;
126155efc227SAndreas Gohr    }
126255efc227SAndreas Gohr
126355efc227SAndreas Gohr    /*************************************************************/
1264276820f7SScrutinizer Auto-Fixer
1265276820f7SScrutinizer Auto-Fixer    /**
1266276820f7SScrutinizer Auto-Fixer     * @param integer $marker
1267276820f7SScrutinizer Auto-Fixer     * @param integer $length
1268f50a239bSTakamura     * @param string $data
1269276820f7SScrutinizer Auto-Fixer     * @param integer $origLength
1270f50a239bSTakamura     *
1271f50a239bSTakamura     * @return bool
1272276820f7SScrutinizer Auto-Fixer     */
12730b17fdc6SAndreas Gohr    function _writeJPEGMarker($marker, $length, &$data, $origLength) {
127455efc227SAndreas Gohr        if ($length <= 0) {
127555efc227SAndreas Gohr            return false;
127655efc227SAndreas Gohr        }
127755efc227SAndreas Gohr
127855efc227SAndreas Gohr        fputs($this->_fpout, chr(0xFF), 1);
127955efc227SAndreas Gohr        fputs($this->_fpout, chr($marker), 1);
128055efc227SAndreas Gohr        fputs($this->_fpout, chr((($length + 2) & 0x0000FF00) >> 8), 1);
128155efc227SAndreas Gohr        fputs($this->_fpout, chr((($length + 2) & 0x000000FF) >> 0), 1);
128255efc227SAndreas Gohr
128355efc227SAndreas Gohr        if (isset($data)) {
128455efc227SAndreas Gohr            // Copy the generated data
128555efc227SAndreas Gohr            fputs($this->_fpout, $data, $length);
128655efc227SAndreas Gohr
128755efc227SAndreas Gohr            if ($origLength > 0) {   // Skip the original data
128855efc227SAndreas Gohr                $result = @fseek($this->_fp, $origLength, SEEK_CUR);
128955efc227SAndreas Gohr                // fseek doesn't seem to like HTTP 'files', but fgetc has no problem
129055efc227SAndreas Gohr                if ($result != 0) {
129155efc227SAndreas Gohr                    for ($i = 0; $i < $origLength; $i++) {
129255efc227SAndreas Gohr                        fgetc($this->_fp);
129355efc227SAndreas Gohr                    }
129455efc227SAndreas Gohr                }
129555efc227SAndreas Gohr            }
12960b17fdc6SAndreas Gohr        } else {
129755efc227SAndreas Gohr            if ($marker == 0xDA) {  // Copy until EOF
129855efc227SAndreas Gohr                while (!feof($this->_fp)) {
1299ed3655c4STom N Harris                    $data = fread($this->_fp, 1024 * 16);
130055efc227SAndreas Gohr                    fputs($this->_fpout, $data, strlen($data));
130155efc227SAndreas Gohr                }
13020b17fdc6SAndreas Gohr            } else { // Copy only $length bytes
1303ed3655c4STom N Harris                $data = @fread($this->_fp, $length);
130455efc227SAndreas Gohr                fputs($this->_fpout, $data, $length);
130555efc227SAndreas Gohr            }
130655efc227SAndreas Gohr        }
130755efc227SAndreas Gohr
130855efc227SAndreas Gohr        return true;
130955efc227SAndreas Gohr    }
131055efc227SAndreas Gohr
131123a34783SAndreas Gohr    /**
131223a34783SAndreas Gohr     * Gets basic info from the file - should work with non-JPEGs
131323a34783SAndreas Gohr     *
131423a34783SAndreas Gohr     * @author  Sebastian Delmont <sdelmont@zonageek.com>
131523a34783SAndreas Gohr     * @author  Andreas Gohr <andi@splitbrain.org>
131623a34783SAndreas Gohr     */
13170b17fdc6SAndreas Gohr    function _parseFileInfo() {
1318639f8f43SAndreas Gohr        if (file_exists($this->_fileName) && is_file($this->_fileName)) {
131955efc227SAndreas Gohr            $this->_info['file'] = array();
13208cbc5ee8SAndreas Gohr            $this->_info['file']['Name'] = utf8_decodeFN(\dokuwiki\Utf8\PhpString::basename($this->_fileName));
132100976812SAndreas Gohr            $this->_info['file']['Path'] = fullpath($this->_fileName);
132255efc227SAndreas Gohr            $this->_info['file']['Size'] = filesize($this->_fileName);
132355efc227SAndreas Gohr            if ($this->_info['file']['Size'] < 1024) {
132455efc227SAndreas Gohr                $this->_info['file']['NiceSize'] = $this->_info['file']['Size'] . 'B';
13250b17fdc6SAndreas Gohr            } elseif ($this->_info['file']['Size'] < (1024 * 1024)) {
132655efc227SAndreas Gohr                $this->_info['file']['NiceSize'] = round($this->_info['file']['Size'] / 1024) . 'KB';
13270b17fdc6SAndreas Gohr            } elseif ($this->_info['file']['Size'] < (1024 * 1024 * 1024)) {
1328fe00a666SAndreas Gohr                $this->_info['file']['NiceSize'] = round($this->_info['file']['Size'] / (1024*1024)) . 'MB';
13290b17fdc6SAndreas Gohr            } else {
133055efc227SAndreas Gohr                $this->_info['file']['NiceSize'] = $this->_info['file']['Size'] . 'B';
133155efc227SAndreas Gohr            }
133255efc227SAndreas Gohr            $this->_info['file']['UnixTime'] = filemtime($this->_fileName);
133355efc227SAndreas Gohr
133455efc227SAndreas Gohr            // get image size directly from file
1335bbdbbeb8SDamien Regad            if ($size = getimagesize($this->_fileName)) {
133655efc227SAndreas Gohr                $this->_info['file']['Width'] = $size[0];
133755efc227SAndreas Gohr                $this->_info['file']['Height'] = $size[1];
1338bbdbbeb8SDamien Regad
133955efc227SAndreas Gohr                // set mime types and formats
134059752844SAnders Sandblad                // http://php.net/manual/en/function.getimagesize.php
134159752844SAnders Sandblad                // http://php.net/manual/en/function.image-type-to-mime-type.php
134255efc227SAndreas Gohr                switch ($size[2]) {
134355efc227SAndreas Gohr                    case 1:
134455efc227SAndreas Gohr                        $this->_info['file']['Mime'] = 'image/gif';
134555efc227SAndreas Gohr                        $this->_info['file']['Format'] = 'GIF';
134655efc227SAndreas Gohr                        break;
134755efc227SAndreas Gohr                    case 2:
134855efc227SAndreas Gohr                        $this->_info['file']['Mime'] = 'image/jpeg';
134955efc227SAndreas Gohr                        $this->_info['file']['Format'] = 'JPEG';
135055efc227SAndreas Gohr                        break;
135155efc227SAndreas Gohr                    case 3:
135255efc227SAndreas Gohr                        $this->_info['file']['Mime'] = 'image/png';
135355efc227SAndreas Gohr                        $this->_info['file']['Format'] = 'PNG';
135455efc227SAndreas Gohr                        break;
135555efc227SAndreas Gohr                    case 4:
135655efc227SAndreas Gohr                        $this->_info['file']['Mime'] = 'application/x-shockwave-flash';
135755efc227SAndreas Gohr                        $this->_info['file']['Format'] = 'SWF';
135855efc227SAndreas Gohr                        break;
135955efc227SAndreas Gohr                    case 5:
136055efc227SAndreas Gohr                        $this->_info['file']['Mime'] = 'image/psd';
136155efc227SAndreas Gohr                        $this->_info['file']['Format'] = 'PSD';
136255efc227SAndreas Gohr                        break;
136355efc227SAndreas Gohr                    case 6:
136455efc227SAndreas Gohr                        $this->_info['file']['Mime'] = 'image/bmp';
136555efc227SAndreas Gohr                        $this->_info['file']['Format'] = 'BMP';
136655efc227SAndreas Gohr                        break;
136755efc227SAndreas Gohr                    case 7:
136855efc227SAndreas Gohr                        $this->_info['file']['Mime'] = 'image/tiff';
136955efc227SAndreas Gohr                        $this->_info['file']['Format'] = 'TIFF (Intel)';
137055efc227SAndreas Gohr                        break;
137155efc227SAndreas Gohr                    case 8:
137255efc227SAndreas Gohr                        $this->_info['file']['Mime'] = 'image/tiff';
137355efc227SAndreas Gohr                        $this->_info['file']['Format'] = 'TIFF (Motorola)';
137455efc227SAndreas Gohr                        break;
137555efc227SAndreas Gohr                    case 9:
137655efc227SAndreas Gohr                        $this->_info['file']['Mime'] = 'application/octet-stream';
137755efc227SAndreas Gohr                        $this->_info['file']['Format'] = 'JPC';
137855efc227SAndreas Gohr                        break;
137955efc227SAndreas Gohr                    case 10:
138055efc227SAndreas Gohr                        $this->_info['file']['Mime'] = 'image/jp2';
138155efc227SAndreas Gohr                        $this->_info['file']['Format'] = 'JP2';
138255efc227SAndreas Gohr                        break;
138355efc227SAndreas Gohr                    case 11:
138455efc227SAndreas Gohr                        $this->_info['file']['Mime'] = 'application/octet-stream';
138555efc227SAndreas Gohr                        $this->_info['file']['Format'] = 'JPX';
138655efc227SAndreas Gohr                        break;
138755efc227SAndreas Gohr                    case 12:
138855efc227SAndreas Gohr                        $this->_info['file']['Mime'] = 'application/octet-stream';
138955efc227SAndreas Gohr                        $this->_info['file']['Format'] = 'JB2';
139055efc227SAndreas Gohr                        break;
139155efc227SAndreas Gohr                    case 13:
139255efc227SAndreas Gohr                        $this->_info['file']['Mime'] = 'application/x-shockwave-flash';
139355efc227SAndreas Gohr                        $this->_info['file']['Format'] = 'SWC';
139455efc227SAndreas Gohr                        break;
139555efc227SAndreas Gohr                    case 14:
139655efc227SAndreas Gohr                        $this->_info['file']['Mime'] = 'image/iff';
139755efc227SAndreas Gohr                        $this->_info['file']['Format'] = 'IFF';
139855efc227SAndreas Gohr                        break;
139955efc227SAndreas Gohr                    case 15:
140055efc227SAndreas Gohr                        $this->_info['file']['Mime'] = 'image/vnd.wap.wbmp';
140155efc227SAndreas Gohr                        $this->_info['file']['Format'] = 'WBMP';
140255efc227SAndreas Gohr                        break;
140355efc227SAndreas Gohr                    case 16:
140455efc227SAndreas Gohr                        $this->_info['file']['Mime'] = 'image/xbm';
140555efc227SAndreas Gohr                        $this->_info['file']['Format'] = 'XBM';
140655efc227SAndreas Gohr                        break;
140755efc227SAndreas Gohr                    default:
140855efc227SAndreas Gohr                        $this->_info['file']['Mime'] = 'image/unknown';
140955efc227SAndreas Gohr                }
1410bbdbbeb8SDamien Regad            }
14110b17fdc6SAndreas Gohr        } else {
141255efc227SAndreas Gohr            $this->_info['file'] = array();
14138cbc5ee8SAndreas Gohr            $this->_info['file']['Name'] = \dokuwiki\Utf8\PhpString::basename($this->_fileName);
141455efc227SAndreas Gohr            $this->_info['file']['Url'] = $this->_fileName;
141555efc227SAndreas Gohr        }
141655efc227SAndreas Gohr
141755efc227SAndreas Gohr        return true;
141855efc227SAndreas Gohr    }
141955efc227SAndreas Gohr
142055efc227SAndreas Gohr    /*************************************************************/
14210b17fdc6SAndreas Gohr    function _parseMarkerJFIF() {
142255efc227SAndreas Gohr        if (!isset($this->_markers)) {
142355efc227SAndreas Gohr            $this->_readJPEG();
142455efc227SAndreas Gohr        }
142555efc227SAndreas Gohr
1426c9c56f8aSasivery        if ($this->_markers == null || $this->_isMarkerDisabled(('jfif'))) {
142755efc227SAndreas Gohr            return false;
142855efc227SAndreas Gohr        }
142955efc227SAndreas Gohr
1430c9c56f8aSasivery        try {
143155efc227SAndreas Gohr            $data = null;
143255efc227SAndreas Gohr            $count = count($this->_markers);
143355efc227SAndreas Gohr            for ($i = 0; $i < $count; $i++) {
143455efc227SAndreas Gohr                if ($this->_markers[$i]['marker'] == 0xE0) {
143555efc227SAndreas Gohr                    $signature = $this->_getFixedString($this->_markers[$i]['data'], 0, 4);
143655efc227SAndreas Gohr                    if ($signature == 'JFIF') {
143755efc227SAndreas Gohr                        $data =& $this->_markers[$i]['data'];
143855efc227SAndreas Gohr                        break;
143955efc227SAndreas Gohr                    }
144055efc227SAndreas Gohr                }
144155efc227SAndreas Gohr            }
144255efc227SAndreas Gohr
144355efc227SAndreas Gohr            if ($data == null) {
144455efc227SAndreas Gohr                $this->_info['jfif'] = false;
144555efc227SAndreas Gohr                return false;
144655efc227SAndreas Gohr            }
144755efc227SAndreas Gohr
144855efc227SAndreas Gohr            $this->_info['jfif'] = array();
144955efc227SAndreas Gohr
145055efc227SAndreas Gohr            $vmaj = $this->_getByte($data, 5);
145155efc227SAndreas Gohr            $vmin = $this->_getByte($data, 6);
145255efc227SAndreas Gohr
145355efc227SAndreas Gohr            $this->_info['jfif']['Version'] = sprintf('%d.%02d', $vmaj, $vmin);
145455efc227SAndreas Gohr
145555efc227SAndreas Gohr            $units = $this->_getByte($data, 7);
145655efc227SAndreas Gohr            switch ($units) {
145755efc227SAndreas Gohr                case 0:
145855efc227SAndreas Gohr                    $this->_info['jfif']['Units'] = 'pixels';
145955efc227SAndreas Gohr                    break;
146055efc227SAndreas Gohr                case 1:
146155efc227SAndreas Gohr                    $this->_info['jfif']['Units'] = 'dpi';
146255efc227SAndreas Gohr                    break;
146355efc227SAndreas Gohr                case 2:
146455efc227SAndreas Gohr                    $this->_info['jfif']['Units'] = 'dpcm';
146555efc227SAndreas Gohr                    break;
146655efc227SAndreas Gohr                default:
146755efc227SAndreas Gohr                    $this->_info['jfif']['Units'] = 'unknown';
146855efc227SAndreas Gohr                    break;
146955efc227SAndreas Gohr            }
147055efc227SAndreas Gohr
147155efc227SAndreas Gohr            $xdens = $this->_getShort($data, 8);
147255efc227SAndreas Gohr            $ydens = $this->_getShort($data, 10);
147355efc227SAndreas Gohr
147455efc227SAndreas Gohr            $this->_info['jfif']['XDensity'] = $xdens;
147555efc227SAndreas Gohr            $this->_info['jfif']['YDensity'] = $ydens;
147655efc227SAndreas Gohr
147755efc227SAndreas Gohr            $thumbx = $this->_getByte($data, 12);
147855efc227SAndreas Gohr            $thumby = $this->_getByte($data, 13);
147955efc227SAndreas Gohr
148055efc227SAndreas Gohr            $this->_info['jfif']['ThumbnailWidth'] = $thumbx;
148155efc227SAndreas Gohr            $this->_info['jfif']['ThumbnailHeight'] = $thumby;
1482c9c56f8aSasivery        } catch(Exception $e) {
1483c9c56f8aSasivery            $this->_handleMarkerParsingException($e);
1484c9c56f8aSasivery            $this->_info['jfif'] = false;
1485c9c56f8aSasivery            return false;
1486c9c56f8aSasivery        }
148755efc227SAndreas Gohr
148855efc227SAndreas Gohr        return true;
148955efc227SAndreas Gohr    }
149055efc227SAndreas Gohr
149155efc227SAndreas Gohr    /*************************************************************/
14920b17fdc6SAndreas Gohr    function _parseMarkerSOF() {
149355efc227SAndreas Gohr        if (!isset($this->_markers)) {
149455efc227SAndreas Gohr            $this->_readJPEG();
149555efc227SAndreas Gohr        }
149655efc227SAndreas Gohr
1497c9c56f8aSasivery        if ($this->_markers == null || $this->_isMarkerDisabled(('sof'))) {
149855efc227SAndreas Gohr            return false;
149955efc227SAndreas Gohr        }
150055efc227SAndreas Gohr
1501c9c56f8aSasivery        try {
150255efc227SAndreas Gohr            $data = null;
150355efc227SAndreas Gohr            $count = count($this->_markers);
150455efc227SAndreas Gohr            for ($i = 0; $i < $count; $i++) {
150555efc227SAndreas Gohr                switch ($this->_markers[$i]['marker']) {
150655efc227SAndreas Gohr                    case 0xC0: // SOF0
150755efc227SAndreas Gohr                    case 0xC1: // SOF1
150855efc227SAndreas Gohr                    case 0xC2: // SOF2
150955efc227SAndreas Gohr                    case 0xC9: // SOF9
151055efc227SAndreas Gohr                        $data =& $this->_markers[$i]['data'];
151155efc227SAndreas Gohr                        $marker = $this->_markers[$i]['marker'];
151255efc227SAndreas Gohr                        break;
151355efc227SAndreas Gohr                }
151455efc227SAndreas Gohr            }
151555efc227SAndreas Gohr
151655efc227SAndreas Gohr            if ($data == null) {
151755efc227SAndreas Gohr                $this->_info['sof'] = false;
151855efc227SAndreas Gohr                return false;
151955efc227SAndreas Gohr            }
152055efc227SAndreas Gohr
152155efc227SAndreas Gohr            $pos = 0;
152255efc227SAndreas Gohr            $this->_info['sof'] = array();
152355efc227SAndreas Gohr
152455efc227SAndreas Gohr            switch ($marker) {
152555efc227SAndreas Gohr                case 0xC0: // SOF0
152655efc227SAndreas Gohr                    $format = 'Baseline';
152755efc227SAndreas Gohr                    break;
152855efc227SAndreas Gohr                case 0xC1: // SOF1
152955efc227SAndreas Gohr                    $format = 'Progessive';
153055efc227SAndreas Gohr                    break;
153155efc227SAndreas Gohr                case 0xC2: // SOF2
153255efc227SAndreas Gohr                    $format = 'Non-baseline';
153355efc227SAndreas Gohr                    break;
153455efc227SAndreas Gohr                case 0xC9: // SOF9
153555efc227SAndreas Gohr                    $format = 'Arithmetic';
153655efc227SAndreas Gohr                    break;
153755efc227SAndreas Gohr                default:
153855efc227SAndreas Gohr                    return false;
153955efc227SAndreas Gohr            }
154055efc227SAndreas Gohr
154155efc227SAndreas Gohr            $this->_info['sof']['Format']          = $format;
154255efc227SAndreas Gohr            $this->_info['sof']['SamplePrecision'] = $this->_getByte($data, $pos + 0);
154355efc227SAndreas Gohr            $this->_info['sof']['ImageHeight']     = $this->_getShort($data, $pos + 1);
154455efc227SAndreas Gohr            $this->_info['sof']['ImageWidth']      = $this->_getShort($data, $pos + 3);
154555efc227SAndreas Gohr            $this->_info['sof']['ColorChannels']   = $this->_getByte($data, $pos + 5);
1546c9c56f8aSasivery        } catch(Exception $e) {
1547c9c56f8aSasivery            $this->_handleMarkerParsingException($e);
1548c9c56f8aSasivery            $this->_info['sof'] = false;
1549c9c56f8aSasivery            return false;
1550c9c56f8aSasivery        }
155155efc227SAndreas Gohr
155255efc227SAndreas Gohr        return true;
155355efc227SAndreas Gohr    }
155455efc227SAndreas Gohr
1555431c7fc8Shakan.sandell    /**
1556431c7fc8Shakan.sandell     * Parses the XMP data
1557431c7fc8Shakan.sandell     *
1558431c7fc8Shakan.sandell     * @author  Hakan Sandell <hakan.sandell@mydata.se>
1559431c7fc8Shakan.sandell     */
15600b17fdc6SAndreas Gohr    function _parseMarkerXmp() {
1561431c7fc8Shakan.sandell        if (!isset($this->_markers)) {
1562431c7fc8Shakan.sandell            $this->_readJPEG();
1563431c7fc8Shakan.sandell        }
1564431c7fc8Shakan.sandell
1565c9c56f8aSasivery        if ($this->_markers == null || $this->_isMarkerDisabled(('xmp'))) {
1566431c7fc8Shakan.sandell            return false;
1567431c7fc8Shakan.sandell        }
1568431c7fc8Shakan.sandell
1569c9c56f8aSasivery        try {
1570431c7fc8Shakan.sandell            $data = null;
1571431c7fc8Shakan.sandell            $count = count($this->_markers);
1572431c7fc8Shakan.sandell            for ($i = 0; $i < $count; $i++) {
1573431c7fc8Shakan.sandell                if ($this->_markers[$i]['marker'] == 0xE1) {
1574431c7fc8Shakan.sandell                    $signature = $this->_getFixedString($this->_markers[$i]['data'], 0, 29);
1575431c7fc8Shakan.sandell                    if ($signature == "http://ns.adobe.com/xap/1.0/\0") {
1576f5891fa4SGerrit Uitslag                        $data = substr($this->_markers[$i]['data'], 29);
1577431c7fc8Shakan.sandell                        break;
1578431c7fc8Shakan.sandell                    }
1579431c7fc8Shakan.sandell                }
1580431c7fc8Shakan.sandell            }
1581431c7fc8Shakan.sandell
1582431c7fc8Shakan.sandell            if ($data == null) {
1583431c7fc8Shakan.sandell                $this->_info['xmp'] = false;
1584431c7fc8Shakan.sandell                return false;
1585431c7fc8Shakan.sandell            }
1586431c7fc8Shakan.sandell
1587431c7fc8Shakan.sandell            $parser = xml_parser_create();
1588431c7fc8Shakan.sandell            xml_parser_set_option($parser, XML_OPTION_CASE_FOLDING, 0);
1589431c7fc8Shakan.sandell            xml_parser_set_option($parser, XML_OPTION_SKIP_WHITE, 1);
1590d8443bf1SHakan Sandell            $result = xml_parse_into_struct($parser, $data, $values, $tags);
1591431c7fc8Shakan.sandell            xml_parser_free($parser);
1592431c7fc8Shakan.sandell
1593d8443bf1SHakan Sandell            if ($result == 0) {
1594d8443bf1SHakan Sandell                $this->_info['xmp'] = false;
1595d8443bf1SHakan Sandell                return false;
1596d8443bf1SHakan Sandell            }
1597d8443bf1SHakan Sandell
1598431c7fc8Shakan.sandell            $this->_info['xmp'] = array();
1599431c7fc8Shakan.sandell            $count = count($values);
1600431c7fc8Shakan.sandell            for ($i = 0; $i < $count; $i++) {
16010b17fdc6SAndreas Gohr                if ($values[$i]['tag'] == 'rdf:Description' && $values[$i]['type'] == 'open') {
1602431c7fc8Shakan.sandell
1603d8443bf1SHakan Sandell                    while ((++$i < $count) && ($values[$i]['tag'] != 'rdf:Description')) {
1604d8443bf1SHakan Sandell                        $this->_parseXmpNode($values, $i, $this->_info['xmp'][$values[$i]['tag']], $count);
1605431c7fc8Shakan.sandell                    }
1606431c7fc8Shakan.sandell                }
1607431c7fc8Shakan.sandell            }
1608c9c56f8aSasivery        } catch (Exception $e) {
1609c9c56f8aSasivery            $this->_handleMarkerParsingException($e);
1610c9c56f8aSasivery            $this->_info['xmp'] = false;
1611c9c56f8aSasivery            return false;
1612c9c56f8aSasivery        }
1613c9c56f8aSasivery
1614431c7fc8Shakan.sandell        return true;
1615431c7fc8Shakan.sandell    }
1616431c7fc8Shakan.sandell
1617431c7fc8Shakan.sandell    /**
1618431c7fc8Shakan.sandell     * Parses XMP nodes by recursion
1619431c7fc8Shakan.sandell     *
1620431c7fc8Shakan.sandell     * @author  Hakan Sandell <hakan.sandell@mydata.se>
1621f50a239bSTakamura     *
1622f50a239bSTakamura     * @param array $values
1623f50a239bSTakamura     * @param int $i
1624f50a239bSTakamura     * @param mixed $meta
1625276820f7SScrutinizer Auto-Fixer     * @param integer $count
1626431c7fc8Shakan.sandell     */
1627d8443bf1SHakan Sandell    function _parseXmpNode($values, &$i, &$meta, $count) {
1628831c10d0SHakan Sandell        if ($values[$i]['type'] == 'close') return;
1629831c10d0SHakan Sandell
16300b17fdc6SAndreas Gohr        if ($values[$i]['type'] == 'complete') {
1631431c7fc8Shakan.sandell            // Simple Type property
1632e3b89425Sasivery            $meta = $values[$i]['value'] ?? '';
1633431c7fc8Shakan.sandell            return;
1634431c7fc8Shakan.sandell        }
1635431c7fc8Shakan.sandell
1636431c7fc8Shakan.sandell        $i++;
1637d8443bf1SHakan Sandell        if ($i >= $count) return;
1638d8443bf1SHakan Sandell
16390b17fdc6SAndreas Gohr        if ($values[$i]['tag'] == 'rdf:Bag' || $values[$i]['tag'] == 'rdf:Seq') {
1640431c7fc8Shakan.sandell            // Array property
1641431c7fc8Shakan.sandell            $meta = array();
16420b17fdc6SAndreas Gohr            while ($values[++$i]['tag'] == 'rdf:li') {
1643d8443bf1SHakan Sandell                $this->_parseXmpNode($values, $i, $meta[], $count);
1644431c7fc8Shakan.sandell            }
1645831c10d0SHakan Sandell            $i++; // skip closing Bag/Seq tag
1646431c7fc8Shakan.sandell
16470b17fdc6SAndreas Gohr        } elseif ($values[$i]['tag'] == 'rdf:Alt') {
1648431c7fc8Shakan.sandell            // Language Alternative property, only the first (default) value is used
1649831c10d0SHakan Sandell            if ($values[$i]['type'] == 'open') {
1650431c7fc8Shakan.sandell                $i++;
1651d8443bf1SHakan Sandell                $this->_parseXmpNode($values, $i, $meta, $count);
1652d8443bf1SHakan Sandell                while ((++$i < $count) && ($values[$i]['tag'] != 'rdf:Alt'));
1653831c10d0SHakan Sandell                $i++; // skip closing Alt tag
1654831c10d0SHakan Sandell            }
1655431c7fc8Shakan.sandell
1656431c7fc8Shakan.sandell        } else {
1657431c7fc8Shakan.sandell            // Structure property
1658431c7fc8Shakan.sandell            $meta = array();
16590b17fdc6SAndreas Gohr            $startTag = $values[$i-1]['tag'];
1660431c7fc8Shakan.sandell            do {
1661d8443bf1SHakan Sandell                $this->_parseXmpNode($values, $i, $meta[$values[$i]['tag']], $count);
1662d8443bf1SHakan Sandell            } while ((++$i < $count) && ($values[$i]['tag'] != $startTag));
1663431c7fc8Shakan.sandell        }
1664431c7fc8Shakan.sandell    }
1665431c7fc8Shakan.sandell
166655efc227SAndreas Gohr    /*************************************************************/
16670b17fdc6SAndreas Gohr    function _parseMarkerExif() {
166855efc227SAndreas Gohr        if (!isset($this->_markers)) {
166955efc227SAndreas Gohr            $this->_readJPEG();
167055efc227SAndreas Gohr        }
167155efc227SAndreas Gohr
1672c9c56f8aSasivery        if ($this->_markers == null || $this->_isMarkerDisabled(('exif'))) {
167355efc227SAndreas Gohr            return false;
167455efc227SAndreas Gohr        }
167555efc227SAndreas Gohr
1676c9c56f8aSasivery        try {
167755efc227SAndreas Gohr            $data = null;
167855efc227SAndreas Gohr            $count = count($this->_markers);
167955efc227SAndreas Gohr            for ($i = 0; $i < $count; $i++) {
168055efc227SAndreas Gohr                if ($this->_markers[$i]['marker'] == 0xE1) {
168155efc227SAndreas Gohr                    $signature = $this->_getFixedString($this->_markers[$i]['data'], 0, 6);
168255efc227SAndreas Gohr                    if ($signature == "Exif\0\0") {
168355efc227SAndreas Gohr                        $data =& $this->_markers[$i]['data'];
168455efc227SAndreas Gohr                        break;
168555efc227SAndreas Gohr                    }
168655efc227SAndreas Gohr                }
168755efc227SAndreas Gohr            }
168855efc227SAndreas Gohr
168955efc227SAndreas Gohr            if ($data == null) {
169055efc227SAndreas Gohr                $this->_info['exif'] = false;
169155efc227SAndreas Gohr                return false;
169255efc227SAndreas Gohr            }
169355efc227SAndreas Gohr            $pos = 6;
169455efc227SAndreas Gohr            $this->_info['exif'] = array();
169555efc227SAndreas Gohr
169655efc227SAndreas Gohr            // We don't increment $pos after this because Exif uses offsets relative to this point
169755efc227SAndreas Gohr
169855efc227SAndreas Gohr            $byteAlign = $this->_getShort($data, $pos + 0);
169955efc227SAndreas Gohr
170055efc227SAndreas Gohr            if ($byteAlign == 0x4949) { // "II"
170155efc227SAndreas Gohr                $isBigEndian = false;
17020b17fdc6SAndreas Gohr            } elseif ($byteAlign == 0x4D4D) { // "MM"
170355efc227SAndreas Gohr                $isBigEndian = true;
17040b17fdc6SAndreas Gohr            } else {
170555efc227SAndreas Gohr                return false; // Unexpected data
170655efc227SAndreas Gohr            }
170755efc227SAndreas Gohr
170855efc227SAndreas Gohr            $alignCheck = $this->_getShort($data, $pos + 2, $isBigEndian);
170955efc227SAndreas Gohr            if ($alignCheck != 0x002A) // That's the expected value
171055efc227SAndreas Gohr                return false; // Unexpected data
171155efc227SAndreas Gohr
171255efc227SAndreas Gohr            if ($isBigEndian) {
171355efc227SAndreas Gohr                $this->_info['exif']['ByteAlign'] = "Big Endian";
17140b17fdc6SAndreas Gohr            } else {
171555efc227SAndreas Gohr                $this->_info['exif']['ByteAlign'] = "Little Endian";
171655efc227SAndreas Gohr            }
171755efc227SAndreas Gohr
171855efc227SAndreas Gohr            $offsetIFD0 = $this->_getLong($data, $pos + 4, $isBigEndian);
171955efc227SAndreas Gohr            if ($offsetIFD0 < 8)
172055efc227SAndreas Gohr                return false; // Unexpected data
172155efc227SAndreas Gohr
172255efc227SAndreas Gohr            $offsetIFD1 = $this->_readIFD($data, $pos, $offsetIFD0, $isBigEndian, 'ifd0');
172355efc227SAndreas Gohr            if ($offsetIFD1 != 0)
172455efc227SAndreas Gohr                $this->_readIFD($data, $pos, $offsetIFD1, $isBigEndian, 'ifd1');
1725c9c56f8aSasivery        } catch(Exception $e) {
1726c9c56f8aSasivery            $this->_handleMarkerParsingException($e);
1727c9c56f8aSasivery            $this->_info['exif'] = false;
1728c9c56f8aSasivery            return false;
1729c9c56f8aSasivery        }
173055efc227SAndreas Gohr
173155efc227SAndreas Gohr        return true;
173255efc227SAndreas Gohr    }
173355efc227SAndreas Gohr
173455efc227SAndreas Gohr    /*************************************************************/
1735276820f7SScrutinizer Auto-Fixer
1736276820f7SScrutinizer Auto-Fixer    /**
1737f50a239bSTakamura     * @param mixed $data
1738276820f7SScrutinizer Auto-Fixer     * @param integer $base
1739f50a239bSTakamura     * @param integer $offset
1740276820f7SScrutinizer Auto-Fixer     * @param boolean $isBigEndian
1741276820f7SScrutinizer Auto-Fixer     * @param string $mode
1742f50a239bSTakamura     *
1743f50a239bSTakamura     * @return int
1744276820f7SScrutinizer Auto-Fixer     */
17450b17fdc6SAndreas Gohr    function _readIFD($data, $base, $offset, $isBigEndian, $mode) {
174655efc227SAndreas Gohr        $EXIFTags = $this->_exifTagNames($mode);
174755efc227SAndreas Gohr
174855efc227SAndreas Gohr        $numEntries = $this->_getShort($data, $base + $offset, $isBigEndian);
174955efc227SAndreas Gohr        $offset += 2;
175055efc227SAndreas Gohr
175155efc227SAndreas Gohr        $exifTIFFOffset = 0;
175255efc227SAndreas Gohr        $exifTIFFLength = 0;
175355efc227SAndreas Gohr        $exifThumbnailOffset = 0;
175455efc227SAndreas Gohr        $exifThumbnailLength = 0;
175555efc227SAndreas Gohr
175655efc227SAndreas Gohr        for ($i = 0; $i < $numEntries; $i++) {
175755efc227SAndreas Gohr            $tag = $this->_getShort($data, $base + $offset, $isBigEndian);
175855efc227SAndreas Gohr            $offset += 2;
175955efc227SAndreas Gohr            $type = $this->_getShort($data, $base + $offset, $isBigEndian);
176055efc227SAndreas Gohr            $offset += 2;
176155efc227SAndreas Gohr            $count = $this->_getLong($data, $base + $offset, $isBigEndian);
176255efc227SAndreas Gohr            $offset += 4;
176355efc227SAndreas Gohr
176455efc227SAndreas Gohr            if (($type < 1) || ($type > 12))
176555efc227SAndreas Gohr                return false; // Unexpected Type
176655efc227SAndreas Gohr
176755efc227SAndreas Gohr            $typeLengths = array( -1, 1, 1, 2, 4, 8, 1, 1, 2, 4, 8, 4, 8 );
176855efc227SAndreas Gohr
176955efc227SAndreas Gohr            $dataLength = $typeLengths[$type] * $count;
177055efc227SAndreas Gohr            if ($dataLength > 4) {
177155efc227SAndreas Gohr                $dataOffset = $this->_getLong($data, $base + $offset, $isBigEndian);
177255efc227SAndreas Gohr                $rawValue = $this->_getFixedString($data, $base + $dataOffset, $dataLength);
17730b17fdc6SAndreas Gohr            } else {
177455efc227SAndreas Gohr                $rawValue = $this->_getFixedString($data, $base + $offset, $dataLength);
177555efc227SAndreas Gohr            }
177655efc227SAndreas Gohr            $offset += 4;
177755efc227SAndreas Gohr
177855efc227SAndreas Gohr            switch ($type) {
177955efc227SAndreas Gohr                case 1:    // UBYTE
178055efc227SAndreas Gohr                    if ($count == 1) {
178155efc227SAndreas Gohr                        $value = $this->_getByte($rawValue, 0);
17820b17fdc6SAndreas Gohr                    } else {
178355efc227SAndreas Gohr                        $value = array();
178455efc227SAndreas Gohr                        for ($j = 0; $j < $count; $j++)
178555efc227SAndreas Gohr                            $value[$j] = $this->_getByte($rawValue, $j);
178655efc227SAndreas Gohr                    }
178755efc227SAndreas Gohr                    break;
178855efc227SAndreas Gohr                case 2:    // ASCII
178955efc227SAndreas Gohr                    $value = $rawValue;
179055efc227SAndreas Gohr                    break;
179155efc227SAndreas Gohr                case 3:    // USHORT
179255efc227SAndreas Gohr                    if ($count == 1) {
179355efc227SAndreas Gohr                        $value = $this->_getShort($rawValue, 0, $isBigEndian);
17940b17fdc6SAndreas Gohr                    } else {
179555efc227SAndreas Gohr                        $value = array();
179655efc227SAndreas Gohr                        for ($j = 0; $j < $count; $j++)
179755efc227SAndreas Gohr                            $value[$j] = $this->_getShort($rawValue, $j * 2, $isBigEndian);
179855efc227SAndreas Gohr                    }
179955efc227SAndreas Gohr                    break;
180055efc227SAndreas Gohr                case 4:    // ULONG
180155efc227SAndreas Gohr                    if ($count == 1) {
180255efc227SAndreas Gohr                        $value = $this->_getLong($rawValue, 0, $isBigEndian);
18030b17fdc6SAndreas Gohr                    } else {
180455efc227SAndreas Gohr                        $value = array();
180555efc227SAndreas Gohr                        for ($j = 0; $j < $count; $j++)
180655efc227SAndreas Gohr                            $value[$j] = $this->_getLong($rawValue, $j * 4, $isBigEndian);
180755efc227SAndreas Gohr                    }
180855efc227SAndreas Gohr                    break;
180955efc227SAndreas Gohr                case 5:    // URATIONAL
181055efc227SAndreas Gohr                    if ($count == 1) {
181155efc227SAndreas Gohr                        $a = $this->_getLong($rawValue, 0, $isBigEndian);
181255efc227SAndreas Gohr                        $b = $this->_getLong($rawValue, 4, $isBigEndian);
181355efc227SAndreas Gohr                        $value = array();
181455efc227SAndreas Gohr                        $value['val'] = 0;
181555efc227SAndreas Gohr                        $value['num'] = $a;
181655efc227SAndreas Gohr                        $value['den'] = $b;
181755efc227SAndreas Gohr                        if (($a != 0) && ($b != 0)) {
181855efc227SAndreas Gohr                            $value['val'] = $a / $b;
181955efc227SAndreas Gohr                        }
18200b17fdc6SAndreas Gohr                    } else {
182155efc227SAndreas Gohr                        $value = array();
182255efc227SAndreas Gohr                        for ($j = 0; $j < $count; $j++) {
182355efc227SAndreas Gohr                            $a = $this->_getLong($rawValue, $j * 8, $isBigEndian);
182455efc227SAndreas Gohr                            $b = $this->_getLong($rawValue, ($j * 8) + 4, $isBigEndian);
182555efc227SAndreas Gohr                            $value = array();
182655efc227SAndreas Gohr                            $value[$j]['val'] = 0;
182755efc227SAndreas Gohr                            $value[$j]['num'] = $a;
182855efc227SAndreas Gohr                            $value[$j]['den'] = $b;
182955efc227SAndreas Gohr                            if (($a != 0) && ($b != 0))
183055efc227SAndreas Gohr                                $value[$j]['val'] = $a / $b;
183155efc227SAndreas Gohr                        }
183255efc227SAndreas Gohr                    }
183355efc227SAndreas Gohr                    break;
183455efc227SAndreas Gohr                case 6:    // SBYTE
183555efc227SAndreas Gohr                    if ($count == 1) {
183655efc227SAndreas Gohr                        $value = $this->_getByte($rawValue, 0);
18370b17fdc6SAndreas Gohr                    } else {
183855efc227SAndreas Gohr                        $value = array();
183955efc227SAndreas Gohr                        for ($j = 0; $j < $count; $j++)
184055efc227SAndreas Gohr                            $value[$j] = $this->_getByte($rawValue, $j);
184155efc227SAndreas Gohr                    }
184255efc227SAndreas Gohr                    break;
184355efc227SAndreas Gohr                case 7:    // UNDEFINED
184455efc227SAndreas Gohr                    $value = $rawValue;
184555efc227SAndreas Gohr                    break;
184655efc227SAndreas Gohr                case 8:    // SSHORT
184755efc227SAndreas Gohr                    if ($count == 1) {
184855efc227SAndreas Gohr                        $value = $this->_getShort($rawValue, 0, $isBigEndian);
18490b17fdc6SAndreas Gohr                    } else {
185055efc227SAndreas Gohr                        $value = array();
185155efc227SAndreas Gohr                        for ($j = 0; $j < $count; $j++)
185255efc227SAndreas Gohr                            $value[$j] = $this->_getShort($rawValue, $j * 2, $isBigEndian);
185355efc227SAndreas Gohr                    }
185455efc227SAndreas Gohr                    break;
185555efc227SAndreas Gohr                case 9:    // SLONG
185655efc227SAndreas Gohr                    if ($count == 1) {
185755efc227SAndreas Gohr                        $value = $this->_getLong($rawValue, 0, $isBigEndian);
18580b17fdc6SAndreas Gohr                    } else {
185955efc227SAndreas Gohr                        $value = array();
186055efc227SAndreas Gohr                        for ($j = 0; $j < $count; $j++)
186155efc227SAndreas Gohr                            $value[$j] = $this->_getLong($rawValue, $j * 4, $isBigEndian);
186255efc227SAndreas Gohr                    }
186355efc227SAndreas Gohr                    break;
186455efc227SAndreas Gohr                case 10:   // SRATIONAL
186555efc227SAndreas Gohr                    if ($count == 1) {
186655efc227SAndreas Gohr                        $a = $this->_getLong($rawValue, 0, $isBigEndian);
186755efc227SAndreas Gohr                        $b = $this->_getLong($rawValue, 4, $isBigEndian);
186855efc227SAndreas Gohr                        $value = array();
186955efc227SAndreas Gohr                        $value['val'] = 0;
187055efc227SAndreas Gohr                        $value['num'] = $a;
187155efc227SAndreas Gohr                        $value['den'] = $b;
187255efc227SAndreas Gohr                        if (($a != 0) && ($b != 0))
187355efc227SAndreas Gohr                            $value['val'] = $a / $b;
18740b17fdc6SAndreas Gohr                    } else {
187555efc227SAndreas Gohr                        $value = array();
187655efc227SAndreas Gohr                        for ($j = 0; $j < $count; $j++) {
187755efc227SAndreas Gohr                            $a = $this->_getLong($rawValue, $j * 8, $isBigEndian);
187855efc227SAndreas Gohr                            $b = $this->_getLong($rawValue, ($j * 8) + 4, $isBigEndian);
187955efc227SAndreas Gohr                            $value = array();
188055efc227SAndreas Gohr                            $value[$j]['val'] = 0;
188155efc227SAndreas Gohr                            $value[$j]['num'] = $a;
188255efc227SAndreas Gohr                            $value[$j]['den'] = $b;
188355efc227SAndreas Gohr                            if (($a != 0) && ($b != 0))
188455efc227SAndreas Gohr                                $value[$j]['val'] = $a / $b;
188555efc227SAndreas Gohr                        }
188655efc227SAndreas Gohr                    }
188755efc227SAndreas Gohr                    break;
188855efc227SAndreas Gohr                case 11:   // FLOAT
188955efc227SAndreas Gohr                    $value = $rawValue;
189055efc227SAndreas Gohr                    break;
189155efc227SAndreas Gohr
189255efc227SAndreas Gohr                case 12:   // DFLOAT
189355efc227SAndreas Gohr                    $value = $rawValue;
189455efc227SAndreas Gohr                    break;
189555efc227SAndreas Gohr                default:
189655efc227SAndreas Gohr                    return false; // Unexpected Type
189755efc227SAndreas Gohr            }
189855efc227SAndreas Gohr
189955efc227SAndreas Gohr            $tagName = '';
190055efc227SAndreas Gohr            if (($mode == 'ifd0') && ($tag == 0x8769)) {  // ExifIFDOffset
190155efc227SAndreas Gohr                $this->_readIFD($data, $base, $value, $isBigEndian, 'exif');
19020b17fdc6SAndreas Gohr            } elseif (($mode == 'ifd0') && ($tag == 0x8825)) {  // GPSIFDOffset
190355efc227SAndreas Gohr                $this->_readIFD($data, $base, $value, $isBigEndian, 'gps');
19040b17fdc6SAndreas Gohr            } elseif (($mode == 'ifd1') && ($tag == 0x0111)) {  // TIFFStripOffsets
190555efc227SAndreas Gohr                $exifTIFFOffset = $value;
19060b17fdc6SAndreas Gohr            } elseif (($mode == 'ifd1') && ($tag == 0x0117)) {  // TIFFStripByteCounts
190755efc227SAndreas Gohr                $exifTIFFLength = $value;
19080b17fdc6SAndreas Gohr            } elseif (($mode == 'ifd1') && ($tag == 0x0201)) {  // TIFFJFIFOffset
190955efc227SAndreas Gohr                $exifThumbnailOffset = $value;
19100b17fdc6SAndreas Gohr            } elseif (($mode == 'ifd1') && ($tag == 0x0202)) {  // TIFFJFIFLength
191155efc227SAndreas Gohr                $exifThumbnailLength = $value;
19120b17fdc6SAndreas Gohr            } elseif (($mode == 'exif') && ($tag == 0xA005)) {  // InteropIFDOffset
191355efc227SAndreas Gohr                $this->_readIFD($data, $base, $value, $isBigEndian, 'interop');
191455efc227SAndreas Gohr            }
191555efc227SAndreas Gohr            // elseif (($mode == 'exif') && ($tag == 0x927C)) {  // MakerNote
191655efc227SAndreas Gohr            // }
191755efc227SAndreas Gohr            else {
191855efc227SAndreas Gohr                if (isset($EXIFTags[$tag])) {
191955efc227SAndreas Gohr                    $tagName = $EXIFTags[$tag];
192055efc227SAndreas Gohr                    if (isset($this->_info['exif'][$tagName])) {
192155efc227SAndreas Gohr                        if (!is_array($this->_info['exif'][$tagName])) {
192255efc227SAndreas Gohr                            $aux = array();
192355efc227SAndreas Gohr                            $aux[0] = $this->_info['exif'][$tagName];
192455efc227SAndreas Gohr                            $this->_info['exif'][$tagName] = $aux;
192555efc227SAndreas Gohr                        }
192655efc227SAndreas Gohr
192755efc227SAndreas Gohr                        $this->_info['exif'][$tagName][count($this->_info['exif'][$tagName])] = $value;
19280b17fdc6SAndreas Gohr                    } else {
192955efc227SAndreas Gohr                        $this->_info['exif'][$tagName] = $value;
193055efc227SAndreas Gohr                    }
193155efc227SAndreas Gohr                }
19320b17fdc6SAndreas Gohr                /*
193355efc227SAndreas Gohr                 else {
19340b17fdc6SAndreas Gohr                    echo sprintf("<h1>Unknown tag %02x (t: %d l: %d) %s in %s</h1>", $tag, $type, $count, $mode, $this->_fileName);
193555efc227SAndreas Gohr                    // Unknown Tags will be ignored!!!
193655efc227SAndreas Gohr                    // That's because the tag might be a pointer (like the Exif tag)
193755efc227SAndreas Gohr                    // and saving it without saving the data it points to might
193855efc227SAndreas Gohr                    // create an invalid file.
193955efc227SAndreas Gohr                }
19400b17fdc6SAndreas Gohr                */
194155efc227SAndreas Gohr            }
194255efc227SAndreas Gohr        }
194355efc227SAndreas Gohr
194455efc227SAndreas Gohr        if (($exifThumbnailOffset > 0) && ($exifThumbnailLength > 0)) {
194555efc227SAndreas Gohr            $this->_info['exif']['JFIFThumbnail'] = $this->_getFixedString($data, $base + $exifThumbnailOffset, $exifThumbnailLength);
194655efc227SAndreas Gohr        }
194755efc227SAndreas Gohr
194855efc227SAndreas Gohr        if (($exifTIFFOffset > 0) && ($exifTIFFLength > 0)) {
194955efc227SAndreas Gohr            $this->_info['exif']['TIFFStrips'] = $this->_getFixedString($data, $base + $exifTIFFOffset, $exifTIFFLength);
195055efc227SAndreas Gohr        }
195155efc227SAndreas Gohr
195255efc227SAndreas Gohr        $nextOffset = $this->_getLong($data, $base + $offset, $isBigEndian);
195355efc227SAndreas Gohr        return $nextOffset;
195455efc227SAndreas Gohr    }
195555efc227SAndreas Gohr
195655efc227SAndreas Gohr    /*************************************************************/
19570b17fdc6SAndreas Gohr    function & _createMarkerExif() {
195855efc227SAndreas Gohr        $data = null;
195955efc227SAndreas Gohr        $count = count($this->_markers);
196055efc227SAndreas Gohr        for ($i = 0; $i < $count; $i++) {
196155efc227SAndreas Gohr            if ($this->_markers[$i]['marker'] == 0xE1) {
196255efc227SAndreas Gohr                $signature = $this->_getFixedString($this->_markers[$i]['data'], 0, 6);
196355efc227SAndreas Gohr                if ($signature == "Exif\0\0") {
196455efc227SAndreas Gohr                    $data =& $this->_markers[$i]['data'];
196555efc227SAndreas Gohr                    break;
196655efc227SAndreas Gohr                }
196755efc227SAndreas Gohr            }
196855efc227SAndreas Gohr        }
196955efc227SAndreas Gohr
197055efc227SAndreas Gohr        if (!isset($this->_info['exif'])) {
197155efc227SAndreas Gohr            return false;
197255efc227SAndreas Gohr        }
197355efc227SAndreas Gohr
197455efc227SAndreas Gohr        $data = "Exif\0\0";
197555efc227SAndreas Gohr        $pos = 6;
197655efc227SAndreas Gohr        $offsetBase = 6;
197755efc227SAndreas Gohr
197855efc227SAndreas Gohr        if (isset($this->_info['exif']['ByteAlign']) && ($this->_info['exif']['ByteAlign'] == "Big Endian")) {
197955efc227SAndreas Gohr            $isBigEndian = true;
198055efc227SAndreas Gohr            $aux = "MM";
198155efc227SAndreas Gohr            $pos = $this->_putString($data, $pos, $aux);
19820b17fdc6SAndreas Gohr        } else {
198355efc227SAndreas Gohr            $isBigEndian = false;
198455efc227SAndreas Gohr            $aux = "II";
198555efc227SAndreas Gohr            $pos = $this->_putString($data, $pos, $aux);
198655efc227SAndreas Gohr        }
198755efc227SAndreas Gohr        $pos = $this->_putShort($data, $pos, 0x002A, $isBigEndian);
198855efc227SAndreas Gohr        $pos = $this->_putLong($data, $pos, 0x00000008, $isBigEndian); // IFD0 Offset is always 8
198955efc227SAndreas Gohr
199055efc227SAndreas Gohr        $ifd0 =& $this->_getIFDEntries($isBigEndian, 'ifd0');
199155efc227SAndreas Gohr        $ifd1 =& $this->_getIFDEntries($isBigEndian, 'ifd1');
199255efc227SAndreas Gohr
199355efc227SAndreas Gohr        $pos = $this->_writeIFD($data, $pos, $offsetBase, $ifd0, $isBigEndian, true);
199455efc227SAndreas Gohr        $pos = $this->_writeIFD($data, $pos, $offsetBase, $ifd1, $isBigEndian, false);
199555efc227SAndreas Gohr
199655efc227SAndreas Gohr        return $data;
199755efc227SAndreas Gohr    }
199855efc227SAndreas Gohr
199955efc227SAndreas Gohr    /*************************************************************/
2000276820f7SScrutinizer Auto-Fixer
2001276820f7SScrutinizer Auto-Fixer    /**
2002f50a239bSTakamura     * @param mixed $data
2003f50a239bSTakamura     * @param integer $pos
2004276820f7SScrutinizer Auto-Fixer     * @param integer $offsetBase
2005f50a239bSTakamura     * @param array $entries
2006276820f7SScrutinizer Auto-Fixer     * @param boolean $isBigEndian
2007276820f7SScrutinizer Auto-Fixer     * @param boolean $hasNext
2008f50a239bSTakamura     *
2009f50a239bSTakamura     * @return mixed
2010276820f7SScrutinizer Auto-Fixer     */
20110b17fdc6SAndreas Gohr    function _writeIFD(&$data, $pos, $offsetBase, &$entries, $isBigEndian, $hasNext) {
201255efc227SAndreas Gohr        $tiffData = null;
201355efc227SAndreas Gohr        $tiffDataOffsetPos = -1;
201455efc227SAndreas Gohr
201555efc227SAndreas Gohr        $entryCount = count($entries);
201655efc227SAndreas Gohr
201755efc227SAndreas Gohr        $dataPos = $pos + 2 + ($entryCount * 12) + 4;
201855efc227SAndreas Gohr        $pos = $this->_putShort($data, $pos, $entryCount, $isBigEndian);
201955efc227SAndreas Gohr
202055efc227SAndreas Gohr        for ($i = 0; $i < $entryCount; $i++) {
202155efc227SAndreas Gohr            $tag = $entries[$i]['tag'];
202255efc227SAndreas Gohr            $type = $entries[$i]['type'];
202355efc227SAndreas Gohr
202455efc227SAndreas Gohr            if ($type == -99) { // SubIFD
202555efc227SAndreas Gohr                $pos = $this->_putShort($data, $pos, $tag, $isBigEndian);
202655efc227SAndreas Gohr                $pos = $this->_putShort($data, $pos, 0x04, $isBigEndian); // LONG
202755efc227SAndreas Gohr                $pos = $this->_putLong($data, $pos, 0x01, $isBigEndian); // Count = 1
202855efc227SAndreas Gohr                $pos = $this->_putLong($data, $pos, $dataPos - $offsetBase, $isBigEndian);
202955efc227SAndreas Gohr
203055efc227SAndreas Gohr                $dataPos = $this->_writeIFD($data, $dataPos, $offsetBase, $entries[$i]['value'], $isBigEndian, false);
20310b17fdc6SAndreas Gohr            } elseif ($type == -98) { // TIFF Data
203255efc227SAndreas Gohr                $pos = $this->_putShort($data, $pos, $tag, $isBigEndian);
203355efc227SAndreas Gohr                $pos = $this->_putShort($data, $pos, 0x04, $isBigEndian); // LONG
203455efc227SAndreas Gohr                $pos = $this->_putLong($data, $pos, 0x01, $isBigEndian); // Count = 1
203555efc227SAndreas Gohr                $tiffDataOffsetPos = $pos;
203655efc227SAndreas Gohr                $pos = $this->_putLong($data, $pos, 0x00, $isBigEndian); // For Now
203755efc227SAndreas Gohr                $tiffData =& $entries[$i]['value'] ;
20380b17fdc6SAndreas Gohr            } else { // Regular Entry
203955efc227SAndreas Gohr                $pos = $this->_putShort($data, $pos, $tag, $isBigEndian);
204055efc227SAndreas Gohr                $pos = $this->_putShort($data, $pos, $type, $isBigEndian);
204155efc227SAndreas Gohr                $pos = $this->_putLong($data, $pos, $entries[$i]['count'], $isBigEndian);
204255efc227SAndreas Gohr                if (strlen($entries[$i]['value']) > 4) {
204355efc227SAndreas Gohr                    $pos = $this->_putLong($data, $pos, $dataPos - $offsetBase, $isBigEndian);
204455efc227SAndreas Gohr                    $dataPos = $this->_putString($data, $dataPos, $entries[$i]['value']);
20450b17fdc6SAndreas Gohr                } else {
204655efc227SAndreas Gohr                    $val = str_pad($entries[$i]['value'], 4, "\0");
204755efc227SAndreas Gohr                    $pos = $this->_putString($data, $pos, $val);
204855efc227SAndreas Gohr                }
204955efc227SAndreas Gohr            }
205055efc227SAndreas Gohr        }
205155efc227SAndreas Gohr
205255efc227SAndreas Gohr        if ($tiffData != null) {
205355efc227SAndreas Gohr            $this->_putLong($data, $tiffDataOffsetPos, $dataPos - $offsetBase, $isBigEndian);
205455efc227SAndreas Gohr            $dataPos = $this->_putString($data, $dataPos, $tiffData);
205555efc227SAndreas Gohr        }
205655efc227SAndreas Gohr
205755efc227SAndreas Gohr        if ($hasNext) {
205855efc227SAndreas Gohr            $pos = $this->_putLong($data, $pos, $dataPos - $offsetBase, $isBigEndian);
20590b17fdc6SAndreas Gohr        } else {
206055efc227SAndreas Gohr            $pos = $this->_putLong($data, $pos, 0, $isBigEndian);
206155efc227SAndreas Gohr        }
206255efc227SAndreas Gohr
206355efc227SAndreas Gohr        return $dataPos;
206455efc227SAndreas Gohr    }
206555efc227SAndreas Gohr
206655efc227SAndreas Gohr    /*************************************************************/
2067276820f7SScrutinizer Auto-Fixer
2068276820f7SScrutinizer Auto-Fixer    /**
2069276820f7SScrutinizer Auto-Fixer     * @param boolean $isBigEndian
2070276820f7SScrutinizer Auto-Fixer     * @param string $mode
2071f50a239bSTakamura     *
2072f50a239bSTakamura     * @return array
2073276820f7SScrutinizer Auto-Fixer     */
20740b17fdc6SAndreas Gohr    function & _getIFDEntries($isBigEndian, $mode) {
207555efc227SAndreas Gohr        $EXIFNames = $this->_exifTagNames($mode);
207655efc227SAndreas Gohr        $EXIFTags = $this->_exifNameTags($mode);
207755efc227SAndreas Gohr        $EXIFTypeInfo = $this->_exifTagTypes($mode);
207855efc227SAndreas Gohr
207955efc227SAndreas Gohr        $ifdEntries = array();
208055efc227SAndreas Gohr        $entryCount = 0;
208155efc227SAndreas Gohr
20829e491c01SAndreas Gohr        foreach($EXIFNames as $tag => $name) {
208355efc227SAndreas Gohr            $type = $EXIFTypeInfo[$tag][0];
208455efc227SAndreas Gohr            $count = $EXIFTypeInfo[$tag][1];
208555efc227SAndreas Gohr            $value = null;
208655efc227SAndreas Gohr
208755efc227SAndreas Gohr            if (($mode == 'ifd0') && ($tag == 0x8769)) {  // ExifIFDOffset
208855efc227SAndreas Gohr                if (isset($this->_info['exif']['EXIFVersion'])) {
208955efc227SAndreas Gohr                    $value =& $this->_getIFDEntries($isBigEndian, "exif");
209055efc227SAndreas Gohr                    $type = -99;
209155efc227SAndreas Gohr                }
209255efc227SAndreas Gohr                else {
209355efc227SAndreas Gohr                    $value = null;
209455efc227SAndreas Gohr                }
20950b17fdc6SAndreas Gohr            } elseif (($mode == 'ifd0') && ($tag == 0x8825)) {  // GPSIFDOffset
209655efc227SAndreas Gohr                if (isset($this->_info['exif']['GPSVersionID'])) {
209755efc227SAndreas Gohr                    $value =& $this->_getIFDEntries($isBigEndian, "gps");
209855efc227SAndreas Gohr                    $type = -99;
20990b17fdc6SAndreas Gohr                } else {
210055efc227SAndreas Gohr                    $value = null;
210155efc227SAndreas Gohr                }
21020b17fdc6SAndreas Gohr            } elseif (($mode == 'ifd1') && ($tag == 0x0111)) {  // TIFFStripOffsets
210355efc227SAndreas Gohr                if (isset($this->_info['exif']['TIFFStrips'])) {
210455efc227SAndreas Gohr                    $value =& $this->_info['exif']['TIFFStrips'];
210555efc227SAndreas Gohr                    $type = -98;
21060b17fdc6SAndreas Gohr                } else {
210755efc227SAndreas Gohr                    $value = null;
210855efc227SAndreas Gohr                }
21090b17fdc6SAndreas Gohr            } elseif (($mode == 'ifd1') && ($tag == 0x0117)) {  // TIFFStripByteCounts
211055efc227SAndreas Gohr                if (isset($this->_info['exif']['TIFFStrips'])) {
211155efc227SAndreas Gohr                    $value = strlen($this->_info['exif']['TIFFStrips']);
21120b17fdc6SAndreas Gohr                } else {
211355efc227SAndreas Gohr                    $value = null;
211455efc227SAndreas Gohr                }
21150b17fdc6SAndreas Gohr            } elseif (($mode == 'ifd1') && ($tag == 0x0201)) {  // TIFFJFIFOffset
211655efc227SAndreas Gohr                if (isset($this->_info['exif']['JFIFThumbnail'])) {
211755efc227SAndreas Gohr                    $value =& $this->_info['exif']['JFIFThumbnail'];
211855efc227SAndreas Gohr                    $type = -98;
21190b17fdc6SAndreas Gohr                } else {
212055efc227SAndreas Gohr                    $value = null;
212155efc227SAndreas Gohr                }
21220b17fdc6SAndreas Gohr            } elseif (($mode == 'ifd1') && ($tag == 0x0202)) {  // TIFFJFIFLength
212355efc227SAndreas Gohr                if (isset($this->_info['exif']['JFIFThumbnail'])) {
212455efc227SAndreas Gohr                    $value = strlen($this->_info['exif']['JFIFThumbnail']);
21250b17fdc6SAndreas Gohr                } else {
212655efc227SAndreas Gohr                    $value = null;
212755efc227SAndreas Gohr                }
21280b17fdc6SAndreas Gohr            } elseif (($mode == 'exif') && ($tag == 0xA005)) {  // InteropIFDOffset
212955efc227SAndreas Gohr                if (isset($this->_info['exif']['InteroperabilityIndex'])) {
213055efc227SAndreas Gohr                    $value =& $this->_getIFDEntries($isBigEndian, "interop");
213155efc227SAndreas Gohr                    $type = -99;
21320b17fdc6SAndreas Gohr                } else {
213355efc227SAndreas Gohr                    $value = null;
213455efc227SAndreas Gohr                }
21350b17fdc6SAndreas Gohr            } elseif (isset($this->_info['exif'][$name])) {
213655efc227SAndreas Gohr                $origValue =& $this->_info['exif'][$name];
213755efc227SAndreas Gohr
213855efc227SAndreas Gohr                // This makes it easier to process variable size elements
213955efc227SAndreas Gohr                if (!is_array($origValue) || isset($origValue['val'])) {
214055efc227SAndreas Gohr                    unset($origValue); // Break the reference
214155efc227SAndreas Gohr                    $origValue = array($this->_info['exif'][$name]);
214255efc227SAndreas Gohr                }
214355efc227SAndreas Gohr                $origCount = count($origValue);
214455efc227SAndreas Gohr
214555efc227SAndreas Gohr                if ($origCount == 0 ) {
214655efc227SAndreas Gohr                    $type = -1;  // To ignore this field
214755efc227SAndreas Gohr                }
214855efc227SAndreas Gohr
214955efc227SAndreas Gohr                $value = " ";
215055efc227SAndreas Gohr
215155efc227SAndreas Gohr                switch ($type) {
215255efc227SAndreas Gohr                    case 1:    // UBYTE
215355efc227SAndreas Gohr                        if ($count == 0) {
215455efc227SAndreas Gohr                            $count = $origCount;
215555efc227SAndreas Gohr                        }
215655efc227SAndreas Gohr
215755efc227SAndreas Gohr                        $j = 0;
215855efc227SAndreas Gohr                        while (($j < $count) && ($j < $origCount)) {
215955efc227SAndreas Gohr
216055efc227SAndreas Gohr                            $this->_putByte($value, $j, $origValue[$j]);
216155efc227SAndreas Gohr                            $j++;
216255efc227SAndreas Gohr                        }
216355efc227SAndreas Gohr
216455efc227SAndreas Gohr                        while ($j < $count) {
216555efc227SAndreas Gohr                            $this->_putByte($value, $j, 0);
216655efc227SAndreas Gohr                            $j++;
216755efc227SAndreas Gohr                        }
216855efc227SAndreas Gohr                        break;
216955efc227SAndreas Gohr                    case 2:    // ASCII
217055efc227SAndreas Gohr                        $v = strval($origValue[0]);
217155efc227SAndreas Gohr                        if (($count != 0) && (strlen($v) > $count)) {
217255efc227SAndreas Gohr                            $v = substr($v, 0, $count);
217355efc227SAndreas Gohr                        }
217455efc227SAndreas Gohr                        elseif (($count > 0) && (strlen($v) < $count)) {
217555efc227SAndreas Gohr                            $v = str_pad($v, $count, "\0");
217655efc227SAndreas Gohr                        }
217755efc227SAndreas Gohr
217855efc227SAndreas Gohr                        $count = strlen($v);
217955efc227SAndreas Gohr
218055efc227SAndreas Gohr                        $this->_putString($value, 0, $v);
218155efc227SAndreas Gohr                        break;
218255efc227SAndreas Gohr                    case 3:    // USHORT
218355efc227SAndreas Gohr                        if ($count == 0) {
218455efc227SAndreas Gohr                            $count = $origCount;
218555efc227SAndreas Gohr                        }
218655efc227SAndreas Gohr
218755efc227SAndreas Gohr                        $j = 0;
218855efc227SAndreas Gohr                        while (($j < $count) && ($j < $origCount)) {
218955efc227SAndreas Gohr                            $this->_putShort($value, $j * 2, $origValue[$j], $isBigEndian);
219055efc227SAndreas Gohr                            $j++;
219155efc227SAndreas Gohr                        }
219255efc227SAndreas Gohr
219355efc227SAndreas Gohr                        while ($j < $count) {
219455efc227SAndreas Gohr                            $this->_putShort($value, $j * 2, 0, $isBigEndian);
219555efc227SAndreas Gohr                            $j++;
219655efc227SAndreas Gohr                        }
219755efc227SAndreas Gohr                        break;
219855efc227SAndreas Gohr                    case 4:    // ULONG
219955efc227SAndreas Gohr                        if ($count == 0) {
220055efc227SAndreas Gohr                            $count = $origCount;
220155efc227SAndreas Gohr                        }
220255efc227SAndreas Gohr
220355efc227SAndreas Gohr                        $j = 0;
220455efc227SAndreas Gohr                        while (($j < $count) && ($j < $origCount)) {
220555efc227SAndreas Gohr                            $this->_putLong($value, $j * 4, $origValue[$j], $isBigEndian);
220655efc227SAndreas Gohr                            $j++;
220755efc227SAndreas Gohr                        }
220855efc227SAndreas Gohr
220955efc227SAndreas Gohr                        while ($j < $count) {
221055efc227SAndreas Gohr                            $this->_putLong($value, $j * 4, 0, $isBigEndian);
221155efc227SAndreas Gohr                            $j++;
221255efc227SAndreas Gohr                        }
221355efc227SAndreas Gohr                        break;
221455efc227SAndreas Gohr                    case 5:    // URATIONAL
221555efc227SAndreas Gohr                        if ($count == 0) {
221655efc227SAndreas Gohr                            $count = $origCount;
221755efc227SAndreas Gohr                        }
221855efc227SAndreas Gohr
221955efc227SAndreas Gohr                        $j = 0;
222055efc227SAndreas Gohr                        while (($j < $count) && ($j < $origCount)) {
222155efc227SAndreas Gohr                            $v = $origValue[$j];
222255efc227SAndreas Gohr                            if (is_array($v)) {
222355efc227SAndreas Gohr                                $a = $v['num'];
222455efc227SAndreas Gohr                                $b = $v['den'];
222555efc227SAndreas Gohr                            }
222655efc227SAndreas Gohr                            else {
222755efc227SAndreas Gohr                                $a = 0;
222855efc227SAndreas Gohr                                $b = 0;
222955efc227SAndreas Gohr                                // TODO: Allow other types and convert them
223055efc227SAndreas Gohr                            }
223155efc227SAndreas Gohr                            $this->_putLong($value, $j * 8, $a, $isBigEndian);
223255efc227SAndreas Gohr                            $this->_putLong($value, ($j * 8) + 4, $b, $isBigEndian);
223355efc227SAndreas Gohr                            $j++;
223455efc227SAndreas Gohr                        }
223555efc227SAndreas Gohr
223655efc227SAndreas Gohr                        while ($j < $count) {
223755efc227SAndreas Gohr                            $this->_putLong($value, $j * 8, 0, $isBigEndian);
223855efc227SAndreas Gohr                            $this->_putLong($value, ($j * 8) + 4, 0, $isBigEndian);
223955efc227SAndreas Gohr                            $j++;
224055efc227SAndreas Gohr                        }
224155efc227SAndreas Gohr                        break;
224255efc227SAndreas Gohr                    case 6:    // SBYTE
224355efc227SAndreas Gohr                        if ($count == 0) {
224455efc227SAndreas Gohr                            $count = $origCount;
224555efc227SAndreas Gohr                        }
224655efc227SAndreas Gohr
224755efc227SAndreas Gohr                        $j = 0;
224855efc227SAndreas Gohr                        while (($j < $count) && ($j < $origCount)) {
224955efc227SAndreas Gohr                            $this->_putByte($value, $j, $origValue[$j]);
225055efc227SAndreas Gohr                            $j++;
225155efc227SAndreas Gohr                        }
225255efc227SAndreas Gohr
225355efc227SAndreas Gohr                        while ($j < $count) {
225455efc227SAndreas Gohr                            $this->_putByte($value, $j, 0);
225555efc227SAndreas Gohr                            $j++;
225655efc227SAndreas Gohr                        }
225755efc227SAndreas Gohr                        break;
225855efc227SAndreas Gohr                    case 7:    // UNDEFINED
225955efc227SAndreas Gohr                        $v = strval($origValue[0]);
226055efc227SAndreas Gohr                        if (($count != 0) && (strlen($v) > $count)) {
226155efc227SAndreas Gohr                            $v = substr($v, 0, $count);
226255efc227SAndreas Gohr                        }
226355efc227SAndreas Gohr                        elseif (($count > 0) && (strlen($v) < $count)) {
226455efc227SAndreas Gohr                            $v = str_pad($v, $count, "\0");
226555efc227SAndreas Gohr                        }
226655efc227SAndreas Gohr
226755efc227SAndreas Gohr                        $count = strlen($v);
226855efc227SAndreas Gohr
226955efc227SAndreas Gohr                        $this->_putString($value, 0, $v);
227055efc227SAndreas Gohr                        break;
227155efc227SAndreas Gohr                    case 8:    // SSHORT
227255efc227SAndreas Gohr                        if ($count == 0) {
227355efc227SAndreas Gohr                            $count = $origCount;
227455efc227SAndreas Gohr                        }
227555efc227SAndreas Gohr
227655efc227SAndreas Gohr                        $j = 0;
227755efc227SAndreas Gohr                        while (($j < $count) && ($j < $origCount)) {
227855efc227SAndreas Gohr                            $this->_putShort($value, $j * 2, $origValue[$j], $isBigEndian);
227955efc227SAndreas Gohr                            $j++;
228055efc227SAndreas Gohr                        }
228155efc227SAndreas Gohr
228255efc227SAndreas Gohr                        while ($j < $count) {
228355efc227SAndreas Gohr                            $this->_putShort($value, $j * 2, 0, $isBigEndian);
228455efc227SAndreas Gohr                            $j++;
228555efc227SAndreas Gohr                        }
228655efc227SAndreas Gohr                        break;
228755efc227SAndreas Gohr                    case 9:    // SLONG
228855efc227SAndreas Gohr                        if ($count == 0) {
228955efc227SAndreas Gohr                            $count = $origCount;
229055efc227SAndreas Gohr                        }
229155efc227SAndreas Gohr
229255efc227SAndreas Gohr                        $j = 0;
229355efc227SAndreas Gohr                        while (($j < $count) && ($j < $origCount)) {
229455efc227SAndreas Gohr                            $this->_putLong($value, $j * 4, $origValue[$j], $isBigEndian);
229555efc227SAndreas Gohr                            $j++;
229655efc227SAndreas Gohr                        }
229755efc227SAndreas Gohr
229855efc227SAndreas Gohr                        while ($j < $count) {
229955efc227SAndreas Gohr                            $this->_putLong($value, $j * 4, 0, $isBigEndian);
230055efc227SAndreas Gohr                            $j++;
230155efc227SAndreas Gohr                        }
230255efc227SAndreas Gohr                        break;
230355efc227SAndreas Gohr                    case 10:   // SRATIONAL
230455efc227SAndreas Gohr                        if ($count == 0) {
230555efc227SAndreas Gohr                            $count = $origCount;
230655efc227SAndreas Gohr                        }
230755efc227SAndreas Gohr
230855efc227SAndreas Gohr                        $j = 0;
230955efc227SAndreas Gohr                        while (($j < $count) && ($j < $origCount)) {
231055efc227SAndreas Gohr                            $v = $origValue[$j];
231155efc227SAndreas Gohr                            if (is_array($v)) {
231255efc227SAndreas Gohr                                $a = $v['num'];
231355efc227SAndreas Gohr                                $b = $v['den'];
231455efc227SAndreas Gohr                            }
231555efc227SAndreas Gohr                            else {
231655efc227SAndreas Gohr                                $a = 0;
231755efc227SAndreas Gohr                                $b = 0;
231855efc227SAndreas Gohr                                // TODO: Allow other types and convert them
231955efc227SAndreas Gohr                            }
232055efc227SAndreas Gohr
232155efc227SAndreas Gohr                            $this->_putLong($value, $j * 8, $a, $isBigEndian);
232255efc227SAndreas Gohr                            $this->_putLong($value, ($j * 8) + 4, $b, $isBigEndian);
232355efc227SAndreas Gohr                            $j++;
232455efc227SAndreas Gohr                        }
232555efc227SAndreas Gohr
232655efc227SAndreas Gohr                        while ($j < $count) {
232755efc227SAndreas Gohr                            $this->_putLong($value, $j * 8, 0, $isBigEndian);
232855efc227SAndreas Gohr                            $this->_putLong($value, ($j * 8) + 4, 0, $isBigEndian);
232955efc227SAndreas Gohr                            $j++;
233055efc227SAndreas Gohr                        }
233155efc227SAndreas Gohr                        break;
233255efc227SAndreas Gohr                    case 11:   // FLOAT
233355efc227SAndreas Gohr                        if ($count == 0) {
233455efc227SAndreas Gohr                            $count = $origCount;
233555efc227SAndreas Gohr                        }
233655efc227SAndreas Gohr
233755efc227SAndreas Gohr                        $j = 0;
233855efc227SAndreas Gohr                        while (($j < $count) && ($j < $origCount)) {
233955efc227SAndreas Gohr                            $v = strval($origValue[$j]);
234055efc227SAndreas Gohr                            if (strlen($v) > 4) {
234155efc227SAndreas Gohr                                $v = substr($v, 0, 4);
234255efc227SAndreas Gohr                            }
234355efc227SAndreas Gohr                            elseif (strlen($v) < 4) {
234455efc227SAndreas Gohr                                $v = str_pad($v, 4, "\0");
234555efc227SAndreas Gohr                            }
234655efc227SAndreas Gohr                            $this->_putString($value, $j * 4, $v);
234755efc227SAndreas Gohr                            $j++;
234855efc227SAndreas Gohr                        }
234955efc227SAndreas Gohr
235055efc227SAndreas Gohr                        while ($j < $count) {
23515aaca723SGerrit Uitslag                            $v = "\0\0\0\0";
23525aaca723SGerrit Uitslag                            $this->_putString($value, $j * 4, $v);
235355efc227SAndreas Gohr                            $j++;
235455efc227SAndreas Gohr                        }
235555efc227SAndreas Gohr                        break;
235655efc227SAndreas Gohr                    case 12:   // DFLOAT
235755efc227SAndreas Gohr                        if ($count == 0) {
235855efc227SAndreas Gohr                            $count = $origCount;
235955efc227SAndreas Gohr                        }
236055efc227SAndreas Gohr
236155efc227SAndreas Gohr                        $j = 0;
236255efc227SAndreas Gohr                        while (($j < $count) && ($j < $origCount)) {
236355efc227SAndreas Gohr                            $v = strval($origValue[$j]);
236455efc227SAndreas Gohr                            if (strlen($v) > 8) {
236555efc227SAndreas Gohr                                $v = substr($v, 0, 8);
236655efc227SAndreas Gohr                            }
236755efc227SAndreas Gohr                            elseif (strlen($v) < 8) {
236855efc227SAndreas Gohr                                $v = str_pad($v, 8, "\0");
236955efc227SAndreas Gohr                            }
237055efc227SAndreas Gohr                            $this->_putString($value, $j * 8, $v);
237155efc227SAndreas Gohr                            $j++;
237255efc227SAndreas Gohr                        }
237355efc227SAndreas Gohr
237455efc227SAndreas Gohr                        while ($j < $count) {
23755aaca723SGerrit Uitslag                            $v = "\0\0\0\0\0\0\0\0";
23765aaca723SGerrit Uitslag                            $this->_putString($value, $j * 8, $v);
237755efc227SAndreas Gohr                            $j++;
237855efc227SAndreas Gohr                        }
237955efc227SAndreas Gohr                        break;
238055efc227SAndreas Gohr                    default:
238155efc227SAndreas Gohr                        $value = null;
238255efc227SAndreas Gohr                        break;
238355efc227SAndreas Gohr                }
238455efc227SAndreas Gohr            }
238555efc227SAndreas Gohr
238655efc227SAndreas Gohr            if ($value != null) {
238755efc227SAndreas Gohr                $ifdEntries[$entryCount] = array();
238855efc227SAndreas Gohr                $ifdEntries[$entryCount]['tag'] = $tag;
238955efc227SAndreas Gohr                $ifdEntries[$entryCount]['type'] = $type;
239055efc227SAndreas Gohr                $ifdEntries[$entryCount]['count'] = $count;
239155efc227SAndreas Gohr                $ifdEntries[$entryCount]['value'] = $value;
239255efc227SAndreas Gohr
239355efc227SAndreas Gohr                $entryCount++;
239455efc227SAndreas Gohr            }
239555efc227SAndreas Gohr        }
239655efc227SAndreas Gohr
239755efc227SAndreas Gohr        return $ifdEntries;
239855efc227SAndreas Gohr    }
2399c9c56f8aSasivery    /*************************************************************/
2400c9c56f8aSasivery    function _handleMarkerParsingException($e) {
2401d40e7d0eSAndreas Gohr        \dokuwiki\ErrorHandler::logException($e, $this->_fileName);
2402c9c56f8aSasivery    }
2403c9c56f8aSasivery
2404c9c56f8aSasivery    /*************************************************************/
2405c9c56f8aSasivery    function _isMarkerDisabled($name) {
2406c9c56f8aSasivery        if (!isset($this->_info)) return false;
2407c9c56f8aSasivery        return isset($this->_info[$name]) && $this->_info[$name] === false;
2408c9c56f8aSasivery    }
240955efc227SAndreas Gohr
241055efc227SAndreas Gohr    /*************************************************************/
24110b17fdc6SAndreas Gohr    function _parseMarkerAdobe() {
241255efc227SAndreas Gohr        if (!isset($this->_markers)) {
241355efc227SAndreas Gohr            $this->_readJPEG();
241455efc227SAndreas Gohr        }
241555efc227SAndreas Gohr
2416c9c56f8aSasivery        if ($this->_markers == null || $this->_isMarkerDisabled('adobe')) {
241755efc227SAndreas Gohr            return false;
241855efc227SAndreas Gohr        }
2419c9c56f8aSasivery        try {
242055efc227SAndreas Gohr            $data = null;
242155efc227SAndreas Gohr            $count = count($this->_markers);
242255efc227SAndreas Gohr            for ($i = 0; $i < $count; $i++) {
242355efc227SAndreas Gohr                if ($this->_markers[$i]['marker'] == 0xED) {
242455efc227SAndreas Gohr                    $signature = $this->_getFixedString($this->_markers[$i]['data'], 0, 14);
242555efc227SAndreas Gohr                    if ($signature == "Photoshop 3.0\0") {
242655efc227SAndreas Gohr                        $data =& $this->_markers[$i]['data'];
242755efc227SAndreas Gohr                        break;
242855efc227SAndreas Gohr                    }
242955efc227SAndreas Gohr                }
243055efc227SAndreas Gohr            }
243155efc227SAndreas Gohr
243255efc227SAndreas Gohr            if ($data == null) {
243355efc227SAndreas Gohr                $this->_info['adobe'] = false;
243455efc227SAndreas Gohr                $this->_info['iptc'] = false;
243555efc227SAndreas Gohr                return false;
243655efc227SAndreas Gohr            }
243755efc227SAndreas Gohr            $pos = 14;
243855efc227SAndreas Gohr            $this->_info['adobe'] = array();
243955efc227SAndreas Gohr            $this->_info['adobe']['raw'] = array();
244055efc227SAndreas Gohr            $this->_info['iptc'] = array();
244155efc227SAndreas Gohr
244255efc227SAndreas Gohr            $datasize = strlen($data);
244355efc227SAndreas Gohr
244455efc227SAndreas Gohr            while ($pos < $datasize) {
244555efc227SAndreas Gohr                $signature = $this->_getFixedString($data, $pos, 4);
244655efc227SAndreas Gohr                if ($signature != '8BIM')
244755efc227SAndreas Gohr                    return false;
244855efc227SAndreas Gohr                $pos += 4;
244955efc227SAndreas Gohr
245055efc227SAndreas Gohr                $type = $this->_getShort($data, $pos);
245155efc227SAndreas Gohr                $pos += 2;
245255efc227SAndreas Gohr
245355efc227SAndreas Gohr                $strlen = $this->_getByte($data, $pos);
245455efc227SAndreas Gohr                $pos += 1;
245555efc227SAndreas Gohr                $header = '';
245655efc227SAndreas Gohr                for ($i = 0; $i < $strlen; $i++) {
24572401f18dSSyntaxseed                    $header .= $data[$pos + $i];
245855efc227SAndreas Gohr                }
245955efc227SAndreas Gohr                $pos += $strlen + 1 - ($strlen % 2);  // The string is padded to even length, counting the length byte itself
246055efc227SAndreas Gohr
246155efc227SAndreas Gohr                $length = $this->_getLong($data, $pos);
246255efc227SAndreas Gohr                $pos += 4;
246355efc227SAndreas Gohr
246455efc227SAndreas Gohr                $basePos = $pos;
246555efc227SAndreas Gohr
246655efc227SAndreas Gohr                switch ($type) {
246755efc227SAndreas Gohr                    case 0x0404: // Caption (IPTC Data)
246855efc227SAndreas Gohr                        $pos = $this->_readIPTC($data, $pos);
246955efc227SAndreas Gohr                        if ($pos == false)
247055efc227SAndreas Gohr                            return false;
247155efc227SAndreas Gohr                        break;
247255efc227SAndreas Gohr                    case 0x040A: // CopyrightFlag
247355efc227SAndreas Gohr                        $this->_info['adobe']['CopyrightFlag'] = $this->_getByte($data, $pos);
247455efc227SAndreas Gohr                        $pos += $length;
247555efc227SAndreas Gohr                        break;
247655efc227SAndreas Gohr                    case 0x040B: // ImageURL
247755efc227SAndreas Gohr                        $this->_info['adobe']['ImageURL'] = $this->_getFixedString($data, $pos, $length);
247855efc227SAndreas Gohr                        $pos += $length;
247955efc227SAndreas Gohr                        break;
248055efc227SAndreas Gohr                    case 0x040C: // Thumbnail
248155efc227SAndreas Gohr                        $aux = $this->_getLong($data, $pos);
248255efc227SAndreas Gohr                        $pos += 4;
248355efc227SAndreas Gohr                        if ($aux == 1) {
248455efc227SAndreas Gohr                            $this->_info['adobe']['ThumbnailWidth'] = $this->_getLong($data, $pos);
248555efc227SAndreas Gohr                            $pos += 4;
248655efc227SAndreas Gohr                            $this->_info['adobe']['ThumbnailHeight'] = $this->_getLong($data, $pos);
248755efc227SAndreas Gohr                            $pos += 4;
248855efc227SAndreas Gohr
248955efc227SAndreas Gohr                            $pos += 16; // Skip some data
249055efc227SAndreas Gohr
249155efc227SAndreas Gohr                            $this->_info['adobe']['ThumbnailData'] = $this->_getFixedString($data, $pos, $length - 28);
249255efc227SAndreas Gohr                            $pos += $length - 28;
249355efc227SAndreas Gohr                        }
249455efc227SAndreas Gohr                        break;
249555efc227SAndreas Gohr                    default:
249655efc227SAndreas Gohr                        break;
249755efc227SAndreas Gohr                }
249855efc227SAndreas Gohr
249955efc227SAndreas Gohr                // We save all blocks, even those we recognized
250055efc227SAndreas Gohr                $label = sprintf('8BIM_0x%04x', $type);
250155efc227SAndreas Gohr                $this->_info['adobe']['raw'][$label] = array();
250255efc227SAndreas Gohr                $this->_info['adobe']['raw'][$label]['type'] = $type;
250355efc227SAndreas Gohr                $this->_info['adobe']['raw'][$label]['header'] = $header;
250455efc227SAndreas Gohr                $this->_info['adobe']['raw'][$label]['data'] =& $this->_getFixedString($data, $basePos, $length);
250555efc227SAndreas Gohr
250655efc227SAndreas Gohr                $pos = $basePos + $length + ($length % 2); // Even padding
250755efc227SAndreas Gohr            }
2508c9c56f8aSasivery        } catch(Exception $e) {
2509c9c56f8aSasivery            $this->_handleMarkerParsingException($e);
2510c9c56f8aSasivery            $this->_info['adobe'] = false;
2511c9c56f8aSasivery            $this->_info['iptc'] = false;
2512c9c56f8aSasivery            return false;
2513c9c56f8aSasivery        }
251455efc227SAndreas Gohr    }
251555efc227SAndreas Gohr
251655efc227SAndreas Gohr    /*************************************************************/
25170b17fdc6SAndreas Gohr    function _readIPTC(&$data, $pos = 0) {
251855efc227SAndreas Gohr        $totalLength = strlen($data);
251955efc227SAndreas Gohr
2520f5891fa4SGerrit Uitslag        $IPTCTags = $this->_iptcTagNames();
252155efc227SAndreas Gohr
252255efc227SAndreas Gohr        while ($pos < ($totalLength - 5)) {
252355efc227SAndreas Gohr            $signature = $this->_getShort($data, $pos);
252455efc227SAndreas Gohr            if ($signature != 0x1C02)
252555efc227SAndreas Gohr                return $pos;
252655efc227SAndreas Gohr            $pos += 2;
252755efc227SAndreas Gohr
252855efc227SAndreas Gohr            $type = $this->_getByte($data, $pos);
252955efc227SAndreas Gohr            $pos += 1;
253055efc227SAndreas Gohr            $length = $this->_getShort($data, $pos);
253155efc227SAndreas Gohr            $pos += 2;
253255efc227SAndreas Gohr
253355efc227SAndreas Gohr            $basePos = $pos;
253455efc227SAndreas Gohr            $label = '';
253555efc227SAndreas Gohr
253655efc227SAndreas Gohr            if (isset($IPTCTags[$type])) {
253755efc227SAndreas Gohr                $label = $IPTCTags[$type];
25380b17fdc6SAndreas Gohr            } else {
253955efc227SAndreas Gohr                $label = sprintf('IPTC_0x%02x', $type);
254055efc227SAndreas Gohr            }
254155efc227SAndreas Gohr
254255efc227SAndreas Gohr            if ($label != '') {
254355efc227SAndreas Gohr                if (isset($this->_info['iptc'][$label])) {
254455efc227SAndreas Gohr                    if (!is_array($this->_info['iptc'][$label])) {
254555efc227SAndreas Gohr                        $aux = array();
254655efc227SAndreas Gohr                        $aux[0] = $this->_info['iptc'][$label];
254755efc227SAndreas Gohr                        $this->_info['iptc'][$label] = $aux;
254855efc227SAndreas Gohr                    }
254955efc227SAndreas Gohr                    $this->_info['iptc'][$label][ count($this->_info['iptc'][$label]) ] = $this->_getFixedString($data, $pos, $length);
25500b17fdc6SAndreas Gohr                } else {
255155efc227SAndreas Gohr                    $this->_info['iptc'][$label] = $this->_getFixedString($data, $pos, $length);
255255efc227SAndreas Gohr                }
255355efc227SAndreas Gohr            }
255455efc227SAndreas Gohr
255555efc227SAndreas Gohr            $pos = $basePos + $length; // No padding
255655efc227SAndreas Gohr        }
255755efc227SAndreas Gohr        return $pos;
255855efc227SAndreas Gohr    }
255955efc227SAndreas Gohr
256055efc227SAndreas Gohr    /*************************************************************/
25610b17fdc6SAndreas Gohr    function & _createMarkerAdobe() {
256255efc227SAndreas Gohr        if (isset($this->_info['iptc'])) {
256355efc227SAndreas Gohr            if (!isset($this->_info['adobe'])) {
256455efc227SAndreas Gohr                $this->_info['adobe'] = array();
256555efc227SAndreas Gohr            }
256655efc227SAndreas Gohr            if (!isset($this->_info['adobe']['raw'])) {
256755efc227SAndreas Gohr                $this->_info['adobe']['raw'] = array();
256855efc227SAndreas Gohr            }
256955efc227SAndreas Gohr            if (!isset($this->_info['adobe']['raw']['8BIM_0x0404'])) {
257055efc227SAndreas Gohr                $this->_info['adobe']['raw']['8BIM_0x0404'] = array();
257155efc227SAndreas Gohr            }
257255efc227SAndreas Gohr            $this->_info['adobe']['raw']['8BIM_0x0404']['type'] = 0x0404;
257355efc227SAndreas Gohr            $this->_info['adobe']['raw']['8BIM_0x0404']['header'] = "Caption";
257455efc227SAndreas Gohr            $this->_info['adobe']['raw']['8BIM_0x0404']['data'] =& $this->_writeIPTC();
257555efc227SAndreas Gohr        }
257655efc227SAndreas Gohr
257755efc227SAndreas Gohr        if (isset($this->_info['adobe']['raw']) && (count($this->_info['adobe']['raw']) > 0)) {
257855efc227SAndreas Gohr            $data = "Photoshop 3.0\0";
257955efc227SAndreas Gohr            $pos = 14;
258055efc227SAndreas Gohr
258155efc227SAndreas Gohr            reset($this->_info['adobe']['raw']);
2582dd5c3e2bSPhy            foreach ($this->_info['adobe']['raw'] as $value){
258355efc227SAndreas Gohr                $pos = $this->_write8BIM(
258455efc227SAndreas Gohr                        $data,
258555efc227SAndreas Gohr                        $pos,
2586dd5c3e2bSPhy                        $value['type'],
2587dd5c3e2bSPhy                        $value['header'],
2588dd5c3e2bSPhy                        $value['data'] );
258955efc227SAndreas Gohr            }
259055efc227SAndreas Gohr        }
259155efc227SAndreas Gohr
259255efc227SAndreas Gohr        return $data;
259355efc227SAndreas Gohr    }
259455efc227SAndreas Gohr
259555efc227SAndreas Gohr    /*************************************************************/
2596276820f7SScrutinizer Auto-Fixer
2597276820f7SScrutinizer Auto-Fixer    /**
2598f50a239bSTakamura     * @param mixed $data
2599276820f7SScrutinizer Auto-Fixer     * @param integer $pos
2600f50a239bSTakamura     *
2601f50a239bSTakamura     * @param string $type
2602f50a239bSTakamura     * @param string $header
2603f50a239bSTakamura     * @param mixed $value
2604f50a239bSTakamura     *
2605f50a239bSTakamura     * @return int|mixed
2606276820f7SScrutinizer Auto-Fixer     */
26070b17fdc6SAndreas Gohr    function _write8BIM(&$data, $pos, $type, $header, &$value) {
260855efc227SAndreas Gohr        $signature = "8BIM";
260955efc227SAndreas Gohr
261055efc227SAndreas Gohr        $pos = $this->_putString($data, $pos, $signature);
261155efc227SAndreas Gohr        $pos = $this->_putShort($data, $pos, $type);
261255efc227SAndreas Gohr
261355efc227SAndreas Gohr        $len = strlen($header);
261455efc227SAndreas Gohr
261555efc227SAndreas Gohr        $pos = $this->_putByte($data, $pos, $len);
261655efc227SAndreas Gohr        $pos = $this->_putString($data, $pos, $header);
261755efc227SAndreas Gohr        if (($len % 2) == 0) {  // Even padding, including the length byte
261855efc227SAndreas Gohr            $pos = $this->_putByte($data, $pos, 0);
261955efc227SAndreas Gohr        }
262055efc227SAndreas Gohr
262155efc227SAndreas Gohr        $len = strlen($value);
262255efc227SAndreas Gohr        $pos = $this->_putLong($data, $pos, $len);
262355efc227SAndreas Gohr        $pos = $this->_putString($data, $pos, $value);
262455efc227SAndreas Gohr        if (($len % 2) != 0) {  // Even padding
262555efc227SAndreas Gohr            $pos = $this->_putByte($data, $pos, 0);
262655efc227SAndreas Gohr        }
262755efc227SAndreas Gohr        return $pos;
262855efc227SAndreas Gohr    }
262955efc227SAndreas Gohr
263055efc227SAndreas Gohr    /*************************************************************/
26310b17fdc6SAndreas Gohr    function & _writeIPTC() {
263255efc227SAndreas Gohr        $data = " ";
263355efc227SAndreas Gohr        $pos = 0;
263455efc227SAndreas Gohr
263555efc227SAndreas Gohr        $IPTCNames =& $this->_iptcNameTags();
263655efc227SAndreas Gohr
26379e491c01SAndreas Gohr        foreach($this->_info['iptc'] as $label => $value) {
263855efc227SAndreas Gohr            $value =& $this->_info['iptc'][$label];
263955efc227SAndreas Gohr            $type = -1;
264055efc227SAndreas Gohr
264155efc227SAndreas Gohr            if (isset($IPTCNames[$label])) {
264255efc227SAndreas Gohr                $type = $IPTCNames[$label];
264355efc227SAndreas Gohr            }
2644*6c16a3a9Sfiwswe            elseif (str_starts_with($label, 'IPTC_0x')) {
264555efc227SAndreas Gohr                $type = hexdec(substr($label, 7, 2));
264655efc227SAndreas Gohr            }
264755efc227SAndreas Gohr
264855efc227SAndreas Gohr            if ($type != -1) {
264955efc227SAndreas Gohr                if (is_array($value)) {
26500b17fdc6SAndreas Gohr                    $vcnt = count($value);
26510b17fdc6SAndreas Gohr                    for ($i = 0; $i < $vcnt; $i++) {
265255efc227SAndreas Gohr                        $pos = $this->_writeIPTCEntry($data, $pos, $type, $value[$i]);
265355efc227SAndreas Gohr                    }
265455efc227SAndreas Gohr                }
265555efc227SAndreas Gohr                else {
265655efc227SAndreas Gohr                    $pos = $this->_writeIPTCEntry($data, $pos, $type, $value);
265755efc227SAndreas Gohr                }
265855efc227SAndreas Gohr            }
265955efc227SAndreas Gohr        }
266055efc227SAndreas Gohr
266155efc227SAndreas Gohr        return $data;
266255efc227SAndreas Gohr    }
266355efc227SAndreas Gohr
266455efc227SAndreas Gohr    /*************************************************************/
2665276820f7SScrutinizer Auto-Fixer
2666276820f7SScrutinizer Auto-Fixer    /**
2667f50a239bSTakamura     * @param mixed $data
2668276820f7SScrutinizer Auto-Fixer     * @param integer $pos
2669f50a239bSTakamura     *
2670f50a239bSTakamura     * @param string $type
2671f50a239bSTakamura     * @param mixed $value
2672f50a239bSTakamura     *
2673f50a239bSTakamura     * @return int|mixed
2674276820f7SScrutinizer Auto-Fixer     */
26750b17fdc6SAndreas Gohr    function _writeIPTCEntry(&$data, $pos, $type, &$value) {
267655efc227SAndreas Gohr        $pos = $this->_putShort($data, $pos, 0x1C02);
267755efc227SAndreas Gohr        $pos = $this->_putByte($data, $pos, $type);
267855efc227SAndreas Gohr        $pos = $this->_putShort($data, $pos, strlen($value));
267955efc227SAndreas Gohr        $pos = $this->_putString($data, $pos, $value);
268055efc227SAndreas Gohr
268155efc227SAndreas Gohr        return $pos;
268255efc227SAndreas Gohr    }
268355efc227SAndreas Gohr
268455efc227SAndreas Gohr    /*************************************************************/
26850b17fdc6SAndreas Gohr    function _exifTagNames($mode) {
268655efc227SAndreas Gohr        $tags = array();
268755efc227SAndreas Gohr
268855efc227SAndreas Gohr        if ($mode == 'ifd0') {
268955efc227SAndreas Gohr            $tags[0x010E] = 'ImageDescription';
269055efc227SAndreas Gohr            $tags[0x010F] = 'Make';
269155efc227SAndreas Gohr            $tags[0x0110] = 'Model';
269255efc227SAndreas Gohr            $tags[0x0112] = 'Orientation';
269355efc227SAndreas Gohr            $tags[0x011A] = 'XResolution';
269455efc227SAndreas Gohr            $tags[0x011B] = 'YResolution';
269555efc227SAndreas Gohr            $tags[0x0128] = 'ResolutionUnit';
269655efc227SAndreas Gohr            $tags[0x0131] = 'Software';
269755efc227SAndreas Gohr            $tags[0x0132] = 'DateTime';
269855efc227SAndreas Gohr            $tags[0x013B] = 'Artist';
269955efc227SAndreas Gohr            $tags[0x013E] = 'WhitePoint';
270055efc227SAndreas Gohr            $tags[0x013F] = 'PrimaryChromaticities';
270155efc227SAndreas Gohr            $tags[0x0211] = 'YCbCrCoefficients';
270255efc227SAndreas Gohr            $tags[0x0212] = 'YCbCrSubSampling';
270355efc227SAndreas Gohr            $tags[0x0213] = 'YCbCrPositioning';
270455efc227SAndreas Gohr            $tags[0x0214] = 'ReferenceBlackWhite';
270555efc227SAndreas Gohr            $tags[0x8298] = 'Copyright';
270655efc227SAndreas Gohr            $tags[0x8769] = 'ExifIFDOffset';
270755efc227SAndreas Gohr            $tags[0x8825] = 'GPSIFDOffset';
270855efc227SAndreas Gohr        }
270955efc227SAndreas Gohr        if ($mode == 'ifd1') {
271055efc227SAndreas Gohr            $tags[0x00FE] = 'TIFFNewSubfileType';
271155efc227SAndreas Gohr            $tags[0x00FF] = 'TIFFSubfileType';
271255efc227SAndreas Gohr            $tags[0x0100] = 'TIFFImageWidth';
271355efc227SAndreas Gohr            $tags[0x0101] = 'TIFFImageHeight';
271455efc227SAndreas Gohr            $tags[0x0102] = 'TIFFBitsPerSample';
271555efc227SAndreas Gohr            $tags[0x0103] = 'TIFFCompression';
271655efc227SAndreas Gohr            $tags[0x0106] = 'TIFFPhotometricInterpretation';
271755efc227SAndreas Gohr            $tags[0x0107] = 'TIFFThreshholding';
271855efc227SAndreas Gohr            $tags[0x0108] = 'TIFFCellWidth';
271955efc227SAndreas Gohr            $tags[0x0109] = 'TIFFCellLength';
272055efc227SAndreas Gohr            $tags[0x010A] = 'TIFFFillOrder';
272155efc227SAndreas Gohr            $tags[0x010E] = 'TIFFImageDescription';
272255efc227SAndreas Gohr            $tags[0x010F] = 'TIFFMake';
272355efc227SAndreas Gohr            $tags[0x0110] = 'TIFFModel';
272455efc227SAndreas Gohr            $tags[0x0111] = 'TIFFStripOffsets';
272555efc227SAndreas Gohr            $tags[0x0112] = 'TIFFOrientation';
272655efc227SAndreas Gohr            $tags[0x0115] = 'TIFFSamplesPerPixel';
272755efc227SAndreas Gohr            $tags[0x0116] = 'TIFFRowsPerStrip';
272855efc227SAndreas Gohr            $tags[0x0117] = 'TIFFStripByteCounts';
272955efc227SAndreas Gohr            $tags[0x0118] = 'TIFFMinSampleValue';
273055efc227SAndreas Gohr            $tags[0x0119] = 'TIFFMaxSampleValue';
273155efc227SAndreas Gohr            $tags[0x011A] = 'TIFFXResolution';
273255efc227SAndreas Gohr            $tags[0x011B] = 'TIFFYResolution';
273355efc227SAndreas Gohr            $tags[0x011C] = 'TIFFPlanarConfiguration';
273455efc227SAndreas Gohr            $tags[0x0122] = 'TIFFGrayResponseUnit';
273555efc227SAndreas Gohr            $tags[0x0123] = 'TIFFGrayResponseCurve';
273655efc227SAndreas Gohr            $tags[0x0128] = 'TIFFResolutionUnit';
273755efc227SAndreas Gohr            $tags[0x0131] = 'TIFFSoftware';
273855efc227SAndreas Gohr            $tags[0x0132] = 'TIFFDateTime';
273955efc227SAndreas Gohr            $tags[0x013B] = 'TIFFArtist';
274055efc227SAndreas Gohr            $tags[0x013C] = 'TIFFHostComputer';
274155efc227SAndreas Gohr            $tags[0x0140] = 'TIFFColorMap';
274255efc227SAndreas Gohr            $tags[0x0152] = 'TIFFExtraSamples';
274355efc227SAndreas Gohr            $tags[0x0201] = 'TIFFJFIFOffset';
274455efc227SAndreas Gohr            $tags[0x0202] = 'TIFFJFIFLength';
274555efc227SAndreas Gohr            $tags[0x0211] = 'TIFFYCbCrCoefficients';
274655efc227SAndreas Gohr            $tags[0x0212] = 'TIFFYCbCrSubSampling';
274755efc227SAndreas Gohr            $tags[0x0213] = 'TIFFYCbCrPositioning';
274855efc227SAndreas Gohr            $tags[0x0214] = 'TIFFReferenceBlackWhite';
274955efc227SAndreas Gohr            $tags[0x8298] = 'TIFFCopyright';
275055efc227SAndreas Gohr            $tags[0x9286] = 'TIFFUserComment';
27510b17fdc6SAndreas Gohr        } elseif ($mode == 'exif') {
275255efc227SAndreas Gohr            $tags[0x829A] = 'ExposureTime';
275355efc227SAndreas Gohr            $tags[0x829D] = 'FNumber';
275455efc227SAndreas Gohr            $tags[0x8822] = 'ExposureProgram';
275555efc227SAndreas Gohr            $tags[0x8824] = 'SpectralSensitivity';
275655efc227SAndreas Gohr            $tags[0x8827] = 'ISOSpeedRatings';
275755efc227SAndreas Gohr            $tags[0x8828] = 'OECF';
275855efc227SAndreas Gohr            $tags[0x9000] = 'EXIFVersion';
27595d00293dSMichael Aschauer            $tags[0x9003] = 'DateTimeOriginal';
27605d00293dSMichael Aschauer            $tags[0x9004] = 'DateTimeDigitized';
276155efc227SAndreas Gohr            $tags[0x9101] = 'ComponentsConfiguration';
276255efc227SAndreas Gohr            $tags[0x9102] = 'CompressedBitsPerPixel';
276355efc227SAndreas Gohr            $tags[0x9201] = 'ShutterSpeedValue';
276455efc227SAndreas Gohr            $tags[0x9202] = 'ApertureValue';
276555efc227SAndreas Gohr            $tags[0x9203] = 'BrightnessValue';
276655efc227SAndreas Gohr            $tags[0x9204] = 'ExposureBiasValue';
276755efc227SAndreas Gohr            $tags[0x9205] = 'MaxApertureValue';
276855efc227SAndreas Gohr            $tags[0x9206] = 'SubjectDistance';
276955efc227SAndreas Gohr            $tags[0x9207] = 'MeteringMode';
277055efc227SAndreas Gohr            $tags[0x9208] = 'LightSource';
277155efc227SAndreas Gohr            $tags[0x9209] = 'Flash';
277255efc227SAndreas Gohr            $tags[0x920A] = 'FocalLength';
277355efc227SAndreas Gohr            $tags[0x927C] = 'MakerNote';
277455efc227SAndreas Gohr            $tags[0x9286] = 'UserComment';
277555efc227SAndreas Gohr            $tags[0x9290] = 'SubSecTime';
277655efc227SAndreas Gohr            $tags[0x9291] = 'SubSecTimeOriginal';
277755efc227SAndreas Gohr            $tags[0x9292] = 'SubSecTimeDigitized';
277855efc227SAndreas Gohr            $tags[0xA000] = 'FlashPixVersion';
277955efc227SAndreas Gohr            $tags[0xA001] = 'ColorSpace';
278055efc227SAndreas Gohr            $tags[0xA002] = 'PixelXDimension';
278155efc227SAndreas Gohr            $tags[0xA003] = 'PixelYDimension';
278255efc227SAndreas Gohr            $tags[0xA004] = 'RelatedSoundFile';
278355efc227SAndreas Gohr            $tags[0xA005] = 'InteropIFDOffset';
278455efc227SAndreas Gohr            $tags[0xA20B] = 'FlashEnergy';
278555efc227SAndreas Gohr            $tags[0xA20C] = 'SpatialFrequencyResponse';
278655efc227SAndreas Gohr            $tags[0xA20E] = 'FocalPlaneXResolution';
278755efc227SAndreas Gohr            $tags[0xA20F] = 'FocalPlaneYResolution';
278855efc227SAndreas Gohr            $tags[0xA210] = 'FocalPlaneResolutionUnit';
278955efc227SAndreas Gohr            $tags[0xA214] = 'SubjectLocation';
279055efc227SAndreas Gohr            $tags[0xA215] = 'ExposureIndex';
279155efc227SAndreas Gohr            $tags[0xA217] = 'SensingMethod';
279255efc227SAndreas Gohr            $tags[0xA300] = 'FileSource';
279355efc227SAndreas Gohr            $tags[0xA301] = 'SceneType';
279455efc227SAndreas Gohr            $tags[0xA302] = 'CFAPattern';
27950b17fdc6SAndreas Gohr        } elseif ($mode == 'interop') {
279655efc227SAndreas Gohr            $tags[0x0001] = 'InteroperabilityIndex';
279755efc227SAndreas Gohr            $tags[0x0002] = 'InteroperabilityVersion';
279855efc227SAndreas Gohr            $tags[0x1000] = 'RelatedImageFileFormat';
279955efc227SAndreas Gohr            $tags[0x1001] = 'RelatedImageWidth';
280055efc227SAndreas Gohr            $tags[0x1002] = 'RelatedImageLength';
28010b17fdc6SAndreas Gohr        } elseif ($mode == 'gps') {
280255efc227SAndreas Gohr            $tags[0x0000] = 'GPSVersionID';
280355efc227SAndreas Gohr            $tags[0x0001] = 'GPSLatitudeRef';
280455efc227SAndreas Gohr            $tags[0x0002] = 'GPSLatitude';
280555efc227SAndreas Gohr            $tags[0x0003] = 'GPSLongitudeRef';
280655efc227SAndreas Gohr            $tags[0x0004] = 'GPSLongitude';
280755efc227SAndreas Gohr            $tags[0x0005] = 'GPSAltitudeRef';
280855efc227SAndreas Gohr            $tags[0x0006] = 'GPSAltitude';
280955efc227SAndreas Gohr            $tags[0x0007] = 'GPSTimeStamp';
281055efc227SAndreas Gohr            $tags[0x0008] = 'GPSSatellites';
281155efc227SAndreas Gohr            $tags[0x0009] = 'GPSStatus';
281255efc227SAndreas Gohr            $tags[0x000A] = 'GPSMeasureMode';
281355efc227SAndreas Gohr            $tags[0x000B] = 'GPSDOP';
281455efc227SAndreas Gohr            $tags[0x000C] = 'GPSSpeedRef';
281555efc227SAndreas Gohr            $tags[0x000D] = 'GPSSpeed';
281655efc227SAndreas Gohr            $tags[0x000E] = 'GPSTrackRef';
281755efc227SAndreas Gohr            $tags[0x000F] = 'GPSTrack';
281855efc227SAndreas Gohr            $tags[0x0010] = 'GPSImgDirectionRef';
281955efc227SAndreas Gohr            $tags[0x0011] = 'GPSImgDirection';
282055efc227SAndreas Gohr            $tags[0x0012] = 'GPSMapDatum';
282155efc227SAndreas Gohr            $tags[0x0013] = 'GPSDestLatitudeRef';
282255efc227SAndreas Gohr            $tags[0x0014] = 'GPSDestLatitude';
282355efc227SAndreas Gohr            $tags[0x0015] = 'GPSDestLongitudeRef';
282455efc227SAndreas Gohr            $tags[0x0016] = 'GPSDestLongitude';
282555efc227SAndreas Gohr            $tags[0x0017] = 'GPSDestBearingRef';
282655efc227SAndreas Gohr            $tags[0x0018] = 'GPSDestBearing';
282755efc227SAndreas Gohr            $tags[0x0019] = 'GPSDestDistanceRef';
282855efc227SAndreas Gohr            $tags[0x001A] = 'GPSDestDistance';
282955efc227SAndreas Gohr        }
283055efc227SAndreas Gohr
283155efc227SAndreas Gohr        return $tags;
283255efc227SAndreas Gohr    }
283355efc227SAndreas Gohr
283455efc227SAndreas Gohr    /*************************************************************/
28350b17fdc6SAndreas Gohr    function _exifTagTypes($mode) {
283655efc227SAndreas Gohr        $tags = array();
283755efc227SAndreas Gohr
283855efc227SAndreas Gohr        if ($mode == 'ifd0') {
283955efc227SAndreas Gohr            $tags[0x010E] = array(2, 0); // ImageDescription -> ASCII, Any
284055efc227SAndreas Gohr            $tags[0x010F] = array(2, 0); // Make -> ASCII, Any
284155efc227SAndreas Gohr            $tags[0x0110] = array(2, 0); // Model -> ASCII, Any
284255efc227SAndreas Gohr            $tags[0x0112] = array(3, 1); // Orientation -> SHORT, 1
284355efc227SAndreas Gohr            $tags[0x011A] = array(5, 1); // XResolution -> RATIONAL, 1
284455efc227SAndreas Gohr            $tags[0x011B] = array(5, 1); // YResolution -> RATIONAL, 1
284555efc227SAndreas Gohr            $tags[0x0128] = array(3, 1); // ResolutionUnit -> SHORT
284655efc227SAndreas Gohr            $tags[0x0131] = array(2, 0); // Software -> ASCII, Any
284755efc227SAndreas Gohr            $tags[0x0132] = array(2, 20); // DateTime -> ASCII, 20
284855efc227SAndreas Gohr            $tags[0x013B] = array(2, 0); // Artist -> ASCII, Any
284955efc227SAndreas Gohr            $tags[0x013E] = array(5, 2); // WhitePoint -> RATIONAL, 2
285055efc227SAndreas Gohr            $tags[0x013F] = array(5, 6); // PrimaryChromaticities -> RATIONAL, 6
285155efc227SAndreas Gohr            $tags[0x0211] = array(5, 3); // YCbCrCoefficients -> RATIONAL, 3
285255efc227SAndreas Gohr            $tags[0x0212] = array(3, 2); // YCbCrSubSampling -> SHORT, 2
285355efc227SAndreas Gohr            $tags[0x0213] = array(3, 1); // YCbCrPositioning -> SHORT, 1
285455efc227SAndreas Gohr            $tags[0x0214] = array(5, 6); // ReferenceBlackWhite -> RATIONAL, 6
285555efc227SAndreas Gohr            $tags[0x8298] = array(2, 0); // Copyright -> ASCII, Any
285655efc227SAndreas Gohr            $tags[0x8769] = array(4, 1); // ExifIFDOffset -> LONG, 1
285755efc227SAndreas Gohr            $tags[0x8825] = array(4, 1); // GPSIFDOffset -> LONG, 1
285855efc227SAndreas Gohr        }
285955efc227SAndreas Gohr        if ($mode == 'ifd1') {
286055efc227SAndreas Gohr            $tags[0x00FE] = array(4, 1); // TIFFNewSubfileType -> LONG, 1
286155efc227SAndreas Gohr            $tags[0x00FF] = array(3, 1); // TIFFSubfileType -> SHORT, 1
286255efc227SAndreas Gohr            $tags[0x0100] = array(4, 1); // TIFFImageWidth -> LONG (or SHORT), 1
286355efc227SAndreas Gohr            $tags[0x0101] = array(4, 1); // TIFFImageHeight -> LONG (or SHORT), 1
286455efc227SAndreas Gohr            $tags[0x0102] = array(3, 3); // TIFFBitsPerSample -> SHORT, 3
286555efc227SAndreas Gohr            $tags[0x0103] = array(3, 1); // TIFFCompression -> SHORT, 1
286655efc227SAndreas Gohr            $tags[0x0106] = array(3, 1); // TIFFPhotometricInterpretation -> SHORT, 1
286755efc227SAndreas Gohr            $tags[0x0107] = array(3, 1); // TIFFThreshholding -> SHORT, 1
286855efc227SAndreas Gohr            $tags[0x0108] = array(3, 1); // TIFFCellWidth -> SHORT, 1
286955efc227SAndreas Gohr            $tags[0x0109] = array(3, 1); // TIFFCellLength -> SHORT, 1
287055efc227SAndreas Gohr            $tags[0x010A] = array(3, 1); // TIFFFillOrder -> SHORT, 1
287155efc227SAndreas Gohr            $tags[0x010E] = array(2, 0); // TIFFImageDescription -> ASCII, Any
287255efc227SAndreas Gohr            $tags[0x010F] = array(2, 0); // TIFFMake -> ASCII, Any
287355efc227SAndreas Gohr            $tags[0x0110] = array(2, 0); // TIFFModel -> ASCII, Any
287455efc227SAndreas Gohr            $tags[0x0111] = array(4, 0); // TIFFStripOffsets -> LONG (or SHORT), Any (one per strip)
287555efc227SAndreas Gohr            $tags[0x0112] = array(3, 1); // TIFFOrientation -> SHORT, 1
287655efc227SAndreas Gohr            $tags[0x0115] = array(3, 1); // TIFFSamplesPerPixel -> SHORT, 1
287755efc227SAndreas Gohr            $tags[0x0116] = array(4, 1); // TIFFRowsPerStrip -> LONG (or SHORT), 1
287855efc227SAndreas Gohr            $tags[0x0117] = array(4, 0); // TIFFStripByteCounts -> LONG (or SHORT), Any (one per strip)
287955efc227SAndreas Gohr            $tags[0x0118] = array(3, 0); // TIFFMinSampleValue -> SHORT, Any (SamplesPerPixel)
288055efc227SAndreas Gohr            $tags[0x0119] = array(3, 0); // TIFFMaxSampleValue -> SHORT, Any (SamplesPerPixel)
288155efc227SAndreas Gohr            $tags[0x011A] = array(5, 1); // TIFFXResolution -> RATIONAL, 1
288255efc227SAndreas Gohr            $tags[0x011B] = array(5, 1); // TIFFYResolution -> RATIONAL, 1
288355efc227SAndreas Gohr            $tags[0x011C] = array(3, 1); // TIFFPlanarConfiguration -> SHORT, 1
288455efc227SAndreas Gohr            $tags[0x0122] = array(3, 1); // TIFFGrayResponseUnit -> SHORT, 1
288555efc227SAndreas Gohr            $tags[0x0123] = array(3, 0); // TIFFGrayResponseCurve -> SHORT, Any (2^BitsPerSample)
288655efc227SAndreas Gohr            $tags[0x0128] = array(3, 1); // TIFFResolutionUnit -> SHORT, 1
288755efc227SAndreas Gohr            $tags[0x0131] = array(2, 0); // TIFFSoftware -> ASCII, Any
288855efc227SAndreas Gohr            $tags[0x0132] = array(2, 20); // TIFFDateTime -> ASCII, 20
288955efc227SAndreas Gohr            $tags[0x013B] = array(2, 0); // TIFFArtist -> ASCII, Any
289055efc227SAndreas Gohr            $tags[0x013C] = array(2, 0); // TIFFHostComputer -> ASCII, Any
289155efc227SAndreas Gohr            $tags[0x0140] = array(3, 0); // TIFFColorMap -> SHORT, Any (3 * 2^BitsPerSample)
289255efc227SAndreas Gohr            $tags[0x0152] = array(3, 0); // TIFFExtraSamples -> SHORT, Any (SamplesPerPixel - 3)
289355efc227SAndreas Gohr            $tags[0x0201] = array(4, 1); // TIFFJFIFOffset -> LONG, 1
289455efc227SAndreas Gohr            $tags[0x0202] = array(4, 1); // TIFFJFIFLength -> LONG, 1
289555efc227SAndreas Gohr            $tags[0x0211] = array(5, 3); // TIFFYCbCrCoefficients -> RATIONAL, 3
289655efc227SAndreas Gohr            $tags[0x0212] = array(3, 2); // TIFFYCbCrSubSampling -> SHORT, 2
289755efc227SAndreas Gohr            $tags[0x0213] = array(3, 1); // TIFFYCbCrPositioning -> SHORT, 1
289855efc227SAndreas Gohr            $tags[0x0214] = array(5, 6); // TIFFReferenceBlackWhite -> RATIONAL, 6
289955efc227SAndreas Gohr            $tags[0x8298] = array(2, 0); // TIFFCopyright -> ASCII, Any
290055efc227SAndreas Gohr            $tags[0x9286] = array(2, 0); // TIFFUserComment -> ASCII, Any
29010b17fdc6SAndreas Gohr        } elseif ($mode == 'exif') {
290255efc227SAndreas Gohr            $tags[0x829A] = array(5, 1); // ExposureTime -> RATIONAL, 1
290355efc227SAndreas Gohr            $tags[0x829D] = array(5, 1); // FNumber -> RATIONAL, 1
290455efc227SAndreas Gohr            $tags[0x8822] = array(3, 1); // ExposureProgram -> SHORT, 1
290555efc227SAndreas Gohr            $tags[0x8824] = array(2, 0); // SpectralSensitivity -> ASCII, Any
290655efc227SAndreas Gohr            $tags[0x8827] = array(3, 0); // ISOSpeedRatings -> SHORT, Any
290755efc227SAndreas Gohr            $tags[0x8828] = array(7, 0); // OECF -> UNDEFINED, Any
290855efc227SAndreas Gohr            $tags[0x9000] = array(7, 4); // EXIFVersion -> UNDEFINED, 4
29095d00293dSMichael Aschauer            $tags[0x9003] = array(2, 20); // DateTimeOriginal -> ASCII, 20
29105d00293dSMichael Aschauer            $tags[0x9004] = array(2, 20); // DateTimeDigitized -> ASCII, 20
291155efc227SAndreas Gohr            $tags[0x9101] = array(7, 4); // ComponentsConfiguration -> UNDEFINED, 4
291255efc227SAndreas Gohr            $tags[0x9102] = array(5, 1); // CompressedBitsPerPixel -> RATIONAL, 1
291355efc227SAndreas Gohr            $tags[0x9201] = array(10, 1); // ShutterSpeedValue -> SRATIONAL, 1
291455efc227SAndreas Gohr            $tags[0x9202] = array(5, 1); // ApertureValue -> RATIONAL, 1
291555efc227SAndreas Gohr            $tags[0x9203] = array(10, 1); // BrightnessValue -> SRATIONAL, 1
291655efc227SAndreas Gohr            $tags[0x9204] = array(10, 1); // ExposureBiasValue -> SRATIONAL, 1
291755efc227SAndreas Gohr            $tags[0x9205] = array(5, 1); // MaxApertureValue -> RATIONAL, 1
291855efc227SAndreas Gohr            $tags[0x9206] = array(5, 1); // SubjectDistance -> RATIONAL, 1
291955efc227SAndreas Gohr            $tags[0x9207] = array(3, 1); // MeteringMode -> SHORT, 1
292055efc227SAndreas Gohr            $tags[0x9208] = array(3, 1); // LightSource -> SHORT, 1
292155efc227SAndreas Gohr            $tags[0x9209] = array(3, 1); // Flash -> SHORT, 1
292255efc227SAndreas Gohr            $tags[0x920A] = array(5, 1); // FocalLength -> RATIONAL, 1
292355efc227SAndreas Gohr            $tags[0x927C] = array(7, 0); // MakerNote -> UNDEFINED, Any
292455efc227SAndreas Gohr            $tags[0x9286] = array(7, 0); // UserComment -> UNDEFINED, Any
292555efc227SAndreas Gohr            $tags[0x9290] = array(2, 0); // SubSecTime -> ASCII, Any
292655efc227SAndreas Gohr            $tags[0x9291] = array(2, 0); // SubSecTimeOriginal -> ASCII, Any
292755efc227SAndreas Gohr            $tags[0x9292] = array(2, 0); // SubSecTimeDigitized -> ASCII, Any
292855efc227SAndreas Gohr            $tags[0xA000] = array(7, 4); // FlashPixVersion -> UNDEFINED, 4
292955efc227SAndreas Gohr            $tags[0xA001] = array(3, 1); // ColorSpace -> SHORT, 1
293055efc227SAndreas Gohr            $tags[0xA002] = array(4, 1); // PixelXDimension -> LONG (or SHORT), 1
293155efc227SAndreas Gohr            $tags[0xA003] = array(4, 1); // PixelYDimension -> LONG (or SHORT), 1
293255efc227SAndreas Gohr            $tags[0xA004] = array(2, 13); // RelatedSoundFile -> ASCII, 13
293355efc227SAndreas Gohr            $tags[0xA005] = array(4, 1); // InteropIFDOffset -> LONG, 1
293455efc227SAndreas Gohr            $tags[0xA20B] = array(5, 1); // FlashEnergy -> RATIONAL, 1
293555efc227SAndreas Gohr            $tags[0xA20C] = array(7, 0); // SpatialFrequencyResponse -> UNDEFINED, Any
293655efc227SAndreas Gohr            $tags[0xA20E] = array(5, 1); // FocalPlaneXResolution -> RATIONAL, 1
293755efc227SAndreas Gohr            $tags[0xA20F] = array(5, 1); // FocalPlaneYResolution -> RATIONAL, 1
293855efc227SAndreas Gohr            $tags[0xA210] = array(3, 1); // FocalPlaneResolutionUnit -> SHORT, 1
293955efc227SAndreas Gohr            $tags[0xA214] = array(3, 2); // SubjectLocation -> SHORT, 2
294055efc227SAndreas Gohr            $tags[0xA215] = array(5, 1); // ExposureIndex -> RATIONAL, 1
294155efc227SAndreas Gohr            $tags[0xA217] = array(3, 1); // SensingMethod -> SHORT, 1
294255efc227SAndreas Gohr            $tags[0xA300] = array(7, 1); // FileSource -> UNDEFINED, 1
294355efc227SAndreas Gohr            $tags[0xA301] = array(7, 1); // SceneType -> UNDEFINED, 1
294455efc227SAndreas Gohr            $tags[0xA302] = array(7, 0); // CFAPattern -> UNDEFINED, Any
29450b17fdc6SAndreas Gohr        } elseif ($mode == 'interop') {
294655efc227SAndreas Gohr            $tags[0x0001] = array(2, 0); // InteroperabilityIndex -> ASCII, Any
294755efc227SAndreas Gohr            $tags[0x0002] = array(7, 4); // InteroperabilityVersion -> UNKNOWN, 4
294855efc227SAndreas Gohr            $tags[0x1000] = array(2, 0); // RelatedImageFileFormat -> ASCII, Any
294955efc227SAndreas Gohr            $tags[0x1001] = array(4, 1); // RelatedImageWidth -> LONG (or SHORT), 1
295055efc227SAndreas Gohr            $tags[0x1002] = array(4, 1); // RelatedImageLength -> LONG (or SHORT), 1
29510b17fdc6SAndreas Gohr        } elseif ($mode == 'gps') {
295255efc227SAndreas Gohr            $tags[0x0000] = array(1, 4); // GPSVersionID -> BYTE, 4
295355efc227SAndreas Gohr            $tags[0x0001] = array(2, 2); // GPSLatitudeRef -> ASCII, 2
295455efc227SAndreas Gohr            $tags[0x0002] = array(5, 3); // GPSLatitude -> RATIONAL, 3
295555efc227SAndreas Gohr            $tags[0x0003] = array(2, 2); // GPSLongitudeRef -> ASCII, 2
295655efc227SAndreas Gohr            $tags[0x0004] = array(5, 3); // GPSLongitude -> RATIONAL, 3
295755efc227SAndreas Gohr            $tags[0x0005] = array(2, 2); // GPSAltitudeRef -> ASCII, 2
295855efc227SAndreas Gohr            $tags[0x0006] = array(5, 1); // GPSAltitude -> RATIONAL, 1
295955efc227SAndreas Gohr            $tags[0x0007] = array(5, 3); // GPSTimeStamp -> RATIONAL, 3
296055efc227SAndreas Gohr            $tags[0x0008] = array(2, 0); // GPSSatellites -> ASCII, Any
296155efc227SAndreas Gohr            $tags[0x0009] = array(2, 2); // GPSStatus -> ASCII, 2
296255efc227SAndreas Gohr            $tags[0x000A] = array(2, 2); // GPSMeasureMode -> ASCII, 2
296355efc227SAndreas Gohr            $tags[0x000B] = array(5, 1); // GPSDOP -> RATIONAL, 1
296455efc227SAndreas Gohr            $tags[0x000C] = array(2, 2); // GPSSpeedRef -> ASCII, 2
296555efc227SAndreas Gohr            $tags[0x000D] = array(5, 1); // GPSSpeed -> RATIONAL, 1
296655efc227SAndreas Gohr            $tags[0x000E] = array(2, 2); // GPSTrackRef -> ASCII, 2
296755efc227SAndreas Gohr            $tags[0x000F] = array(5, 1); // GPSTrack -> RATIONAL, 1
296855efc227SAndreas Gohr            $tags[0x0010] = array(2, 2); // GPSImgDirectionRef -> ASCII, 2
296955efc227SAndreas Gohr            $tags[0x0011] = array(5, 1); // GPSImgDirection -> RATIONAL, 1
297055efc227SAndreas Gohr            $tags[0x0012] = array(2, 0); // GPSMapDatum -> ASCII, Any
297155efc227SAndreas Gohr            $tags[0x0013] = array(2, 2); // GPSDestLatitudeRef -> ASCII, 2
297255efc227SAndreas Gohr            $tags[0x0014] = array(5, 3); // GPSDestLatitude -> RATIONAL, 3
297355efc227SAndreas Gohr            $tags[0x0015] = array(2, 2); // GPSDestLongitudeRef -> ASCII, 2
297455efc227SAndreas Gohr            $tags[0x0016] = array(5, 3); // GPSDestLongitude -> RATIONAL, 3
297555efc227SAndreas Gohr            $tags[0x0017] = array(2, 2); // GPSDestBearingRef -> ASCII, 2
297655efc227SAndreas Gohr            $tags[0x0018] = array(5, 1); // GPSDestBearing -> RATIONAL, 1
297755efc227SAndreas Gohr            $tags[0x0019] = array(2, 2); // GPSDestDistanceRef -> ASCII, 2
297855efc227SAndreas Gohr            $tags[0x001A] = array(5, 1); // GPSDestDistance -> RATIONAL, 1
297955efc227SAndreas Gohr        }
298055efc227SAndreas Gohr
298155efc227SAndreas Gohr        return $tags;
298255efc227SAndreas Gohr    }
298355efc227SAndreas Gohr
298455efc227SAndreas Gohr    /*************************************************************/
29850b17fdc6SAndreas Gohr    function _exifNameTags($mode) {
298655efc227SAndreas Gohr        $tags = $this->_exifTagNames($mode);
298755efc227SAndreas Gohr        return $this->_names2Tags($tags);
298855efc227SAndreas Gohr    }
298955efc227SAndreas Gohr
299055efc227SAndreas Gohr    /*************************************************************/
29910b17fdc6SAndreas Gohr    function _iptcTagNames() {
299255efc227SAndreas Gohr        $tags = array();
299355efc227SAndreas Gohr        $tags[0x14] = 'SuplementalCategories';
299455efc227SAndreas Gohr        $tags[0x19] = 'Keywords';
299555efc227SAndreas Gohr        $tags[0x78] = 'Caption';
299655efc227SAndreas Gohr        $tags[0x7A] = 'CaptionWriter';
299755efc227SAndreas Gohr        $tags[0x69] = 'Headline';
299855efc227SAndreas Gohr        $tags[0x28] = 'SpecialInstructions';
299955efc227SAndreas Gohr        $tags[0x0F] = 'Category';
300055efc227SAndreas Gohr        $tags[0x50] = 'Byline';
300155efc227SAndreas Gohr        $tags[0x55] = 'BylineTitle';
300255efc227SAndreas Gohr        $tags[0x6E] = 'Credit';
300355efc227SAndreas Gohr        $tags[0x73] = 'Source';
300455efc227SAndreas Gohr        $tags[0x74] = 'CopyrightNotice';
300555efc227SAndreas Gohr        $tags[0x05] = 'ObjectName';
300655efc227SAndreas Gohr        $tags[0x5A] = 'City';
3007493531a8SJoe Lapp        $tags[0x5C] = 'Sublocation';
300855efc227SAndreas Gohr        $tags[0x5F] = 'ProvinceState';
300955efc227SAndreas Gohr        $tags[0x65] = 'CountryName';
301055efc227SAndreas Gohr        $tags[0x67] = 'OriginalTransmissionReference';
301155efc227SAndreas Gohr        $tags[0x37] = 'DateCreated';
301255efc227SAndreas Gohr        $tags[0x0A] = 'CopyrightFlag';
301355efc227SAndreas Gohr
301455efc227SAndreas Gohr        return $tags;
301555efc227SAndreas Gohr    }
301655efc227SAndreas Gohr
301755efc227SAndreas Gohr    /*************************************************************/
30180b17fdc6SAndreas Gohr    function & _iptcNameTags() {
301955efc227SAndreas Gohr        $tags = $this->_iptcTagNames();
302055efc227SAndreas Gohr        return $this->_names2Tags($tags);
302155efc227SAndreas Gohr    }
302255efc227SAndreas Gohr
302355efc227SAndreas Gohr    /*************************************************************/
30240b17fdc6SAndreas Gohr    function _names2Tags($tags2Names) {
302555efc227SAndreas Gohr        $names2Tags = array();
30269e491c01SAndreas Gohr
30279e491c01SAndreas Gohr        foreach($tags2Names as $tag => $name) {
302855efc227SAndreas Gohr            $names2Tags[$name] = $tag;
302955efc227SAndreas Gohr        }
303055efc227SAndreas Gohr
303155efc227SAndreas Gohr        return $names2Tags;
303255efc227SAndreas Gohr    }
303355efc227SAndreas Gohr
303455efc227SAndreas Gohr    /*************************************************************/
3035276820f7SScrutinizer Auto-Fixer
3036276820f7SScrutinizer Auto-Fixer    /**
3037f50a239bSTakamura     * @param $data
3038276820f7SScrutinizer Auto-Fixer     * @param integer $pos
3039f50a239bSTakamura     *
3040f50a239bSTakamura     * @return int
3041276820f7SScrutinizer Auto-Fixer     */
30420b17fdc6SAndreas Gohr    function _getByte(&$data, $pos) {
3043c9c56f8aSasivery        if (!isset($data[$pos])) {
3044c9c56f8aSasivery            throw new Exception("Requested byte at ".$pos.". Reading outside of file's boundaries.");
3045c9c56f8aSasivery        }
3046c9c56f8aSasivery
30472401f18dSSyntaxseed        return ord($data[$pos]);
304855efc227SAndreas Gohr    }
304955efc227SAndreas Gohr
305055efc227SAndreas Gohr    /*************************************************************/
3051276820f7SScrutinizer Auto-Fixer
3052276820f7SScrutinizer Auto-Fixer    /**
3053f50a239bSTakamura     * @param mixed $data
3054276820f7SScrutinizer Auto-Fixer     * @param integer $pos
3055f50a239bSTakamura     *
3056f50a239bSTakamura     * @param mixed $val
3057f50a239bSTakamura     *
3058f50a239bSTakamura     * @return int
3059276820f7SScrutinizer Auto-Fixer     */
30600b17fdc6SAndreas Gohr    function _putByte(&$data, $pos, $val) {
306155efc227SAndreas Gohr        $val = intval($val);
306255efc227SAndreas Gohr
30632401f18dSSyntaxseed        $data[$pos] = chr($val);
306455efc227SAndreas Gohr
306555efc227SAndreas Gohr        return $pos + 1;
306655efc227SAndreas Gohr    }
306755efc227SAndreas Gohr
306855efc227SAndreas Gohr    /*************************************************************/
30690b17fdc6SAndreas Gohr    function _getShort(&$data, $pos, $bigEndian = true) {
3070c9c56f8aSasivery        if (!isset($data[$pos]) || !isset($data[$pos + 1])) {
3071c9c56f8aSasivery            throw new Exception("Requested short at ".$pos.". Reading outside of file's boundaries.");
3072c9c56f8aSasivery        }
3073c9c56f8aSasivery
307455efc227SAndreas Gohr        if ($bigEndian) {
30752401f18dSSyntaxseed            return (ord($data[$pos]) << 8)
30762401f18dSSyntaxseed                + ord($data[$pos + 1]);
30770b17fdc6SAndreas Gohr        } else {
30782401f18dSSyntaxseed            return ord($data[$pos])
30792401f18dSSyntaxseed                + (ord($data[$pos + 1]) << 8);
308055efc227SAndreas Gohr        }
308155efc227SAndreas Gohr    }
308255efc227SAndreas Gohr
308355efc227SAndreas Gohr    /*************************************************************/
30840b17fdc6SAndreas Gohr    function _putShort(&$data, $pos = 0, $val = 0, $bigEndian = true) {
308555efc227SAndreas Gohr        $val = intval($val);
308655efc227SAndreas Gohr
308755efc227SAndreas Gohr        if ($bigEndian) {
30882401f18dSSyntaxseed            $data[$pos + 0] = chr(($val & 0x0000FF00) >> 8);
30892401f18dSSyntaxseed            $data[$pos + 1] = chr(($val & 0x000000FF) >> 0);
30900b17fdc6SAndreas Gohr        } else {
30912401f18dSSyntaxseed            $data[$pos + 0] = chr(($val & 0x00FF) >> 0);
30922401f18dSSyntaxseed            $data[$pos + 1] = chr(($val & 0xFF00) >> 8);
309355efc227SAndreas Gohr        }
309455efc227SAndreas Gohr
309555efc227SAndreas Gohr        return $pos + 2;
309655efc227SAndreas Gohr    }
309755efc227SAndreas Gohr
309855efc227SAndreas Gohr    /*************************************************************/
3099276820f7SScrutinizer Auto-Fixer
3100276820f7SScrutinizer Auto-Fixer    /**
3101f50a239bSTakamura     * @param mixed $data
3102276820f7SScrutinizer Auto-Fixer     * @param integer $pos
3103f50a239bSTakamura     *
3104f50a239bSTakamura     * @param bool $bigEndian
3105f50a239bSTakamura     *
3106f50a239bSTakamura     * @return int
3107276820f7SScrutinizer Auto-Fixer     */
31080b17fdc6SAndreas Gohr    function _getLong(&$data, $pos, $bigEndian = true) {
3109c9c56f8aSasivery        // Assume that if the start and end bytes are defined, the bytes inbetween are defined as well.
3110c9c56f8aSasivery        if (!isset($data[$pos]) || !isset($data[$pos + 3])){
3111c9c56f8aSasivery            throw new Exception("Requested long at ".$pos.". Reading outside of file's boundaries.");
3112c9c56f8aSasivery        }
311355efc227SAndreas Gohr        if ($bigEndian) {
31142401f18dSSyntaxseed            return (ord($data[$pos]) << 24)
31152401f18dSSyntaxseed                + (ord($data[$pos + 1]) << 16)
31162401f18dSSyntaxseed                + (ord($data[$pos + 2]) << 8)
31172401f18dSSyntaxseed                + ord($data[$pos + 3]);
31180b17fdc6SAndreas Gohr        } else {
31192401f18dSSyntaxseed            return ord($data[$pos])
31202401f18dSSyntaxseed                + (ord($data[$pos + 1]) << 8)
31212401f18dSSyntaxseed                + (ord($data[$pos + 2]) << 16)
31222401f18dSSyntaxseed                + (ord($data[$pos + 3]) << 24);
312355efc227SAndreas Gohr        }
312455efc227SAndreas Gohr    }
312555efc227SAndreas Gohr
312655efc227SAndreas Gohr    /*************************************************************/
3127276820f7SScrutinizer Auto-Fixer
3128276820f7SScrutinizer Auto-Fixer    /**
3129f50a239bSTakamura     * @param mixed $data
3130276820f7SScrutinizer Auto-Fixer     * @param integer $pos
3131f50a239bSTakamura     *
3132f50a239bSTakamura     * @param mixed $val
3133f50a239bSTakamura     * @param bool $bigEndian
3134f50a239bSTakamura     *
3135f50a239bSTakamura     * @return int
3136276820f7SScrutinizer Auto-Fixer     */
31370b17fdc6SAndreas Gohr    function _putLong(&$data, $pos, $val, $bigEndian = true) {
313855efc227SAndreas Gohr        $val = intval($val);
313955efc227SAndreas Gohr
314055efc227SAndreas Gohr        if ($bigEndian) {
31412401f18dSSyntaxseed            $data[$pos + 0] = chr(($val & 0xFF000000) >> 24);
31422401f18dSSyntaxseed            $data[$pos + 1] = chr(($val & 0x00FF0000) >> 16);
31432401f18dSSyntaxseed            $data[$pos + 2] = chr(($val & 0x0000FF00) >> 8);
31442401f18dSSyntaxseed            $data[$pos + 3] = chr(($val & 0x000000FF) >> 0);
31450b17fdc6SAndreas Gohr        } else {
31462401f18dSSyntaxseed            $data[$pos + 0] = chr(($val & 0x000000FF) >> 0);
31472401f18dSSyntaxseed            $data[$pos + 1] = chr(($val & 0x0000FF00) >> 8);
31482401f18dSSyntaxseed            $data[$pos + 2] = chr(($val & 0x00FF0000) >> 16);
31492401f18dSSyntaxseed            $data[$pos + 3] = chr(($val & 0xFF000000) >> 24);
315055efc227SAndreas Gohr        }
315155efc227SAndreas Gohr
315255efc227SAndreas Gohr        return $pos + 4;
315355efc227SAndreas Gohr    }
315455efc227SAndreas Gohr
315555efc227SAndreas Gohr    /*************************************************************/
31560b17fdc6SAndreas Gohr    function & _getNullString(&$data, $pos) {
315755efc227SAndreas Gohr        $str = '';
315855efc227SAndreas Gohr        $max = strlen($data);
315955efc227SAndreas Gohr
316055efc227SAndreas Gohr        while ($pos < $max) {
3161c9c56f8aSasivery            if (!isset($data[$pos])) {
3162c9c56f8aSasivery                throw new Exception("Requested null-terminated string at offset ".$pos.". File terminated before the null-byte.");
3163c9c56f8aSasivery            }
31642401f18dSSyntaxseed            if (ord($data[$pos]) == 0) {
316555efc227SAndreas Gohr                return $str;
31660b17fdc6SAndreas Gohr            } else {
31672401f18dSSyntaxseed                $str .= $data[$pos];
316855efc227SAndreas Gohr            }
316955efc227SAndreas Gohr            $pos++;
317055efc227SAndreas Gohr        }
317155efc227SAndreas Gohr
317255efc227SAndreas Gohr        return $str;
317355efc227SAndreas Gohr    }
317455efc227SAndreas Gohr
317555efc227SAndreas Gohr    /*************************************************************/
31760b17fdc6SAndreas Gohr    function & _getFixedString(&$data, $pos, $length = -1) {
317755efc227SAndreas Gohr        if ($length == -1) {
317855efc227SAndreas Gohr            $length = strlen($data) - $pos;
317955efc227SAndreas Gohr        }
318055efc227SAndreas Gohr
318117dd401eSChristopher Smith        $rv = substr($data, $pos, $length);
318236fb33ceSasivery        if (strlen($rv) != $length) {
3183d40e7d0eSAndreas Gohr            throw new ErrorException(sprintf(
3184d40e7d0eSAndreas Gohr                "JPEGMeta failed parsing image metadata of %s. Got %d instead of %d bytes at offset %d.",
3185d40e7d0eSAndreas Gohr                $this->_fileName, strlen($rv), $length, $pos
3186d40e7d0eSAndreas Gohr            ), 0, E_WARNING);
318736fb33ceSasivery        }
318817dd401eSChristopher Smith        return $rv;
318955efc227SAndreas Gohr    }
319055efc227SAndreas Gohr
319155efc227SAndreas Gohr    /*************************************************************/
31920b17fdc6SAndreas Gohr    function _putString(&$data, $pos, &$str) {
319355efc227SAndreas Gohr        $len = strlen($str);
319455efc227SAndreas Gohr        for ($i = 0; $i < $len; $i++) {
31952401f18dSSyntaxseed            $data[$pos + $i] = $str[$i];
319655efc227SAndreas Gohr        }
319755efc227SAndreas Gohr
319855efc227SAndreas Gohr        return $pos + $len;
319955efc227SAndreas Gohr    }
320055efc227SAndreas Gohr
320155efc227SAndreas Gohr    /*************************************************************/
32020b17fdc6SAndreas Gohr    function _hexDump(&$data, $start = 0, $length = -1) {
320355efc227SAndreas Gohr        if (($length == -1) || (($length + $start) > strlen($data))) {
320455efc227SAndreas Gohr            $end = strlen($data);
32050b17fdc6SAndreas Gohr        } else {
320655efc227SAndreas Gohr            $end = $start + $length;
320755efc227SAndreas Gohr        }
320855efc227SAndreas Gohr
320955efc227SAndreas Gohr        $ascii = '';
321055efc227SAndreas Gohr        $count = 0;
321155efc227SAndreas Gohr
321255efc227SAndreas Gohr        echo "<tt>\n";
321355efc227SAndreas Gohr
321455efc227SAndreas Gohr        while ($start < $end) {
321555efc227SAndreas Gohr            if (($count % 16) == 0) {
321655efc227SAndreas Gohr                echo sprintf('%04d', $count) . ': ';
321755efc227SAndreas Gohr            }
321855efc227SAndreas Gohr
32192401f18dSSyntaxseed            $c = ord($data[$start]);
322055efc227SAndreas Gohr            $count++;
322155efc227SAndreas Gohr            $start++;
322255efc227SAndreas Gohr
322355efc227SAndreas Gohr            $aux = dechex($c);
322455efc227SAndreas Gohr            if (strlen($aux) == 1)
322555efc227SAndreas Gohr                echo '0';
322655efc227SAndreas Gohr            echo $aux . ' ';
322755efc227SAndreas Gohr
322855efc227SAndreas Gohr            if ($c == 60)
322955efc227SAndreas Gohr                $ascii .= '&lt;';
323055efc227SAndreas Gohr            elseif ($c == 62)
323155efc227SAndreas Gohr                $ascii .= '&gt;';
323255efc227SAndreas Gohr            elseif ($c == 32)
3233e260f93bSAnika Henke                $ascii .= '&#160;';
323455efc227SAndreas Gohr            elseif ($c > 32)
323555efc227SAndreas Gohr                $ascii .= chr($c);
323655efc227SAndreas Gohr            else
323755efc227SAndreas Gohr                $ascii .= '.';
323855efc227SAndreas Gohr
323955efc227SAndreas Gohr            if (($count % 4) == 0) {
324055efc227SAndreas Gohr                echo ' - ';
324155efc227SAndreas Gohr            }
324255efc227SAndreas Gohr
324355efc227SAndreas Gohr            if (($count % 16) == 0) {
324455efc227SAndreas Gohr                echo ': ' . $ascii . "<br>\n";
324555efc227SAndreas Gohr                $ascii = '';
324655efc227SAndreas Gohr            }
324755efc227SAndreas Gohr        }
324855efc227SAndreas Gohr
324955efc227SAndreas Gohr        if ($ascii != '') {
325055efc227SAndreas Gohr            while (($count % 16) != 0) {
325155efc227SAndreas Gohr                echo '-- ';
325255efc227SAndreas Gohr                $count++;
325355efc227SAndreas Gohr                if (($count % 4) == 0) {
325455efc227SAndreas Gohr                    echo ' - ';
325555efc227SAndreas Gohr                }
325655efc227SAndreas Gohr            }
325755efc227SAndreas Gohr            echo ': ' . $ascii . "<br>\n";
325855efc227SAndreas Gohr        }
325955efc227SAndreas Gohr
326055efc227SAndreas Gohr        echo "</tt>\n";
326155efc227SAndreas Gohr    }
326255efc227SAndreas Gohr
326355efc227SAndreas Gohr    /*****************************************************************/
326455efc227SAndreas Gohr}
326555efc227SAndreas Gohr
326655efc227SAndreas Gohr/* vim: set expandtab tabstop=4 shiftwidth=4: */
3267