1*e3723e1fSApple OSS Distributions #include <sys/cdefs.h>
2*e3723e1fSApple OSS Distributions #include <sys/proc_internal.h>
3*e3723e1fSApple OSS Distributions #include <sys/vnode_internal.h>
4*e3723e1fSApple OSS Distributions #include <sys/ubc_internal.h>
5*e3723e1fSApple OSS Distributions #include <sys/file_internal.h>
6*e3723e1fSApple OSS Distributions #include <sys/vnode.h>
7*e3723e1fSApple OSS Distributions
8*e3723e1fSApple OSS Distributions // From vm_parameter_validation_kern.c
9*e3723e1fSApple OSS Distributions void testprintf(const char *format, ...);
10*e3723e1fSApple OSS Distributions
11*e3723e1fSApple OSS Distributions struct file_control_return {
12*e3723e1fSApple OSS Distributions void * control;
13*e3723e1fSApple OSS Distributions struct fileproc * fp;
14*e3723e1fSApple OSS Distributions struct vnode * vp;
15*e3723e1fSApple OSS Distributions int fd;
16*e3723e1fSApple OSS Distributions };
17*e3723e1fSApple OSS Distributions struct file_control_return get_control_from_fd(int fd);
18*e3723e1fSApple OSS Distributions
19*e3723e1fSApple OSS Distributions struct file_control_return
get_control_from_fd(int fd)20*e3723e1fSApple OSS Distributions get_control_from_fd(int fd)
21*e3723e1fSApple OSS Distributions {
22*e3723e1fSApple OSS Distributions struct fileproc *fp = NULL;
23*e3723e1fSApple OSS Distributions struct vnode *vp = NULL;
24*e3723e1fSApple OSS Distributions size_t file_size;
25*e3723e1fSApple OSS Distributions off_t fs;
26*e3723e1fSApple OSS Distributions memory_object_control_t file_control = NULL;
27*e3723e1fSApple OSS Distributions int error;
28*e3723e1fSApple OSS Distributions struct file_control_return ret = {NULL, NULL, NULL, fd};
29*e3723e1fSApple OSS Distributions
30*e3723e1fSApple OSS Distributions
31*e3723e1fSApple OSS Distributions proc_t p = current_proc();
32*e3723e1fSApple OSS Distributions
33*e3723e1fSApple OSS Distributions /* get file structure from file descriptor */
34*e3723e1fSApple OSS Distributions error = fp_get_ftype(p, fd, DTYPE_VNODE, EINVAL, &fp);
35*e3723e1fSApple OSS Distributions if (error) {
36*e3723e1fSApple OSS Distributions testprintf("%s: [%d(%s)]: fp_get_ftype() failed, error %d\n",
37*e3723e1fSApple OSS Distributions __func__, proc_getpid(p), p->p_comm, error);
38*e3723e1fSApple OSS Distributions return ret;
39*e3723e1fSApple OSS Distributions }
40*e3723e1fSApple OSS Distributions ret.fp = fp;
41*e3723e1fSApple OSS Distributions
42*e3723e1fSApple OSS Distributions /* We need at least read permission on the file */
43*e3723e1fSApple OSS Distributions if (!(fp->fp_glob->fg_flag & FREAD)) {
44*e3723e1fSApple OSS Distributions testprintf("%s: [%d(%s)]: not readable\n",
45*e3723e1fSApple OSS Distributions __func__, proc_getpid(p), p->p_comm);
46*e3723e1fSApple OSS Distributions return ret;
47*e3723e1fSApple OSS Distributions }
48*e3723e1fSApple OSS Distributions
49*e3723e1fSApple OSS Distributions /* Get the vnode from file structure */
50*e3723e1fSApple OSS Distributions vp = (struct vnode *)fp_get_data(fp);
51*e3723e1fSApple OSS Distributions error = vnode_getwithref(vp);
52*e3723e1fSApple OSS Distributions if (error) {
53*e3723e1fSApple OSS Distributions testprintf("%s: [%d(%s)]: failed to get vnode, error %d\n",
54*e3723e1fSApple OSS Distributions __func__, proc_getpid(p), p->p_comm, error);
55*e3723e1fSApple OSS Distributions return ret;
56*e3723e1fSApple OSS Distributions }
57*e3723e1fSApple OSS Distributions ret.vp = vp;
58*e3723e1fSApple OSS Distributions
59*e3723e1fSApple OSS Distributions /* Make sure the vnode is a regular file */
60*e3723e1fSApple OSS Distributions if (vp->v_type != VREG) {
61*e3723e1fSApple OSS Distributions testprintf("%s: [%d(%s)]: vnode not VREG\n",
62*e3723e1fSApple OSS Distributions __func__, proc_getpid(p), p->p_comm);
63*e3723e1fSApple OSS Distributions return ret;
64*e3723e1fSApple OSS Distributions }
65*e3723e1fSApple OSS Distributions
66*e3723e1fSApple OSS Distributions /* get vnode size */
67*e3723e1fSApple OSS Distributions error = vnode_size(vp, &fs, vfs_context_current());
68*e3723e1fSApple OSS Distributions if (error) {
69*e3723e1fSApple OSS Distributions return ret;
70*e3723e1fSApple OSS Distributions }
71*e3723e1fSApple OSS Distributions file_size = fs;
72*e3723e1fSApple OSS Distributions
73*e3723e1fSApple OSS Distributions /* get the file's memory object handle */
74*e3723e1fSApple OSS Distributions file_control = ubc_getobject(vp, UBC_HOLDOBJECT);
75*e3723e1fSApple OSS Distributions if (file_control == MEMORY_OBJECT_CONTROL_NULL) {
76*e3723e1fSApple OSS Distributions testprintf("%s: [%d(%s)]: no memory object\n",
77*e3723e1fSApple OSS Distributions __func__, proc_getpid(p), p->p_comm);
78*e3723e1fSApple OSS Distributions return ret;
79*e3723e1fSApple OSS Distributions }
80*e3723e1fSApple OSS Distributions ret.control = file_control;
81*e3723e1fSApple OSS Distributions
82*e3723e1fSApple OSS Distributions return ret;
83*e3723e1fSApple OSS Distributions }
84*e3723e1fSApple OSS Distributions
85*e3723e1fSApple OSS Distributions void cleanup_control_related_data(struct file_control_return info);
86*e3723e1fSApple OSS Distributions
87*e3723e1fSApple OSS Distributions void
cleanup_control_related_data(struct file_control_return info)88*e3723e1fSApple OSS Distributions cleanup_control_related_data(struct file_control_return info)
89*e3723e1fSApple OSS Distributions {
90*e3723e1fSApple OSS Distributions if (info.fp != NULL) {
91*e3723e1fSApple OSS Distributions /* release the file descriptor */
92*e3723e1fSApple OSS Distributions fp_drop(current_proc(), info.fd, info.fp, 0);
93*e3723e1fSApple OSS Distributions }
94*e3723e1fSApple OSS Distributions if (info.vp != NULL) {
95*e3723e1fSApple OSS Distributions (void)vnode_put(info.vp);
96*e3723e1fSApple OSS Distributions }
97*e3723e1fSApple OSS Distributions }
98