Understanding lseek


 
Thread Tools Search this Thread
Top Forums UNIX for Dummies Questions & Answers Understanding lseek
# 1  
Old 09-24-2012
Understanding lseek

I tried to use lseek system call to determine the number of bytes in a file. To do so, I used open system call with O_APPEND flag to open a file. As lseek returns the current offset so I called lseek for opened file with offset as zero and whence as SEEK_CUR. So I guess it must return the number of bytes as the file is ready to append and lseek seeks only 0 bytes. But result is showing 0 bytes. Please correct me if I have understood wrong.

Program is as following
Code:
#include <stdlib.h>
#include <stdio.h>
#include <unistd.h>
#include <fnctl.h>

int main(void)
{
   int fd;
   
   if((fd = open("./example.txt", O_RDWR | O_APPEND)) < 0) {
       perror("File open: ");
       exit(1);
   }
   printf("File size in bytes: %ld\n", lseek(fd, 0, SEEK_CUR));
   close(fd);

   exit(0);
}

Program gives the output as zero, though the file 'example.txt' do exist with so data in it.

Thanks.

Last edited by jim mcnamara; 09-24-2012 at 02:43 PM..
# 2  
Old 09-24-2012
File pointers do not necesarily have any relevance to file size. Use fstat() on an open file. lseek() is meant to move file pointers, other uses may not work as you found out.

Code:
#include <sys/stat.h>
size_t filesize(int fd)
{
    struct stat st;
    if(fstat(fd, &st)==-1)
    {
        perror("Cannot stat file");
        exit(1);
    }
    return st.st_size;
}

// usage someplace else in your code

FILE *in=fopen(somefile.dat, "a");
size_t sz=filesize(fileno(in));

# 3  
Old 09-24-2012
Quote:
Originally Posted by Deepak Raj
I tried to use lseek system call to determine the number of bytes in a file. To do so, I used open system call with O_APPEND flag to open a file. As lseek returns the current offset so I called lseek for opened file with offset as zero and whence as SEEK_CUR. So I guess it must return the number of bytes as the file is ready to append and lseek seeks only 0 bytes. But result is showing 0 bytes. Please correct me if I have understood wrong.
Your entire line of reasoning depends on the open() syscall placing the file pointer at the end of the file. That is not a given.

From the POSIX open() system call manual page
Quote:
O_APPEND
If set, the file offset shall be set to the end of the file prior to each write.
Additionally, keep in mind that writes from other processes won't adjust your descriptor's offset.

Regards,
Alister

---------- Post updated at 02:32 PM ---------- Previous update was at 02:18 PM ----------

Quote:
Originally Posted by Deepak Raj
Code:
   printf("File size in bytes: %ld\n", lseek(fd, 0, SEEK_CUR));

That's not a safe use of printf. You're assuming that off_t and long are the same width. I suggest something like:
Code:
printf("lseek return value: %jd\n", (intmax_t) lseek(fd, 0, SEEK_CUR));

Regards,
Alister
# 4  
Old 09-25-2012
Thanks Jim. It's an nice and simple idea for identifying the file size.

Quote:
Originally Posted by jim mcnamara
File pointers do not necesarily have any relevance to file size. Use fstat() on an open file. lseek() is meant to move file pointers, other uses may not work as you found out.

Code:
#include <sys/stat.h>
size_t filesize(int fd)
{
    struct stat st;
    if(fstat(fd, &st)==-1)
    {
        perror("Cannot stat file");
        exit(1);
    }
    return st.st_size;
}

// usage someplace else in your code

FILE *in=fopen(somefile.dat, "a");
size_t sz=filesize(fileno(in));

---------- Post updated at 02:17 PM ---------- Previous update was at 01:19 PM ----------

Thanks Alister

That was helpful. Open syscall used with O_APPEND option doesn't position the offset to end of the file. Rather offset is positioned to the 'seeked' place just before any write operation occurs. This is what I observed with with the following code.

#include <unistd.h>
#include <stdio.h>
#include <fcntl.h>

int main(void)
{
int fd;

if((fd = open("./example.c", O_WRONLY | APPEND)) < 0) {
perror("Open failed: ");
exit(1);
}
printf("Offset after append flag: %jd\n", (intmax_t) lseek(fd, 0, SEEK_CUR));
if(write(fd, "end", 3) != 3)
perror("Write failed: ");
printf("Offset after write: %jd\n", (intmax_t) lseek(fd, 0, SEEK_CUR));
close(fd);

exit(0);
}

And Thanks for correcting me with format specifier %jd and type conversion of lseek's return value. It helps with compatibility issue, that's what I found out. It will be appreciable if you can explain it.

Regards
Deepak

---------- Post updated at 02:24 PM ---------- Previous update was at 02:17 PM ----------

Hi

Okay, now I thought of a way to use lseek to know number of bytes in file. I only need to simply replace whence SEEK_CUR to SEEK_END. I guess, it works because we can't make sure that offset is set to EOF using O_APPEND, but SEEK_END with zero as offset will set it to EOF, surly. Correct me if I'm incorrect.

Regards
Deepak
 
Login or Register to Ask a Question

Previous Thread | Next Thread

9 More Discussions You Might Find Interesting

1. Programming

Problem with lseek call.

The following code: int fd; if (fd = open("mem", O_RDONLY) == -1) return 1; if (lseek(fd, 0, SEEK_SET) == -1) { char *buf = malloc(512); buf = strerror(errno); printf("%s\n", buf); return 1; } always returns with "illegal seek"... (2 Replies)
Discussion started by: Sir_Tomasz
2 Replies

2. AIX

Backup: The lseek call failed

Hi, We are facing issues while backing up our 1205 GB filesystem on LTO5 Tape. During backup the "backup: The lseek call failed." messages were generated, I want to know why these messages were generating AIX version is: 6100-08-00-0000 backup: The date of this level 0 backup is Mon Mar 11... (4 Replies)
Discussion started by: m_raheelahmed
4 Replies

3. Programming

what is the main difference between difference between using nonatomic lseek and O_APPEND

I think both write at the end of the file ...... but is there a sharp difference between those 2 instruction ..... thank you this is my 3rd question today forgive me :D (1 Reply)
Discussion started by: fwrlfo
1 Replies

4. UNIX for Dummies Questions & Answers

Lseek implementation

Hi everybody, i've been googling for ages now and gotten kinda desperate... The question, however, might be rather trivial for the experts: What is it exactly, i.e. physically, the POSIX function (for a file) "lseek" does? Does it trigger some kind of synchronization on disk? Is it just for the... (4 Replies)
Discussion started by: Humudituu
4 Replies

5. UNIX for Dummies Questions & Answers

lseek() equivalent

I know there is lseek() function that will allow to write or read from certain position in the file. Is there similar function that will let do same but for array rather then file? (9 Replies)
Discussion started by: joker40
9 Replies

6. UNIX for Dummies Questions & Answers

understanding {%/*}/

Hi Gurus: I am trying to understand the following line of code.I did enough of googling to understand but no luck.Please help me understand the follow chunk of code: X=$0 MOD=${X%/*}/env.ksh X is the current script from which I am trying to execute. Say if X=test.ksh $MOD is echoing :... (3 Replies)
Discussion started by: vemana
3 Replies

7. Programming

Hairy Problem! lseek over 4G

recently my project needs me to lseek a position over 4G size.... i found in linux or unix the parameters are all ulong 32 bits...the limit dooms the movement of a position over 4G I was told that i should lseek64 to meet my need... but i have no idea where i can get the function neither by "man... (8 Replies)
Discussion started by: macroideal
8 Replies

8. Shell Programming and Scripting

need help understanding mv

I just started shell coding and I'm a bit confused on how 'mv' works can someone explain to me how it works and if i did this correctly. Thanks. echo "Enter Name of the first file:" read file1 #echo $file1 if ; then echo "Sorry, file does not exist." exit 1 ... (16 Replies)
Discussion started by: taiL
16 Replies

9. Programming

lseek in c

sir, i used lssek as this lseek(fp,-10,2); i am not getting any output i dont now why can you explan sir.. Thanks in advance, Arunkumar (4 Replies)
Discussion started by: arunkumar_mca
4 Replies
Login or Register to Ask a Question