2007-11-17 04:07:11 +03:00
/*
2005-03-18 16:13:38 +03:00
* Catia VFS module
*
* Implement a fixed mapping of forbidden NT characters in filenames that are
* used a lot by the CAD package Catia .
*
* Yes , this a BAD BAD UGLY INCOMPLETE hack , but it helps quite some people
* out there . Catia V4 on AIX uses characters like " <*$ a *lot*, all forbidden
* under Windows . . .
*
* Copyright ( C ) Volker Lendecke , 2005
*
* This program is free software ; you can redistribute it and / or modify
* it under the terms of the GNU General Public License as published by
2007-07-09 23:25:36 +04:00
* the Free Software Foundation ; either version 3 of the License , or
2005-03-18 16:13:38 +03:00
* ( at your option ) any later version .
2007-11-17 04:07:11 +03:00
*
2005-03-18 16:13:38 +03:00
* This program is distributed in the hope that it will be useful ,
* but WITHOUT ANY WARRANTY ; without even the implied warranty of
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE . See the
* GNU General Public License for more details .
2007-11-17 04:07:11 +03:00
*
2005-03-18 16:13:38 +03:00
* You should have received a copy of the GNU General Public License
2007-07-10 09:23:25 +04:00
* along with this program ; if not , see < http : //www.gnu.org/licenses/>.
2005-03-18 16:13:38 +03:00
*/
# include "includes.h"
2007-11-17 04:07:11 +03:00
static char * catia_string_replace ( TALLOC_CTX * ctx ,
const char * s ,
unsigned char oldc ,
unsigned char newc )
2005-03-18 16:13:38 +03:00
{
2007-11-17 04:07:11 +03:00
smb_ucs2_t * tmpbuf = NULL ;
smb_ucs2_t * ptr = NULL ;
smb_ucs2_t old = oldc ;
char * ret = NULL ;
2008-04-30 01:36:24 +04:00
size_t converted_size ;
2007-11-17 04:07:11 +03:00
if ( ! s ) {
return NULL ;
}
2008-04-30 01:36:24 +04:00
if ( ! push_ucs2_talloc ( ctx , & tmpbuf , s , & converted_size ) ) {
2007-11-17 04:07:11 +03:00
return NULL ;
}
ptr = tmpbuf ;
for ( ; * ptr ; ptr + + ) {
if ( * ptr = = old ) {
* ptr = newc ;
}
}
2008-04-30 01:36:24 +04:00
if ( ! pull_ucs2_talloc ( ctx , & ret , tmpbuf , & converted_size ) ) {
2007-11-17 04:07:11 +03:00
TALLOC_FREE ( tmpbuf ) ;
return NULL ;
}
TALLOC_FREE ( tmpbuf ) ;
return ret ;
2005-03-18 16:13:38 +03:00
}
2007-11-17 04:07:11 +03:00
static char * from_unix ( TALLOC_CTX * ctx , const char * s )
2005-03-18 16:13:38 +03:00
{
2007-11-17 04:07:11 +03:00
char * ret = catia_string_replace ( ctx , s , ' \x22 ' , ' \xa8 ' ) ;
ret = catia_string_replace ( ctx , ret , ' \x2a ' , ' \xa4 ' ) ;
ret = catia_string_replace ( ctx , ret , ' \x2f ' , ' \xf8 ' ) ;
ret = catia_string_replace ( ctx , ret , ' \x3a ' , ' \xf7 ' ) ;
ret = catia_string_replace ( ctx , ret , ' \x3c ' , ' \xab ' ) ;
ret = catia_string_replace ( ctx , ret , ' \x3e ' , ' \xbb ' ) ;
ret = catia_string_replace ( ctx , ret , ' \x3f ' , ' \xbf ' ) ;
ret = catia_string_replace ( ctx , ret , ' \x5c ' , ' \xff ' ) ;
ret = catia_string_replace ( ctx , ret , ' \x7c ' , ' \xa6 ' ) ;
return catia_string_replace ( ctx , ret , ' ' , ' \xb1 ' ) ;
2005-03-18 16:13:38 +03:00
}
2007-11-17 04:07:11 +03:00
static char * to_unix ( TALLOC_CTX * ctx , const char * s )
2005-03-18 16:13:38 +03:00
{
2007-11-17 04:07:11 +03:00
char * ret = catia_string_replace ( ctx , s , ' \xa8 ' , ' \x22 ' ) ;
ret = catia_string_replace ( ctx , ret , ' \xa4 ' , ' \x2a ' ) ;
ret = catia_string_replace ( ctx , ret , ' \xf8 ' , ' \x2f ' ) ;
ret = catia_string_replace ( ctx , ret , ' \xf7 ' , ' \x3a ' ) ;
ret = catia_string_replace ( ctx , ret , ' \xab ' , ' \x3c ' ) ;
ret = catia_string_replace ( ctx , ret , ' \xbb ' , ' \x3e ' ) ;
ret = catia_string_replace ( ctx , ret , ' \xbf ' , ' \x3f ' ) ;
ret = catia_string_replace ( ctx , ret , ' \xff ' , ' \x5c ' ) ;
ret = catia_string_replace ( ctx , ret , ' \xa6 ' , ' \x7c ' ) ;
return catia_string_replace ( ctx , ret , ' \xb1 ' , ' ' ) ;
2005-03-18 16:13:38 +03:00
}
2006-07-11 22:01:26 +04:00
static SMB_STRUCT_DIR * catia_opendir ( vfs_handle_struct * handle ,
const char * fname , const char * mask , uint32 attr )
2005-03-18 16:13:38 +03:00
{
2007-11-17 04:07:11 +03:00
char * name = to_unix ( talloc_tos ( ) , fname ) ;
2005-03-18 16:13:38 +03:00
2007-11-17 04:07:11 +03:00
if ( ! name ) {
errno = ENOMEM ;
return NULL ;
}
2006-07-11 22:01:26 +04:00
return SMB_VFS_NEXT_OPENDIR ( handle , name , mask , attr ) ;
2005-03-18 16:13:38 +03:00
}
2007-11-17 04:07:11 +03:00
static SMB_STRUCT_DIRENT * catia_readdir ( vfs_handle_struct * handle ,
2009-07-19 04:32:44 +04:00
SMB_STRUCT_DIR * dirp ,
SMB_STRUCT_STAT * sbuf )
2005-03-18 16:13:38 +03:00
{
2009-01-23 07:14:38 +03:00
SMB_STRUCT_DIRENT * result = NULL ;
SMB_STRUCT_DIRENT * newdirent = NULL ;
2007-11-17 04:07:11 +03:00
char * newname ;
size_t newnamelen ;
2009-01-23 07:14:38 +03:00
result = SMB_VFS_NEXT_READDIR ( handle , dirp , NULL ) ;
2007-11-17 04:07:11 +03:00
if ( result = = NULL ) {
return result ;
}
newname = from_unix ( talloc_tos ( ) , result - > d_name ) ;
if ( ! newname ) {
return NULL ;
}
newnamelen = strlen ( newname ) + 1 ;
newdirent = ( SMB_STRUCT_DIRENT * ) TALLOC_ARRAY ( talloc_tos ( ) ,
char ,
sizeof ( SMB_STRUCT_DIRENT ) +
newnamelen ) ;
if ( ! newdirent ) {
return NULL ;
}
memcpy ( newdirent , result , sizeof ( SMB_STRUCT_DIRENT ) ) ;
memcpy ( & newdirent - > d_name , newname , newnamelen ) ;
return newdirent ;
2005-03-18 16:13:38 +03:00
}
2006-07-11 22:01:26 +04:00
static int catia_open ( vfs_handle_struct * handle ,
2009-06-16 23:01:13 +04:00
struct smb_filename * smb_fname ,
2007-11-17 04:07:11 +03:00
files_struct * fsp ,
int flags ,
mode_t mode )
2005-03-18 16:13:38 +03:00
{
2009-06-16 23:01:13 +04:00
char * name ;
char * tmp_base_name ;
int ret ;
2005-03-18 16:13:38 +03:00
2009-06-16 23:01:13 +04:00
name = to_unix ( talloc_tos ( ) , smb_fname - > base_name ) ;
2007-11-17 04:07:11 +03:00
if ( ! name ) {
errno = ENOMEM ;
return - 1 ;
}
2009-06-16 23:01:13 +04:00
tmp_base_name = smb_fname - > base_name ;
smb_fname - > base_name = name ;
2009-07-07 22:40:39 +04:00
ret = SMB_VFS_NEXT_OPEN ( handle , smb_fname , fsp , flags , mode ) ;
2009-06-16 23:01:13 +04:00
smb_fname - > base_name = tmp_base_name ;
TALLOC_FREE ( name ) ;
return ret ;
2005-03-18 16:13:38 +03:00
}
2006-07-11 22:01:26 +04:00
static int catia_rename ( vfs_handle_struct * handle ,
2009-07-01 04:04:38 +04:00
const struct smb_filename * smb_fname_src ,
const struct smb_filename * smb_fname_dst )
2005-03-18 16:13:38 +03:00
{
2007-11-17 04:07:11 +03:00
TALLOC_CTX * ctx = talloc_tos ( ) ;
2009-07-01 04:04:38 +04:00
char * oname = NULL ;
char * nname = NULL ;
struct smb_filename * smb_fname_src_tmp = NULL ;
struct smb_filename * smb_fname_dst_tmp = NULL ;
NTSTATUS status ;
int ret = - 1 ;
oname = to_unix ( ctx , smb_fname_src - > base_name ) ;
nname = to_unix ( ctx , smb_fname_dst - > base_name ) ;
2007-11-17 04:07:11 +03:00
if ( ! oname | | ! nname ) {
errno = ENOMEM ;
2009-07-01 04:04:38 +04:00
goto out ;
}
/* Setup temporary smb_filename structs. */
status = copy_smb_filename ( talloc_tos ( ) , smb_fname_src ,
& smb_fname_src_tmp ) ;
if ( ! NT_STATUS_IS_OK ( status ) ) {
errno = map_errno_from_nt_status ( status ) ;
goto out ;
}
status = copy_smb_filename ( talloc_tos ( ) , smb_fname_dst ,
& smb_fname_dst_tmp ) ;
if ( ! NT_STATUS_IS_OK ( status ) ) {
errno = map_errno_from_nt_status ( status ) ;
goto out ;
2007-11-17 04:07:11 +03:00
}
2005-03-18 16:13:38 +03:00
2009-07-01 04:04:38 +04:00
smb_fname_src_tmp - > base_name = oname ;
smb_fname_dst_tmp - > base_name = nname ;
DEBUG ( 10 , ( " converted old name: %s \n " ,
smb_fname_str_dbg ( smb_fname_src_tmp ) ) ) ;
DEBUG ( 10 , ( " converted new name: %s \n " ,
smb_fname_str_dbg ( smb_fname_dst_tmp ) ) ) ;
ret = SMB_VFS_NEXT_RENAME ( handle , smb_fname_src_tmp ,
smb_fname_dst_tmp ) ;
out :
TALLOC_FREE ( oname ) ;
2009-07-07 22:40:39 +04:00
TALLOC_FREE ( nname ) ;
2009-07-01 04:04:38 +04:00
TALLOC_FREE ( smb_fname_src_tmp ) ;
TALLOC_FREE ( smb_fname_dst_tmp ) ;
return ret ;
2005-03-18 16:13:38 +03:00
}
2006-07-11 22:01:26 +04:00
static int catia_stat ( vfs_handle_struct * handle ,
2009-06-23 02:26:56 +04:00
struct smb_filename * smb_fname )
2005-03-18 16:13:38 +03:00
{
2009-06-23 02:26:56 +04:00
char * name ;
char * tmp_base_name ;
int ret ;
2005-03-18 16:13:38 +03:00
2009-06-23 02:26:56 +04:00
name = to_unix ( talloc_tos ( ) , smb_fname - > base_name ) ;
2007-11-17 04:07:11 +03:00
if ( ! name ) {
errno = ENOMEM ;
return - 1 ;
}
2009-06-23 02:26:56 +04:00
tmp_base_name = smb_fname - > base_name ;
smb_fname - > base_name = name ;
ret = SMB_VFS_NEXT_STAT ( handle , smb_fname ) ;
smb_fname - > base_name = tmp_base_name ;
TALLOC_FREE ( name ) ;
return ret ;
2005-03-18 16:13:38 +03:00
}
2006-07-11 22:01:26 +04:00
static int catia_lstat ( vfs_handle_struct * handle ,
2009-06-23 02:26:56 +04:00
struct smb_filename * smb_fname )
2005-03-18 16:13:38 +03:00
{
2009-06-23 02:26:56 +04:00
char * name ;
char * tmp_base_name ;
int ret ;
2005-03-18 16:13:38 +03:00
2009-06-23 02:26:56 +04:00
name = to_unix ( talloc_tos ( ) , smb_fname - > base_name ) ;
2007-11-17 04:07:11 +03:00
if ( ! name ) {
errno = ENOMEM ;
return - 1 ;
}
2009-06-23 02:26:56 +04:00
tmp_base_name = smb_fname - > base_name ;
smb_fname - > base_name = name ;
ret = SMB_VFS_NEXT_LSTAT ( handle , smb_fname ) ;
smb_fname - > base_name = tmp_base_name ;
TALLOC_FREE ( name ) ;
return ret ;
2005-03-18 16:13:38 +03:00
}
2009-07-02 20:27:44 +04:00
static int catia_unlink ( vfs_handle_struct * handle ,
const struct smb_filename * smb_fname )
2005-03-18 16:13:38 +03:00
{
2009-07-02 20:27:44 +04:00
struct smb_filename * smb_fname_tmp = NULL ;
char * name = NULL ;
NTSTATUS status ;
int ret ;
2005-03-18 16:13:38 +03:00
2009-07-02 20:27:44 +04:00
name = to_unix ( talloc_tos ( ) , smb_fname - > base_name ) ;
2007-11-17 04:07:11 +03:00
if ( ! name ) {
errno = ENOMEM ;
return - 1 ;
}
2009-07-02 20:27:44 +04:00
/* Setup temporary smb_filename structs. */
status = copy_smb_filename ( talloc_tos ( ) , smb_fname ,
& smb_fname_tmp ) ;
if ( ! NT_STATUS_IS_OK ( status ) ) {
errno = map_errno_from_nt_status ( status ) ;
return - 1 ;
}
smb_fname_tmp - > base_name = name ;
ret = SMB_VFS_NEXT_UNLINK ( handle , smb_fname_tmp ) ;
TALLOC_FREE ( smb_fname_tmp ) ;
return ret ;
2005-03-18 16:13:38 +03:00
}
2006-07-11 22:01:26 +04:00
static int catia_chmod ( vfs_handle_struct * handle ,
2005-03-18 16:13:38 +03:00
const char * path , mode_t mode )
{
2007-11-17 04:07:11 +03:00
char * name = to_unix ( talloc_tos ( ) , path ) ;
2005-03-18 16:13:38 +03:00
2007-11-17 04:07:11 +03:00
if ( ! name ) {
errno = ENOMEM ;
return - 1 ;
}
2006-07-11 22:01:26 +04:00
return SMB_VFS_NEXT_CHMOD ( handle , name , mode ) ;
2005-03-18 16:13:38 +03:00
}
2006-07-11 22:01:26 +04:00
static int catia_chown ( vfs_handle_struct * handle ,
2005-03-18 16:13:38 +03:00
const char * path , uid_t uid , gid_t gid )
{
2007-11-17 04:07:11 +03:00
char * name = to_unix ( talloc_tos ( ) , path ) ;
2005-03-18 16:13:38 +03:00
2007-11-17 04:07:11 +03:00
if ( ! name ) {
errno = ENOMEM ;
return - 1 ;
}
2006-07-11 22:01:26 +04:00
return SMB_VFS_NEXT_CHOWN ( handle , name , uid , gid ) ;
2005-03-18 16:13:38 +03:00
}
2007-05-24 03:55:12 +04:00
static int catia_lchown ( vfs_handle_struct * handle ,
const char * path , uid_t uid , gid_t gid )
{
2007-11-17 04:07:11 +03:00
char * name = to_unix ( talloc_tos ( ) , path ) ;
2007-05-24 03:55:12 +04:00
2007-11-17 04:07:11 +03:00
if ( ! name ) {
errno = ENOMEM ;
return - 1 ;
}
2007-05-24 03:55:12 +04:00
return SMB_VFS_NEXT_LCHOWN ( handle , name , uid , gid ) ;
}
2006-07-11 22:01:26 +04:00
static int catia_chdir ( vfs_handle_struct * handle ,
2005-03-18 16:13:38 +03:00
const char * path )
{
2007-11-17 04:07:11 +03:00
char * name = to_unix ( talloc_tos ( ) , path ) ;
2005-03-18 16:13:38 +03:00
2007-11-17 04:07:11 +03:00
if ( ! name ) {
errno = ENOMEM ;
return - 1 ;
}
2006-07-11 22:01:26 +04:00
return SMB_VFS_NEXT_CHDIR ( handle , name ) ;
2005-03-18 16:13:38 +03:00
}
/* VFS operations structure */
static vfs_op_tuple catia_op_tuples [ ] = {
/* Directory operations */
2007-11-17 04:07:11 +03:00
{ SMB_VFS_OP ( catia_opendir ) , SMB_VFS_OP_OPENDIR ,
2005-03-18 16:13:38 +03:00
SMB_VFS_LAYER_TRANSPARENT } ,
2007-11-17 04:07:11 +03:00
{ SMB_VFS_OP ( catia_readdir ) , SMB_VFS_OP_READDIR ,
2005-03-18 16:13:38 +03:00
SMB_VFS_LAYER_TRANSPARENT } ,
/* File operations */
2007-11-17 04:07:11 +03:00
{ SMB_VFS_OP ( catia_open ) , SMB_VFS_OP_OPEN ,
2005-03-18 16:13:38 +03:00
SMB_VFS_LAYER_TRANSPARENT } ,
2007-11-17 04:07:11 +03:00
{ SMB_VFS_OP ( catia_rename ) , SMB_VFS_OP_RENAME ,
2005-03-18 16:13:38 +03:00
SMB_VFS_LAYER_TRANSPARENT } ,
2007-11-17 04:07:11 +03:00
{ SMB_VFS_OP ( catia_stat ) , SMB_VFS_OP_STAT ,
2005-03-18 16:13:38 +03:00
SMB_VFS_LAYER_TRANSPARENT } ,
2007-11-17 04:07:11 +03:00
{ SMB_VFS_OP ( catia_lstat ) , SMB_VFS_OP_LSTAT ,
2005-03-18 16:13:38 +03:00
SMB_VFS_LAYER_TRANSPARENT } ,
2007-11-17 04:07:11 +03:00
{ SMB_VFS_OP ( catia_unlink ) , SMB_VFS_OP_UNLINK ,
2005-03-18 16:13:38 +03:00
SMB_VFS_LAYER_TRANSPARENT } ,
2007-11-17 04:07:11 +03:00
{ SMB_VFS_OP ( catia_chmod ) , SMB_VFS_OP_CHMOD ,
2005-03-18 16:13:38 +03:00
SMB_VFS_LAYER_TRANSPARENT } ,
2007-11-17 04:07:11 +03:00
{ SMB_VFS_OP ( catia_chown ) , SMB_VFS_OP_CHOWN ,
2007-05-24 03:55:12 +04:00
SMB_VFS_LAYER_TRANSPARENT } ,
2007-11-17 04:07:11 +03:00
{ SMB_VFS_OP ( catia_lchown ) , SMB_VFS_OP_LCHOWN ,
2005-03-18 16:13:38 +03:00
SMB_VFS_LAYER_TRANSPARENT } ,
2007-11-17 04:07:11 +03:00
{ SMB_VFS_OP ( catia_chdir ) , SMB_VFS_OP_CHDIR ,
2005-03-18 16:13:38 +03:00
SMB_VFS_LAYER_TRANSPARENT } ,
2007-11-17 04:07:11 +03:00
{ NULL , SMB_VFS_OP_NOOP ,
2005-03-18 16:13:38 +03:00
SMB_VFS_LAYER_NOOP }
} ;
2006-12-19 23:16:52 +03:00
NTSTATUS vfs_catia_init ( void ) ;
2006-07-11 22:01:26 +04:00
NTSTATUS vfs_catia_init ( void )
2005-03-18 16:13:38 +03:00
{
2007-11-17 04:07:11 +03:00
return smb_register_vfs ( SMB_VFS_INTERFACE_VERSION , " catia " ,
2005-03-18 16:13:38 +03:00
catia_op_tuples ) ;
}