xref: /dokuwiki/inc/JpegMeta.php (revision e260f93b6cea05bc39bbd77b9db5bdc0c2c424bf)
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;
4555efc227SAndreas Gohr    var $_type = 'unknown';
4655efc227SAndreas Gohr
4755efc227SAndreas Gohr    var $_markers;
4855efc227SAndreas Gohr    var $_info;
4955efc227SAndreas Gohr
5055efc227SAndreas Gohr
5155efc227SAndreas Gohr    /**
5255efc227SAndreas Gohr     * Constructor
5355efc227SAndreas Gohr     *
5455efc227SAndreas Gohr     * @author Sebastian Delmont <sdelmont@zonageek.com>
5555efc227SAndreas Gohr     */
560b17fdc6SAndreas Gohr    function JpegMeta($fileName) {
5755efc227SAndreas Gohr
5855efc227SAndreas Gohr        $this->_fileName = $fileName;
5955efc227SAndreas Gohr
6055efc227SAndreas Gohr        $this->_fp = null;
6155efc227SAndreas Gohr        $this->_type = 'unknown';
6255efc227SAndreas Gohr
6355efc227SAndreas Gohr        unset($this->_info);
6455efc227SAndreas Gohr        unset($this->_markers);
6555efc227SAndreas Gohr    }
6655efc227SAndreas Gohr
6755efc227SAndreas Gohr    /**
6855efc227SAndreas Gohr     * Returns all gathered info as multidim array
6955efc227SAndreas Gohr     *
7055efc227SAndreas Gohr     * @author Sebastian Delmont <sdelmont@zonageek.com>
7155efc227SAndreas Gohr     */
720b17fdc6SAndreas Gohr    function & getRawInfo() {
7355efc227SAndreas Gohr        $this->_parseAll();
7455efc227SAndreas Gohr
7555efc227SAndreas Gohr        if ($this->_markers == null) {
7655efc227SAndreas Gohr            return false;
7755efc227SAndreas Gohr        }
7855efc227SAndreas Gohr
7955efc227SAndreas Gohr        return $this->_info;
8055efc227SAndreas Gohr    }
8155efc227SAndreas Gohr
8255efc227SAndreas Gohr    /**
8355efc227SAndreas Gohr     * Returns basic image info
8455efc227SAndreas Gohr     *
8555efc227SAndreas Gohr     * @author Sebastian Delmont <sdelmont@zonageek.com>
8655efc227SAndreas Gohr     */
870b17fdc6SAndreas Gohr    function & getBasicInfo() {
8855efc227SAndreas Gohr        $this->_parseAll();
8955efc227SAndreas Gohr
9055efc227SAndreas Gohr        $info = array();
9155efc227SAndreas Gohr
9255efc227SAndreas Gohr        if ($this->_markers == null) {
9355efc227SAndreas Gohr            return false;
9455efc227SAndreas Gohr        }
9555efc227SAndreas Gohr
9655efc227SAndreas Gohr        $info['Name'] = $this->_info['file']['Name'];
9755efc227SAndreas Gohr        if (isset($this->_info['file']['Url'])) {
9855efc227SAndreas Gohr            $info['Url'] = $this->_info['file']['Url'];
9955efc227SAndreas Gohr            $info['NiceSize'] = "???KB";
1000b17fdc6SAndreas Gohr        } else {
10155efc227SAndreas Gohr            $info['Size'] = $this->_info['file']['Size'];
10255efc227SAndreas Gohr            $info['NiceSize'] = $this->_info['file']['NiceSize'];
10355efc227SAndreas Gohr        }
10455efc227SAndreas Gohr
10555efc227SAndreas Gohr        if (@isset($this->_info['sof']['Format'])) {
10655efc227SAndreas Gohr            $info['Format'] = $this->_info['sof']['Format'] . " JPEG";
1070b17fdc6SAndreas Gohr        } else {
10855efc227SAndreas Gohr            $info['Format'] = $this->_info['sof']['Format'] . " JPEG";
10955efc227SAndreas Gohr        }
11055efc227SAndreas Gohr
11155efc227SAndreas Gohr        if (@isset($this->_info['sof']['ColorChannels'])) {
11255efc227SAndreas Gohr            $info['ColorMode'] = ($this->_info['sof']['ColorChannels'] > 1) ? "Color" : "B&W";
11355efc227SAndreas Gohr        }
11455efc227SAndreas Gohr
11555efc227SAndreas Gohr        $info['Width'] = $this->getWidth();
11655efc227SAndreas Gohr        $info['Height'] = $this->getHeight();
11755efc227SAndreas Gohr        $info['DimStr'] = $this->getDimStr();
11855efc227SAndreas Gohr
11955efc227SAndreas Gohr        $dates = $this->getDates();
12055efc227SAndreas Gohr
12155efc227SAndreas Gohr        $info['DateTime'] = $dates['EarliestTime'];
12255efc227SAndreas Gohr        $info['DateTimeStr'] = $dates['EarliestTimeStr'];
12355efc227SAndreas Gohr
12455efc227SAndreas Gohr        $info['HasThumbnail'] = $this->hasThumbnail();
12555efc227SAndreas Gohr
12655efc227SAndreas Gohr        return $info;
12755efc227SAndreas Gohr    }
12855efc227SAndreas Gohr
12955efc227SAndreas Gohr
13055efc227SAndreas Gohr    /**
13155efc227SAndreas Gohr     * Convinience function to access nearly all available Data
13255efc227SAndreas Gohr     * through one function
13355efc227SAndreas Gohr     *
13455efc227SAndreas Gohr     * @author Andreas Gohr <andi@splitbrain.org>
13555efc227SAndreas Gohr     */
1360b17fdc6SAndreas Gohr    function getField($fields) {
13755efc227SAndreas Gohr        if(!is_array($fields)) $fields = array($fields);
13855efc227SAndreas Gohr        $info = false;
13955efc227SAndreas Gohr        foreach($fields as $field){
14055efc227SAndreas Gohr            if(strtolower(substr($field,0,5)) == 'iptc.'){
14155efc227SAndreas Gohr                $info = $this->getIPTCField(substr($field,5));
14255efc227SAndreas Gohr            }elseif(strtolower(substr($field,0,5)) == 'exif.'){
14355efc227SAndreas Gohr                $info = $this->getExifField(substr($field,5));
144431c7fc8Shakan.sandell            }elseif(strtolower(substr($field,0,4)) == 'xmp.'){
145431c7fc8Shakan.sandell                $info = $this->getXmpField(substr($field,4));
14655efc227SAndreas Gohr            }elseif(strtolower(substr($field,0,5)) == 'file.'){
14755efc227SAndreas Gohr                $info = $this->getFileField(substr($field,5));
14855efc227SAndreas Gohr            }elseif(strtolower(substr($field,0,5)) == 'date.'){
14955efc227SAndreas Gohr                $info = $this->getDateField(substr($field,5));
15055efc227SAndreas Gohr            }elseif(strtolower($field) == 'simple.camera'){
15155efc227SAndreas Gohr                $info = $this->getCamera();
15255efc227SAndreas Gohr            }elseif(strtolower($field) == 'simple.raw'){
15355efc227SAndreas Gohr                return $this->getRawInfo();
15455efc227SAndreas Gohr            }elseif(strtolower($field) == 'simple.title'){
15555efc227SAndreas Gohr                $info = $this->getTitle();
1566db72d46SJoe Lapp            }elseif(strtolower($field) == 'simple.shutterspeed'){
1576db72d46SJoe Lapp                $info = $this->getShutterSpeed();
15855efc227SAndreas Gohr            }else{
15955efc227SAndreas Gohr                $info = $this->getExifField($field);
16055efc227SAndreas Gohr            }
16155efc227SAndreas Gohr            if($info != false) break;
16255efc227SAndreas Gohr        }
16355efc227SAndreas Gohr
16455efc227SAndreas Gohr        if($info === false)  $info = $alt;
16555efc227SAndreas Gohr        if(is_array($info)){
16655efc227SAndreas Gohr            if(isset($info['val'])){
16755efc227SAndreas Gohr                $info = $info['val'];
16855efc227SAndreas Gohr            }else{
16955efc227SAndreas Gohr                $info = join(', ',$info);
17055efc227SAndreas Gohr            }
17155efc227SAndreas Gohr        }
17255efc227SAndreas Gohr        return trim($info);
17355efc227SAndreas Gohr    }
17455efc227SAndreas Gohr
17555efc227SAndreas Gohr    /**
17636df6fa3SAndreas Gohr     * Convinience function to set nearly all available Data
17736df6fa3SAndreas Gohr     * through one function
17836df6fa3SAndreas Gohr     *
17936df6fa3SAndreas Gohr     * @author Andreas Gohr <andi@splitbrain.org>
18036df6fa3SAndreas Gohr     */
1810b17fdc6SAndreas Gohr    function setField($field, $value) {
18236df6fa3SAndreas Gohr        if(strtolower(substr($field,0,5)) == 'iptc.'){
18336df6fa3SAndreas Gohr            return $this->setIPTCField(substr($field,5),$value);
18436df6fa3SAndreas Gohr        }elseif(strtolower(substr($field,0,5)) == 'exif.'){
18536df6fa3SAndreas Gohr            return $this->setExifField(substr($field,5),$value);
18636df6fa3SAndreas Gohr        }else{
18736df6fa3SAndreas Gohr            return $this->setExifField($field,$value);
18836df6fa3SAndreas Gohr        }
18936df6fa3SAndreas Gohr    }
19036df6fa3SAndreas Gohr
19136df6fa3SAndreas Gohr    /**
19236df6fa3SAndreas Gohr     * Convinience function to delete nearly all available Data
19336df6fa3SAndreas Gohr     * through one function
19436df6fa3SAndreas Gohr     *
19536df6fa3SAndreas Gohr     * @author Andreas Gohr <andi@splitbrain.org>
19636df6fa3SAndreas Gohr     */
1970b17fdc6SAndreas Gohr    function deleteField($field) {
19836df6fa3SAndreas Gohr        if(strtolower(substr($field,0,5)) == 'iptc.'){
19936df6fa3SAndreas Gohr            return $this->deleteIPTCField(substr($field,5));
20036df6fa3SAndreas Gohr        }elseif(strtolower(substr($field,0,5)) == 'exif.'){
20136df6fa3SAndreas Gohr            return $this->deleteExifField(substr($field,5));
20236df6fa3SAndreas Gohr        }else{
20336df6fa3SAndreas Gohr            return $this->deleteExifField($field);
20436df6fa3SAndreas Gohr        }
20536df6fa3SAndreas Gohr    }
20636df6fa3SAndreas Gohr
20736df6fa3SAndreas Gohr    /**
20855efc227SAndreas Gohr     * Return a date field
20955efc227SAndreas Gohr     *
21055efc227SAndreas Gohr     * @author Andreas Gohr <andi@splitbrain.org>
21155efc227SAndreas Gohr     */
2120b17fdc6SAndreas Gohr    function getDateField($field) {
21355efc227SAndreas Gohr        if (!isset($this->_info['dates'])) {
21455efc227SAndreas Gohr            $this->_info['dates'] = $this->getDates();
21555efc227SAndreas Gohr        }
21655efc227SAndreas Gohr
21755efc227SAndreas Gohr        if (isset($this->_info['dates'][$field])) {
21855efc227SAndreas Gohr            return $this->_info['dates'][$field];
21955efc227SAndreas Gohr        }
22055efc227SAndreas Gohr
22155efc227SAndreas Gohr        return false;
22255efc227SAndreas Gohr    }
22355efc227SAndreas Gohr
22455efc227SAndreas Gohr    /**
22555efc227SAndreas Gohr     * Return a file info field
22655efc227SAndreas Gohr     *
22755efc227SAndreas Gohr     * @author Andreas Gohr <andi@splitbrain.org>
22855efc227SAndreas Gohr     */
2290b17fdc6SAndreas Gohr    function getFileField($field) {
23055efc227SAndreas Gohr        if (!isset($this->_info['file'])) {
23155efc227SAndreas Gohr            $this->_parseFileInfo();
23255efc227SAndreas Gohr        }
23355efc227SAndreas Gohr
23455efc227SAndreas Gohr        if (isset($this->_info['file'][$field])) {
23555efc227SAndreas Gohr            return $this->_info['file'][$field];
23655efc227SAndreas Gohr        }
23755efc227SAndreas Gohr
23855efc227SAndreas Gohr        return false;
23955efc227SAndreas Gohr    }
24055efc227SAndreas Gohr
24155efc227SAndreas Gohr    /**
24255efc227SAndreas Gohr     * Return the camera info (Maker and Model)
24355efc227SAndreas Gohr     *
24455efc227SAndreas Gohr     * @author Andreas Gohr <andi@splitbrain.org>
24555efc227SAndreas Gohr     * @todo   handle makernotes
24655efc227SAndreas Gohr     */
24755efc227SAndreas Gohr    function getCamera(){
24855efc227SAndreas Gohr        $make  = $this->getField(array('Exif.Make','Exif.TIFFMake'));
24955efc227SAndreas Gohr        $model = $this->getField(array('Exif.Model','Exif.TIFFModel'));
25055efc227SAndreas Gohr        $cam = trim("$make $model");
25155efc227SAndreas Gohr        if(empty($cam)) return false;
25255efc227SAndreas Gohr        return $cam;
25355efc227SAndreas Gohr    }
25455efc227SAndreas Gohr
25555efc227SAndreas Gohr    /**
2566db72d46SJoe Lapp     * Return shutter speed as a ratio
2576db72d46SJoe Lapp     *
2586db72d46SJoe Lapp     * @author Joe Lapp <joe.lapp@pobox.com>
2596db72d46SJoe Lapp     */
2600b17fdc6SAndreas Gohr    function getShutterSpeed() {
2616db72d46SJoe Lapp        if (!isset($this->_info['exif'])) {
2626db72d46SJoe Lapp            $this->_parseMarkerExif();
2636db72d46SJoe Lapp        }
2646db72d46SJoe Lapp        if(!isset($this->_info['exif']['ExposureTime'])){
2656db72d46SJoe Lapp            return '';
2666db72d46SJoe Lapp        }
2676db72d46SJoe Lapp
2686db72d46SJoe Lapp        $field = $this->_info['exif']['ExposureTime'];
2696db72d46SJoe Lapp        if($field['den'] == 1) return $field['num'];
2706db72d46SJoe Lapp        return $field['num'].'/'.$field['den'];
2716db72d46SJoe Lapp    }
2726db72d46SJoe Lapp
2736db72d46SJoe Lapp    /**
27455efc227SAndreas Gohr     * Return an EXIF field
27555efc227SAndreas Gohr     *
27655efc227SAndreas Gohr     * @author Sebastian Delmont <sdelmont@zonageek.com>
27755efc227SAndreas Gohr     */
2780b17fdc6SAndreas Gohr    function getExifField($field) {
27955efc227SAndreas Gohr        if (!isset($this->_info['exif'])) {
28055efc227SAndreas Gohr            $this->_parseMarkerExif();
28155efc227SAndreas Gohr        }
28255efc227SAndreas Gohr
28355efc227SAndreas Gohr        if ($this->_markers == null) {
28455efc227SAndreas Gohr            return false;
28555efc227SAndreas Gohr        }
28655efc227SAndreas Gohr
28755efc227SAndreas Gohr        if (isset($this->_info['exif'][$field])) {
28855efc227SAndreas Gohr            return $this->_info['exif'][$field];
28955efc227SAndreas Gohr        }
29055efc227SAndreas Gohr
29155efc227SAndreas Gohr        return false;
29255efc227SAndreas Gohr    }
29355efc227SAndreas Gohr
29455efc227SAndreas Gohr    /**
295431c7fc8Shakan.sandell     * Return an XMP field
296431c7fc8Shakan.sandell     *
297431c7fc8Shakan.sandell     * @author Hakan Sandell <hakan.sandell@mydata.se>
298431c7fc8Shakan.sandell     */
2990b17fdc6SAndreas Gohr    function getXmpField($field) {
300431c7fc8Shakan.sandell        if (!isset($this->_info['xmp'])) {
301431c7fc8Shakan.sandell            $this->_parseMarkerXmp();
302431c7fc8Shakan.sandell        }
303431c7fc8Shakan.sandell
304431c7fc8Shakan.sandell        if ($this->_markers == null) {
305431c7fc8Shakan.sandell            return false;
306431c7fc8Shakan.sandell        }
307431c7fc8Shakan.sandell
308431c7fc8Shakan.sandell        if (isset($this->_info['xmp'][$field])) {
309431c7fc8Shakan.sandell            return $this->_info['xmp'][$field];
310431c7fc8Shakan.sandell        }
311431c7fc8Shakan.sandell
312431c7fc8Shakan.sandell        return false;
313431c7fc8Shakan.sandell    }
314431c7fc8Shakan.sandell
315431c7fc8Shakan.sandell    /**
31655efc227SAndreas Gohr     * Return an Adobe Field
31755efc227SAndreas Gohr     *
31855efc227SAndreas Gohr     * @author Sebastian Delmont <sdelmont@zonageek.com>
31955efc227SAndreas Gohr     */
3200b17fdc6SAndreas Gohr    function getAdobeField($field) {
32155efc227SAndreas Gohr        if (!isset($this->_info['adobe'])) {
32255efc227SAndreas Gohr            $this->_parseMarkerAdobe();
32355efc227SAndreas Gohr        }
32455efc227SAndreas Gohr
32555efc227SAndreas Gohr        if ($this->_markers == null) {
32655efc227SAndreas Gohr            return false;
32755efc227SAndreas Gohr        }
32855efc227SAndreas Gohr
32955efc227SAndreas Gohr        if (isset($this->_info['adobe'][$field])) {
33055efc227SAndreas Gohr            return $this->_info['adobe'][$field];
33155efc227SAndreas Gohr        }
33255efc227SAndreas Gohr
33355efc227SAndreas Gohr        return false;
33455efc227SAndreas Gohr    }
33555efc227SAndreas Gohr
33655efc227SAndreas Gohr    /**
33755efc227SAndreas Gohr     * Return an IPTC field
33855efc227SAndreas Gohr     *
33955efc227SAndreas Gohr     * @author Sebastian Delmont <sdelmont@zonageek.com>
34055efc227SAndreas Gohr     */
3410b17fdc6SAndreas Gohr    function getIPTCField($field) {
34255efc227SAndreas Gohr        if (!isset($this->_info['iptc'])) {
34355efc227SAndreas Gohr            $this->_parseMarkerAdobe();
34455efc227SAndreas Gohr        }
34555efc227SAndreas Gohr
34655efc227SAndreas Gohr        if ($this->_markers == null) {
34755efc227SAndreas Gohr            return false;
34855efc227SAndreas Gohr        }
34955efc227SAndreas Gohr
35055efc227SAndreas Gohr        if (isset($this->_info['iptc'][$field])) {
35155efc227SAndreas Gohr            return $this->_info['iptc'][$field];
35255efc227SAndreas Gohr        }
35355efc227SAndreas Gohr
35455efc227SAndreas Gohr        return false;
35555efc227SAndreas Gohr    }
35655efc227SAndreas Gohr
35755efc227SAndreas Gohr    /**
35855efc227SAndreas Gohr     * Set an EXIF field
35955efc227SAndreas Gohr     *
36055efc227SAndreas Gohr     * @author Sebastian Delmont <sdelmont@zonageek.com>
361b5a81756SJoe Lapp     * @author Joe Lapp <joe.lapp@pobox.com>
36255efc227SAndreas Gohr     */
3630b17fdc6SAndreas Gohr    function setExifField($field, $value) {
36455efc227SAndreas Gohr        if (!isset($this->_info['exif'])) {
36555efc227SAndreas Gohr            $this->_parseMarkerExif();
36655efc227SAndreas Gohr        }
36755efc227SAndreas Gohr
36855efc227SAndreas Gohr        if ($this->_markers == null) {
36955efc227SAndreas Gohr            return false;
37055efc227SAndreas Gohr        }
37155efc227SAndreas Gohr
37255efc227SAndreas Gohr        if ($this->_info['exif'] == false) {
37355efc227SAndreas Gohr            $this->_info['exif'] = array();
37455efc227SAndreas Gohr        }
37555efc227SAndreas Gohr
376b5a81756SJoe Lapp        // make sure datetimes are in correct format
377b5a81756SJoe Lapp        if(strlen($field) >= 8 && strtolower(substr($field, 0, 8)) == 'datetime') {
378b5a81756SJoe Lapp            if(strlen($value) < 8 || $value{4} != ':' || $value{7} != ':') {
379b5a81756SJoe Lapp                $value = date('Y:m:d H:i:s', strtotime($value));
380b5a81756SJoe Lapp            }
381b5a81756SJoe Lapp        }
382b5a81756SJoe Lapp
38355efc227SAndreas Gohr        $this->_info['exif'][$field] = $value;
38455efc227SAndreas Gohr
38555efc227SAndreas Gohr        return true;
38655efc227SAndreas Gohr    }
38755efc227SAndreas Gohr
38855efc227SAndreas Gohr    /**
38955efc227SAndreas Gohr     * Set an Adobe Field
39055efc227SAndreas Gohr     *
39155efc227SAndreas Gohr     * @author Sebastian Delmont <sdelmont@zonageek.com>
39255efc227SAndreas Gohr     */
3930b17fdc6SAndreas Gohr    function setAdobeField($field, $value) {
39455efc227SAndreas Gohr        if (!isset($this->_info['adobe'])) {
39555efc227SAndreas Gohr            $this->_parseMarkerAdobe();
39655efc227SAndreas Gohr        }
39755efc227SAndreas Gohr
39855efc227SAndreas Gohr        if ($this->_markers == null) {
39955efc227SAndreas Gohr            return false;
40055efc227SAndreas Gohr        }
40155efc227SAndreas Gohr
40255efc227SAndreas Gohr        if ($this->_info['adobe'] == false) {
40355efc227SAndreas Gohr            $this->_info['adobe'] = array();
40455efc227SAndreas Gohr        }
40555efc227SAndreas Gohr
40655efc227SAndreas Gohr        $this->_info['adobe'][$field] = $value;
40755efc227SAndreas Gohr
40855efc227SAndreas Gohr        return true;
40955efc227SAndreas Gohr    }
41055efc227SAndreas Gohr
41155efc227SAndreas Gohr    /**
41223a34783SAndreas Gohr     * Calculates the multiplier needed to resize the image to the given
41323a34783SAndreas Gohr     * dimensions
41423a34783SAndreas Gohr     *
41523a34783SAndreas Gohr     * @author Andreas Gohr <andi@splitbrain.org>
41623a34783SAndreas Gohr     */
41723a34783SAndreas Gohr    function getResizeRatio($maxwidth,$maxheight=0){
41823a34783SAndreas Gohr        if(!$maxheight) $maxheight = $maxwidth;
41923a34783SAndreas Gohr
42023a34783SAndreas Gohr        $w = $this->getField('File.Width');
42123a34783SAndreas Gohr        $h = $this->getField('File.Height');
42223a34783SAndreas Gohr
42323a34783SAndreas Gohr        $ratio = 1;
42423a34783SAndreas Gohr        if($w >= $h){
42523a34783SAndreas Gohr            if($w >= $maxwidth){
42623a34783SAndreas Gohr                $ratio = $maxwidth/$w;
42723a34783SAndreas Gohr            }elseif($h > $maxheight){
42823a34783SAndreas Gohr                $ratio = $maxheight/$h;
42923a34783SAndreas Gohr            }
43023a34783SAndreas Gohr        }else{
43123a34783SAndreas Gohr            if($h >= $maxheight){
43223a34783SAndreas Gohr                $ratio = $maxheight/$h;
43323a34783SAndreas Gohr            }elseif($w > $maxwidth){
43423a34783SAndreas Gohr                $ratio = $maxwidth/$w;
43523a34783SAndreas Gohr            }
43623a34783SAndreas Gohr        }
43723a34783SAndreas Gohr        return $ratio;
43823a34783SAndreas Gohr    }
43923a34783SAndreas Gohr
44023a34783SAndreas Gohr
44123a34783SAndreas Gohr    /**
44255efc227SAndreas Gohr     * Set an IPTC field
44355efc227SAndreas Gohr     *
44455efc227SAndreas Gohr     * @author Sebastian Delmont <sdelmont@zonageek.com>
44555efc227SAndreas Gohr     */
4460b17fdc6SAndreas Gohr    function setIPTCField($field, $value) {
44755efc227SAndreas Gohr        if (!isset($this->_info['iptc'])) {
44855efc227SAndreas Gohr            $this->_parseMarkerAdobe();
44955efc227SAndreas Gohr        }
45055efc227SAndreas Gohr
45155efc227SAndreas Gohr        if ($this->_markers == null) {
45255efc227SAndreas Gohr            return false;
45355efc227SAndreas Gohr        }
45455efc227SAndreas Gohr
45555efc227SAndreas Gohr        if ($this->_info['iptc'] == false) {
45655efc227SAndreas Gohr            $this->_info['iptc'] = array();
45755efc227SAndreas Gohr        }
45855efc227SAndreas Gohr
45955efc227SAndreas Gohr        $this->_info['iptc'][$field] = $value;
46055efc227SAndreas Gohr
46155efc227SAndreas Gohr        return true;
46255efc227SAndreas Gohr    }
46355efc227SAndreas Gohr
46455efc227SAndreas Gohr    /**
46555efc227SAndreas Gohr     * Delete an EXIF field
46655efc227SAndreas Gohr     *
46755efc227SAndreas Gohr     * @author Sebastian Delmont <sdelmont@zonageek.com>
46855efc227SAndreas Gohr     */
4690b17fdc6SAndreas Gohr    function deleteExifField($field) {
47055efc227SAndreas Gohr        if (!isset($this->_info['exif'])) {
47155efc227SAndreas Gohr            $this->_parseMarkerAdobe();
47255efc227SAndreas Gohr        }
47355efc227SAndreas Gohr
47455efc227SAndreas Gohr        if ($this->_markers == null) {
47555efc227SAndreas Gohr            return false;
47655efc227SAndreas Gohr        }
47755efc227SAndreas Gohr
47855efc227SAndreas Gohr        if ($this->_info['exif'] != false) {
47955efc227SAndreas Gohr            unset($this->_info['exif'][$field]);
48055efc227SAndreas Gohr        }
48155efc227SAndreas Gohr
48255efc227SAndreas Gohr        return true;
48355efc227SAndreas Gohr    }
48455efc227SAndreas Gohr
48555efc227SAndreas Gohr    /**
48655efc227SAndreas Gohr     * Delete an Adobe field
48755efc227SAndreas Gohr     *
48855efc227SAndreas Gohr     * @author Sebastian Delmont <sdelmont@zonageek.com>
48955efc227SAndreas Gohr     */
4900b17fdc6SAndreas Gohr    function deleteAdobeField($field) {
49155efc227SAndreas Gohr        if (!isset($this->_info['adobe'])) {
49255efc227SAndreas Gohr            $this->_parseMarkerAdobe();
49355efc227SAndreas Gohr        }
49455efc227SAndreas Gohr
49555efc227SAndreas Gohr        if ($this->_markers == null) {
49655efc227SAndreas Gohr            return false;
49755efc227SAndreas Gohr        }
49855efc227SAndreas Gohr
49955efc227SAndreas Gohr        if ($this->_info['adobe'] != false) {
50055efc227SAndreas Gohr            unset($this->_info['adobe'][$field]);
50155efc227SAndreas Gohr        }
50255efc227SAndreas Gohr
50355efc227SAndreas Gohr        return true;
50455efc227SAndreas Gohr    }
50555efc227SAndreas Gohr
50655efc227SAndreas Gohr    /**
50755efc227SAndreas Gohr     * Delete an IPTC field
50855efc227SAndreas Gohr     *
50955efc227SAndreas Gohr     * @author Sebastian Delmont <sdelmont@zonageek.com>
51055efc227SAndreas Gohr     */
5110b17fdc6SAndreas Gohr    function deleteIPTCField($field) {
51255efc227SAndreas Gohr        if (!isset($this->_info['iptc'])) {
51355efc227SAndreas Gohr            $this->_parseMarkerAdobe();
51455efc227SAndreas Gohr        }
51555efc227SAndreas Gohr
51655efc227SAndreas Gohr        if ($this->_markers == null) {
51755efc227SAndreas Gohr            return false;
51855efc227SAndreas Gohr        }
51955efc227SAndreas Gohr
52055efc227SAndreas Gohr        if ($this->_info['iptc'] != false) {
52155efc227SAndreas Gohr            unset($this->_info['iptc'][$field]);
52255efc227SAndreas Gohr        }
52355efc227SAndreas Gohr
52455efc227SAndreas Gohr        return true;
52555efc227SAndreas Gohr    }
52655efc227SAndreas Gohr
52755efc227SAndreas Gohr    /**
52855efc227SAndreas Gohr     * Get the image's title, tries various fields
52955efc227SAndreas Gohr     *
53055efc227SAndreas Gohr     * @param int $max  maximum number chars (keeps words)
53155efc227SAndreas Gohr     * @author Andreas Gohr <andi@splitbrain.org>
53255efc227SAndreas Gohr     */
53355efc227SAndreas Gohr    function getTitle($max=80){
53455efc227SAndreas Gohr        $cap = '';
53555efc227SAndreas Gohr
53655efc227SAndreas Gohr        // try various fields
53755efc227SAndreas Gohr        $cap = $this->getField(array('Iptc.Headline',
53855efc227SAndreas Gohr                    'Iptc.Caption',
539431c7fc8Shakan.sandell                    'Xmp.dc:title',
54055efc227SAndreas Gohr                    'Exif.UserComment',
54155efc227SAndreas Gohr                    'Exif.TIFFUserComment',
5422684e50aSAndreas Gohr                    'Exif.TIFFImageDescription',
5432684e50aSAndreas Gohr                    'File.Name'));
54455efc227SAndreas Gohr        if (empty($cap)) return false;
54555efc227SAndreas Gohr
54655efc227SAndreas Gohr        if(!$max) return $cap;
54755efc227SAndreas Gohr        // Shorten to 80 chars (keeping words)
54855efc227SAndreas Gohr        $new = preg_replace('/\n.+$/','',wordwrap($cap, $max));
54955efc227SAndreas Gohr        if($new != $cap) $new .= '...';
55055efc227SAndreas Gohr
55155efc227SAndreas Gohr        return $new;
55255efc227SAndreas Gohr    }
55355efc227SAndreas Gohr
55455efc227SAndreas Gohr    /**
55555efc227SAndreas Gohr     * Gather various date fields
55655efc227SAndreas Gohr     *
55755efc227SAndreas Gohr     * @author Sebastian Delmont <sdelmont@zonageek.com>
55855efc227SAndreas Gohr     */
5590b17fdc6SAndreas Gohr    function getDates() {
56055efc227SAndreas Gohr        $this->_parseAll();
56155efc227SAndreas Gohr        if ($this->_markers == null) {
562a73b5b7eSAndreas Gohr            if (@isset($this->_info['file']['UnixTime'])) {
563a73b5b7eSAndreas Gohr                $dates['FileModified'] = $this->_info['file']['UnixTime'];
564a73b5b7eSAndreas Gohr                $dates['Time'] = $this->_info['file']['UnixTime'];
565a73b5b7eSAndreas Gohr                $dates['TimeSource'] = 'FileModified';
566a73b5b7eSAndreas Gohr                $dates['TimeStr'] = date("Y-m-d H:i:s", $this->_info['file']['UnixTime']);
567a73b5b7eSAndreas Gohr                $dates['EarliestTime'] = $this->_info['file']['UnixTime'];
568a73b5b7eSAndreas Gohr                $dates['EarliestTimeSource'] = 'FileModified';
569a73b5b7eSAndreas Gohr                $dates['EarliestTimeStr'] = date("Y-m-d H:i:s", $this->_info['file']['UnixTime']);
570a73b5b7eSAndreas Gohr                $dates['LatestTime'] = $this->_info['file']['UnixTime'];
571a73b5b7eSAndreas Gohr                $dates['LatestTimeSource'] = 'FileModified';
572a73b5b7eSAndreas Gohr                $dates['LatestTimeStr'] = date("Y-m-d H:i:s", $this->_info['file']['UnixTime']);
573a73b5b7eSAndreas Gohr                return $dates;
574a73b5b7eSAndreas Gohr            }
57555efc227SAndreas Gohr            return false;
57655efc227SAndreas Gohr        }
57755efc227SAndreas Gohr
57855efc227SAndreas Gohr        $dates = array();
57955efc227SAndreas Gohr
58055efc227SAndreas Gohr        $latestTime = 0;
58155efc227SAndreas Gohr        $latestTimeSource = "";
58255efc227SAndreas Gohr        $earliestTime = time();
58355efc227SAndreas Gohr        $earliestTimeSource = "";
58455efc227SAndreas Gohr
58555efc227SAndreas Gohr        if (@isset($this->_info['exif']['DateTime'])) {
58655efc227SAndreas Gohr            $dates['ExifDateTime'] = $this->_info['exif']['DateTime'];
58755efc227SAndreas Gohr
58855efc227SAndreas Gohr            $aux = $this->_info['exif']['DateTime'];
58955efc227SAndreas Gohr            $aux{4} = "-";
59055efc227SAndreas Gohr            $aux{7} = "-";
59155efc227SAndreas Gohr            $t = strtotime($aux);
59255efc227SAndreas Gohr
5932114dafdSAndreas Gohr            if ($t && $t > $latestTime) {
59455efc227SAndreas Gohr                $latestTime = $t;
59555efc227SAndreas Gohr                $latestTimeSource = "ExifDateTime";
59655efc227SAndreas Gohr            }
59755efc227SAndreas Gohr
5982114dafdSAndreas Gohr            if ($t && $t < $earliestTime) {
59955efc227SAndreas Gohr                $earliestTime = $t;
60055efc227SAndreas Gohr                $earliestTimeSource = "ExifDateTime";
60155efc227SAndreas Gohr            }
60255efc227SAndreas Gohr        }
60355efc227SAndreas Gohr
60455efc227SAndreas Gohr        if (@isset($this->_info['exif']['DateTimeOriginal'])) {
60555efc227SAndreas Gohr            $dates['ExifDateTimeOriginal'] = $this->_info['exif']['DateTime'];
60655efc227SAndreas Gohr
60755efc227SAndreas Gohr            $aux = $this->_info['exif']['DateTimeOriginal'];
60855efc227SAndreas Gohr            $aux{4} = "-";
60955efc227SAndreas Gohr            $aux{7} = "-";
61055efc227SAndreas Gohr            $t = strtotime($aux);
61155efc227SAndreas Gohr
6122114dafdSAndreas Gohr            if ($t && $t > $latestTime) {
61355efc227SAndreas Gohr                $latestTime = $t;
61455efc227SAndreas Gohr                $latestTimeSource = "ExifDateTimeOriginal";
61555efc227SAndreas Gohr            }
61655efc227SAndreas Gohr
6172114dafdSAndreas Gohr            if ($t && $t < $earliestTime) {
61855efc227SAndreas Gohr                $earliestTime = $t;
61955efc227SAndreas Gohr                $earliestTimeSource = "ExifDateTimeOriginal";
62055efc227SAndreas Gohr            }
62155efc227SAndreas Gohr        }
62255efc227SAndreas Gohr
62355efc227SAndreas Gohr        if (@isset($this->_info['exif']['DateTimeDigitized'])) {
62455efc227SAndreas Gohr            $dates['ExifDateTimeDigitized'] = $this->_info['exif']['DateTime'];
62555efc227SAndreas Gohr
62655efc227SAndreas Gohr            $aux = $this->_info['exif']['DateTimeDigitized'];
62755efc227SAndreas Gohr            $aux{4} = "-";
62855efc227SAndreas Gohr            $aux{7} = "-";
62955efc227SAndreas Gohr            $t = strtotime($aux);
63055efc227SAndreas Gohr
6312114dafdSAndreas Gohr            if ($t && $t > $latestTime) {
63255efc227SAndreas Gohr                $latestTime = $t;
63355efc227SAndreas Gohr                $latestTimeSource = "ExifDateTimeDigitized";
63455efc227SAndreas Gohr            }
63555efc227SAndreas Gohr
6362114dafdSAndreas Gohr            if ($t && $t < $earliestTime) {
63755efc227SAndreas Gohr                $earliestTime = $t;
63855efc227SAndreas Gohr                $earliestTimeSource = "ExifDateTimeDigitized";
63955efc227SAndreas Gohr            }
64055efc227SAndreas Gohr        }
64155efc227SAndreas Gohr
64255efc227SAndreas Gohr        if (@isset($this->_info['iptc']['DateCreated'])) {
64355efc227SAndreas Gohr            $dates['IPTCDateCreated'] = $this->_info['iptc']['DateCreated'];
64455efc227SAndreas Gohr
64555efc227SAndreas Gohr            $aux = $this->_info['iptc']['DateCreated'];
64655efc227SAndreas Gohr            $aux = substr($aux, 0, 4) . "-" . substr($aux, 4, 2) . "-" . substr($aux, 6, 2);
64755efc227SAndreas Gohr            $t = strtotime($aux);
64855efc227SAndreas Gohr
6492114dafdSAndreas Gohr            if ($t && $t > $latestTime) {
65055efc227SAndreas Gohr                $latestTime = $t;
65155efc227SAndreas Gohr                $latestTimeSource = "IPTCDateCreated";
65255efc227SAndreas Gohr            }
65355efc227SAndreas Gohr
6542114dafdSAndreas Gohr            if ($t && $t < $earliestTime) {
65555efc227SAndreas Gohr                $earliestTime = $t;
65655efc227SAndreas Gohr                $earliestTimeSource = "IPTCDateCreated";
65755efc227SAndreas Gohr            }
65855efc227SAndreas Gohr        }
65955efc227SAndreas Gohr
66055efc227SAndreas Gohr        if (@isset($this->_info['file']['UnixTime'])) {
66155efc227SAndreas Gohr            $dates['FileModified'] = $this->_info['file']['UnixTime'];
66255efc227SAndreas Gohr
66355efc227SAndreas Gohr            $t = $this->_info['file']['UnixTime'];
66455efc227SAndreas Gohr
6652114dafdSAndreas Gohr            if ($t && $t > $latestTime) {
66655efc227SAndreas Gohr                $latestTime = $t;
66755efc227SAndreas Gohr                $latestTimeSource = "FileModified";
66855efc227SAndreas Gohr            }
66955efc227SAndreas Gohr
6702114dafdSAndreas Gohr            if ($t && $t < $earliestTime) {
67155efc227SAndreas Gohr                $earliestTime = $t;
67255efc227SAndreas Gohr                $earliestTimeSource = "FileModified";
67355efc227SAndreas Gohr            }
67455efc227SAndreas Gohr        }
67555efc227SAndreas Gohr
67655efc227SAndreas Gohr        $dates['Time'] = $earliestTime;
67755efc227SAndreas Gohr        $dates['TimeSource'] = $earliestTimeSource;
67855efc227SAndreas Gohr        $dates['TimeStr'] = date("Y-m-d H:i:s", $earliestTime);
67955efc227SAndreas Gohr        $dates['EarliestTime'] = $earliestTime;
68055efc227SAndreas Gohr        $dates['EarliestTimeSource'] = $earliestTimeSource;
68155efc227SAndreas Gohr        $dates['EarliestTimeStr'] = date("Y-m-d H:i:s", $earliestTime);
68255efc227SAndreas Gohr        $dates['LatestTime'] = $latestTime;
68355efc227SAndreas Gohr        $dates['LatestTimeSource'] = $latestTimeSource;
68455efc227SAndreas Gohr        $dates['LatestTimeStr'] = date("Y-m-d H:i:s", $latestTime);
68555efc227SAndreas Gohr
68655efc227SAndreas Gohr        return $dates;
68755efc227SAndreas Gohr    }
68855efc227SAndreas Gohr
68955efc227SAndreas Gohr    /**
69055efc227SAndreas Gohr     * Get the image width, tries various fields
69155efc227SAndreas Gohr     *
69255efc227SAndreas Gohr     * @author Sebastian Delmont <sdelmont@zonageek.com>
69355efc227SAndreas Gohr     */
6940b17fdc6SAndreas Gohr    function getWidth() {
69555efc227SAndreas Gohr        if (!isset($this->_info['sof'])) {
69655efc227SAndreas Gohr            $this->_parseMarkerSOF();
69755efc227SAndreas Gohr        }
69855efc227SAndreas Gohr
69955efc227SAndreas Gohr        if ($this->_markers == null) {
70055efc227SAndreas Gohr            return false;
70155efc227SAndreas Gohr        }
70255efc227SAndreas Gohr
70355efc227SAndreas Gohr        if (isset($this->_info['sof']['ImageWidth'])) {
70455efc227SAndreas Gohr            return $this->_info['sof']['ImageWidth'];
70555efc227SAndreas Gohr        }
70655efc227SAndreas Gohr
70755efc227SAndreas Gohr        if (!isset($this->_info['exif'])) {
70855efc227SAndreas Gohr            $this->_parseMarkerExif();
70955efc227SAndreas Gohr        }
71055efc227SAndreas Gohr
71155efc227SAndreas Gohr        if (isset($this->_info['exif']['PixelXDimension'])) {
71255efc227SAndreas Gohr            return $this->_info['exif']['PixelXDimension'];
71355efc227SAndreas Gohr        }
71455efc227SAndreas Gohr
71555efc227SAndreas Gohr        return false;
71655efc227SAndreas Gohr    }
71755efc227SAndreas Gohr
71855efc227SAndreas Gohr    /**
71955efc227SAndreas Gohr     * Get the image height, tries various fields
72055efc227SAndreas Gohr     *
72155efc227SAndreas Gohr     * @author Sebastian Delmont <sdelmont@zonageek.com>
72255efc227SAndreas Gohr     */
7230b17fdc6SAndreas Gohr    function getHeight() {
72455efc227SAndreas Gohr        if (!isset($this->_info['sof'])) {
72555efc227SAndreas Gohr            $this->_parseMarkerSOF();
72655efc227SAndreas Gohr        }
72755efc227SAndreas Gohr
72855efc227SAndreas Gohr        if ($this->_markers == null) {
72955efc227SAndreas Gohr            return false;
73055efc227SAndreas Gohr        }
73155efc227SAndreas Gohr
73255efc227SAndreas Gohr        if (isset($this->_info['sof']['ImageHeight'])) {
73355efc227SAndreas Gohr            return $this->_info['sof']['ImageHeight'];
73455efc227SAndreas Gohr        }
73555efc227SAndreas Gohr
73655efc227SAndreas Gohr        if (!isset($this->_info['exif'])) {
73755efc227SAndreas Gohr            $this->_parseMarkerExif();
73855efc227SAndreas Gohr        }
73955efc227SAndreas Gohr
74055efc227SAndreas Gohr        if (isset($this->_info['exif']['PixelYDimension'])) {
74155efc227SAndreas Gohr            return $this->_info['exif']['PixelYDimension'];
74255efc227SAndreas Gohr        }
74355efc227SAndreas Gohr
74455efc227SAndreas Gohr        return false;
74555efc227SAndreas Gohr    }
74655efc227SAndreas Gohr
74755efc227SAndreas Gohr    /**
74855efc227SAndreas Gohr     * Get an dimension string for use in img tag
74955efc227SAndreas Gohr     *
75055efc227SAndreas Gohr     * @author Sebastian Delmont <sdelmont@zonageek.com>
75155efc227SAndreas Gohr     */
7520b17fdc6SAndreas Gohr    function getDimStr() {
75355efc227SAndreas Gohr        if ($this->_markers == null) {
75455efc227SAndreas Gohr            return false;
75555efc227SAndreas Gohr        }
75655efc227SAndreas Gohr
75755efc227SAndreas Gohr        $w = $this->getWidth();
75855efc227SAndreas Gohr        $h = $this->getHeight();
75955efc227SAndreas Gohr
76055efc227SAndreas Gohr        return "width='" . $w . "' height='" . $h . "'";
76155efc227SAndreas Gohr    }
76255efc227SAndreas Gohr
76355efc227SAndreas Gohr    /**
76455efc227SAndreas Gohr     * Checks for an embedded thumbnail
76555efc227SAndreas Gohr     *
76655efc227SAndreas Gohr     * @author Sebastian Delmont <sdelmont@zonageek.com>
76755efc227SAndreas Gohr     */
7680b17fdc6SAndreas Gohr    function hasThumbnail($which = 'any') {
76955efc227SAndreas Gohr        if (($which == 'any') || ($which == 'exif')) {
77055efc227SAndreas Gohr            if (!isset($this->_info['exif'])) {
77155efc227SAndreas Gohr                $this->_parseMarkerExif();
77255efc227SAndreas Gohr            }
77355efc227SAndreas Gohr
77455efc227SAndreas Gohr            if ($this->_markers == null) {
77555efc227SAndreas Gohr                return false;
77655efc227SAndreas Gohr            }
77755efc227SAndreas Gohr
77855efc227SAndreas Gohr            if (isset($this->_info['exif']) && is_array($this->_info['exif'])) {
77955efc227SAndreas Gohr                if (isset($this->_info['exif']['JFIFThumbnail'])) {
78055efc227SAndreas Gohr                    return 'exif';
78155efc227SAndreas Gohr                }
78255efc227SAndreas Gohr            }
78355efc227SAndreas Gohr        }
78455efc227SAndreas Gohr
78555efc227SAndreas Gohr        if ($which == 'adobe') {
78655efc227SAndreas Gohr            if (!isset($this->_info['adobe'])) {
78755efc227SAndreas Gohr                $this->_parseMarkerAdobe();
78855efc227SAndreas Gohr            }
78955efc227SAndreas Gohr
79055efc227SAndreas Gohr            if ($this->_markers == null) {
79155efc227SAndreas Gohr                return false;
79255efc227SAndreas Gohr            }
79355efc227SAndreas Gohr
79455efc227SAndreas Gohr            if (isset($this->_info['adobe']) && is_array($this->_info['adobe'])) {
79555efc227SAndreas Gohr                if (isset($this->_info['adobe']['ThumbnailData'])) {
79655efc227SAndreas Gohr                    return 'exif';
79755efc227SAndreas Gohr                }
79855efc227SAndreas Gohr            }
79955efc227SAndreas Gohr        }
80055efc227SAndreas Gohr
80155efc227SAndreas Gohr        return false;
80255efc227SAndreas Gohr    }
80355efc227SAndreas Gohr
80455efc227SAndreas Gohr    /**
80555efc227SAndreas Gohr     * Send embedded thumbnail to browser
80655efc227SAndreas Gohr     *
80755efc227SAndreas Gohr     * @author Sebastian Delmont <sdelmont@zonageek.com>
80855efc227SAndreas Gohr     */
8090b17fdc6SAndreas Gohr    function sendThumbnail($which = 'any') {
81055efc227SAndreas Gohr        $data = null;
81155efc227SAndreas Gohr
81255efc227SAndreas Gohr        if (($which == 'any') || ($which == 'exif')) {
81355efc227SAndreas Gohr            if (!isset($this->_info['exif'])) {
81455efc227SAndreas Gohr                $this->_parseMarkerExif();
81555efc227SAndreas Gohr            }
81655efc227SAndreas Gohr
81755efc227SAndreas Gohr            if ($this->_markers == null) {
81855efc227SAndreas Gohr                return false;
81955efc227SAndreas Gohr            }
82055efc227SAndreas Gohr
82155efc227SAndreas Gohr            if (isset($this->_info['exif']) && is_array($this->_info['exif'])) {
82255efc227SAndreas Gohr                if (isset($this->_info['exif']['JFIFThumbnail'])) {
82355efc227SAndreas Gohr                    $data =& $this->_info['exif']['JFIFThumbnail'];
82455efc227SAndreas Gohr                }
82555efc227SAndreas Gohr            }
82655efc227SAndreas Gohr        }
82755efc227SAndreas Gohr
82855efc227SAndreas Gohr        if (($which == 'adobe') || ($data == null)){
82955efc227SAndreas Gohr            if (!isset($this->_info['adobe'])) {
83055efc227SAndreas Gohr                $this->_parseMarkerAdobe();
83155efc227SAndreas Gohr            }
83255efc227SAndreas Gohr
83355efc227SAndreas Gohr            if ($this->_markers == null) {
83455efc227SAndreas Gohr                return false;
83555efc227SAndreas Gohr            }
83655efc227SAndreas Gohr
83755efc227SAndreas Gohr            if (isset($this->_info['adobe']) && is_array($this->_info['adobe'])) {
83855efc227SAndreas Gohr                if (isset($this->_info['adobe']['ThumbnailData'])) {
83955efc227SAndreas Gohr                    $data =& $this->_info['adobe']['ThumbnailData'];
84055efc227SAndreas Gohr                }
84155efc227SAndreas Gohr            }
84255efc227SAndreas Gohr        }
84355efc227SAndreas Gohr
84455efc227SAndreas Gohr        if ($data != null) {
84555efc227SAndreas Gohr            header("Content-type: image/jpeg");
84655efc227SAndreas Gohr            echo $data;
84755efc227SAndreas Gohr            return true;
84855efc227SAndreas Gohr        }
84955efc227SAndreas Gohr
85055efc227SAndreas Gohr        return false;
85155efc227SAndreas Gohr    }
85255efc227SAndreas Gohr
85355efc227SAndreas Gohr    /**
85455efc227SAndreas Gohr     * Save changed Metadata
85555efc227SAndreas Gohr     *
85655efc227SAndreas Gohr     * @author Sebastian Delmont <sdelmont@zonageek.com>
85736df6fa3SAndreas Gohr     * @author Andreas Gohr <andi@splitbrain.org>
85855efc227SAndreas Gohr     */
85955efc227SAndreas Gohr    function save($fileName = "") {
86055efc227SAndreas Gohr        if ($fileName == "") {
86136df6fa3SAndreas Gohr            $tmpName = tempnam(dirname($this->_fileName),'_metatemp_');
86255efc227SAndreas Gohr            $this->_writeJPEG($tmpName);
86336df6fa3SAndreas Gohr            if (@file_exists($tmpName)) {
864bf5e5a5bSAndreas Gohr                return io_rename($tmpName, $this->_fileName);
86555efc227SAndreas Gohr            }
86636df6fa3SAndreas Gohr        } else {
86736df6fa3SAndreas Gohr            return $this->_writeJPEG($fileName);
86855efc227SAndreas Gohr        }
86936df6fa3SAndreas Gohr        return false;
87055efc227SAndreas Gohr    }
87155efc227SAndreas Gohr
87255efc227SAndreas Gohr    /*************************************************************/
87355efc227SAndreas Gohr    /* PRIVATE FUNCTIONS (Internal Use Only!)                    */
87455efc227SAndreas Gohr    /*************************************************************/
87555efc227SAndreas Gohr
87655efc227SAndreas Gohr    /*************************************************************/
8770b17fdc6SAndreas Gohr    function _dispose() {
87855efc227SAndreas Gohr        $this->_fileName = $fileName;
87955efc227SAndreas Gohr
88055efc227SAndreas Gohr        $this->_fp = null;
88155efc227SAndreas Gohr        $this->_type = 'unknown';
88255efc227SAndreas Gohr
88355efc227SAndreas Gohr        unset($this->_markers);
88455efc227SAndreas Gohr        unset($this->_info);
88555efc227SAndreas Gohr    }
88655efc227SAndreas Gohr
88755efc227SAndreas Gohr    /*************************************************************/
8880b17fdc6SAndreas Gohr    function _readJPEG() {
88955efc227SAndreas Gohr        unset($this->_markers);
890a73b5b7eSAndreas Gohr        //unset($this->_info);
89155efc227SAndreas Gohr        $this->_markers = array();
892a73b5b7eSAndreas Gohr        //$this->_info = array();
89355efc227SAndreas Gohr
89455efc227SAndreas Gohr        $this->_fp = @fopen($this->_fileName, 'rb');
89555efc227SAndreas Gohr        if ($this->_fp) {
89655efc227SAndreas Gohr            if (file_exists($this->_fileName)) {
89755efc227SAndreas Gohr                $this->_type = 'file';
89855efc227SAndreas Gohr            }
89955efc227SAndreas Gohr            else {
90055efc227SAndreas Gohr                $this->_type = 'url';
90155efc227SAndreas Gohr            }
9020b17fdc6SAndreas Gohr        } else {
90355efc227SAndreas Gohr            $this->_fp = null;
90455efc227SAndreas Gohr            return false;  // ERROR: Can't open file
90555efc227SAndreas Gohr        }
90655efc227SAndreas Gohr
90755efc227SAndreas Gohr        // Check for the JPEG signature
90855efc227SAndreas Gohr        $c1 = ord(fgetc($this->_fp));
90955efc227SAndreas Gohr        $c2 = ord(fgetc($this->_fp));
91055efc227SAndreas Gohr
91155efc227SAndreas Gohr        if ($c1 != 0xFF || $c2 != 0xD8) {   // (0xFF + SOI)
91255efc227SAndreas Gohr            $this->_markers = null;
91355efc227SAndreas Gohr            return false;  // ERROR: File is not a JPEG
91455efc227SAndreas Gohr        }
91555efc227SAndreas Gohr
91655efc227SAndreas Gohr        $count = 0;
91755efc227SAndreas Gohr
91855efc227SAndreas Gohr        $done = false;
91955efc227SAndreas Gohr        $ok = true;
92055efc227SAndreas Gohr
92155efc227SAndreas Gohr        while (!$done) {
92255efc227SAndreas Gohr            $capture = false;
92355efc227SAndreas Gohr
92455efc227SAndreas Gohr            // First, skip any non 0xFF bytes
92555efc227SAndreas Gohr            $discarded = 0;
92655efc227SAndreas Gohr            $c = ord(fgetc($this->_fp));
92755efc227SAndreas Gohr            while (!feof($this->_fp) && ($c != 0xFF)) {
92855efc227SAndreas Gohr                $discarded++;
92955efc227SAndreas Gohr                $c = ord(fgetc($this->_fp));
93055efc227SAndreas Gohr            }
93155efc227SAndreas Gohr            // Then skip all 0xFF until the marker byte
93255efc227SAndreas Gohr            do {
93355efc227SAndreas Gohr                $marker = ord(fgetc($this->_fp));
93455efc227SAndreas Gohr            } while (!feof($this->_fp) && ($marker == 0xFF));
93555efc227SAndreas Gohr
93655efc227SAndreas Gohr            if (feof($this->_fp)) {
93755efc227SAndreas Gohr                return false; // ERROR: Unexpected EOF
93855efc227SAndreas Gohr            }
93955efc227SAndreas Gohr            if ($discarded != 0) {
94055efc227SAndreas Gohr                return false; // ERROR: Extraneous data
94155efc227SAndreas Gohr            }
94255efc227SAndreas Gohr
94355efc227SAndreas Gohr            $length = ord(fgetc($this->_fp)) * 256 + ord(fgetc($this->_fp));
94455efc227SAndreas Gohr            if (feof($this->_fp)) {
94555efc227SAndreas Gohr                return false; // ERROR: Unexpected EOF
94655efc227SAndreas Gohr            }
94755efc227SAndreas Gohr            if ($length < 2) {
94855efc227SAndreas Gohr                return false; // ERROR: Extraneous data
94955efc227SAndreas Gohr            }
95055efc227SAndreas Gohr            $length = $length - 2; // The length we got counts itself
95155efc227SAndreas Gohr
95255efc227SAndreas Gohr            switch ($marker) {
95355efc227SAndreas Gohr                case 0xC0:    // SOF0
95455efc227SAndreas Gohr                case 0xC1:    // SOF1
95555efc227SAndreas Gohr                case 0xC2:    // SOF2
95655efc227SAndreas Gohr                case 0xC9:    // SOF9
95755efc227SAndreas Gohr                case 0xE0:    // APP0: JFIF data
958431c7fc8Shakan.sandell                case 0xE1:    // APP1: EXIF or XMP data
95955efc227SAndreas Gohr                case 0xED:    // APP13: IPTC / Photoshop data
96055efc227SAndreas Gohr                    $capture = true;
96155efc227SAndreas Gohr                    break;
96255efc227SAndreas Gohr                case 0xDA:    // SOS: Start of scan... the image itself and the last block on the file
96355efc227SAndreas Gohr                    $capture = false;
96455efc227SAndreas Gohr                    $length = -1;  // This field has no length... it includes all data until EOF
96555efc227SAndreas Gohr                    $done = true;
96655efc227SAndreas Gohr                    break;
96755efc227SAndreas Gohr                default:
96855efc227SAndreas Gohr                    $capture = true;//false;
96955efc227SAndreas Gohr                    break;
97055efc227SAndreas Gohr            }
97155efc227SAndreas Gohr
97255efc227SAndreas Gohr            $this->_markers[$count] = array();
97355efc227SAndreas Gohr            $this->_markers[$count]['marker'] = $marker;
97455efc227SAndreas Gohr            $this->_markers[$count]['length'] = $length;
97555efc227SAndreas Gohr
97655efc227SAndreas Gohr            if ($capture) {
977ed3655c4STom N Harris                if ($length)
97855efc227SAndreas Gohr                    $this->_markers[$count]['data'] =& fread($this->_fp, $length);
979ed3655c4STom N Harris                else
980ed3655c4STom N Harris                    $this->_markers[$count]['data'] = "";
98155efc227SAndreas Gohr            }
98255efc227SAndreas Gohr            elseif (!$done) {
98355efc227SAndreas Gohr                $result = @fseek($this->_fp, $length, SEEK_CUR);
98455efc227SAndreas Gohr                // fseek doesn't seem to like HTTP 'files', but fgetc has no problem
98555efc227SAndreas Gohr                if (!($result === 0)) {
98655efc227SAndreas Gohr                    for ($i = 0; $i < $length; $i++) {
98755efc227SAndreas Gohr                        fgetc($this->_fp);
98855efc227SAndreas Gohr                    }
98955efc227SAndreas Gohr                }
99055efc227SAndreas Gohr            }
99155efc227SAndreas Gohr            $count++;
99255efc227SAndreas Gohr        }
99355efc227SAndreas Gohr
99455efc227SAndreas Gohr        if ($this->_fp) {
99555efc227SAndreas Gohr            fclose($this->_fp);
99655efc227SAndreas Gohr            $this->_fp = null;
99755efc227SAndreas Gohr        }
99855efc227SAndreas Gohr
99955efc227SAndreas Gohr        return $ok;
100055efc227SAndreas Gohr    }
100155efc227SAndreas Gohr
100255efc227SAndreas Gohr    /*************************************************************/
10030b17fdc6SAndreas Gohr    function _parseAll() {
10041017ae2eSAndreas Gohr        if (!isset($this->_info['file'])) {
10051017ae2eSAndreas Gohr            $this->_parseFileInfo();
10061017ae2eSAndreas Gohr        }
100755efc227SAndreas Gohr        if (!isset($this->_markers)) {
100855efc227SAndreas Gohr            $this->_readJPEG();
100955efc227SAndreas Gohr        }
101055efc227SAndreas Gohr
101155efc227SAndreas Gohr        if ($this->_markers == null) {
101255efc227SAndreas Gohr            return false;
101355efc227SAndreas Gohr        }
101455efc227SAndreas Gohr
101555efc227SAndreas Gohr        if (!isset($this->_info['jfif'])) {
101655efc227SAndreas Gohr            $this->_parseMarkerJFIF();
101755efc227SAndreas Gohr        }
101855efc227SAndreas Gohr        if (!isset($this->_info['jpeg'])) {
101955efc227SAndreas Gohr            $this->_parseMarkerSOF();
102055efc227SAndreas Gohr        }
102155efc227SAndreas Gohr        if (!isset($this->_info['exif'])) {
102255efc227SAndreas Gohr            $this->_parseMarkerExif();
102355efc227SAndreas Gohr        }
1024431c7fc8Shakan.sandell        if (!isset($this->_info['xmp'])) {
1025431c7fc8Shakan.sandell            $this->_parseMarkerXmp();
1026431c7fc8Shakan.sandell        }
102755efc227SAndreas Gohr        if (!isset($this->_info['adobe'])) {
102855efc227SAndreas Gohr            $this->_parseMarkerAdobe();
102955efc227SAndreas Gohr        }
103055efc227SAndreas Gohr    }
103155efc227SAndreas Gohr
103255efc227SAndreas Gohr    /*************************************************************/
10330b17fdc6SAndreas Gohr    function _writeJPEG($outputName) {
103455efc227SAndreas Gohr        $this->_parseAll();
103555efc227SAndreas Gohr
103655efc227SAndreas Gohr        $wroteEXIF = false;
103755efc227SAndreas Gohr        $wroteAdobe = false;
103855efc227SAndreas Gohr
103955efc227SAndreas Gohr        $this->_fp = @fopen($this->_fileName, 'r');
104055efc227SAndreas Gohr        if ($this->_fp) {
104155efc227SAndreas Gohr            if (file_exists($this->_fileName)) {
104255efc227SAndreas Gohr                $this->_type = 'file';
104355efc227SAndreas Gohr            }
104455efc227SAndreas Gohr            else {
104555efc227SAndreas Gohr                $this->_type = 'url';
104655efc227SAndreas Gohr            }
10470b17fdc6SAndreas Gohr        } else {
104855efc227SAndreas Gohr            $this->_fp = null;
104955efc227SAndreas Gohr            return false;  // ERROR: Can't open file
105055efc227SAndreas Gohr        }
105155efc227SAndreas Gohr
105255efc227SAndreas Gohr        $this->_fpout = fopen($outputName, 'wb');
10530b17fdc6SAndreas Gohr        if (!$this->_fpout) {
105455efc227SAndreas Gohr            $this->_fpout = null;
105555efc227SAndreas Gohr            fclose($this->_fp);
105655efc227SAndreas Gohr            $this->_fp = null;
105755efc227SAndreas Gohr            return false;  // ERROR: Can't open output file
105855efc227SAndreas Gohr        }
105955efc227SAndreas Gohr
106055efc227SAndreas Gohr        // Check for the JPEG signature
106155efc227SAndreas Gohr        $c1 = ord(fgetc($this->_fp));
106255efc227SAndreas Gohr        $c2 = ord(fgetc($this->_fp));
106355efc227SAndreas Gohr
106455efc227SAndreas Gohr        if ($c1 != 0xFF || $c2 != 0xD8) {   // (0xFF + SOI)
106555efc227SAndreas Gohr            return false;  // ERROR: File is not a JPEG
106655efc227SAndreas Gohr        }
106755efc227SAndreas Gohr
106855efc227SAndreas Gohr        fputs($this->_fpout, chr(0xFF), 1);
106955efc227SAndreas Gohr        fputs($this->_fpout, chr(0xD8), 1); // (0xFF + SOI)
107055efc227SAndreas Gohr
107155efc227SAndreas Gohr        $count = 0;
107255efc227SAndreas Gohr
107355efc227SAndreas Gohr        $done = false;
107455efc227SAndreas Gohr        $ok = true;
107555efc227SAndreas Gohr
107655efc227SAndreas Gohr        while (!$done) {
107755efc227SAndreas Gohr            // First, skip any non 0xFF bytes
107855efc227SAndreas Gohr            $discarded = 0;
107955efc227SAndreas Gohr            $c = ord(fgetc($this->_fp));
108055efc227SAndreas Gohr            while (!feof($this->_fp) && ($c != 0xFF)) {
108155efc227SAndreas Gohr                $discarded++;
108255efc227SAndreas Gohr                $c = ord(fgetc($this->_fp));
108355efc227SAndreas Gohr            }
108455efc227SAndreas Gohr            // Then skip all 0xFF until the marker byte
108555efc227SAndreas Gohr            do {
108655efc227SAndreas Gohr                $marker = ord(fgetc($this->_fp));
108755efc227SAndreas Gohr            } while (!feof($this->_fp) && ($marker == 0xFF));
108855efc227SAndreas Gohr
108955efc227SAndreas Gohr            if (feof($this->_fp)) {
109055efc227SAndreas Gohr                $ok = false;
109155efc227SAndreas Gohr                break; // ERROR: Unexpected EOF
109255efc227SAndreas Gohr            }
109355efc227SAndreas Gohr            if ($discarded != 0) {
109455efc227SAndreas Gohr                $ok = false;
109555efc227SAndreas Gohr                break; // ERROR: Extraneous data
109655efc227SAndreas Gohr            }
109755efc227SAndreas Gohr
109855efc227SAndreas Gohr            $length = ord(fgetc($this->_fp)) * 256 + ord(fgetc($this->_fp));
109955efc227SAndreas Gohr            if (feof($this->_fp)) {
110055efc227SAndreas Gohr                $ok = false;
110155efc227SAndreas Gohr                break; // ERROR: Unexpected EOF
110255efc227SAndreas Gohr            }
110355efc227SAndreas Gohr            if ($length < 2) {
110455efc227SAndreas Gohr                $ok = false;
110555efc227SAndreas Gohr                break; // ERROR: Extraneous data
110655efc227SAndreas Gohr            }
110755efc227SAndreas Gohr            $length = $length - 2; // The length we got counts itself
110855efc227SAndreas Gohr
110955efc227SAndreas Gohr            unset($data);
111055efc227SAndreas Gohr            if ($marker == 0xE1) { // APP1: EXIF data
111155efc227SAndreas Gohr                $data =& $this->_createMarkerEXIF();
111255efc227SAndreas Gohr                $wroteEXIF = true;
111355efc227SAndreas Gohr            }
111455efc227SAndreas Gohr            elseif ($marker == 0xED) { // APP13: IPTC / Photoshop data
111555efc227SAndreas Gohr                $data =& $this->_createMarkerAdobe();
111655efc227SAndreas Gohr                $wroteAdobe = true;
111755efc227SAndreas Gohr            }
111855efc227SAndreas Gohr            elseif ($marker == 0xDA) { // SOS: Start of scan... the image itself and the last block on the file
111955efc227SAndreas Gohr                $done = true;
112055efc227SAndreas Gohr            }
112155efc227SAndreas Gohr
112255efc227SAndreas Gohr            if (!$wroteEXIF && (($marker < 0xE0) || ($marker > 0xEF))) {
112355efc227SAndreas Gohr                if (isset($this->_info['exif']) && is_array($this->_info['exif'])) {
112455efc227SAndreas Gohr                    $exif =& $this->_createMarkerEXIF();
112555efc227SAndreas Gohr                    $this->_writeJPEGMarker(0xE1, strlen($exif), $exif, 0);
112655efc227SAndreas Gohr                    unset($exif);
112755efc227SAndreas Gohr                }
112855efc227SAndreas Gohr                $wroteEXIF = true;
112955efc227SAndreas Gohr            }
113055efc227SAndreas Gohr
113155efc227SAndreas Gohr            if (!$wroteAdobe && (($marker < 0xE0) || ($marker > 0xEF))) {
113255efc227SAndreas Gohr                if ((isset($this->_info['adobe']) && is_array($this->_info['adobe']))
113355efc227SAndreas Gohr                        || (isset($this->_info['iptc']) && is_array($this->_info['iptc']))) {
113455efc227SAndreas Gohr                    $adobe =& $this->_createMarkerAdobe();
113555efc227SAndreas Gohr                    $this->_writeJPEGMarker(0xED, strlen($adobe), $adobe, 0);
113655efc227SAndreas Gohr                    unset($adobe);
113755efc227SAndreas Gohr                }
113855efc227SAndreas Gohr                $wroteAdobe = true;
113955efc227SAndreas Gohr            }
114055efc227SAndreas Gohr
114155efc227SAndreas Gohr            $origLength = $length;
114255efc227SAndreas Gohr            if (isset($data)) {
114355efc227SAndreas Gohr                $length = strlen($data);
114455efc227SAndreas Gohr            }
114555efc227SAndreas Gohr
114655efc227SAndreas Gohr            if ($marker != -1) {
114755efc227SAndreas Gohr                $this->_writeJPEGMarker($marker, $length, $data, $origLength);
114855efc227SAndreas Gohr            }
114955efc227SAndreas Gohr        }
115055efc227SAndreas Gohr
115155efc227SAndreas Gohr        if ($this->_fp) {
115255efc227SAndreas Gohr            fclose($this->_fp);
115355efc227SAndreas Gohr            $this->_fp = null;
115455efc227SAndreas Gohr        }
115555efc227SAndreas Gohr
115655efc227SAndreas Gohr        if ($this->_fpout) {
115755efc227SAndreas Gohr            fclose($this->_fpout);
115855efc227SAndreas Gohr            $this->_fpout = null;
115955efc227SAndreas Gohr        }
116055efc227SAndreas Gohr
116155efc227SAndreas Gohr        return $ok;
116255efc227SAndreas Gohr    }
116355efc227SAndreas Gohr
116455efc227SAndreas Gohr    /*************************************************************/
11650b17fdc6SAndreas Gohr    function _writeJPEGMarker($marker, $length, &$data, $origLength) {
116655efc227SAndreas Gohr        if ($length <= 0) {
116755efc227SAndreas Gohr            return false;
116855efc227SAndreas Gohr        }
116955efc227SAndreas Gohr
117055efc227SAndreas Gohr        fputs($this->_fpout, chr(0xFF), 1);
117155efc227SAndreas Gohr        fputs($this->_fpout, chr($marker), 1);
117255efc227SAndreas Gohr        fputs($this->_fpout, chr((($length + 2) & 0x0000FF00) >> 8), 1);
117355efc227SAndreas Gohr        fputs($this->_fpout, chr((($length + 2) & 0x000000FF) >> 0), 1);
117455efc227SAndreas Gohr
117555efc227SAndreas Gohr        if (isset($data)) {
117655efc227SAndreas Gohr            // Copy the generated data
117755efc227SAndreas Gohr            fputs($this->_fpout, $data, $length);
117855efc227SAndreas Gohr
117955efc227SAndreas Gohr            if ($origLength > 0) {   // Skip the original data
118055efc227SAndreas Gohr                $result = @fseek($this->_fp, $origLength, SEEK_CUR);
118155efc227SAndreas Gohr                // fseek doesn't seem to like HTTP 'files', but fgetc has no problem
118255efc227SAndreas Gohr                if ($result != 0) {
118355efc227SAndreas Gohr                    for ($i = 0; $i < $origLength; $i++) {
118455efc227SAndreas Gohr                        fgetc($this->_fp);
118555efc227SAndreas Gohr                    }
118655efc227SAndreas Gohr                }
118755efc227SAndreas Gohr            }
11880b17fdc6SAndreas Gohr        } else {
118955efc227SAndreas Gohr            if ($marker == 0xDA) {  // Copy until EOF
119055efc227SAndreas Gohr                while (!feof($this->_fp)) {
1191ed3655c4STom N Harris                    $data = fread($this->_fp, 1024 * 16);
119255efc227SAndreas Gohr                    fputs($this->_fpout, $data, strlen($data));
119355efc227SAndreas Gohr                }
11940b17fdc6SAndreas Gohr            } else { // Copy only $length bytes
1195ed3655c4STom N Harris                $data = @fread($this->_fp, $length);
119655efc227SAndreas Gohr                fputs($this->_fpout, $data, $length);
119755efc227SAndreas Gohr            }
119855efc227SAndreas Gohr        }
119955efc227SAndreas Gohr
120055efc227SAndreas Gohr        return true;
120155efc227SAndreas Gohr    }
120255efc227SAndreas Gohr
120323a34783SAndreas Gohr    /**
120423a34783SAndreas Gohr     * Gets basic info from the file - should work with non-JPEGs
120523a34783SAndreas Gohr     *
120623a34783SAndreas Gohr     * @author  Sebastian Delmont <sdelmont@zonageek.com>
120723a34783SAndreas Gohr     * @author  Andreas Gohr <andi@splitbrain.org>
120823a34783SAndreas Gohr     */
12090b17fdc6SAndreas Gohr    function _parseFileInfo() {
1210639f8f43SAndreas Gohr        if (file_exists($this->_fileName) && is_file($this->_fileName)) {
121155efc227SAndreas Gohr            $this->_info['file'] = array();
121255efc227SAndreas Gohr            $this->_info['file']['Name'] = basename($this->_fileName);
121300976812SAndreas Gohr            $this->_info['file']['Path'] = fullpath($this->_fileName);
121455efc227SAndreas Gohr            $this->_info['file']['Size'] = filesize($this->_fileName);
121555efc227SAndreas Gohr            if ($this->_info['file']['Size'] < 1024) {
121655efc227SAndreas Gohr                $this->_info['file']['NiceSize'] = $this->_info['file']['Size'] . 'B';
12170b17fdc6SAndreas Gohr            } elseif ($this->_info['file']['Size'] < (1024 * 1024)) {
121855efc227SAndreas Gohr                $this->_info['file']['NiceSize'] = round($this->_info['file']['Size'] / 1024) . 'KB';
12190b17fdc6SAndreas Gohr            } elseif ($this->_info['file']['Size'] < (1024 * 1024 * 1024)) {
1220fe00a666SAndreas Gohr                $this->_info['file']['NiceSize'] = round($this->_info['file']['Size'] / (1024*1024)) . 'MB';
12210b17fdc6SAndreas Gohr            } else {
122255efc227SAndreas Gohr                $this->_info['file']['NiceSize'] = $this->_info['file']['Size'] . 'B';
122355efc227SAndreas Gohr            }
122455efc227SAndreas Gohr            $this->_info['file']['UnixTime'] = filemtime($this->_fileName);
122555efc227SAndreas Gohr
122655efc227SAndreas Gohr            // get image size directly from file
122755efc227SAndreas Gohr            $size = getimagesize($this->_fileName);
122855efc227SAndreas Gohr            $this->_info['file']['Width']  = $size[0];
122955efc227SAndreas Gohr            $this->_info['file']['Height'] = $size[1];
123055efc227SAndreas Gohr            // set mime types and formats
123155efc227SAndreas Gohr            // http://www.php.net/manual/en/function.getimagesize.php
123255efc227SAndreas Gohr            // http://www.php.net/manual/en/function.image-type-to-mime-type.php
123355efc227SAndreas Gohr            switch ($size[2]){
123455efc227SAndreas Gohr                case 1:
123555efc227SAndreas Gohr                    $this->_info['file']['Mime']   = 'image/gif';
123655efc227SAndreas Gohr                    $this->_info['file']['Format'] = 'GIF';
123755efc227SAndreas Gohr                    break;
123855efc227SAndreas Gohr                case 2:
123955efc227SAndreas Gohr                    $this->_info['file']['Mime']   = 'image/jpeg';
124055efc227SAndreas Gohr                    $this->_info['file']['Format'] = 'JPEG';
124155efc227SAndreas Gohr                    break;
124255efc227SAndreas Gohr                case 3:
124355efc227SAndreas Gohr                    $this->_info['file']['Mime']   = 'image/png';
124455efc227SAndreas Gohr                    $this->_info['file']['Format'] = 'PNG';
124555efc227SAndreas Gohr                    break;
124655efc227SAndreas Gohr                case 4:
124755efc227SAndreas Gohr                    $this->_info['file']['Mime']   = 'application/x-shockwave-flash';
124855efc227SAndreas Gohr                    $this->_info['file']['Format'] = 'SWF';
124955efc227SAndreas Gohr                    break;
125055efc227SAndreas Gohr                case 5:
125155efc227SAndreas Gohr                    $this->_info['file']['Mime']   = 'image/psd';
125255efc227SAndreas Gohr                    $this->_info['file']['Format'] = 'PSD';
125355efc227SAndreas Gohr                    break;
125455efc227SAndreas Gohr                case 6:
125555efc227SAndreas Gohr                    $this->_info['file']['Mime']   = 'image/bmp';
125655efc227SAndreas Gohr                    $this->_info['file']['Format'] = 'BMP';
125755efc227SAndreas Gohr                    break;
125855efc227SAndreas Gohr                case 7:
125955efc227SAndreas Gohr                    $this->_info['file']['Mime']   = 'image/tiff';
126055efc227SAndreas Gohr                    $this->_info['file']['Format'] = 'TIFF (Intel)';
126155efc227SAndreas Gohr                    break;
126255efc227SAndreas Gohr                case 8:
126355efc227SAndreas Gohr                    $this->_info['file']['Mime']   = 'image/tiff';
126455efc227SAndreas Gohr                    $this->_info['file']['Format'] = 'TIFF (Motorola)';
126555efc227SAndreas Gohr                    break;
126655efc227SAndreas Gohr                case 9:
126755efc227SAndreas Gohr                    $this->_info['file']['Mime']   = 'application/octet-stream';
126855efc227SAndreas Gohr                    $this->_info['file']['Format'] = 'JPC';
126955efc227SAndreas Gohr                    break;
127055efc227SAndreas Gohr                case 10:
127155efc227SAndreas Gohr                    $this->_info['file']['Mime']   = 'image/jp2';
127255efc227SAndreas Gohr                    $this->_info['file']['Format'] = 'JP2';
127355efc227SAndreas Gohr                    break;
127455efc227SAndreas Gohr                case 11:
127555efc227SAndreas Gohr                    $this->_info['file']['Mime']   = 'application/octet-stream';
127655efc227SAndreas Gohr                    $this->_info['file']['Format'] = 'JPX';
127755efc227SAndreas Gohr                    break;
127855efc227SAndreas Gohr                case 12:
127955efc227SAndreas Gohr                    $this->_info['file']['Mime']   = 'application/octet-stream';
128055efc227SAndreas Gohr                    $this->_info['file']['Format'] = 'JB2';
128155efc227SAndreas Gohr                    break;
128255efc227SAndreas Gohr                case 13:
128355efc227SAndreas Gohr                    $this->_info['file']['Mime']   = 'application/x-shockwave-flash';
128455efc227SAndreas Gohr                    $this->_info['file']['Format'] = 'SWC';
128555efc227SAndreas Gohr                    break;
128655efc227SAndreas Gohr                case 14:
128755efc227SAndreas Gohr                    $this->_info['file']['Mime']   = 'image/iff';
128855efc227SAndreas Gohr                    $this->_info['file']['Format'] = 'IFF';
128955efc227SAndreas Gohr                    break;
129055efc227SAndreas Gohr                case 15:
129155efc227SAndreas Gohr                    $this->_info['file']['Mime']   = 'image/vnd.wap.wbmp';
129255efc227SAndreas Gohr                    $this->_info['file']['Format'] = 'WBMP';
129355efc227SAndreas Gohr                    break;
129455efc227SAndreas Gohr                case 16:
129555efc227SAndreas Gohr                    $this->_info['file']['Mime']   = 'image/xbm';
129655efc227SAndreas Gohr                    $this->_info['file']['Format'] = 'XBM';
129755efc227SAndreas Gohr                    break;
129855efc227SAndreas Gohr                default:
129955efc227SAndreas Gohr                    $this->_info['file']['Mime']   = 'image/unknown';
130055efc227SAndreas Gohr            }
13010b17fdc6SAndreas Gohr        } else {
130255efc227SAndreas Gohr            $this->_info['file'] = array();
130355efc227SAndreas Gohr            $this->_info['file']['Name'] = basename($this->_fileName);
130455efc227SAndreas Gohr            $this->_info['file']['Url'] = $this->_fileName;
130555efc227SAndreas Gohr        }
130655efc227SAndreas Gohr
130755efc227SAndreas Gohr        return true;
130855efc227SAndreas Gohr    }
130955efc227SAndreas Gohr
131055efc227SAndreas Gohr    /*************************************************************/
13110b17fdc6SAndreas Gohr    function _parseMarkerJFIF() {
131255efc227SAndreas Gohr        if (!isset($this->_markers)) {
131355efc227SAndreas Gohr            $this->_readJPEG();
131455efc227SAndreas Gohr        }
131555efc227SAndreas Gohr
131655efc227SAndreas Gohr        if ($this->_markers == null) {
131755efc227SAndreas Gohr            return false;
131855efc227SAndreas Gohr        }
131955efc227SAndreas Gohr
132055efc227SAndreas Gohr        $data = null;
132155efc227SAndreas Gohr        $count = count($this->_markers);
132255efc227SAndreas Gohr        for ($i = 0; $i < $count; $i++) {
132355efc227SAndreas Gohr            if ($this->_markers[$i]['marker'] == 0xE0) {
132455efc227SAndreas Gohr                $signature = $this->_getFixedString($this->_markers[$i]['data'], 0, 4);
132555efc227SAndreas Gohr                if ($signature == 'JFIF') {
132655efc227SAndreas Gohr                    $data =& $this->_markers[$i]['data'];
132755efc227SAndreas Gohr                    break;
132855efc227SAndreas Gohr                }
132955efc227SAndreas Gohr            }
133055efc227SAndreas Gohr        }
133155efc227SAndreas Gohr
133255efc227SAndreas Gohr        if ($data == null) {
133355efc227SAndreas Gohr            $this->_info['jfif'] = false;
133455efc227SAndreas Gohr            return false;
133555efc227SAndreas Gohr        }
133655efc227SAndreas Gohr
133755efc227SAndreas Gohr        $pos = 0;
133855efc227SAndreas Gohr        $this->_info['jfif'] = array();
133955efc227SAndreas Gohr
134055efc227SAndreas Gohr        $vmaj = $this->_getByte($data, 5);
134155efc227SAndreas Gohr        $vmin = $this->_getByte($data, 6);
134255efc227SAndreas Gohr
134355efc227SAndreas Gohr        $this->_info['jfif']['Version'] = sprintf('%d.%02d', $vmaj, $vmin);
134455efc227SAndreas Gohr
134555efc227SAndreas Gohr        $units = $this->_getByte($data, 7);
134655efc227SAndreas Gohr        switch ($units) {
134755efc227SAndreas Gohr            case 0:
134855efc227SAndreas Gohr                $this->_info['jfif']['Units'] = 'pixels';
134955efc227SAndreas Gohr                break;
135055efc227SAndreas Gohr            case 1:
135155efc227SAndreas Gohr                $this->_info['jfif']['Units'] = 'dpi';
135255efc227SAndreas Gohr                break;
135355efc227SAndreas Gohr            case 2:
135455efc227SAndreas Gohr                $this->_info['jfif']['Units'] = 'dpcm';
135555efc227SAndreas Gohr                break;
135655efc227SAndreas Gohr            default:
135755efc227SAndreas Gohr                $this->_info['jfif']['Units'] = 'unknown';
135855efc227SAndreas Gohr                break;
135955efc227SAndreas Gohr        }
136055efc227SAndreas Gohr
136155efc227SAndreas Gohr        $xdens = $this->_getShort($data, 8);
136255efc227SAndreas Gohr        $ydens = $this->_getShort($data, 10);
136355efc227SAndreas Gohr
136455efc227SAndreas Gohr        $this->_info['jfif']['XDensity'] = $xdens;
136555efc227SAndreas Gohr        $this->_info['jfif']['YDensity'] = $ydens;
136655efc227SAndreas Gohr
136755efc227SAndreas Gohr        $thumbx = $this->_getByte($data, 12);
136855efc227SAndreas Gohr        $thumby = $this->_getByte($data, 13);
136955efc227SAndreas Gohr
137055efc227SAndreas Gohr        $this->_info['jfif']['ThumbnailWidth'] = $thumbx;
137155efc227SAndreas Gohr        $this->_info['jfif']['ThumbnailHeight'] = $thumby;
137255efc227SAndreas Gohr
137355efc227SAndreas Gohr        return true;
137455efc227SAndreas Gohr    }
137555efc227SAndreas Gohr
137655efc227SAndreas Gohr    /*************************************************************/
13770b17fdc6SAndreas Gohr    function _parseMarkerSOF() {
137855efc227SAndreas Gohr        if (!isset($this->_markers)) {
137955efc227SAndreas Gohr            $this->_readJPEG();
138055efc227SAndreas Gohr        }
138155efc227SAndreas Gohr
138255efc227SAndreas Gohr        if ($this->_markers == null) {
138355efc227SAndreas Gohr            return false;
138455efc227SAndreas Gohr        }
138555efc227SAndreas Gohr
138655efc227SAndreas Gohr        $data = null;
138755efc227SAndreas Gohr        $count = count($this->_markers);
138855efc227SAndreas Gohr        for ($i = 0; $i < $count; $i++) {
138955efc227SAndreas Gohr            switch ($this->_markers[$i]['marker']) {
139055efc227SAndreas Gohr                case 0xC0: // SOF0
139155efc227SAndreas Gohr                case 0xC1: // SOF1
139255efc227SAndreas Gohr                case 0xC2: // SOF2
139355efc227SAndreas Gohr                case 0xC9: // SOF9
139455efc227SAndreas Gohr                    $data =& $this->_markers[$i]['data'];
139555efc227SAndreas Gohr                    $marker = $this->_markers[$i]['marker'];
139655efc227SAndreas Gohr                    break;
139755efc227SAndreas Gohr            }
139855efc227SAndreas Gohr        }
139955efc227SAndreas Gohr
140055efc227SAndreas Gohr        if ($data == null) {
140155efc227SAndreas Gohr            $this->_info['sof'] = false;
140255efc227SAndreas Gohr            return false;
140355efc227SAndreas Gohr        }
140455efc227SAndreas Gohr
140555efc227SAndreas Gohr        $pos = 0;
140655efc227SAndreas Gohr        $this->_info['sof'] = array();
140755efc227SAndreas Gohr
140855efc227SAndreas Gohr        switch ($marker) {
140955efc227SAndreas Gohr            case 0xC0: // SOF0
141055efc227SAndreas Gohr                $format = 'Baseline';
141155efc227SAndreas Gohr                break;
141255efc227SAndreas Gohr            case 0xC1: // SOF1
141355efc227SAndreas Gohr                $format = 'Progessive';
141455efc227SAndreas Gohr                break;
141555efc227SAndreas Gohr            case 0xC2: // SOF2
141655efc227SAndreas Gohr                $format = 'Non-baseline';
141755efc227SAndreas Gohr                break;
141855efc227SAndreas Gohr            case 0xC9: // SOF9
141955efc227SAndreas Gohr                $format = 'Arithmetic';
142055efc227SAndreas Gohr                break;
142155efc227SAndreas Gohr            default:
142255efc227SAndreas Gohr                return false;
142355efc227SAndreas Gohr                break;
142455efc227SAndreas Gohr        }
142555efc227SAndreas Gohr
142655efc227SAndreas Gohr        $this->_info['sof']['Format']          = $format;
142755efc227SAndreas Gohr        $this->_info['sof']['SamplePrecision'] = $this->_getByte($data, $pos + 0);
142855efc227SAndreas Gohr        $this->_info['sof']['ImageHeight']     = $this->_getShort($data, $pos + 1);
142955efc227SAndreas Gohr        $this->_info['sof']['ImageWidth']      = $this->_getShort($data, $pos + 3);
143055efc227SAndreas Gohr        $this->_info['sof']['ColorChannels']   = $this->_getByte($data, $pos + 5);
143155efc227SAndreas Gohr
143255efc227SAndreas Gohr        return true;
143355efc227SAndreas Gohr    }
143455efc227SAndreas Gohr
1435431c7fc8Shakan.sandell    /**
1436431c7fc8Shakan.sandell     * Parses the XMP data
1437431c7fc8Shakan.sandell     *
1438431c7fc8Shakan.sandell     * @author  Hakan Sandell <hakan.sandell@mydata.se>
1439431c7fc8Shakan.sandell     */
14400b17fdc6SAndreas Gohr    function _parseMarkerXmp() {
1441431c7fc8Shakan.sandell        if (!isset($this->_markers)) {
1442431c7fc8Shakan.sandell            $this->_readJPEG();
1443431c7fc8Shakan.sandell        }
1444431c7fc8Shakan.sandell
1445431c7fc8Shakan.sandell        if ($this->_markers == null) {
1446431c7fc8Shakan.sandell            return false;
1447431c7fc8Shakan.sandell        }
1448431c7fc8Shakan.sandell
1449431c7fc8Shakan.sandell        $data = null;
1450431c7fc8Shakan.sandell        $count = count($this->_markers);
1451431c7fc8Shakan.sandell        for ($i = 0; $i < $count; $i++) {
1452431c7fc8Shakan.sandell            if ($this->_markers[$i]['marker'] == 0xE1) {
1453431c7fc8Shakan.sandell                $signature = $this->_getFixedString($this->_markers[$i]['data'], 0, 29);
1454431c7fc8Shakan.sandell                if ($signature == "http://ns.adobe.com/xap/1.0/\0") {
1455431c7fc8Shakan.sandell                    $data =& substr($this->_markers[$i]['data'], 29);
1456431c7fc8Shakan.sandell                    break;
1457431c7fc8Shakan.sandell                }
1458431c7fc8Shakan.sandell            }
1459431c7fc8Shakan.sandell        }
1460431c7fc8Shakan.sandell
1461431c7fc8Shakan.sandell        if ($data == null) {
1462431c7fc8Shakan.sandell            $this->_info['xmp'] = false;
1463431c7fc8Shakan.sandell            return false;
1464431c7fc8Shakan.sandell        }
1465431c7fc8Shakan.sandell
1466431c7fc8Shakan.sandell        $parser = xml_parser_create();
1467431c7fc8Shakan.sandell        xml_parser_set_option($parser, XML_OPTION_CASE_FOLDING, 0);
1468431c7fc8Shakan.sandell        xml_parser_set_option($parser, XML_OPTION_SKIP_WHITE, 1);
1469d8443bf1SHakan Sandell        $result = xml_parse_into_struct($parser, $data, $values, $tags);
1470431c7fc8Shakan.sandell        xml_parser_free($parser);
1471431c7fc8Shakan.sandell
1472d8443bf1SHakan Sandell        if ($result == 0) {
1473d8443bf1SHakan Sandell            $this->_info['xmp'] = false;
1474d8443bf1SHakan Sandell            return false;
1475d8443bf1SHakan Sandell        }
1476d8443bf1SHakan Sandell
1477431c7fc8Shakan.sandell        $this->_info['xmp'] = array();
1478431c7fc8Shakan.sandell        $count = count($values);
1479431c7fc8Shakan.sandell        for ($i = 0; $i < $count; $i++) {
14800b17fdc6SAndreas Gohr            if ($values[$i]['tag'] == 'rdf:Description' && $values[$i]['type'] == 'open') {
1481431c7fc8Shakan.sandell
1482d8443bf1SHakan Sandell                while ((++$i < $count) && ($values[$i]['tag'] != 'rdf:Description')) {
1483d8443bf1SHakan Sandell                    $this->_parseXmpNode($values, $i, $this->_info['xmp'][$values[$i]['tag']], $count);
1484431c7fc8Shakan.sandell                }
1485431c7fc8Shakan.sandell            }
1486431c7fc8Shakan.sandell        }
1487431c7fc8Shakan.sandell        return true;
1488431c7fc8Shakan.sandell    }
1489431c7fc8Shakan.sandell
1490431c7fc8Shakan.sandell    /**
1491431c7fc8Shakan.sandell     * Parses XMP nodes by recursion
1492431c7fc8Shakan.sandell     *
1493431c7fc8Shakan.sandell     * @author  Hakan Sandell <hakan.sandell@mydata.se>
1494431c7fc8Shakan.sandell     */
1495d8443bf1SHakan Sandell    function _parseXmpNode($values, &$i, &$meta, $count) {
1496831c10d0SHakan Sandell        if ($values[$i]['type'] == 'close') return;
1497831c10d0SHakan Sandell
14980b17fdc6SAndreas Gohr        if ($values[$i]['type'] == 'complete') {
1499431c7fc8Shakan.sandell            // Simple Type property
15000b17fdc6SAndreas Gohr            $meta = $values[$i]['value'];
1501431c7fc8Shakan.sandell            return;
1502431c7fc8Shakan.sandell        }
1503431c7fc8Shakan.sandell
1504431c7fc8Shakan.sandell        $i++;
1505d8443bf1SHakan Sandell        if ($i >= $count) return;
1506d8443bf1SHakan Sandell
15070b17fdc6SAndreas Gohr        if ($values[$i]['tag'] == 'rdf:Bag' || $values[$i]['tag'] == 'rdf:Seq') {
1508431c7fc8Shakan.sandell            // Array property
1509431c7fc8Shakan.sandell            $meta = array();
15100b17fdc6SAndreas Gohr            while ($values[++$i]['tag'] == 'rdf:li') {
1511d8443bf1SHakan Sandell                $this->_parseXmpNode($values, $i, $meta[], $count);
1512431c7fc8Shakan.sandell            }
1513831c10d0SHakan Sandell            $i++; // skip closing Bag/Seq tag
1514431c7fc8Shakan.sandell
15150b17fdc6SAndreas Gohr        } elseif ($values[$i]['tag'] == 'rdf:Alt') {
1516431c7fc8Shakan.sandell            // Language Alternative property, only the first (default) value is used
1517831c10d0SHakan Sandell            if ($values[$i]['type'] == 'open') {
1518431c7fc8Shakan.sandell                $i++;
1519d8443bf1SHakan Sandell                $this->_parseXmpNode($values, $i, $meta, $count);
1520d8443bf1SHakan Sandell                while ((++$i < $count) && ($values[$i]['tag'] != 'rdf:Alt'));
1521831c10d0SHakan Sandell                $i++; // skip closing Alt tag
1522831c10d0SHakan Sandell            }
1523431c7fc8Shakan.sandell
1524431c7fc8Shakan.sandell        } else {
1525431c7fc8Shakan.sandell            // Structure property
1526431c7fc8Shakan.sandell            $meta = array();
15270b17fdc6SAndreas Gohr            $startTag = $values[$i-1]['tag'];
1528431c7fc8Shakan.sandell            do {
1529d8443bf1SHakan Sandell                $this->_parseXmpNode($values, $i, $meta[$values[$i]['tag']], $count);
1530d8443bf1SHakan Sandell            } while ((++$i < $count) && ($values[$i]['tag'] != $startTag));
1531431c7fc8Shakan.sandell        }
1532431c7fc8Shakan.sandell    }
1533431c7fc8Shakan.sandell
153455efc227SAndreas Gohr    /*************************************************************/
15350b17fdc6SAndreas Gohr    function _parseMarkerExif() {
153655efc227SAndreas Gohr        if (!isset($this->_markers)) {
153755efc227SAndreas Gohr            $this->_readJPEG();
153855efc227SAndreas Gohr        }
153955efc227SAndreas Gohr
154055efc227SAndreas Gohr        if ($this->_markers == null) {
154155efc227SAndreas Gohr            return false;
154255efc227SAndreas Gohr        }
154355efc227SAndreas Gohr
154455efc227SAndreas Gohr        $data = null;
154555efc227SAndreas Gohr        $count = count($this->_markers);
154655efc227SAndreas Gohr        for ($i = 0; $i < $count; $i++) {
154755efc227SAndreas Gohr            if ($this->_markers[$i]['marker'] == 0xE1) {
154855efc227SAndreas Gohr                $signature = $this->_getFixedString($this->_markers[$i]['data'], 0, 6);
154955efc227SAndreas Gohr                if ($signature == "Exif\0\0") {
155055efc227SAndreas Gohr                    $data =& $this->_markers[$i]['data'];
155155efc227SAndreas Gohr                    break;
155255efc227SAndreas Gohr                }
155355efc227SAndreas Gohr            }
155455efc227SAndreas Gohr        }
155555efc227SAndreas Gohr
155655efc227SAndreas Gohr        if ($data == null) {
155755efc227SAndreas Gohr            $this->_info['exif'] = false;
155855efc227SAndreas Gohr            return false;
155955efc227SAndreas Gohr        }
156055efc227SAndreas Gohr        $pos = 6;
156155efc227SAndreas Gohr        $this->_info['exif'] = array();
156255efc227SAndreas Gohr
156355efc227SAndreas Gohr        // We don't increment $pos after this because Exif uses offsets relative to this point
156455efc227SAndreas Gohr
156555efc227SAndreas Gohr        $byteAlign = $this->_getShort($data, $pos + 0);
156655efc227SAndreas Gohr
156755efc227SAndreas Gohr        if ($byteAlign == 0x4949) { // "II"
156855efc227SAndreas Gohr            $isBigEndian = false;
15690b17fdc6SAndreas Gohr        } elseif ($byteAlign == 0x4D4D) { // "MM"
157055efc227SAndreas Gohr            $isBigEndian = true;
15710b17fdc6SAndreas Gohr        } else {
157255efc227SAndreas Gohr            return false; // Unexpected data
157355efc227SAndreas Gohr        }
157455efc227SAndreas Gohr
157555efc227SAndreas Gohr        $alignCheck = $this->_getShort($data, $pos + 2, $isBigEndian);
157655efc227SAndreas Gohr        if ($alignCheck != 0x002A) // That's the expected value
157755efc227SAndreas Gohr            return false; // Unexpected data
157855efc227SAndreas Gohr
157955efc227SAndreas Gohr        if ($isBigEndian) {
158055efc227SAndreas Gohr            $this->_info['exif']['ByteAlign'] = "Big Endian";
15810b17fdc6SAndreas Gohr        } else {
158255efc227SAndreas Gohr            $this->_info['exif']['ByteAlign'] = "Little Endian";
158355efc227SAndreas Gohr        }
158455efc227SAndreas Gohr
158555efc227SAndreas Gohr        $offsetIFD0 = $this->_getLong($data, $pos + 4, $isBigEndian);
158655efc227SAndreas Gohr        if ($offsetIFD0 < 8)
158755efc227SAndreas Gohr            return false; // Unexpected data
158855efc227SAndreas Gohr
158955efc227SAndreas Gohr        $offsetIFD1 = $this->_readIFD($data, $pos, $offsetIFD0, $isBigEndian, 'ifd0');
159055efc227SAndreas Gohr        if ($offsetIFD1 != 0)
159155efc227SAndreas Gohr            $this->_readIFD($data, $pos, $offsetIFD1, $isBigEndian, 'ifd1');
159255efc227SAndreas Gohr
159355efc227SAndreas Gohr        return true;
159455efc227SAndreas Gohr    }
159555efc227SAndreas Gohr
159655efc227SAndreas Gohr    /*************************************************************/
15970b17fdc6SAndreas Gohr    function _readIFD($data, $base, $offset, $isBigEndian, $mode) {
159855efc227SAndreas Gohr        $EXIFTags = $this->_exifTagNames($mode);
159955efc227SAndreas Gohr
160055efc227SAndreas Gohr        $numEntries = $this->_getShort($data, $base + $offset, $isBigEndian);
160155efc227SAndreas Gohr        $offset += 2;
160255efc227SAndreas Gohr
160355efc227SAndreas Gohr        $exifTIFFOffset = 0;
160455efc227SAndreas Gohr        $exifTIFFLength = 0;
160555efc227SAndreas Gohr        $exifThumbnailOffset = 0;
160655efc227SAndreas Gohr        $exifThumbnailLength = 0;
160755efc227SAndreas Gohr
160855efc227SAndreas Gohr        for ($i = 0; $i < $numEntries; $i++) {
160955efc227SAndreas Gohr            $tag = $this->_getShort($data, $base + $offset, $isBigEndian);
161055efc227SAndreas Gohr            $offset += 2;
161155efc227SAndreas Gohr            $type = $this->_getShort($data, $base + $offset, $isBigEndian);
161255efc227SAndreas Gohr            $offset += 2;
161355efc227SAndreas Gohr            $count = $this->_getLong($data, $base + $offset, $isBigEndian);
161455efc227SAndreas Gohr            $offset += 4;
161555efc227SAndreas Gohr
161655efc227SAndreas Gohr            if (($type < 1) || ($type > 12))
161755efc227SAndreas Gohr                return false; // Unexpected Type
161855efc227SAndreas Gohr
161955efc227SAndreas Gohr            $typeLengths = array( -1, 1, 1, 2, 4, 8, 1, 1, 2, 4, 8, 4, 8 );
162055efc227SAndreas Gohr
162155efc227SAndreas Gohr            $dataLength = $typeLengths[$type] * $count;
162255efc227SAndreas Gohr            if ($dataLength > 4) {
162355efc227SAndreas Gohr                $dataOffset = $this->_getLong($data, $base + $offset, $isBigEndian);
162455efc227SAndreas Gohr                $rawValue = $this->_getFixedString($data, $base + $dataOffset, $dataLength);
16250b17fdc6SAndreas Gohr            } else {
162655efc227SAndreas Gohr                $rawValue = $this->_getFixedString($data, $base + $offset, $dataLength);
162755efc227SAndreas Gohr            }
162855efc227SAndreas Gohr            $offset += 4;
162955efc227SAndreas Gohr
163055efc227SAndreas Gohr            switch ($type) {
163155efc227SAndreas Gohr                case 1:    // UBYTE
163255efc227SAndreas Gohr                    if ($count == 1) {
163355efc227SAndreas Gohr                        $value = $this->_getByte($rawValue, 0);
16340b17fdc6SAndreas Gohr                    } else {
163555efc227SAndreas Gohr                        $value = array();
163655efc227SAndreas Gohr                        for ($j = 0; $j < $count; $j++)
163755efc227SAndreas Gohr                            $value[$j] = $this->_getByte($rawValue, $j);
163855efc227SAndreas Gohr                    }
163955efc227SAndreas Gohr                    break;
164055efc227SAndreas Gohr                case 2:    // ASCII
164155efc227SAndreas Gohr                    $value = $rawValue;
164255efc227SAndreas Gohr                    break;
164355efc227SAndreas Gohr                case 3:    // USHORT
164455efc227SAndreas Gohr                    if ($count == 1) {
164555efc227SAndreas Gohr                        $value = $this->_getShort($rawValue, 0, $isBigEndian);
16460b17fdc6SAndreas Gohr                    } else {
164755efc227SAndreas Gohr                        $value = array();
164855efc227SAndreas Gohr                        for ($j = 0; $j < $count; $j++)
164955efc227SAndreas Gohr                            $value[$j] = $this->_getShort($rawValue, $j * 2, $isBigEndian);
165055efc227SAndreas Gohr                    }
165155efc227SAndreas Gohr                    break;
165255efc227SAndreas Gohr                case 4:    // ULONG
165355efc227SAndreas Gohr                    if ($count == 1) {
165455efc227SAndreas Gohr                        $value = $this->_getLong($rawValue, 0, $isBigEndian);
16550b17fdc6SAndreas Gohr                    } else {
165655efc227SAndreas Gohr                        $value = array();
165755efc227SAndreas Gohr                        for ($j = 0; $j < $count; $j++)
165855efc227SAndreas Gohr                            $value[$j] = $this->_getLong($rawValue, $j * 4, $isBigEndian);
165955efc227SAndreas Gohr                    }
166055efc227SAndreas Gohr                    break;
166155efc227SAndreas Gohr                case 5:    // URATIONAL
166255efc227SAndreas Gohr                    if ($count == 1) {
166355efc227SAndreas Gohr                        $a = $this->_getLong($rawValue, 0, $isBigEndian);
166455efc227SAndreas Gohr                        $b = $this->_getLong($rawValue, 4, $isBigEndian);
166555efc227SAndreas Gohr                        $value = array();
166655efc227SAndreas Gohr                        $value['val'] = 0;
166755efc227SAndreas Gohr                        $value['num'] = $a;
166855efc227SAndreas Gohr                        $value['den'] = $b;
166955efc227SAndreas Gohr                        if (($a != 0) && ($b != 0)) {
167055efc227SAndreas Gohr                            $value['val'] = $a / $b;
167155efc227SAndreas Gohr                        }
16720b17fdc6SAndreas Gohr                    } else {
167355efc227SAndreas Gohr                        $value = array();
167455efc227SAndreas Gohr                        for ($j = 0; $j < $count; $j++) {
167555efc227SAndreas Gohr                            $a = $this->_getLong($rawValue, $j * 8, $isBigEndian);
167655efc227SAndreas Gohr                            $b = $this->_getLong($rawValue, ($j * 8) + 4, $isBigEndian);
167755efc227SAndreas Gohr                            $value = array();
167855efc227SAndreas Gohr                            $value[$j]['val'] = 0;
167955efc227SAndreas Gohr                            $value[$j]['num'] = $a;
168055efc227SAndreas Gohr                            $value[$j]['den'] = $b;
168155efc227SAndreas Gohr                            if (($a != 0) && ($b != 0))
168255efc227SAndreas Gohr                                $value[$j]['val'] = $a / $b;
168355efc227SAndreas Gohr                        }
168455efc227SAndreas Gohr                    }
168555efc227SAndreas Gohr                    break;
168655efc227SAndreas Gohr                case 6:    // SBYTE
168755efc227SAndreas Gohr                    if ($count == 1) {
168855efc227SAndreas Gohr                        $value = $this->_getByte($rawValue, 0);
16890b17fdc6SAndreas Gohr                    } else {
169055efc227SAndreas Gohr                        $value = array();
169155efc227SAndreas Gohr                        for ($j = 0; $j < $count; $j++)
169255efc227SAndreas Gohr                            $value[$j] = $this->_getByte($rawValue, $j);
169355efc227SAndreas Gohr                    }
169455efc227SAndreas Gohr                    break;
169555efc227SAndreas Gohr                case 7:    // UNDEFINED
169655efc227SAndreas Gohr                    $value = $rawValue;
169755efc227SAndreas Gohr                    break;
169855efc227SAndreas Gohr                case 8:    // SSHORT
169955efc227SAndreas Gohr                    if ($count == 1) {
170055efc227SAndreas Gohr                        $value = $this->_getShort($rawValue, 0, $isBigEndian);
17010b17fdc6SAndreas Gohr                    } else {
170255efc227SAndreas Gohr                        $value = array();
170355efc227SAndreas Gohr                        for ($j = 0; $j < $count; $j++)
170455efc227SAndreas Gohr                            $value[$j] = $this->_getShort($rawValue, $j * 2, $isBigEndian);
170555efc227SAndreas Gohr                    }
170655efc227SAndreas Gohr                    break;
170755efc227SAndreas Gohr                case 9:    // SLONG
170855efc227SAndreas Gohr                    if ($count == 1) {
170955efc227SAndreas Gohr                        $value = $this->_getLong($rawValue, 0, $isBigEndian);
17100b17fdc6SAndreas Gohr                    } else {
171155efc227SAndreas Gohr                        $value = array();
171255efc227SAndreas Gohr                        for ($j = 0; $j < $count; $j++)
171355efc227SAndreas Gohr                            $value[$j] = $this->_getLong($rawValue, $j * 4, $isBigEndian);
171455efc227SAndreas Gohr                    }
171555efc227SAndreas Gohr                    break;
171655efc227SAndreas Gohr                case 10:   // SRATIONAL
171755efc227SAndreas Gohr                    if ($count == 1) {
171855efc227SAndreas Gohr                        $a = $this->_getLong($rawValue, 0, $isBigEndian);
171955efc227SAndreas Gohr                        $b = $this->_getLong($rawValue, 4, $isBigEndian);
172055efc227SAndreas Gohr                        $value = array();
172155efc227SAndreas Gohr                        $value['val'] = 0;
172255efc227SAndreas Gohr                        $value['num'] = $a;
172355efc227SAndreas Gohr                        $value['den'] = $b;
172455efc227SAndreas Gohr                        if (($a != 0) && ($b != 0))
172555efc227SAndreas Gohr                            $value['val'] = $a / $b;
17260b17fdc6SAndreas Gohr                    } else {
172755efc227SAndreas Gohr                        $value = array();
172855efc227SAndreas Gohr                        for ($j = 0; $j < $count; $j++) {
172955efc227SAndreas Gohr                            $a = $this->_getLong($rawValue, $j * 8, $isBigEndian);
173055efc227SAndreas Gohr                            $b = $this->_getLong($rawValue, ($j * 8) + 4, $isBigEndian);
173155efc227SAndreas Gohr                            $value = array();
173255efc227SAndreas Gohr                            $value[$j]['val'] = 0;
173355efc227SAndreas Gohr                            $value[$j]['num'] = $a;
173455efc227SAndreas Gohr                            $value[$j]['den'] = $b;
173555efc227SAndreas Gohr                            if (($a != 0) && ($b != 0))
173655efc227SAndreas Gohr                                $value[$j]['val'] = $a / $b;
173755efc227SAndreas Gohr                        }
173855efc227SAndreas Gohr                    }
173955efc227SAndreas Gohr                    break;
174055efc227SAndreas Gohr                case 11:   // FLOAT
174155efc227SAndreas Gohr                    $value = $rawValue;
174255efc227SAndreas Gohr                    break;
174355efc227SAndreas Gohr
174455efc227SAndreas Gohr                case 12:   // DFLOAT
174555efc227SAndreas Gohr                    $value = $rawValue;
174655efc227SAndreas Gohr                    break;
174755efc227SAndreas Gohr                default:
174855efc227SAndreas Gohr                    return false; // Unexpected Type
174955efc227SAndreas Gohr            }
175055efc227SAndreas Gohr
175155efc227SAndreas Gohr            $tagName = '';
175255efc227SAndreas Gohr            if (($mode == 'ifd0') && ($tag == 0x8769)) {  // ExifIFDOffset
175355efc227SAndreas Gohr                $this->_readIFD($data, $base, $value, $isBigEndian, 'exif');
17540b17fdc6SAndreas Gohr            } elseif (($mode == 'ifd0') && ($tag == 0x8825)) {  // GPSIFDOffset
175555efc227SAndreas Gohr                $this->_readIFD($data, $base, $value, $isBigEndian, 'gps');
17560b17fdc6SAndreas Gohr            } elseif (($mode == 'ifd1') && ($tag == 0x0111)) {  // TIFFStripOffsets
175755efc227SAndreas Gohr                $exifTIFFOffset = $value;
17580b17fdc6SAndreas Gohr            } elseif (($mode == 'ifd1') && ($tag == 0x0117)) {  // TIFFStripByteCounts
175955efc227SAndreas Gohr                $exifTIFFLength = $value;
17600b17fdc6SAndreas Gohr            } elseif (($mode == 'ifd1') && ($tag == 0x0201)) {  // TIFFJFIFOffset
176155efc227SAndreas Gohr                $exifThumbnailOffset = $value;
17620b17fdc6SAndreas Gohr            } elseif (($mode == 'ifd1') && ($tag == 0x0202)) {  // TIFFJFIFLength
176355efc227SAndreas Gohr                $exifThumbnailLength = $value;
17640b17fdc6SAndreas Gohr            } elseif (($mode == 'exif') && ($tag == 0xA005)) {  // InteropIFDOffset
176555efc227SAndreas Gohr                $this->_readIFD($data, $base, $value, $isBigEndian, 'interop');
176655efc227SAndreas Gohr            }
176755efc227SAndreas Gohr            // elseif (($mode == 'exif') && ($tag == 0x927C)) {  // MakerNote
176855efc227SAndreas Gohr            // }
176955efc227SAndreas Gohr            else {
177055efc227SAndreas Gohr                if (isset($EXIFTags[$tag])) {
177155efc227SAndreas Gohr                    $tagName = $EXIFTags[$tag];
177255efc227SAndreas Gohr                    if (isset($this->_info['exif'][$tagName])) {
177355efc227SAndreas Gohr                        if (!is_array($this->_info['exif'][$tagName])) {
177455efc227SAndreas Gohr                            $aux = array();
177555efc227SAndreas Gohr                            $aux[0] = $this->_info['exif'][$tagName];
177655efc227SAndreas Gohr                            $this->_info['exif'][$tagName] = $aux;
177755efc227SAndreas Gohr                        }
177855efc227SAndreas Gohr
177955efc227SAndreas Gohr                        $this->_info['exif'][$tagName][count($this->_info['exif'][$tagName])] = $value;
17800b17fdc6SAndreas Gohr                    } else {
178155efc227SAndreas Gohr                        $this->_info['exif'][$tagName] = $value;
178255efc227SAndreas Gohr                    }
178355efc227SAndreas Gohr                }
17840b17fdc6SAndreas Gohr                /*
178555efc227SAndreas Gohr                 else {
17860b17fdc6SAndreas Gohr                    echo sprintf("<h1>Unknown tag %02x (t: %d l: %d) %s in %s</h1>", $tag, $type, $count, $mode, $this->_fileName);
178755efc227SAndreas Gohr                    // Unknown Tags will be ignored!!!
178855efc227SAndreas Gohr                    // That's because the tag might be a pointer (like the Exif tag)
178955efc227SAndreas Gohr                    // and saving it without saving the data it points to might
179055efc227SAndreas Gohr                    // create an invalid file.
179155efc227SAndreas Gohr                }
17920b17fdc6SAndreas Gohr                */
179355efc227SAndreas Gohr            }
179455efc227SAndreas Gohr        }
179555efc227SAndreas Gohr
179655efc227SAndreas Gohr        if (($exifThumbnailOffset > 0) && ($exifThumbnailLength > 0)) {
179755efc227SAndreas Gohr            $this->_info['exif']['JFIFThumbnail'] = $this->_getFixedString($data, $base + $exifThumbnailOffset, $exifThumbnailLength);
179855efc227SAndreas Gohr        }
179955efc227SAndreas Gohr
180055efc227SAndreas Gohr        if (($exifTIFFOffset > 0) && ($exifTIFFLength > 0)) {
180155efc227SAndreas Gohr            $this->_info['exif']['TIFFStrips'] = $this->_getFixedString($data, $base + $exifTIFFOffset, $exifTIFFLength);
180255efc227SAndreas Gohr        }
180355efc227SAndreas Gohr
180455efc227SAndreas Gohr        $nextOffset = $this->_getLong($data, $base + $offset, $isBigEndian);
180555efc227SAndreas Gohr        return $nextOffset;
180655efc227SAndreas Gohr    }
180755efc227SAndreas Gohr
180855efc227SAndreas Gohr    /*************************************************************/
18090b17fdc6SAndreas Gohr    function & _createMarkerExif() {
181055efc227SAndreas Gohr        $data = null;
181155efc227SAndreas Gohr        $count = count($this->_markers);
181255efc227SAndreas Gohr        for ($i = 0; $i < $count; $i++) {
181355efc227SAndreas Gohr            if ($this->_markers[$i]['marker'] == 0xE1) {
181455efc227SAndreas Gohr                $signature = $this->_getFixedString($this->_markers[$i]['data'], 0, 6);
181555efc227SAndreas Gohr                if ($signature == "Exif\0\0") {
181655efc227SAndreas Gohr                    $data =& $this->_markers[$i]['data'];
181755efc227SAndreas Gohr                    break;
181855efc227SAndreas Gohr                }
181955efc227SAndreas Gohr            }
182055efc227SAndreas Gohr        }
182155efc227SAndreas Gohr
182255efc227SAndreas Gohr        if (!isset($this->_info['exif'])) {
182355efc227SAndreas Gohr            return false;
182455efc227SAndreas Gohr        }
182555efc227SAndreas Gohr
182655efc227SAndreas Gohr        $data = "Exif\0\0";
182755efc227SAndreas Gohr        $pos = 6;
182855efc227SAndreas Gohr        $offsetBase = 6;
182955efc227SAndreas Gohr
183055efc227SAndreas Gohr        if (isset($this->_info['exif']['ByteAlign']) && ($this->_info['exif']['ByteAlign'] == "Big Endian")) {
183155efc227SAndreas Gohr            $isBigEndian = true;
183255efc227SAndreas Gohr            $aux = "MM";
183355efc227SAndreas Gohr            $pos = $this->_putString($data, $pos, $aux);
18340b17fdc6SAndreas Gohr        } else {
183555efc227SAndreas Gohr            $isBigEndian = false;
183655efc227SAndreas Gohr            $aux = "II";
183755efc227SAndreas Gohr            $pos = $this->_putString($data, $pos, $aux);
183855efc227SAndreas Gohr        }
183955efc227SAndreas Gohr        $pos = $this->_putShort($data, $pos, 0x002A, $isBigEndian);
184055efc227SAndreas Gohr        $pos = $this->_putLong($data, $pos, 0x00000008, $isBigEndian); // IFD0 Offset is always 8
184155efc227SAndreas Gohr
184255efc227SAndreas Gohr        $ifd0 =& $this->_getIFDEntries($isBigEndian, 'ifd0');
184355efc227SAndreas Gohr        $ifd1 =& $this->_getIFDEntries($isBigEndian, 'ifd1');
184455efc227SAndreas Gohr
184555efc227SAndreas Gohr        $pos = $this->_writeIFD($data, $pos, $offsetBase, $ifd0, $isBigEndian, true);
184655efc227SAndreas Gohr        $pos = $this->_writeIFD($data, $pos, $offsetBase, $ifd1, $isBigEndian, false);
184755efc227SAndreas Gohr
184855efc227SAndreas Gohr        return $data;
184955efc227SAndreas Gohr    }
185055efc227SAndreas Gohr
185155efc227SAndreas Gohr    /*************************************************************/
18520b17fdc6SAndreas Gohr    function _writeIFD(&$data, $pos, $offsetBase, &$entries, $isBigEndian, $hasNext) {
185355efc227SAndreas Gohr        $tiffData = null;
185455efc227SAndreas Gohr        $tiffDataOffsetPos = -1;
185555efc227SAndreas Gohr
185655efc227SAndreas Gohr        $entryCount = count($entries);
185755efc227SAndreas Gohr
185855efc227SAndreas Gohr        $dataPos = $pos + 2 + ($entryCount * 12) + 4;
185955efc227SAndreas Gohr        $pos = $this->_putShort($data, $pos, $entryCount, $isBigEndian);
186055efc227SAndreas Gohr
186155efc227SAndreas Gohr        for ($i = 0; $i < $entryCount; $i++) {
186255efc227SAndreas Gohr            $tag = $entries[$i]['tag'];
186355efc227SAndreas Gohr            $type = $entries[$i]['type'];
186455efc227SAndreas Gohr
186555efc227SAndreas Gohr            if ($type == -99) { // SubIFD
186655efc227SAndreas Gohr                $pos = $this->_putShort($data, $pos, $tag, $isBigEndian);
186755efc227SAndreas Gohr                $pos = $this->_putShort($data, $pos, 0x04, $isBigEndian); // LONG
186855efc227SAndreas Gohr                $pos = $this->_putLong($data, $pos, 0x01, $isBigEndian); // Count = 1
186955efc227SAndreas Gohr                $pos = $this->_putLong($data, $pos, $dataPos - $offsetBase, $isBigEndian);
187055efc227SAndreas Gohr
187155efc227SAndreas Gohr                $dataPos = $this->_writeIFD($data, $dataPos, $offsetBase, $entries[$i]['value'], $isBigEndian, false);
18720b17fdc6SAndreas Gohr            } elseif ($type == -98) { // TIFF Data
187355efc227SAndreas Gohr                $pos = $this->_putShort($data, $pos, $tag, $isBigEndian);
187455efc227SAndreas Gohr                $pos = $this->_putShort($data, $pos, 0x04, $isBigEndian); // LONG
187555efc227SAndreas Gohr                $pos = $this->_putLong($data, $pos, 0x01, $isBigEndian); // Count = 1
187655efc227SAndreas Gohr                $tiffDataOffsetPos = $pos;
187755efc227SAndreas Gohr                $pos = $this->_putLong($data, $pos, 0x00, $isBigEndian); // For Now
187855efc227SAndreas Gohr                $tiffData =& $entries[$i]['value'] ;
18790b17fdc6SAndreas Gohr            } else { // Regular Entry
188055efc227SAndreas Gohr                $pos = $this->_putShort($data, $pos, $tag, $isBigEndian);
188155efc227SAndreas Gohr                $pos = $this->_putShort($data, $pos, $type, $isBigEndian);
188255efc227SAndreas Gohr                $pos = $this->_putLong($data, $pos, $entries[$i]['count'], $isBigEndian);
188355efc227SAndreas Gohr                if (strlen($entries[$i]['value']) > 4) {
188455efc227SAndreas Gohr                    $pos = $this->_putLong($data, $pos, $dataPos - $offsetBase, $isBigEndian);
188555efc227SAndreas Gohr                    $dataPos = $this->_putString($data, $dataPos, $entries[$i]['value']);
18860b17fdc6SAndreas Gohr                } else {
188755efc227SAndreas Gohr                    $val = str_pad($entries[$i]['value'], 4, "\0");
188855efc227SAndreas Gohr                    $pos = $this->_putString($data, $pos, $val);
188955efc227SAndreas Gohr                }
189055efc227SAndreas Gohr            }
189155efc227SAndreas Gohr        }
189255efc227SAndreas Gohr
189355efc227SAndreas Gohr        if ($tiffData != null) {
189455efc227SAndreas Gohr            $this->_putLong($data, $tiffDataOffsetPos, $dataPos - $offsetBase, $isBigEndian);
189555efc227SAndreas Gohr            $dataPos = $this->_putString($data, $dataPos, $tiffData);
189655efc227SAndreas Gohr        }
189755efc227SAndreas Gohr
189855efc227SAndreas Gohr        if ($hasNext) {
189955efc227SAndreas Gohr            $pos = $this->_putLong($data, $pos, $dataPos - $offsetBase, $isBigEndian);
19000b17fdc6SAndreas Gohr        } else {
190155efc227SAndreas Gohr            $pos = $this->_putLong($data, $pos, 0, $isBigEndian);
190255efc227SAndreas Gohr        }
190355efc227SAndreas Gohr
190455efc227SAndreas Gohr        return $dataPos;
190555efc227SAndreas Gohr    }
190655efc227SAndreas Gohr
190755efc227SAndreas Gohr    /*************************************************************/
19080b17fdc6SAndreas Gohr    function & _getIFDEntries($isBigEndian, $mode) {
190955efc227SAndreas Gohr        $EXIFNames = $this->_exifTagNames($mode);
191055efc227SAndreas Gohr        $EXIFTags = $this->_exifNameTags($mode);
191155efc227SAndreas Gohr        $EXIFTypeInfo = $this->_exifTagTypes($mode);
191255efc227SAndreas Gohr
191355efc227SAndreas Gohr        $ifdEntries = array();
191455efc227SAndreas Gohr        $entryCount = 0;
191555efc227SAndreas Gohr
191655efc227SAndreas Gohr        reset($EXIFNames);
191755efc227SAndreas Gohr        while (list($tag, $name) = each($EXIFNames)) {
191855efc227SAndreas Gohr            $type = $EXIFTypeInfo[$tag][0];
191955efc227SAndreas Gohr            $count = $EXIFTypeInfo[$tag][1];
192055efc227SAndreas Gohr            $value = null;
192155efc227SAndreas Gohr
192255efc227SAndreas Gohr            if (($mode == 'ifd0') && ($tag == 0x8769)) {  // ExifIFDOffset
192355efc227SAndreas Gohr                if (isset($this->_info['exif']['EXIFVersion'])) {
192455efc227SAndreas Gohr                    $value =& $this->_getIFDEntries($isBigEndian, "exif");
192555efc227SAndreas Gohr                    $type = -99;
192655efc227SAndreas Gohr                }
192755efc227SAndreas Gohr                else {
192855efc227SAndreas Gohr                    $value = null;
192955efc227SAndreas Gohr                }
19300b17fdc6SAndreas Gohr            } elseif (($mode == 'ifd0') && ($tag == 0x8825)) {  // GPSIFDOffset
193155efc227SAndreas Gohr                if (isset($this->_info['exif']['GPSVersionID'])) {
193255efc227SAndreas Gohr                    $value =& $this->_getIFDEntries($isBigEndian, "gps");
193355efc227SAndreas Gohr                    $type = -99;
19340b17fdc6SAndreas Gohr                } else {
193555efc227SAndreas Gohr                    $value = null;
193655efc227SAndreas Gohr                }
19370b17fdc6SAndreas Gohr            } elseif (($mode == 'ifd1') && ($tag == 0x0111)) {  // TIFFStripOffsets
193855efc227SAndreas Gohr                if (isset($this->_info['exif']['TIFFStrips'])) {
193955efc227SAndreas Gohr                    $value =& $this->_info['exif']['TIFFStrips'];
194055efc227SAndreas Gohr                    $type = -98;
19410b17fdc6SAndreas Gohr                } else {
194255efc227SAndreas Gohr                    $value = null;
194355efc227SAndreas Gohr                }
19440b17fdc6SAndreas Gohr            } elseif (($mode == 'ifd1') && ($tag == 0x0117)) {  // TIFFStripByteCounts
194555efc227SAndreas Gohr                if (isset($this->_info['exif']['TIFFStrips'])) {
194655efc227SAndreas Gohr                    $value = strlen($this->_info['exif']['TIFFStrips']);
19470b17fdc6SAndreas Gohr                } else {
194855efc227SAndreas Gohr                    $value = null;
194955efc227SAndreas Gohr                }
19500b17fdc6SAndreas Gohr            } elseif (($mode == 'ifd1') && ($tag == 0x0201)) {  // TIFFJFIFOffset
195155efc227SAndreas Gohr                if (isset($this->_info['exif']['JFIFThumbnail'])) {
195255efc227SAndreas Gohr                    $value =& $this->_info['exif']['JFIFThumbnail'];
195355efc227SAndreas Gohr                    $type = -98;
19540b17fdc6SAndreas Gohr                } else {
195555efc227SAndreas Gohr                    $value = null;
195655efc227SAndreas Gohr                }
19570b17fdc6SAndreas Gohr            } elseif (($mode == 'ifd1') && ($tag == 0x0202)) {  // TIFFJFIFLength
195855efc227SAndreas Gohr                if (isset($this->_info['exif']['JFIFThumbnail'])) {
195955efc227SAndreas Gohr                    $value = strlen($this->_info['exif']['JFIFThumbnail']);
19600b17fdc6SAndreas Gohr                } else {
196155efc227SAndreas Gohr                    $value = null;
196255efc227SAndreas Gohr                }
19630b17fdc6SAndreas Gohr            } elseif (($mode == 'exif') && ($tag == 0xA005)) {  // InteropIFDOffset
196455efc227SAndreas Gohr                if (isset($this->_info['exif']['InteroperabilityIndex'])) {
196555efc227SAndreas Gohr                    $value =& $this->_getIFDEntries($isBigEndian, "interop");
196655efc227SAndreas Gohr                    $type = -99;
19670b17fdc6SAndreas Gohr                } else {
196855efc227SAndreas Gohr                    $value = null;
196955efc227SAndreas Gohr                }
19700b17fdc6SAndreas Gohr            } elseif (isset($this->_info['exif'][$name])) {
197155efc227SAndreas Gohr                $origValue =& $this->_info['exif'][$name];
197255efc227SAndreas Gohr
197355efc227SAndreas Gohr                // This makes it easier to process variable size elements
197455efc227SAndreas Gohr                if (!is_array($origValue) || isset($origValue['val'])) {
197555efc227SAndreas Gohr                    unset($origValue); // Break the reference
197655efc227SAndreas Gohr                    $origValue = array($this->_info['exif'][$name]);
197755efc227SAndreas Gohr                }
197855efc227SAndreas Gohr                $origCount = count($origValue);
197955efc227SAndreas Gohr
198055efc227SAndreas Gohr                if ($origCount == 0 ) {
198155efc227SAndreas Gohr                    $type = -1;  // To ignore this field
198255efc227SAndreas Gohr                }
198355efc227SAndreas Gohr
198455efc227SAndreas Gohr                $value = " ";
198555efc227SAndreas Gohr
198655efc227SAndreas Gohr                switch ($type) {
198755efc227SAndreas Gohr                    case 1:    // UBYTE
198855efc227SAndreas Gohr                        if ($count == 0) {
198955efc227SAndreas Gohr                            $count = $origCount;
199055efc227SAndreas Gohr                        }
199155efc227SAndreas Gohr
199255efc227SAndreas Gohr                        $j = 0;
199355efc227SAndreas Gohr                        while (($j < $count) && ($j < $origCount)) {
199455efc227SAndreas Gohr
199555efc227SAndreas Gohr                            $this->_putByte($value, $j, $origValue[$j]);
199655efc227SAndreas Gohr                            $j++;
199755efc227SAndreas Gohr                        }
199855efc227SAndreas Gohr
199955efc227SAndreas Gohr                        while ($j < $count) {
200055efc227SAndreas Gohr                            $this->_putByte($value, $j, 0);
200155efc227SAndreas Gohr                            $j++;
200255efc227SAndreas Gohr                        }
200355efc227SAndreas Gohr                        break;
200455efc227SAndreas Gohr                    case 2:    // ASCII
200555efc227SAndreas Gohr                        $v = strval($origValue[0]);
200655efc227SAndreas Gohr                        if (($count != 0) && (strlen($v) > $count)) {
200755efc227SAndreas Gohr                            $v = substr($v, 0, $count);
200855efc227SAndreas Gohr                        }
200955efc227SAndreas Gohr                        elseif (($count > 0) && (strlen($v) < $count)) {
201055efc227SAndreas Gohr                            $v = str_pad($v, $count, "\0");
201155efc227SAndreas Gohr                        }
201255efc227SAndreas Gohr
201355efc227SAndreas Gohr                        $count = strlen($v);
201455efc227SAndreas Gohr
201555efc227SAndreas Gohr                        $this->_putString($value, 0, $v);
201655efc227SAndreas Gohr                        break;
201755efc227SAndreas Gohr                    case 3:    // USHORT
201855efc227SAndreas Gohr                        if ($count == 0) {
201955efc227SAndreas Gohr                            $count = $origCount;
202055efc227SAndreas Gohr                        }
202155efc227SAndreas Gohr
202255efc227SAndreas Gohr                        $j = 0;
202355efc227SAndreas Gohr                        while (($j < $count) && ($j < $origCount)) {
202455efc227SAndreas Gohr                            $this->_putShort($value, $j * 2, $origValue[$j], $isBigEndian);
202555efc227SAndreas Gohr                            $j++;
202655efc227SAndreas Gohr                        }
202755efc227SAndreas Gohr
202855efc227SAndreas Gohr                        while ($j < $count) {
202955efc227SAndreas Gohr                            $this->_putShort($value, $j * 2, 0, $isBigEndian);
203055efc227SAndreas Gohr                            $j++;
203155efc227SAndreas Gohr                        }
203255efc227SAndreas Gohr                        break;
203355efc227SAndreas Gohr                    case 4:    // ULONG
203455efc227SAndreas Gohr                        if ($count == 0) {
203555efc227SAndreas Gohr                            $count = $origCount;
203655efc227SAndreas Gohr                        }
203755efc227SAndreas Gohr
203855efc227SAndreas Gohr                        $j = 0;
203955efc227SAndreas Gohr                        while (($j < $count) && ($j < $origCount)) {
204055efc227SAndreas Gohr                            $this->_putLong($value, $j * 4, $origValue[$j], $isBigEndian);
204155efc227SAndreas Gohr                            $j++;
204255efc227SAndreas Gohr                        }
204355efc227SAndreas Gohr
204455efc227SAndreas Gohr                        while ($j < $count) {
204555efc227SAndreas Gohr                            $this->_putLong($value, $j * 4, 0, $isBigEndian);
204655efc227SAndreas Gohr                            $j++;
204755efc227SAndreas Gohr                        }
204855efc227SAndreas Gohr                        break;
204955efc227SAndreas Gohr                    case 5:    // URATIONAL
205055efc227SAndreas Gohr                        if ($count == 0) {
205155efc227SAndreas Gohr                            $count = $origCount;
205255efc227SAndreas Gohr                        }
205355efc227SAndreas Gohr
205455efc227SAndreas Gohr                        $j = 0;
205555efc227SAndreas Gohr                        while (($j < $count) && ($j < $origCount)) {
205655efc227SAndreas Gohr                            $v = $origValue[$j];
205755efc227SAndreas Gohr                            if (is_array($v)) {
205855efc227SAndreas Gohr                                $a = $v['num'];
205955efc227SAndreas Gohr                                $b = $v['den'];
206055efc227SAndreas Gohr                            }
206155efc227SAndreas Gohr                            else {
206255efc227SAndreas Gohr                                $a = 0;
206355efc227SAndreas Gohr                                $b = 0;
206455efc227SAndreas Gohr                                // TODO: Allow other types and convert them
206555efc227SAndreas Gohr                            }
206655efc227SAndreas Gohr                            $this->_putLong($value, $j * 8, $a, $isBigEndian);
206755efc227SAndreas Gohr                            $this->_putLong($value, ($j * 8) + 4, $b, $isBigEndian);
206855efc227SAndreas Gohr                            $j++;
206955efc227SAndreas Gohr                        }
207055efc227SAndreas Gohr
207155efc227SAndreas Gohr                        while ($j < $count) {
207255efc227SAndreas Gohr                            $this->_putLong($value, $j * 8, 0, $isBigEndian);
207355efc227SAndreas Gohr                            $this->_putLong($value, ($j * 8) + 4, 0, $isBigEndian);
207455efc227SAndreas Gohr                            $j++;
207555efc227SAndreas Gohr                        }
207655efc227SAndreas Gohr                        break;
207755efc227SAndreas Gohr                    case 6:    // SBYTE
207855efc227SAndreas Gohr                        if ($count == 0) {
207955efc227SAndreas Gohr                            $count = $origCount;
208055efc227SAndreas Gohr                        }
208155efc227SAndreas Gohr
208255efc227SAndreas Gohr                        $j = 0;
208355efc227SAndreas Gohr                        while (($j < $count) && ($j < $origCount)) {
208455efc227SAndreas Gohr                            $this->_putByte($value, $j, $origValue[$j]);
208555efc227SAndreas Gohr                            $j++;
208655efc227SAndreas Gohr                        }
208755efc227SAndreas Gohr
208855efc227SAndreas Gohr                        while ($j < $count) {
208955efc227SAndreas Gohr                            $this->_putByte($value, $j, 0);
209055efc227SAndreas Gohr                            $j++;
209155efc227SAndreas Gohr                        }
209255efc227SAndreas Gohr                        break;
209355efc227SAndreas Gohr                    case 7:    // UNDEFINED
209455efc227SAndreas Gohr                        $v = strval($origValue[0]);
209555efc227SAndreas Gohr                        if (($count != 0) && (strlen($v) > $count)) {
209655efc227SAndreas Gohr                            $v = substr($v, 0, $count);
209755efc227SAndreas Gohr                        }
209855efc227SAndreas Gohr                        elseif (($count > 0) && (strlen($v) < $count)) {
209955efc227SAndreas Gohr                            $v = str_pad($v, $count, "\0");
210055efc227SAndreas Gohr                        }
210155efc227SAndreas Gohr
210255efc227SAndreas Gohr                        $count = strlen($v);
210355efc227SAndreas Gohr
210455efc227SAndreas Gohr                        $this->_putString($value, 0, $v);
210555efc227SAndreas Gohr                        break;
210655efc227SAndreas Gohr                    case 8:    // SSHORT
210755efc227SAndreas Gohr                        if ($count == 0) {
210855efc227SAndreas Gohr                            $count = $origCount;
210955efc227SAndreas Gohr                        }
211055efc227SAndreas Gohr
211155efc227SAndreas Gohr                        $j = 0;
211255efc227SAndreas Gohr                        while (($j < $count) && ($j < $origCount)) {
211355efc227SAndreas Gohr                            $this->_putShort($value, $j * 2, $origValue[$j], $isBigEndian);
211455efc227SAndreas Gohr                            $j++;
211555efc227SAndreas Gohr                        }
211655efc227SAndreas Gohr
211755efc227SAndreas Gohr                        while ($j < $count) {
211855efc227SAndreas Gohr                            $this->_putShort($value, $j * 2, 0, $isBigEndian);
211955efc227SAndreas Gohr                            $j++;
212055efc227SAndreas Gohr                        }
212155efc227SAndreas Gohr                        break;
212255efc227SAndreas Gohr                    case 9:    // SLONG
212355efc227SAndreas Gohr                        if ($count == 0) {
212455efc227SAndreas Gohr                            $count = $origCount;
212555efc227SAndreas Gohr                        }
212655efc227SAndreas Gohr
212755efc227SAndreas Gohr                        $j = 0;
212855efc227SAndreas Gohr                        while (($j < $count) && ($j < $origCount)) {
212955efc227SAndreas Gohr                            $this->_putLong($value, $j * 4, $origValue[$j], $isBigEndian);
213055efc227SAndreas Gohr                            $j++;
213155efc227SAndreas Gohr                        }
213255efc227SAndreas Gohr
213355efc227SAndreas Gohr                        while ($j < $count) {
213455efc227SAndreas Gohr                            $this->_putLong($value, $j * 4, 0, $isBigEndian);
213555efc227SAndreas Gohr                            $j++;
213655efc227SAndreas Gohr                        }
213755efc227SAndreas Gohr                        break;
213855efc227SAndreas Gohr                    case 10:   // SRATIONAL
213955efc227SAndreas Gohr                        if ($count == 0) {
214055efc227SAndreas Gohr                            $count = $origCount;
214155efc227SAndreas Gohr                        }
214255efc227SAndreas Gohr
214355efc227SAndreas Gohr                        $j = 0;
214455efc227SAndreas Gohr                        while (($j < $count) && ($j < $origCount)) {
214555efc227SAndreas Gohr                            $v = $origValue[$j];
214655efc227SAndreas Gohr                            if (is_array($v)) {
214755efc227SAndreas Gohr                                $a = $v['num'];
214855efc227SAndreas Gohr                                $b = $v['den'];
214955efc227SAndreas Gohr                            }
215055efc227SAndreas Gohr                            else {
215155efc227SAndreas Gohr                                $a = 0;
215255efc227SAndreas Gohr                                $b = 0;
215355efc227SAndreas Gohr                                // TODO: Allow other types and convert them
215455efc227SAndreas Gohr                            }
215555efc227SAndreas Gohr
215655efc227SAndreas Gohr                            $this->_putLong($value, $j * 8, $a, $isBigEndian);
215755efc227SAndreas Gohr                            $this->_putLong($value, ($j * 8) + 4, $b, $isBigEndian);
215855efc227SAndreas Gohr                            $j++;
215955efc227SAndreas Gohr                        }
216055efc227SAndreas Gohr
216155efc227SAndreas Gohr                        while ($j < $count) {
216255efc227SAndreas Gohr                            $this->_putLong($value, $j * 8, 0, $isBigEndian);
216355efc227SAndreas Gohr                            $this->_putLong($value, ($j * 8) + 4, 0, $isBigEndian);
216455efc227SAndreas Gohr                            $j++;
216555efc227SAndreas Gohr                        }
216655efc227SAndreas Gohr                        break;
216755efc227SAndreas Gohr                    case 11:   // FLOAT
216855efc227SAndreas Gohr                        if ($count == 0) {
216955efc227SAndreas Gohr                            $count = $origCount;
217055efc227SAndreas Gohr                        }
217155efc227SAndreas Gohr
217255efc227SAndreas Gohr                        $j = 0;
217355efc227SAndreas Gohr                        while (($j < $count) && ($j < $origCount)) {
217455efc227SAndreas Gohr                            $v = strval($origValue[$j]);
217555efc227SAndreas Gohr                            if (strlen($v) > 4) {
217655efc227SAndreas Gohr                                $v = substr($v, 0, 4);
217755efc227SAndreas Gohr                            }
217855efc227SAndreas Gohr                            elseif (strlen($v) < 4) {
217955efc227SAndreas Gohr                                $v = str_pad($v, 4, "\0");
218055efc227SAndreas Gohr                            }
218155efc227SAndreas Gohr                            $this->_putString($value, $j * 4, $v);
218255efc227SAndreas Gohr                            $j++;
218355efc227SAndreas Gohr                        }
218455efc227SAndreas Gohr
218555efc227SAndreas Gohr                        while ($j < $count) {
218655efc227SAndreas Gohr                            $this->_putString($value, $j * 4, "\0\0\0\0");
218755efc227SAndreas Gohr                            $j++;
218855efc227SAndreas Gohr                        }
218955efc227SAndreas Gohr                        break;
219055efc227SAndreas Gohr                    case 12:   // DFLOAT
219155efc227SAndreas Gohr                        if ($count == 0) {
219255efc227SAndreas Gohr                            $count = $origCount;
219355efc227SAndreas Gohr                        }
219455efc227SAndreas Gohr
219555efc227SAndreas Gohr                        $j = 0;
219655efc227SAndreas Gohr                        while (($j < $count) && ($j < $origCount)) {
219755efc227SAndreas Gohr                            $v = strval($origValue[$j]);
219855efc227SAndreas Gohr                            if (strlen($v) > 8) {
219955efc227SAndreas Gohr                                $v = substr($v, 0, 8);
220055efc227SAndreas Gohr                            }
220155efc227SAndreas Gohr                            elseif (strlen($v) < 8) {
220255efc227SAndreas Gohr                                $v = str_pad($v, 8, "\0");
220355efc227SAndreas Gohr                            }
220455efc227SAndreas Gohr                            $this->_putString($value, $j * 8, $v);
220555efc227SAndreas Gohr                            $j++;
220655efc227SAndreas Gohr                        }
220755efc227SAndreas Gohr
220855efc227SAndreas Gohr                        while ($j < $count) {
220955efc227SAndreas Gohr                            $this->_putString($value, $j * 8, "\0\0\0\0\0\0\0\0");
221055efc227SAndreas Gohr                            $j++;
221155efc227SAndreas Gohr                        }
221255efc227SAndreas Gohr                        break;
221355efc227SAndreas Gohr                    default:
221455efc227SAndreas Gohr                        $value = null;
221555efc227SAndreas Gohr                        break;
221655efc227SAndreas Gohr                }
221755efc227SAndreas Gohr            }
221855efc227SAndreas Gohr
221955efc227SAndreas Gohr            if ($value != null) {
222055efc227SAndreas Gohr                $ifdEntries[$entryCount] = array();
222155efc227SAndreas Gohr                $ifdEntries[$entryCount]['tag'] = $tag;
222255efc227SAndreas Gohr                $ifdEntries[$entryCount]['type'] = $type;
222355efc227SAndreas Gohr                $ifdEntries[$entryCount]['count'] = $count;
222455efc227SAndreas Gohr                $ifdEntries[$entryCount]['value'] = $value;
222555efc227SAndreas Gohr
222655efc227SAndreas Gohr                $entryCount++;
222755efc227SAndreas Gohr            }
222855efc227SAndreas Gohr        }
222955efc227SAndreas Gohr
223055efc227SAndreas Gohr        return $ifdEntries;
223155efc227SAndreas Gohr    }
223255efc227SAndreas Gohr
223355efc227SAndreas Gohr    /*************************************************************/
22340b17fdc6SAndreas Gohr    function _parseMarkerAdobe() {
223555efc227SAndreas Gohr        if (!isset($this->_markers)) {
223655efc227SAndreas Gohr            $this->_readJPEG();
223755efc227SAndreas Gohr        }
223855efc227SAndreas Gohr
223955efc227SAndreas Gohr        if ($this->_markers == null) {
224055efc227SAndreas Gohr            return false;
224155efc227SAndreas Gohr        }
224255efc227SAndreas Gohr
224355efc227SAndreas Gohr        $data = null;
224455efc227SAndreas Gohr        $count = count($this->_markers);
224555efc227SAndreas Gohr        for ($i = 0; $i < $count; $i++) {
224655efc227SAndreas Gohr            if ($this->_markers[$i]['marker'] == 0xED) {
224755efc227SAndreas Gohr                $signature = $this->_getFixedString($this->_markers[$i]['data'], 0, 14);
224855efc227SAndreas Gohr                if ($signature == "Photoshop 3.0\0") {
224955efc227SAndreas Gohr                    $data =& $this->_markers[$i]['data'];
225055efc227SAndreas Gohr                    break;
225155efc227SAndreas Gohr                }
225255efc227SAndreas Gohr            }
225355efc227SAndreas Gohr        }
225455efc227SAndreas Gohr
225555efc227SAndreas Gohr        if ($data == null) {
225655efc227SAndreas Gohr            $this->_info['adobe'] = false;
225755efc227SAndreas Gohr            $this->_info['iptc'] = false;
225855efc227SAndreas Gohr            return false;
225955efc227SAndreas Gohr        }
226055efc227SAndreas Gohr        $pos = 14;
226155efc227SAndreas Gohr        $this->_info['adobe'] = array();
226255efc227SAndreas Gohr        $this->_info['adobe']['raw'] = array();
226355efc227SAndreas Gohr        $this->_info['iptc'] = array();
226455efc227SAndreas Gohr
226555efc227SAndreas Gohr        $datasize = strlen($data);
226655efc227SAndreas Gohr
226755efc227SAndreas Gohr        while ($pos < $datasize) {
226855efc227SAndreas Gohr            $signature = $this->_getFixedString($data, $pos, 4);
226955efc227SAndreas Gohr            if ($signature != '8BIM')
227055efc227SAndreas Gohr                return false;
227155efc227SAndreas Gohr            $pos += 4;
227255efc227SAndreas Gohr
227355efc227SAndreas Gohr            $type = $this->_getShort($data, $pos);
227455efc227SAndreas Gohr            $pos += 2;
227555efc227SAndreas Gohr
227655efc227SAndreas Gohr            $strlen = $this->_getByte($data, $pos);
227755efc227SAndreas Gohr            $pos += 1;
227855efc227SAndreas Gohr            $header = '';
227955efc227SAndreas Gohr            for ($i = 0; $i < $strlen; $i++) {
228055efc227SAndreas Gohr                $header .= $data{$pos + $i};
228155efc227SAndreas Gohr            }
228255efc227SAndreas Gohr            $pos += $strlen + 1 - ($strlen % 2);  // The string is padded to even length, counting the length byte itself
228355efc227SAndreas Gohr
228455efc227SAndreas Gohr            $length = $this->_getLong($data, $pos);
228555efc227SAndreas Gohr            $pos += 4;
228655efc227SAndreas Gohr
228755efc227SAndreas Gohr            $basePos = $pos;
228855efc227SAndreas Gohr
228955efc227SAndreas Gohr            switch ($type) {
229055efc227SAndreas Gohr                case 0x0404: // Caption (IPTC Data)
229155efc227SAndreas Gohr                    $pos = $this->_readIPTC($data, $pos);
229255efc227SAndreas Gohr                    if ($pos == false)
229355efc227SAndreas Gohr                        return false;
229455efc227SAndreas Gohr                    break;
229555efc227SAndreas Gohr                case 0x040A: // CopyrightFlag
229655efc227SAndreas Gohr                    $this->_info['adobe']['CopyrightFlag'] = $this->_getByte($data, $pos);
229755efc227SAndreas Gohr                    $pos += $length;
229855efc227SAndreas Gohr                    break;
229955efc227SAndreas Gohr                case 0x040B: // ImageURL
230055efc227SAndreas Gohr                    $this->_info['adobe']['ImageURL'] = $this->_getFixedString($data, $pos, $length);
230155efc227SAndreas Gohr                    $pos += $length;
230255efc227SAndreas Gohr                    break;
230355efc227SAndreas Gohr                case 0x040C: // Thumbnail
230455efc227SAndreas Gohr                    $aux = $this->_getLong($data, $pos);
230555efc227SAndreas Gohr                    $pos += 4;
230655efc227SAndreas Gohr                    if ($aux == 1) {
230755efc227SAndreas Gohr                        $this->_info['adobe']['ThumbnailWidth'] = $this->_getLong($data, $pos);
230855efc227SAndreas Gohr                        $pos += 4;
230955efc227SAndreas Gohr                        $this->_info['adobe']['ThumbnailHeight'] = $this->_getLong($data, $pos);
231055efc227SAndreas Gohr                        $pos += 4;
231155efc227SAndreas Gohr
231255efc227SAndreas Gohr                        $pos += 16; // Skip some data
231355efc227SAndreas Gohr
231455efc227SAndreas Gohr                        $this->_info['adobe']['ThumbnailData'] = $this->_getFixedString($data, $pos, $length - 28);
231555efc227SAndreas Gohr                        $pos += $length - 28;
231655efc227SAndreas Gohr                    }
231755efc227SAndreas Gohr                    break;
231855efc227SAndreas Gohr                default:
231955efc227SAndreas Gohr                    break;
232055efc227SAndreas Gohr            }
232155efc227SAndreas Gohr
232255efc227SAndreas Gohr            // We save all blocks, even those we recognized
232355efc227SAndreas Gohr            $label = sprintf('8BIM_0x%04x', $type);
232455efc227SAndreas Gohr            $this->_info['adobe']['raw'][$label] = array();
232555efc227SAndreas Gohr            $this->_info['adobe']['raw'][$label]['type'] = $type;
232655efc227SAndreas Gohr            $this->_info['adobe']['raw'][$label]['header'] = $header;
232755efc227SAndreas Gohr            $this->_info['adobe']['raw'][$label]['data'] =& $this->_getFixedString($data, $basePos, $length);
232855efc227SAndreas Gohr
232955efc227SAndreas Gohr            $pos = $basePos + $length + ($length % 2); // Even padding
233055efc227SAndreas Gohr        }
233155efc227SAndreas Gohr
233255efc227SAndreas Gohr    }
233355efc227SAndreas Gohr
233455efc227SAndreas Gohr    /*************************************************************/
23350b17fdc6SAndreas Gohr    function _readIPTC(&$data, $pos = 0) {
233655efc227SAndreas Gohr        $totalLength = strlen($data);
233755efc227SAndreas Gohr
233855efc227SAndreas Gohr        $IPTCTags =& $this->_iptcTagNames();
233955efc227SAndreas Gohr
234055efc227SAndreas Gohr        while ($pos < ($totalLength - 5)) {
234155efc227SAndreas Gohr            $signature = $this->_getShort($data, $pos);
234255efc227SAndreas Gohr            if ($signature != 0x1C02)
234355efc227SAndreas Gohr                return $pos;
234455efc227SAndreas Gohr            $pos += 2;
234555efc227SAndreas Gohr
234655efc227SAndreas Gohr            $type = $this->_getByte($data, $pos);
234755efc227SAndreas Gohr            $pos += 1;
234855efc227SAndreas Gohr            $length = $this->_getShort($data, $pos);
234955efc227SAndreas Gohr            $pos += 2;
235055efc227SAndreas Gohr
235155efc227SAndreas Gohr            $basePos = $pos;
235255efc227SAndreas Gohr            $label = '';
235355efc227SAndreas Gohr
235455efc227SAndreas Gohr            if (isset($IPTCTags[$type])) {
235555efc227SAndreas Gohr                $label = $IPTCTags[$type];
23560b17fdc6SAndreas Gohr            } else {
235755efc227SAndreas Gohr                $label = sprintf('IPTC_0x%02x', $type);
235855efc227SAndreas Gohr            }
235955efc227SAndreas Gohr
236055efc227SAndreas Gohr            if ($label != '') {
236155efc227SAndreas Gohr                if (isset($this->_info['iptc'][$label])) {
236255efc227SAndreas Gohr                    if (!is_array($this->_info['iptc'][$label])) {
236355efc227SAndreas Gohr                        $aux = array();
236455efc227SAndreas Gohr                        $aux[0] = $this->_info['iptc'][$label];
236555efc227SAndreas Gohr                        $this->_info['iptc'][$label] = $aux;
236655efc227SAndreas Gohr                    }
236755efc227SAndreas Gohr                    $this->_info['iptc'][$label][ count($this->_info['iptc'][$label]) ] = $this->_getFixedString($data, $pos, $length);
23680b17fdc6SAndreas Gohr                } else {
236955efc227SAndreas Gohr                    $this->_info['iptc'][$label] = $this->_getFixedString($data, $pos, $length);
237055efc227SAndreas Gohr                }
237155efc227SAndreas Gohr            }
237255efc227SAndreas Gohr
237355efc227SAndreas Gohr            $pos = $basePos + $length; // No padding
237455efc227SAndreas Gohr        }
237555efc227SAndreas Gohr        return $pos;
237655efc227SAndreas Gohr    }
237755efc227SAndreas Gohr
237855efc227SAndreas Gohr    /*************************************************************/
23790b17fdc6SAndreas Gohr    function & _createMarkerAdobe() {
238055efc227SAndreas Gohr        if (isset($this->_info['iptc'])) {
238155efc227SAndreas Gohr            if (!isset($this->_info['adobe'])) {
238255efc227SAndreas Gohr                $this->_info['adobe'] = array();
238355efc227SAndreas Gohr            }
238455efc227SAndreas Gohr            if (!isset($this->_info['adobe']['raw'])) {
238555efc227SAndreas Gohr                $this->_info['adobe']['raw'] = array();
238655efc227SAndreas Gohr            }
238755efc227SAndreas Gohr            if (!isset($this->_info['adobe']['raw']['8BIM_0x0404'])) {
238855efc227SAndreas Gohr                $this->_info['adobe']['raw']['8BIM_0x0404'] = array();
238955efc227SAndreas Gohr            }
239055efc227SAndreas Gohr            $this->_info['adobe']['raw']['8BIM_0x0404']['type'] = 0x0404;
239155efc227SAndreas Gohr            $this->_info['adobe']['raw']['8BIM_0x0404']['header'] = "Caption";
239255efc227SAndreas Gohr            $this->_info['adobe']['raw']['8BIM_0x0404']['data'] =& $this->_writeIPTC();
239355efc227SAndreas Gohr        }
239455efc227SAndreas Gohr
239555efc227SAndreas Gohr        if (isset($this->_info['adobe']['raw']) && (count($this->_info['adobe']['raw']) > 0)) {
239655efc227SAndreas Gohr            $data = "Photoshop 3.0\0";
239755efc227SAndreas Gohr            $pos = 14;
239855efc227SAndreas Gohr
239955efc227SAndreas Gohr            reset($this->_info['adobe']['raw']);
240055efc227SAndreas Gohr            while (list($key) = each($this->_info['adobe']['raw'])) {
240155efc227SAndreas Gohr                $pos = $this->_write8BIM(
240255efc227SAndreas Gohr                        $data,
240355efc227SAndreas Gohr                        $pos,
240455efc227SAndreas Gohr                        $this->_info['adobe']['raw'][$key]['type'],
240555efc227SAndreas Gohr                        $this->_info['adobe']['raw'][$key]['header'],
240655efc227SAndreas Gohr                        $this->_info['adobe']['raw'][$key]['data'] );
240755efc227SAndreas Gohr            }
240855efc227SAndreas Gohr        }
240955efc227SAndreas Gohr
241055efc227SAndreas Gohr        return $data;
241155efc227SAndreas Gohr    }
241255efc227SAndreas Gohr
241355efc227SAndreas Gohr    /*************************************************************/
24140b17fdc6SAndreas Gohr    function _write8BIM(&$data, $pos, $type, $header, &$value) {
241555efc227SAndreas Gohr        $signature = "8BIM";
241655efc227SAndreas Gohr
241755efc227SAndreas Gohr        $pos = $this->_putString($data, $pos, $signature);
241855efc227SAndreas Gohr        $pos = $this->_putShort($data, $pos, $type);
241955efc227SAndreas Gohr
242055efc227SAndreas Gohr        $len = strlen($header);
242155efc227SAndreas Gohr
242255efc227SAndreas Gohr        $pos = $this->_putByte($data, $pos, $len);
242355efc227SAndreas Gohr        $pos = $this->_putString($data, $pos, $header);
242455efc227SAndreas Gohr        if (($len % 2) == 0) {  // Even padding, including the length byte
242555efc227SAndreas Gohr            $pos = $this->_putByte($data, $pos, 0);
242655efc227SAndreas Gohr        }
242755efc227SAndreas Gohr
242855efc227SAndreas Gohr        $len = strlen($value);
242955efc227SAndreas Gohr        $pos = $this->_putLong($data, $pos, $len);
243055efc227SAndreas Gohr        $pos = $this->_putString($data, $pos, $value);
243155efc227SAndreas Gohr        if (($len % 2) != 0) {  // Even padding
243255efc227SAndreas Gohr            $pos = $this->_putByte($data, $pos, 0);
243355efc227SAndreas Gohr        }
243455efc227SAndreas Gohr        return $pos;
243555efc227SAndreas Gohr    }
243655efc227SAndreas Gohr
243755efc227SAndreas Gohr    /*************************************************************/
24380b17fdc6SAndreas Gohr    function & _writeIPTC() {
243955efc227SAndreas Gohr        $data = " ";
244055efc227SAndreas Gohr        $pos = 0;
244155efc227SAndreas Gohr
244255efc227SAndreas Gohr        $IPTCNames =& $this->_iptcNameTags();
244355efc227SAndreas Gohr
244455efc227SAndreas Gohr        reset($this->_info['iptc']);
244555efc227SAndreas Gohr
244655efc227SAndreas Gohr        while (list($label) = each($this->_info['iptc'])) {
244755efc227SAndreas Gohr            $value =& $this->_info['iptc'][$label];
244855efc227SAndreas Gohr            $type = -1;
244955efc227SAndreas Gohr
245055efc227SAndreas Gohr            if (isset($IPTCNames[$label])) {
245155efc227SAndreas Gohr                $type = $IPTCNames[$label];
245255efc227SAndreas Gohr            }
245355efc227SAndreas Gohr            elseif (substr($label, 0, 7) == "IPTC_0x") {
245455efc227SAndreas Gohr                $type = hexdec(substr($label, 7, 2));
245555efc227SAndreas Gohr            }
245655efc227SAndreas Gohr
245755efc227SAndreas Gohr            if ($type != -1) {
245855efc227SAndreas Gohr                if (is_array($value)) {
24590b17fdc6SAndreas Gohr                    $vcnt = count($value);
24600b17fdc6SAndreas Gohr                    for ($i = 0; $i < $vcnt; $i++) {
246155efc227SAndreas Gohr                        $pos = $this->_writeIPTCEntry($data, $pos, $type, $value[$i]);
246255efc227SAndreas Gohr                    }
246355efc227SAndreas Gohr                }
246455efc227SAndreas Gohr                else {
246555efc227SAndreas Gohr                    $pos = $this->_writeIPTCEntry($data, $pos, $type, $value);
246655efc227SAndreas Gohr                }
246755efc227SAndreas Gohr            }
246855efc227SAndreas Gohr        }
246955efc227SAndreas Gohr
247055efc227SAndreas Gohr        return $data;
247155efc227SAndreas Gohr    }
247255efc227SAndreas Gohr
247355efc227SAndreas Gohr    /*************************************************************/
24740b17fdc6SAndreas Gohr    function _writeIPTCEntry(&$data, $pos, $type, &$value) {
247555efc227SAndreas Gohr        $pos = $this->_putShort($data, $pos, 0x1C02);
247655efc227SAndreas Gohr        $pos = $this->_putByte($data, $pos, $type);
247755efc227SAndreas Gohr        $pos = $this->_putShort($data, $pos, strlen($value));
247855efc227SAndreas Gohr        $pos = $this->_putString($data, $pos, $value);
247955efc227SAndreas Gohr
248055efc227SAndreas Gohr        return $pos;
248155efc227SAndreas Gohr    }
248255efc227SAndreas Gohr
248355efc227SAndreas Gohr    /*************************************************************/
24840b17fdc6SAndreas Gohr    function _exifTagNames($mode) {
248555efc227SAndreas Gohr        $tags = array();
248655efc227SAndreas Gohr
248755efc227SAndreas Gohr        if ($mode == 'ifd0') {
248855efc227SAndreas Gohr            $tags[0x010E] = 'ImageDescription';
248955efc227SAndreas Gohr            $tags[0x010F] = 'Make';
249055efc227SAndreas Gohr            $tags[0x0110] = 'Model';
249155efc227SAndreas Gohr            $tags[0x0112] = 'Orientation';
249255efc227SAndreas Gohr            $tags[0x011A] = 'XResolution';
249355efc227SAndreas Gohr            $tags[0x011B] = 'YResolution';
249455efc227SAndreas Gohr            $tags[0x0128] = 'ResolutionUnit';
249555efc227SAndreas Gohr            $tags[0x0131] = 'Software';
249655efc227SAndreas Gohr            $tags[0x0132] = 'DateTime';
249755efc227SAndreas Gohr            $tags[0x013B] = 'Artist';
249855efc227SAndreas Gohr            $tags[0x013E] = 'WhitePoint';
249955efc227SAndreas Gohr            $tags[0x013F] = 'PrimaryChromaticities';
250055efc227SAndreas Gohr            $tags[0x0211] = 'YCbCrCoefficients';
250155efc227SAndreas Gohr            $tags[0x0212] = 'YCbCrSubSampling';
250255efc227SAndreas Gohr            $tags[0x0213] = 'YCbCrPositioning';
250355efc227SAndreas Gohr            $tags[0x0214] = 'ReferenceBlackWhite';
250455efc227SAndreas Gohr            $tags[0x8298] = 'Copyright';
250555efc227SAndreas Gohr            $tags[0x8769] = 'ExifIFDOffset';
250655efc227SAndreas Gohr            $tags[0x8825] = 'GPSIFDOffset';
250755efc227SAndreas Gohr        }
250855efc227SAndreas Gohr        if ($mode == 'ifd1') {
250955efc227SAndreas Gohr            $tags[0x00FE] = 'TIFFNewSubfileType';
251055efc227SAndreas Gohr            $tags[0x00FF] = 'TIFFSubfileType';
251155efc227SAndreas Gohr            $tags[0x0100] = 'TIFFImageWidth';
251255efc227SAndreas Gohr            $tags[0x0101] = 'TIFFImageHeight';
251355efc227SAndreas Gohr            $tags[0x0102] = 'TIFFBitsPerSample';
251455efc227SAndreas Gohr            $tags[0x0103] = 'TIFFCompression';
251555efc227SAndreas Gohr            $tags[0x0106] = 'TIFFPhotometricInterpretation';
251655efc227SAndreas Gohr            $tags[0x0107] = 'TIFFThreshholding';
251755efc227SAndreas Gohr            $tags[0x0108] = 'TIFFCellWidth';
251855efc227SAndreas Gohr            $tags[0x0109] = 'TIFFCellLength';
251955efc227SAndreas Gohr            $tags[0x010A] = 'TIFFFillOrder';
252055efc227SAndreas Gohr            $tags[0x010E] = 'TIFFImageDescription';
252155efc227SAndreas Gohr            $tags[0x010F] = 'TIFFMake';
252255efc227SAndreas Gohr            $tags[0x0110] = 'TIFFModel';
252355efc227SAndreas Gohr            $tags[0x0111] = 'TIFFStripOffsets';
252455efc227SAndreas Gohr            $tags[0x0112] = 'TIFFOrientation';
252555efc227SAndreas Gohr            $tags[0x0115] = 'TIFFSamplesPerPixel';
252655efc227SAndreas Gohr            $tags[0x0116] = 'TIFFRowsPerStrip';
252755efc227SAndreas Gohr            $tags[0x0117] = 'TIFFStripByteCounts';
252855efc227SAndreas Gohr            $tags[0x0118] = 'TIFFMinSampleValue';
252955efc227SAndreas Gohr            $tags[0x0119] = 'TIFFMaxSampleValue';
253055efc227SAndreas Gohr            $tags[0x011A] = 'TIFFXResolution';
253155efc227SAndreas Gohr            $tags[0x011B] = 'TIFFYResolution';
253255efc227SAndreas Gohr            $tags[0x011C] = 'TIFFPlanarConfiguration';
253355efc227SAndreas Gohr            $tags[0x0122] = 'TIFFGrayResponseUnit';
253455efc227SAndreas Gohr            $tags[0x0123] = 'TIFFGrayResponseCurve';
253555efc227SAndreas Gohr            $tags[0x0128] = 'TIFFResolutionUnit';
253655efc227SAndreas Gohr            $tags[0x0131] = 'TIFFSoftware';
253755efc227SAndreas Gohr            $tags[0x0132] = 'TIFFDateTime';
253855efc227SAndreas Gohr            $tags[0x013B] = 'TIFFArtist';
253955efc227SAndreas Gohr            $tags[0x013C] = 'TIFFHostComputer';
254055efc227SAndreas Gohr            $tags[0x0140] = 'TIFFColorMap';
254155efc227SAndreas Gohr            $tags[0x0152] = 'TIFFExtraSamples';
254255efc227SAndreas Gohr            $tags[0x0201] = 'TIFFJFIFOffset';
254355efc227SAndreas Gohr            $tags[0x0202] = 'TIFFJFIFLength';
254455efc227SAndreas Gohr            $tags[0x0211] = 'TIFFYCbCrCoefficients';
254555efc227SAndreas Gohr            $tags[0x0212] = 'TIFFYCbCrSubSampling';
254655efc227SAndreas Gohr            $tags[0x0213] = 'TIFFYCbCrPositioning';
254755efc227SAndreas Gohr            $tags[0x0214] = 'TIFFReferenceBlackWhite';
254855efc227SAndreas Gohr            $tags[0x8298] = 'TIFFCopyright';
254955efc227SAndreas Gohr            $tags[0x9286] = 'TIFFUserComment';
25500b17fdc6SAndreas Gohr        } elseif ($mode == 'exif') {
255155efc227SAndreas Gohr            $tags[0x829A] = 'ExposureTime';
255255efc227SAndreas Gohr            $tags[0x829D] = 'FNumber';
255355efc227SAndreas Gohr            $tags[0x8822] = 'ExposureProgram';
255455efc227SAndreas Gohr            $tags[0x8824] = 'SpectralSensitivity';
255555efc227SAndreas Gohr            $tags[0x8827] = 'ISOSpeedRatings';
255655efc227SAndreas Gohr            $tags[0x8828] = 'OECF';
255755efc227SAndreas Gohr            $tags[0x9000] = 'EXIFVersion';
255855efc227SAndreas Gohr            $tags[0x9003] = 'DatetimeOriginal';
255955efc227SAndreas Gohr            $tags[0x9004] = 'DatetimeDigitized';
256055efc227SAndreas Gohr            $tags[0x9101] = 'ComponentsConfiguration';
256155efc227SAndreas Gohr            $tags[0x9102] = 'CompressedBitsPerPixel';
256255efc227SAndreas Gohr            $tags[0x9201] = 'ShutterSpeedValue';
256355efc227SAndreas Gohr            $tags[0x9202] = 'ApertureValue';
256455efc227SAndreas Gohr            $tags[0x9203] = 'BrightnessValue';
256555efc227SAndreas Gohr            $tags[0x9204] = 'ExposureBiasValue';
256655efc227SAndreas Gohr            $tags[0x9205] = 'MaxApertureValue';
256755efc227SAndreas Gohr            $tags[0x9206] = 'SubjectDistance';
256855efc227SAndreas Gohr            $tags[0x9207] = 'MeteringMode';
256955efc227SAndreas Gohr            $tags[0x9208] = 'LightSource';
257055efc227SAndreas Gohr            $tags[0x9209] = 'Flash';
257155efc227SAndreas Gohr            $tags[0x920A] = 'FocalLength';
257255efc227SAndreas Gohr            $tags[0x927C] = 'MakerNote';
257355efc227SAndreas Gohr            $tags[0x9286] = 'UserComment';
257455efc227SAndreas Gohr            $tags[0x9290] = 'SubSecTime';
257555efc227SAndreas Gohr            $tags[0x9291] = 'SubSecTimeOriginal';
257655efc227SAndreas Gohr            $tags[0x9292] = 'SubSecTimeDigitized';
257755efc227SAndreas Gohr            $tags[0xA000] = 'FlashPixVersion';
257855efc227SAndreas Gohr            $tags[0xA001] = 'ColorSpace';
257955efc227SAndreas Gohr            $tags[0xA002] = 'PixelXDimension';
258055efc227SAndreas Gohr            $tags[0xA003] = 'PixelYDimension';
258155efc227SAndreas Gohr            $tags[0xA004] = 'RelatedSoundFile';
258255efc227SAndreas Gohr            $tags[0xA005] = 'InteropIFDOffset';
258355efc227SAndreas Gohr            $tags[0xA20B] = 'FlashEnergy';
258455efc227SAndreas Gohr            $tags[0xA20C] = 'SpatialFrequencyResponse';
258555efc227SAndreas Gohr            $tags[0xA20E] = 'FocalPlaneXResolution';
258655efc227SAndreas Gohr            $tags[0xA20F] = 'FocalPlaneYResolution';
258755efc227SAndreas Gohr            $tags[0xA210] = 'FocalPlaneResolutionUnit';
258855efc227SAndreas Gohr            $tags[0xA214] = 'SubjectLocation';
258955efc227SAndreas Gohr            $tags[0xA215] = 'ExposureIndex';
259055efc227SAndreas Gohr            $tags[0xA217] = 'SensingMethod';
259155efc227SAndreas Gohr            $tags[0xA300] = 'FileSource';
259255efc227SAndreas Gohr            $tags[0xA301] = 'SceneType';
259355efc227SAndreas Gohr            $tags[0xA302] = 'CFAPattern';
25940b17fdc6SAndreas Gohr        } elseif ($mode == 'interop') {
259555efc227SAndreas Gohr            $tags[0x0001] = 'InteroperabilityIndex';
259655efc227SAndreas Gohr            $tags[0x0002] = 'InteroperabilityVersion';
259755efc227SAndreas Gohr            $tags[0x1000] = 'RelatedImageFileFormat';
259855efc227SAndreas Gohr            $tags[0x1001] = 'RelatedImageWidth';
259955efc227SAndreas Gohr            $tags[0x1002] = 'RelatedImageLength';
26000b17fdc6SAndreas Gohr        } elseif ($mode == 'gps') {
260155efc227SAndreas Gohr            $tags[0x0000] = 'GPSVersionID';
260255efc227SAndreas Gohr            $tags[0x0001] = 'GPSLatitudeRef';
260355efc227SAndreas Gohr            $tags[0x0002] = 'GPSLatitude';
260455efc227SAndreas Gohr            $tags[0x0003] = 'GPSLongitudeRef';
260555efc227SAndreas Gohr            $tags[0x0004] = 'GPSLongitude';
260655efc227SAndreas Gohr            $tags[0x0005] = 'GPSAltitudeRef';
260755efc227SAndreas Gohr            $tags[0x0006] = 'GPSAltitude';
260855efc227SAndreas Gohr            $tags[0x0007] = 'GPSTimeStamp';
260955efc227SAndreas Gohr            $tags[0x0008] = 'GPSSatellites';
261055efc227SAndreas Gohr            $tags[0x0009] = 'GPSStatus';
261155efc227SAndreas Gohr            $tags[0x000A] = 'GPSMeasureMode';
261255efc227SAndreas Gohr            $tags[0x000B] = 'GPSDOP';
261355efc227SAndreas Gohr            $tags[0x000C] = 'GPSSpeedRef';
261455efc227SAndreas Gohr            $tags[0x000D] = 'GPSSpeed';
261555efc227SAndreas Gohr            $tags[0x000E] = 'GPSTrackRef';
261655efc227SAndreas Gohr            $tags[0x000F] = 'GPSTrack';
261755efc227SAndreas Gohr            $tags[0x0010] = 'GPSImgDirectionRef';
261855efc227SAndreas Gohr            $tags[0x0011] = 'GPSImgDirection';
261955efc227SAndreas Gohr            $tags[0x0012] = 'GPSMapDatum';
262055efc227SAndreas Gohr            $tags[0x0013] = 'GPSDestLatitudeRef';
262155efc227SAndreas Gohr            $tags[0x0014] = 'GPSDestLatitude';
262255efc227SAndreas Gohr            $tags[0x0015] = 'GPSDestLongitudeRef';
262355efc227SAndreas Gohr            $tags[0x0016] = 'GPSDestLongitude';
262455efc227SAndreas Gohr            $tags[0x0017] = 'GPSDestBearingRef';
262555efc227SAndreas Gohr            $tags[0x0018] = 'GPSDestBearing';
262655efc227SAndreas Gohr            $tags[0x0019] = 'GPSDestDistanceRef';
262755efc227SAndreas Gohr            $tags[0x001A] = 'GPSDestDistance';
262855efc227SAndreas Gohr        }
262955efc227SAndreas Gohr
263055efc227SAndreas Gohr        return $tags;
263155efc227SAndreas Gohr    }
263255efc227SAndreas Gohr
263355efc227SAndreas Gohr    /*************************************************************/
26340b17fdc6SAndreas Gohr    function _exifTagTypes($mode) {
263555efc227SAndreas Gohr        $tags = array();
263655efc227SAndreas Gohr
263755efc227SAndreas Gohr        if ($mode == 'ifd0') {
263855efc227SAndreas Gohr            $tags[0x010E] = array(2, 0); // ImageDescription -> ASCII, Any
263955efc227SAndreas Gohr            $tags[0x010F] = array(2, 0); // Make -> ASCII, Any
264055efc227SAndreas Gohr            $tags[0x0110] = array(2, 0); // Model -> ASCII, Any
264155efc227SAndreas Gohr            $tags[0x0112] = array(3, 1); // Orientation -> SHORT, 1
264255efc227SAndreas Gohr            $tags[0x011A] = array(5, 1); // XResolution -> RATIONAL, 1
264355efc227SAndreas Gohr            $tags[0x011B] = array(5, 1); // YResolution -> RATIONAL, 1
264455efc227SAndreas Gohr            $tags[0x0128] = array(3, 1); // ResolutionUnit -> SHORT
264555efc227SAndreas Gohr            $tags[0x0131] = array(2, 0); // Software -> ASCII, Any
264655efc227SAndreas Gohr            $tags[0x0132] = array(2, 20); // DateTime -> ASCII, 20
264755efc227SAndreas Gohr            $tags[0x013B] = array(2, 0); // Artist -> ASCII, Any
264855efc227SAndreas Gohr            $tags[0x013E] = array(5, 2); // WhitePoint -> RATIONAL, 2
264955efc227SAndreas Gohr            $tags[0x013F] = array(5, 6); // PrimaryChromaticities -> RATIONAL, 6
265055efc227SAndreas Gohr            $tags[0x0211] = array(5, 3); // YCbCrCoefficients -> RATIONAL, 3
265155efc227SAndreas Gohr            $tags[0x0212] = array(3, 2); // YCbCrSubSampling -> SHORT, 2
265255efc227SAndreas Gohr            $tags[0x0213] = array(3, 1); // YCbCrPositioning -> SHORT, 1
265355efc227SAndreas Gohr            $tags[0x0214] = array(5, 6); // ReferenceBlackWhite -> RATIONAL, 6
265455efc227SAndreas Gohr            $tags[0x8298] = array(2, 0); // Copyright -> ASCII, Any
265555efc227SAndreas Gohr            $tags[0x8769] = array(4, 1); // ExifIFDOffset -> LONG, 1
265655efc227SAndreas Gohr            $tags[0x8825] = array(4, 1); // GPSIFDOffset -> LONG, 1
265755efc227SAndreas Gohr        }
265855efc227SAndreas Gohr        if ($mode == 'ifd1') {
265955efc227SAndreas Gohr            $tags[0x00FE] = array(4, 1); // TIFFNewSubfileType -> LONG, 1
266055efc227SAndreas Gohr            $tags[0x00FF] = array(3, 1); // TIFFSubfileType -> SHORT, 1
266155efc227SAndreas Gohr            $tags[0x0100] = array(4, 1); // TIFFImageWidth -> LONG (or SHORT), 1
266255efc227SAndreas Gohr            $tags[0x0101] = array(4, 1); // TIFFImageHeight -> LONG (or SHORT), 1
266355efc227SAndreas Gohr            $tags[0x0102] = array(3, 3); // TIFFBitsPerSample -> SHORT, 3
266455efc227SAndreas Gohr            $tags[0x0103] = array(3, 1); // TIFFCompression -> SHORT, 1
266555efc227SAndreas Gohr            $tags[0x0106] = array(3, 1); // TIFFPhotometricInterpretation -> SHORT, 1
266655efc227SAndreas Gohr            $tags[0x0107] = array(3, 1); // TIFFThreshholding -> SHORT, 1
266755efc227SAndreas Gohr            $tags[0x0108] = array(3, 1); // TIFFCellWidth -> SHORT, 1
266855efc227SAndreas Gohr            $tags[0x0109] = array(3, 1); // TIFFCellLength -> SHORT, 1
266955efc227SAndreas Gohr            $tags[0x010A] = array(3, 1); // TIFFFillOrder -> SHORT, 1
267055efc227SAndreas Gohr            $tags[0x010E] = array(2, 0); // TIFFImageDescription -> ASCII, Any
267155efc227SAndreas Gohr            $tags[0x010F] = array(2, 0); // TIFFMake -> ASCII, Any
267255efc227SAndreas Gohr            $tags[0x0110] = array(2, 0); // TIFFModel -> ASCII, Any
267355efc227SAndreas Gohr            $tags[0x0111] = array(4, 0); // TIFFStripOffsets -> LONG (or SHORT), Any (one per strip)
267455efc227SAndreas Gohr            $tags[0x0112] = array(3, 1); // TIFFOrientation -> SHORT, 1
267555efc227SAndreas Gohr            $tags[0x0115] = array(3, 1); // TIFFSamplesPerPixel -> SHORT, 1
267655efc227SAndreas Gohr            $tags[0x0116] = array(4, 1); // TIFFRowsPerStrip -> LONG (or SHORT), 1
267755efc227SAndreas Gohr            $tags[0x0117] = array(4, 0); // TIFFStripByteCounts -> LONG (or SHORT), Any (one per strip)
267855efc227SAndreas Gohr            $tags[0x0118] = array(3, 0); // TIFFMinSampleValue -> SHORT, Any (SamplesPerPixel)
267955efc227SAndreas Gohr            $tags[0x0119] = array(3, 0); // TIFFMaxSampleValue -> SHORT, Any (SamplesPerPixel)
268055efc227SAndreas Gohr            $tags[0x011A] = array(5, 1); // TIFFXResolution -> RATIONAL, 1
268155efc227SAndreas Gohr            $tags[0x011B] = array(5, 1); // TIFFYResolution -> RATIONAL, 1
268255efc227SAndreas Gohr            $tags[0x011C] = array(3, 1); // TIFFPlanarConfiguration -> SHORT, 1
268355efc227SAndreas Gohr            $tags[0x0122] = array(3, 1); // TIFFGrayResponseUnit -> SHORT, 1
268455efc227SAndreas Gohr            $tags[0x0123] = array(3, 0); // TIFFGrayResponseCurve -> SHORT, Any (2^BitsPerSample)
268555efc227SAndreas Gohr            $tags[0x0128] = array(3, 1); // TIFFResolutionUnit -> SHORT, 1
268655efc227SAndreas Gohr            $tags[0x0131] = array(2, 0); // TIFFSoftware -> ASCII, Any
268755efc227SAndreas Gohr            $tags[0x0132] = array(2, 20); // TIFFDateTime -> ASCII, 20
268855efc227SAndreas Gohr            $tags[0x013B] = array(2, 0); // TIFFArtist -> ASCII, Any
268955efc227SAndreas Gohr            $tags[0x013C] = array(2, 0); // TIFFHostComputer -> ASCII, Any
269055efc227SAndreas Gohr            $tags[0x0140] = array(3, 0); // TIFFColorMap -> SHORT, Any (3 * 2^BitsPerSample)
269155efc227SAndreas Gohr            $tags[0x0152] = array(3, 0); // TIFFExtraSamples -> SHORT, Any (SamplesPerPixel - 3)
269255efc227SAndreas Gohr            $tags[0x0201] = array(4, 1); // TIFFJFIFOffset -> LONG, 1
269355efc227SAndreas Gohr            $tags[0x0202] = array(4, 1); // TIFFJFIFLength -> LONG, 1
269455efc227SAndreas Gohr            $tags[0x0211] = array(5, 3); // TIFFYCbCrCoefficients -> RATIONAL, 3
269555efc227SAndreas Gohr            $tags[0x0212] = array(3, 2); // TIFFYCbCrSubSampling -> SHORT, 2
269655efc227SAndreas Gohr            $tags[0x0213] = array(3, 1); // TIFFYCbCrPositioning -> SHORT, 1
269755efc227SAndreas Gohr            $tags[0x0214] = array(5, 6); // TIFFReferenceBlackWhite -> RATIONAL, 6
269855efc227SAndreas Gohr            $tags[0x8298] = array(2, 0); // TIFFCopyright -> ASCII, Any
269955efc227SAndreas Gohr            $tags[0x9286] = array(2, 0); // TIFFUserComment -> ASCII, Any
27000b17fdc6SAndreas Gohr        } elseif ($mode == 'exif') {
270155efc227SAndreas Gohr            $tags[0x829A] = array(5, 1); // ExposureTime -> RATIONAL, 1
270255efc227SAndreas Gohr            $tags[0x829D] = array(5, 1); // FNumber -> RATIONAL, 1
270355efc227SAndreas Gohr            $tags[0x8822] = array(3, 1); // ExposureProgram -> SHORT, 1
270455efc227SAndreas Gohr            $tags[0x8824] = array(2, 0); // SpectralSensitivity -> ASCII, Any
270555efc227SAndreas Gohr            $tags[0x8827] = array(3, 0); // ISOSpeedRatings -> SHORT, Any
270655efc227SAndreas Gohr            $tags[0x8828] = array(7, 0); // OECF -> UNDEFINED, Any
270755efc227SAndreas Gohr            $tags[0x9000] = array(7, 4); // EXIFVersion -> UNDEFINED, 4
270855efc227SAndreas Gohr            $tags[0x9003] = array(2, 20); // DatetimeOriginal -> ASCII, 20
270955efc227SAndreas Gohr            $tags[0x9004] = array(2, 20); // DatetimeDigitized -> ASCII, 20
271055efc227SAndreas Gohr            $tags[0x9101] = array(7, 4); // ComponentsConfiguration -> UNDEFINED, 4
271155efc227SAndreas Gohr            $tags[0x9102] = array(5, 1); // CompressedBitsPerPixel -> RATIONAL, 1
271255efc227SAndreas Gohr            $tags[0x9201] = array(10, 1); // ShutterSpeedValue -> SRATIONAL, 1
271355efc227SAndreas Gohr            $tags[0x9202] = array(5, 1); // ApertureValue -> RATIONAL, 1
271455efc227SAndreas Gohr            $tags[0x9203] = array(10, 1); // BrightnessValue -> SRATIONAL, 1
271555efc227SAndreas Gohr            $tags[0x9204] = array(10, 1); // ExposureBiasValue -> SRATIONAL, 1
271655efc227SAndreas Gohr            $tags[0x9205] = array(5, 1); // MaxApertureValue -> RATIONAL, 1
271755efc227SAndreas Gohr            $tags[0x9206] = array(5, 1); // SubjectDistance -> RATIONAL, 1
271855efc227SAndreas Gohr            $tags[0x9207] = array(3, 1); // MeteringMode -> SHORT, 1
271955efc227SAndreas Gohr            $tags[0x9208] = array(3, 1); // LightSource -> SHORT, 1
272055efc227SAndreas Gohr            $tags[0x9209] = array(3, 1); // Flash -> SHORT, 1
272155efc227SAndreas Gohr            $tags[0x920A] = array(5, 1); // FocalLength -> RATIONAL, 1
272255efc227SAndreas Gohr            $tags[0x927C] = array(7, 0); // MakerNote -> UNDEFINED, Any
272355efc227SAndreas Gohr            $tags[0x9286] = array(7, 0); // UserComment -> UNDEFINED, Any
272455efc227SAndreas Gohr            $tags[0x9290] = array(2, 0); // SubSecTime -> ASCII, Any
272555efc227SAndreas Gohr            $tags[0x9291] = array(2, 0); // SubSecTimeOriginal -> ASCII, Any
272655efc227SAndreas Gohr            $tags[0x9292] = array(2, 0); // SubSecTimeDigitized -> ASCII, Any
272755efc227SAndreas Gohr            $tags[0xA000] = array(7, 4); // FlashPixVersion -> UNDEFINED, 4
272855efc227SAndreas Gohr            $tags[0xA001] = array(3, 1); // ColorSpace -> SHORT, 1
272955efc227SAndreas Gohr            $tags[0xA002] = array(4, 1); // PixelXDimension -> LONG (or SHORT), 1
273055efc227SAndreas Gohr            $tags[0xA003] = array(4, 1); // PixelYDimension -> LONG (or SHORT), 1
273155efc227SAndreas Gohr            $tags[0xA004] = array(2, 13); // RelatedSoundFile -> ASCII, 13
273255efc227SAndreas Gohr            $tags[0xA005] = array(4, 1); // InteropIFDOffset -> LONG, 1
273355efc227SAndreas Gohr            $tags[0xA20B] = array(5, 1); // FlashEnergy -> RATIONAL, 1
273455efc227SAndreas Gohr            $tags[0xA20C] = array(7, 0); // SpatialFrequencyResponse -> UNDEFINED, Any
273555efc227SAndreas Gohr            $tags[0xA20E] = array(5, 1); // FocalPlaneXResolution -> RATIONAL, 1
273655efc227SAndreas Gohr            $tags[0xA20F] = array(5, 1); // FocalPlaneYResolution -> RATIONAL, 1
273755efc227SAndreas Gohr            $tags[0xA210] = array(3, 1); // FocalPlaneResolutionUnit -> SHORT, 1
273855efc227SAndreas Gohr            $tags[0xA214] = array(3, 2); // SubjectLocation -> SHORT, 2
273955efc227SAndreas Gohr            $tags[0xA215] = array(5, 1); // ExposureIndex -> RATIONAL, 1
274055efc227SAndreas Gohr            $tags[0xA217] = array(3, 1); // SensingMethod -> SHORT, 1
274155efc227SAndreas Gohr            $tags[0xA300] = array(7, 1); // FileSource -> UNDEFINED, 1
274255efc227SAndreas Gohr            $tags[0xA301] = array(7, 1); // SceneType -> UNDEFINED, 1
274355efc227SAndreas Gohr            $tags[0xA302] = array(7, 0); // CFAPattern -> UNDEFINED, Any
27440b17fdc6SAndreas Gohr        } elseif ($mode == 'interop') {
274555efc227SAndreas Gohr            $tags[0x0001] = array(2, 0); // InteroperabilityIndex -> ASCII, Any
274655efc227SAndreas Gohr            $tags[0x0002] = array(7, 4); // InteroperabilityVersion -> UNKNOWN, 4
274755efc227SAndreas Gohr            $tags[0x1000] = array(2, 0); // RelatedImageFileFormat -> ASCII, Any
274855efc227SAndreas Gohr            $tags[0x1001] = array(4, 1); // RelatedImageWidth -> LONG (or SHORT), 1
274955efc227SAndreas Gohr            $tags[0x1002] = array(4, 1); // RelatedImageLength -> LONG (or SHORT), 1
27500b17fdc6SAndreas Gohr        } elseif ($mode == 'gps') {
275155efc227SAndreas Gohr            $tags[0x0000] = array(1, 4); // GPSVersionID -> BYTE, 4
275255efc227SAndreas Gohr            $tags[0x0001] = array(2, 2); // GPSLatitudeRef -> ASCII, 2
275355efc227SAndreas Gohr            $tags[0x0002] = array(5, 3); // GPSLatitude -> RATIONAL, 3
275455efc227SAndreas Gohr            $tags[0x0003] = array(2, 2); // GPSLongitudeRef -> ASCII, 2
275555efc227SAndreas Gohr            $tags[0x0004] = array(5, 3); // GPSLongitude -> RATIONAL, 3
275655efc227SAndreas Gohr            $tags[0x0005] = array(2, 2); // GPSAltitudeRef -> ASCII, 2
275755efc227SAndreas Gohr            $tags[0x0006] = array(5, 1); // GPSAltitude -> RATIONAL, 1
275855efc227SAndreas Gohr            $tags[0x0007] = array(5, 3); // GPSTimeStamp -> RATIONAL, 3
275955efc227SAndreas Gohr            $tags[0x0008] = array(2, 0); // GPSSatellites -> ASCII, Any
276055efc227SAndreas Gohr            $tags[0x0009] = array(2, 2); // GPSStatus -> ASCII, 2
276155efc227SAndreas Gohr            $tags[0x000A] = array(2, 2); // GPSMeasureMode -> ASCII, 2
276255efc227SAndreas Gohr            $tags[0x000B] = array(5, 1); // GPSDOP -> RATIONAL, 1
276355efc227SAndreas Gohr            $tags[0x000C] = array(2, 2); // GPSSpeedRef -> ASCII, 2
276455efc227SAndreas Gohr            $tags[0x000D] = array(5, 1); // GPSSpeed -> RATIONAL, 1
276555efc227SAndreas Gohr            $tags[0x000E] = array(2, 2); // GPSTrackRef -> ASCII, 2
276655efc227SAndreas Gohr            $tags[0x000F] = array(5, 1); // GPSTrack -> RATIONAL, 1
276755efc227SAndreas Gohr            $tags[0x0010] = array(2, 2); // GPSImgDirectionRef -> ASCII, 2
276855efc227SAndreas Gohr            $tags[0x0011] = array(5, 1); // GPSImgDirection -> RATIONAL, 1
276955efc227SAndreas Gohr            $tags[0x0012] = array(2, 0); // GPSMapDatum -> ASCII, Any
277055efc227SAndreas Gohr            $tags[0x0013] = array(2, 2); // GPSDestLatitudeRef -> ASCII, 2
277155efc227SAndreas Gohr            $tags[0x0014] = array(5, 3); // GPSDestLatitude -> RATIONAL, 3
277255efc227SAndreas Gohr            $tags[0x0015] = array(2, 2); // GPSDestLongitudeRef -> ASCII, 2
277355efc227SAndreas Gohr            $tags[0x0016] = array(5, 3); // GPSDestLongitude -> RATIONAL, 3
277455efc227SAndreas Gohr            $tags[0x0017] = array(2, 2); // GPSDestBearingRef -> ASCII, 2
277555efc227SAndreas Gohr            $tags[0x0018] = array(5, 1); // GPSDestBearing -> RATIONAL, 1
277655efc227SAndreas Gohr            $tags[0x0019] = array(2, 2); // GPSDestDistanceRef -> ASCII, 2
277755efc227SAndreas Gohr            $tags[0x001A] = array(5, 1); // GPSDestDistance -> RATIONAL, 1
277855efc227SAndreas Gohr        }
277955efc227SAndreas Gohr
278055efc227SAndreas Gohr        return $tags;
278155efc227SAndreas Gohr    }
278255efc227SAndreas Gohr
278355efc227SAndreas Gohr    /*************************************************************/
27840b17fdc6SAndreas Gohr    function _exifNameTags($mode) {
278555efc227SAndreas Gohr        $tags = $this->_exifTagNames($mode);
278655efc227SAndreas Gohr        return $this->_names2Tags($tags);
278755efc227SAndreas Gohr    }
278855efc227SAndreas Gohr
278955efc227SAndreas Gohr    /*************************************************************/
27900b17fdc6SAndreas Gohr    function _iptcTagNames() {
279155efc227SAndreas Gohr        $tags = array();
279255efc227SAndreas Gohr        $tags[0x14] = 'SuplementalCategories';
279355efc227SAndreas Gohr        $tags[0x19] = 'Keywords';
279455efc227SAndreas Gohr        $tags[0x78] = 'Caption';
279555efc227SAndreas Gohr        $tags[0x7A] = 'CaptionWriter';
279655efc227SAndreas Gohr        $tags[0x69] = 'Headline';
279755efc227SAndreas Gohr        $tags[0x28] = 'SpecialInstructions';
279855efc227SAndreas Gohr        $tags[0x0F] = 'Category';
279955efc227SAndreas Gohr        $tags[0x50] = 'Byline';
280055efc227SAndreas Gohr        $tags[0x55] = 'BylineTitle';
280155efc227SAndreas Gohr        $tags[0x6E] = 'Credit';
280255efc227SAndreas Gohr        $tags[0x73] = 'Source';
280355efc227SAndreas Gohr        $tags[0x74] = 'CopyrightNotice';
280455efc227SAndreas Gohr        $tags[0x05] = 'ObjectName';
280555efc227SAndreas Gohr        $tags[0x5A] = 'City';
2806493531a8SJoe Lapp        $tags[0x5C] = 'Sublocation';
280755efc227SAndreas Gohr        $tags[0x5F] = 'ProvinceState';
280855efc227SAndreas Gohr        $tags[0x65] = 'CountryName';
280955efc227SAndreas Gohr        $tags[0x67] = 'OriginalTransmissionReference';
281055efc227SAndreas Gohr        $tags[0x37] = 'DateCreated';
281155efc227SAndreas Gohr        $tags[0x0A] = 'CopyrightFlag';
281255efc227SAndreas Gohr
281355efc227SAndreas Gohr        return $tags;
281455efc227SAndreas Gohr    }
281555efc227SAndreas Gohr
281655efc227SAndreas Gohr    /*************************************************************/
28170b17fdc6SAndreas Gohr    function & _iptcNameTags() {
281855efc227SAndreas Gohr        $tags = $this->_iptcTagNames();
281955efc227SAndreas Gohr        return $this->_names2Tags($tags);
282055efc227SAndreas Gohr    }
282155efc227SAndreas Gohr
282255efc227SAndreas Gohr    /*************************************************************/
28230b17fdc6SAndreas Gohr    function _names2Tags($tags2Names) {
282455efc227SAndreas Gohr        $names2Tags = array();
282555efc227SAndreas Gohr        reset($tags2Names);
282655efc227SAndreas Gohr        while (list($tag, $name) = each($tags2Names)) {
282755efc227SAndreas Gohr            $names2Tags[$name] = $tag;
282855efc227SAndreas Gohr        }
282955efc227SAndreas Gohr
283055efc227SAndreas Gohr        return $names2Tags;
283155efc227SAndreas Gohr    }
283255efc227SAndreas Gohr
283355efc227SAndreas Gohr    /*************************************************************/
28340b17fdc6SAndreas Gohr    function _getByte(&$data, $pos) {
283555efc227SAndreas Gohr        return ord($data{$pos});
283655efc227SAndreas Gohr    }
283755efc227SAndreas Gohr
283855efc227SAndreas Gohr    /*************************************************************/
28390b17fdc6SAndreas Gohr    function _putByte(&$data, $pos, $val) {
284055efc227SAndreas Gohr        $val = intval($val);
284155efc227SAndreas Gohr
284255efc227SAndreas Gohr        $data{$pos} = chr($val);
284355efc227SAndreas Gohr
284455efc227SAndreas Gohr        return $pos + 1;
284555efc227SAndreas Gohr    }
284655efc227SAndreas Gohr
284755efc227SAndreas Gohr    /*************************************************************/
28480b17fdc6SAndreas Gohr    function _getShort(&$data, $pos, $bigEndian = true) {
284955efc227SAndreas Gohr        if ($bigEndian) {
285055efc227SAndreas Gohr            return (ord($data{$pos}) << 8)
285155efc227SAndreas Gohr                + ord($data{$pos + 1});
28520b17fdc6SAndreas Gohr        } else {
285355efc227SAndreas Gohr            return ord($data{$pos})
285455efc227SAndreas Gohr                + (ord($data{$pos + 1}) << 8);
285555efc227SAndreas Gohr        }
285655efc227SAndreas Gohr    }
285755efc227SAndreas Gohr
285855efc227SAndreas Gohr    /*************************************************************/
28590b17fdc6SAndreas Gohr    function _putShort(&$data, $pos = 0, $val = 0, $bigEndian = true) {
286055efc227SAndreas Gohr        $val = intval($val);
286155efc227SAndreas Gohr
286255efc227SAndreas Gohr        if ($bigEndian) {
286355efc227SAndreas Gohr            $data{$pos + 0} = chr(($val & 0x0000FF00) >> 8);
286455efc227SAndreas Gohr            $data{$pos + 1} = chr(($val & 0x000000FF) >> 0);
28650b17fdc6SAndreas Gohr        } else {
286655efc227SAndreas Gohr            $data{$pos + 0} = chr(($val & 0x00FF) >> 0);
286755efc227SAndreas Gohr            $data{$pos + 1} = chr(($val & 0xFF00) >> 8);
286855efc227SAndreas Gohr        }
286955efc227SAndreas Gohr
287055efc227SAndreas Gohr        return $pos + 2;
287155efc227SAndreas Gohr    }
287255efc227SAndreas Gohr
287355efc227SAndreas Gohr    /*************************************************************/
28740b17fdc6SAndreas Gohr    function _getLong(&$data, $pos, $bigEndian = true) {
287555efc227SAndreas Gohr        if ($bigEndian) {
287655efc227SAndreas Gohr            return (ord($data{$pos}) << 24)
287755efc227SAndreas Gohr                + (ord($data{$pos + 1}) << 16)
287855efc227SAndreas Gohr                + (ord($data{$pos + 2}) << 8)
287955efc227SAndreas Gohr                + ord($data{$pos + 3});
28800b17fdc6SAndreas Gohr        } else {
288155efc227SAndreas Gohr            return ord($data{$pos})
288255efc227SAndreas Gohr                + (ord($data{$pos + 1}) << 8)
288355efc227SAndreas Gohr                + (ord($data{$pos + 2}) << 16)
288455efc227SAndreas Gohr                + (ord($data{$pos + 3}) << 24);
288555efc227SAndreas Gohr        }
288655efc227SAndreas Gohr    }
288755efc227SAndreas Gohr
288855efc227SAndreas Gohr    /*************************************************************/
28890b17fdc6SAndreas Gohr    function _putLong(&$data, $pos, $val, $bigEndian = true) {
289055efc227SAndreas Gohr        $val = intval($val);
289155efc227SAndreas Gohr
289255efc227SAndreas Gohr        if ($bigEndian) {
289355efc227SAndreas Gohr            $data{$pos + 0} = chr(($val & 0xFF000000) >> 24);
289455efc227SAndreas Gohr            $data{$pos + 1} = chr(($val & 0x00FF0000) >> 16);
289555efc227SAndreas Gohr            $data{$pos + 2} = chr(($val & 0x0000FF00) >> 8);
289655efc227SAndreas Gohr            $data{$pos + 3} = chr(($val & 0x000000FF) >> 0);
28970b17fdc6SAndreas Gohr        } else {
289855efc227SAndreas Gohr            $data{$pos + 0} = chr(($val & 0x000000FF) >> 0);
289955efc227SAndreas Gohr            $data{$pos + 1} = chr(($val & 0x0000FF00) >> 8);
290055efc227SAndreas Gohr            $data{$pos + 2} = chr(($val & 0x00FF0000) >> 16);
290155efc227SAndreas Gohr            $data{$pos + 3} = chr(($val & 0xFF000000) >> 24);
290255efc227SAndreas Gohr        }
290355efc227SAndreas Gohr
290455efc227SAndreas Gohr        return $pos + 4;
290555efc227SAndreas Gohr    }
290655efc227SAndreas Gohr
290755efc227SAndreas Gohr    /*************************************************************/
29080b17fdc6SAndreas Gohr    function & _getNullString(&$data, $pos) {
290955efc227SAndreas Gohr        $str = '';
291055efc227SAndreas Gohr        $max = strlen($data);
291155efc227SAndreas Gohr
291255efc227SAndreas Gohr        while ($pos < $max) {
291355efc227SAndreas Gohr            if (ord($data{$pos}) == 0) {
291455efc227SAndreas Gohr                return $str;
29150b17fdc6SAndreas Gohr            } else {
291655efc227SAndreas Gohr                $str .= $data{$pos};
291755efc227SAndreas Gohr            }
291855efc227SAndreas Gohr            $pos++;
291955efc227SAndreas Gohr        }
292055efc227SAndreas Gohr
292155efc227SAndreas Gohr        return $str;
292255efc227SAndreas Gohr    }
292355efc227SAndreas Gohr
292455efc227SAndreas Gohr    /*************************************************************/
29250b17fdc6SAndreas Gohr    function & _getFixedString(&$data, $pos, $length = -1) {
292655efc227SAndreas Gohr        if ($length == -1) {
292755efc227SAndreas Gohr            $length = strlen($data) - $pos;
292855efc227SAndreas Gohr        }
292955efc227SAndreas Gohr
293055efc227SAndreas Gohr        return substr($data, $pos, $length);
293155efc227SAndreas Gohr    }
293255efc227SAndreas Gohr
293355efc227SAndreas Gohr    /*************************************************************/
29340b17fdc6SAndreas Gohr    function _putString(&$data, $pos, &$str) {
293555efc227SAndreas Gohr        $len = strlen($str);
293655efc227SAndreas Gohr        for ($i = 0; $i < $len; $i++) {
293755efc227SAndreas Gohr            $data{$pos + $i} = $str{$i};
293855efc227SAndreas Gohr        }
293955efc227SAndreas Gohr
294055efc227SAndreas Gohr        return $pos + $len;
294155efc227SAndreas Gohr    }
294255efc227SAndreas Gohr
294355efc227SAndreas Gohr    /*************************************************************/
29440b17fdc6SAndreas Gohr    function _hexDump(&$data, $start = 0, $length = -1) {
294555efc227SAndreas Gohr        if (($length == -1) || (($length + $start) > strlen($data))) {
294655efc227SAndreas Gohr            $end = strlen($data);
29470b17fdc6SAndreas Gohr        } else {
294855efc227SAndreas Gohr            $end = $start + $length;
294955efc227SAndreas Gohr        }
295055efc227SAndreas Gohr
295155efc227SAndreas Gohr        $ascii = '';
295255efc227SAndreas Gohr        $count = 0;
295355efc227SAndreas Gohr
295455efc227SAndreas Gohr        echo "<tt>\n";
295555efc227SAndreas Gohr
295655efc227SAndreas Gohr        while ($start < $end) {
295755efc227SAndreas Gohr            if (($count % 16) == 0) {
295855efc227SAndreas Gohr                echo sprintf('%04d', $count) . ': ';
295955efc227SAndreas Gohr            }
296055efc227SAndreas Gohr
296155efc227SAndreas Gohr            $c = ord($data{$start});
296255efc227SAndreas Gohr            $count++;
296355efc227SAndreas Gohr            $start++;
296455efc227SAndreas Gohr
296555efc227SAndreas Gohr            $aux = dechex($c);
296655efc227SAndreas Gohr            if (strlen($aux) == 1)
296755efc227SAndreas Gohr                echo '0';
296855efc227SAndreas Gohr            echo $aux . ' ';
296955efc227SAndreas Gohr
297055efc227SAndreas Gohr            if ($c == 60)
297155efc227SAndreas Gohr                $ascii .= '&lt;';
297255efc227SAndreas Gohr            elseif ($c == 62)
297355efc227SAndreas Gohr                $ascii .= '&gt;';
297455efc227SAndreas Gohr            elseif ($c == 32)
2975*e260f93bSAnika Henke                $ascii .= '&#160;';
297655efc227SAndreas Gohr            elseif ($c > 32)
297755efc227SAndreas Gohr                $ascii .= chr($c);
297855efc227SAndreas Gohr            else
297955efc227SAndreas Gohr                $ascii .= '.';
298055efc227SAndreas Gohr
298155efc227SAndreas Gohr            if (($count % 4) == 0) {
298255efc227SAndreas Gohr                echo ' - ';
298355efc227SAndreas Gohr            }
298455efc227SAndreas Gohr
298555efc227SAndreas Gohr            if (($count % 16) == 0) {
298655efc227SAndreas Gohr                echo ': ' . $ascii . "<br>\n";
298755efc227SAndreas Gohr                $ascii = '';
298855efc227SAndreas Gohr            }
298955efc227SAndreas Gohr        }
299055efc227SAndreas Gohr
299155efc227SAndreas Gohr        if ($ascii != '') {
299255efc227SAndreas Gohr            while (($count % 16) != 0) {
299355efc227SAndreas Gohr                echo '-- ';
299455efc227SAndreas Gohr                $count++;
299555efc227SAndreas Gohr                if (($count % 4) == 0) {
299655efc227SAndreas Gohr                    echo ' - ';
299755efc227SAndreas Gohr                }
299855efc227SAndreas Gohr            }
299955efc227SAndreas Gohr            echo ': ' . $ascii . "<br>\n";
300055efc227SAndreas Gohr        }
300155efc227SAndreas Gohr
300255efc227SAndreas Gohr        echo "</tt>\n";
300355efc227SAndreas Gohr    }
300455efc227SAndreas Gohr
300555efc227SAndreas Gohr    /*****************************************************************/
300655efc227SAndreas Gohr}
300755efc227SAndreas Gohr
300855efc227SAndreas Gohr/* vim: set expandtab tabstop=4 shiftwidth=4: */
3009